/********************************************************************** Freeciv - Copyright (C) 2005 - The Freeciv Project This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. ***********************************************************************/ #ifdef HAVE_CONFIG_H #include #endif #include /* USHRT_MAX */ /* utility */ #include "fcintl.h" #include "hash.h" #include "log.h" #include "shared.h" #include "support.h" /* common */ #include "events.h" #include "game.h" #include "government.h" #include "map.h" #include "movement.h" #include "nation.h" #include "terrain.h" #include "unitlist.h" /* generator */ #include "utilities.h" /* server */ #include "citytools.h" #include "cityturn.h" #include "connecthand.h" #include "gamehand.h" #include "hand_gen.h" #include "maphand.h" #include "plrhand.h" #include "notify.h" #include "srv_main.h" #include "stdinhand.h" #include "techtools.h" #include "unittools.h" #include "edithand.h" /* This table holds pointers to tiles for which expensive * checks (e.g. assign_continent_numbers) have been left * until after a sequence of edits is complete. */ static struct hash_table *unfixed_tile_table; /* Array of size player_slot_count() indexed by player * number to tell whether a given player has fog of war * disabled in edit mode. */ static bool *unfogged_players; /**************************************************************************** Initialize data structures required for edit mode. ****************************************************************************/ void edithand_init(void) { if (unfixed_tile_table != NULL) { hash_free(unfixed_tile_table); } unfixed_tile_table = hash_new(hash_fval_keyval, hash_fcmp_keyval); if (unfogged_players != NULL) { free(unfogged_players); } unfogged_players = fc_calloc(player_slot_count(), sizeof(bool)); } /**************************************************************************** Free all memory used by data structures required for edit mode. ****************************************************************************/ void edithand_free(void) { if (unfixed_tile_table != NULL) { hash_free(unfixed_tile_table); unfixed_tile_table = NULL; } if (unfogged_players != NULL) { free(unfogged_players); unfogged_players = NULL; } } /**************************************************************************** Do the potentially slow checks required after some tile's terrain changes. ****************************************************************************/ static void check_edited_tile_terrains(void) { if (hash_num_entries(unfixed_tile_table) < 1) { return; } hash_keys_iterate(unfixed_tile_table, ptile) { fix_tile_on_terrain_change(ptile, FALSE); } hash_keys_iterate_end; hash_delete_all_entries(unfixed_tile_table); assign_continent_numbers(); send_all_known_tiles(NULL); } /**************************************************************************** Do any necessary checks after leaving edit mode to ensure that the game is in a consistent state. ****************************************************************************/ static void check_leaving_edit_mode(void) { bool unfogged; conn_list_do_buffer(game.est_connections); players_iterate(pplayer) { unfogged = unfogged_players[player_number(pplayer)]; if (unfogged && game.info.fogofwar) { enable_fog_of_war_player(pplayer); } else if (!unfogged && !game.info.fogofwar) { disable_fog_of_war_player(pplayer); } } players_iterate_end; /* Clear the whole array. */ memset(unfogged_players, 0, player_slot_count() * sizeof(bool)); check_edited_tile_terrains(); conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handles a request by the client to enter edit mode. ****************************************************************************/ void handle_edit_mode(struct connection *pc, bool is_edit_mode) { if (!can_conn_enable_editing(pc)) { return; } if (!game.info.is_edit_mode && is_edit_mode) { /* Someone could be cheating! Warn people. */ notify_conn(NULL, NULL, E_SETTING, ftc_editor, _(" *** Server set to edit mode by %s! *** "), conn_description(pc)); } if (game.info.is_edit_mode && !is_edit_mode) { notify_conn(NULL, NULL, E_SETTING, ftc_editor, _(" *** Edit mode cancelled by %s. *** "), conn_description(pc)); check_leaving_edit_mode(); } if (game.info.is_edit_mode != is_edit_mode) { game.info.is_edit_mode = is_edit_mode; /* We make startpos information available to clients only when in edit * mode. */ whole_map_iterate(ptile) { if (map_has_startpos(ptile)) { send_tile_info(NULL, ptile, TRUE); } } whole_map_iterate_end; send_game_info(NULL); } } /**************************************************************************** Base function to edit the terrain property of a tile. Returns TRUE if the terrain has changed. ****************************************************************************/ static bool edit_tile_terrain_handling(struct tile *ptile, struct terrain *pterrain, bool send_tile_info) { struct terrain *old_terrain = tile_terrain(ptile); if (old_terrain == pterrain) { return FALSE; } tile_change_terrain(ptile, pterrain); if (need_to_fix_terrain_change(old_terrain, pterrain)) { hash_insert(unfixed_tile_table, ptile, ptile); } if (send_tile_info) { update_tile_knowledge(ptile); } return TRUE; } /**************************************************************************** Base function to edit the resource property of a tile. Returns TRUE if the resource has changed. ****************************************************************************/ static bool edit_tile_resource_handling(struct tile *ptile, struct resource *presource, bool send_tile_info) { if (presource == tile_resource(ptile)) { return FALSE; } if (NULL != presource && !terrain_has_resource(tile_terrain(ptile), presource)) { return FALSE; } tile_set_resource(ptile, presource); if (send_tile_info) { update_tile_knowledge(ptile); } return TRUE; } /**************************************************************************** Base function to edit the special property of a tile. Returns TRUE if the special state has changed. ****************************************************************************/ static bool edit_tile_special_handling(struct tile *ptile, enum tile_special_type special, bool remove_mode, bool send_tile_info) { if (remove_mode) { if (!tile_has_special(ptile, special)) { return FALSE; } tile_remove_special(ptile, special); } else { if (tile_has_special(ptile, special) || !is_native_tile_to_special(special, ptile)) { return FALSE; } tile_add_special(ptile, special); } if (send_tile_info) { update_tile_knowledge(ptile); } return TRUE; } /**************************************************************************** Base function to edit the base property of a tile. Returns TRUE if the base state has changed. ****************************************************************************/ static bool edit_tile_base_handling(struct tile *ptile, struct base_type *pbase, bool remove_mode, bool send_tile_info) { if (remove_mode) { if (!tile_has_base(ptile, pbase)) { return FALSE; } tile_remove_base(ptile, pbase); } else { if (tile_has_base(ptile, pbase) || !is_native_tile_to_base(pbase, ptile)) { return FALSE; } tile_add_base(ptile, pbase); } if (send_tile_info) { update_tile_knowledge(ptile); } return TRUE; } /**************************************************************************** Handles a client request to change the terrain of the tile at the given x, y coordinates. The 'size' parameter indicates that all tiles in a square of "radius" 'size' should be affected. So size=1 corresponds to the single tile case. ****************************************************************************/ void handle_edit_tile_terrain(struct connection *pc, int tile, Terrain_type_id terrain, int size) { struct terrain *pterrain; struct tile *ptile_center; ptile_center = index_to_tile(tile); if (!ptile_center) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit the tile because %d is not a valid " "tile index on this map!"), tile); return; } pterrain = terrain_by_number(terrain); if (!pterrain) { notify_conn(pc->self, ptile_center, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." the tile because"... */ _("Cannot modify terrain for the tile %s because " "%d is not a valid terrain id."), tile_link(ptile_center), terrain); return; } conn_list_do_buffer(game.est_connections); square_iterate(ptile_center, size - 1, ptile) { edit_tile_terrain_handling(ptile, pterrain, TRUE); } square_iterate_end; conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handle a request to change one or more tiles' resources. ****************************************************************************/ void handle_edit_tile_resource(struct connection *pc, int tile, Resource_type_id resource, int size) { struct resource *presource; struct tile *ptile_center; ptile_center = index_to_tile(tile); if (!ptile_center) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit the tile because %d is not a valid " "tile index on this map!"), tile); return; } presource = resource_by_number(resource); /* May be NULL. */ conn_list_do_buffer(game.est_connections); square_iterate(ptile_center, size - 1, ptile) { edit_tile_resource_handling(ptile, presource, TRUE); } square_iterate_end; conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handle a request to change one or more tiles' specials. The 'remove' argument controls whether to remove or add the given special of type 'special' from the tile. ****************************************************************************/ void handle_edit_tile_special(struct connection *pc, int tile, enum tile_special_type special, bool remove, int size) { struct tile *ptile_center; ptile_center = index_to_tile(tile); if (!ptile_center) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit the tile because %d is not a valid " "tile index on this map!"), tile); return; } if (!(0 <= special && special < S_LAST)) { notify_conn(pc->self, ptile_center, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." the tile because"... */ _("Cannot modify specials for the tile %s because " "%d is not a valid terrain special id."), tile_link(ptile_center), special); return; } conn_list_do_buffer(game.est_connections); square_iterate(ptile_center, size - 1, ptile) { edit_tile_special_handling(ptile, special, remove, TRUE); } square_iterate_end; conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handle a request to change the military base at one or more than one tile. ****************************************************************************/ void handle_edit_tile_base(struct connection *pc, int tile, Base_type_id id, bool remove, int size) { struct tile *ptile_center; struct base_type *pbase; ptile_center = index_to_tile(tile); if (!ptile_center) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit the tile because %d is not a valid " "tile index on this map!"), tile); return; } pbase = base_by_number(id); if (!pbase) { notify_conn(pc->self, ptile_center, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." the tile because"... */ _("Cannot modify base for the tile %s because " "%d is not a valid base type id."), tile_link(ptile_center), id); return; } conn_list_do_buffer(game.est_connections); square_iterate(ptile_center, size - 1, ptile) { edit_tile_base_handling(ptile, pbase, remove, TRUE); } square_iterate_end; conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handles tile information from the client, to make edits to tiles. ****************************************************************************/ void handle_edit_tile(struct connection *pc, struct packet_edit_tile *packet) { struct tile *ptile; bool changed = FALSE; ptile = index_to_tile(packet->tile); if (!ptile) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit the tile because %d is not a valid " "tile index on this map!"), packet->tile); return; } /* Handle changes in specials. */ if (!BV_ARE_EQUAL(packet->specials, ptile->special)) { tile_special_type_iterate(spe) { edit_tile_special_handling(ptile, spe, !BV_ISSET(packet->specials, spe), FALSE); } tile_special_type_iterate_end; changed = TRUE; } /* Handle changes in bases. */ if (!(BV_ARE_EQUAL(packet->bases, ptile->bases))) { base_type_iterate(pbase) { edit_tile_base_handling(ptile, pbase, !BV_ISSET(packet->bases, base_number(pbase)), FALSE); } base_type_iterate_end; changed = TRUE; } /* TODO: Handle more property edits. */ /* Send the new state to all affected. */ if (changed) { update_tile_knowledge(ptile); send_tile_info(NULL, ptile, FALSE); } } /**************************************************************************** Handle a request to create 'count' units of type 'utid' at the tile given by the x, y coordinates and owned by player with number 'owner'. ****************************************************************************/ void handle_edit_unit_create(struct connection *pc, int owner, int tile, Unit_type_id utid, int count, int tag) { struct tile *ptile; struct unit_type *punittype; struct player *pplayer; struct city *homecity; struct unit *punit; bool coastal; int id, i; ptile = index_to_tile(tile); if (!ptile) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot create units because %d is not a valid " "tile index on this map!"), tile); return; } punittype = utype_by_number(utid); if (!punittype) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." at because"... */ _("Cannot create a unit at %s because the " "given unit type id %d is invalid."), tile_link(ptile), utid); return; } pplayer = valid_player_by_number(owner); if (!pplayer) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." type at "... */ _("Cannot create a unit of type %s at %s " "because the given owner's player id %d is " "invalid."), utype_name_translation(punittype), tile_link(ptile), owner); return; } if (is_non_allied_unit_tile(ptile, pplayer) || (tile_city(ptile) && !pplayers_allied(pplayer, city_owner(tile_city(ptile))))) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." type on enemy tile * "... */ _("Cannot create unit of type %s on enemy tile " "%s."), utype_name_translation(punittype), tile_link(ptile)); return; } if (!can_exist_at_tile(punittype, ptile)) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." type on the terrain at * "... */ _("Cannot create a unit of type %s on the terrain " "at %s."), utype_name_translation(punittype), tile_link(ptile)); return; } if (count > 0 && !pplayer->is_alive) { pplayer->is_alive = TRUE; send_player_info(pplayer, NULL); } /* FIXME: Make this more general? */ coastal = is_sailing_unittype(punittype); homecity = find_closest_owned_city(pplayer, ptile, coastal, NULL); id = homecity ? homecity->id : 0; conn_list_do_buffer(game.est_connections); map_show_circle(pplayer, ptile, punittype->vision_radius_sq); for (i = 0; i < count; i++) { /* As far as I can see create_unit is guaranteed to * never return NULL. */ punit = create_unit(pplayer, ptile, punittype, 0, id, -1); if (tag > 0) { dsend_packet_edit_object_created(pc, tag, punit->id); } } conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Remove 'count' units of type 'utid' owned by player number 'owner' at tile (x, y). ****************************************************************************/ void handle_edit_unit_remove(struct connection *pc, int owner, int tile, Unit_type_id utid, int count) { struct tile *ptile; struct unit_type *punittype; struct player *pplayer; int i; ptile = index_to_tile(tile); if (!ptile) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot remove units because %d is not a valid " "tile index on this map!"), tile); return; } punittype = utype_by_number(utid); if (!punittype) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." at because"... */ _("Cannot remove a unit at %s because the " "given unit type id %d is invalid."), tile_link(ptile), utid); return; } pplayer = valid_player_by_number(owner); if (!pplayer) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." type at * because"... */ _("Cannot remove a unit of type %s at %s " "because the given owner's player id %d is " "invalid."), utype_name_translation(punittype), tile_link(ptile), owner); return; } i = 0; unit_list_iterate_safe(ptile->units, punit) { if (i >= count) { break; } if (unit_type(punit) != punittype || unit_owner(punit) != pplayer) { continue; } wipe_unit(punit); i++; } unit_list_iterate_safe_end; } /**************************************************************************** Handle a request to remove a unit given by its id. ****************************************************************************/ void handle_edit_unit_remove_by_id(struct connection *pc, Unit_type_id id) { struct unit *punit; punit = game_find_unit_by_number(id); if (!punit) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No such unit (ID %d)."), id); return; } wipe_unit(punit); } /**************************************************************************** Handles unit information from the client, to make edits to units. ****************************************************************************/ void handle_edit_unit(struct connection *pc, struct packet_edit_unit *packet) { struct tile *ptile; struct unit_type *putype; struct player *pplayer; struct unit *punit; int id; bool changed = FALSE; int moves_left, fuel, hp; id = packet->id; punit = game_find_unit_by_number(id); if (!punit) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No such unit (ID %d)."), id); return; } ptile = unit_tile(punit); putype = unit_type(punit); pplayer = unit_owner(punit); moves_left = CLIP(0, packet->moves_left, putype->move_rate); if (moves_left != punit->moves_left) { punit->moves_left = moves_left; changed = TRUE; } fuel = CLIP(0, packet->fuel, utype_fuel(putype)); if (fuel != punit->fuel) { punit->fuel = fuel; changed = TRUE; } if (packet->moved != punit->moved) { punit->moved = packet->moved; changed = TRUE; } if (packet->done_moving != punit->done_moving) { punit->done_moving = packet->done_moving; changed = TRUE; } hp = CLIP(1, packet->hp, putype->hp); if (hp != punit->hp) { punit->hp = hp; changed = TRUE; } if (packet->veteran != punit->veteran && !unit_has_type_flag(punit, F_NO_VETERAN)) { int v = packet->veteran; if (putype->veteran[v].name[0] == '\0') { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Invalid veteran level %d for unit %d (%s)."), v, id, unit_link(punit)); } else { punit->veteran = v; changed = TRUE; } } /* TODO: Handle more property edits. */ /* Send the new state to all affected. */ if (changed) { send_unit_info(NULL, punit); } } /**************************************************************************** Allows the editing client to create a city at the given position and of size 'size'. ****************************************************************************/ void handle_edit_city_create(struct connection *pc, int owner, int tile, int size, int tag) { struct tile *ptile; struct city *pcity; struct player *pplayer; ptile = index_to_tile(tile); if (!ptile) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot create a city because %d is not a valid " "tile index on this map!"), tile); return; } pplayer = player_by_number(owner); if (!pplayer) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." at because"... */ _("Cannot create a city at %s because the " "given owner's player id %d is invalid"), tile_link(ptile), owner); return; } if (is_enemy_unit_tile(ptile, pplayer) != NULL || !city_can_be_built_here(ptile, NULL)) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." at ." */ _("A city may not be built at %s."), tile_link(ptile)); return; } if (!pplayer->is_alive) { pplayer->is_alive = TRUE; send_player_info(pplayer, NULL); } conn_list_do_buffer(game.est_connections); map_show_tile(pplayer, ptile); create_city(pplayer, ptile, city_name_suggestion(pplayer, ptile)); pcity = tile_city(ptile); if (size > 1) { /* FIXME: Slow and inefficient for large size changes. */ city_change_size(pcity, CLIP(1, size, MAX_CITY_SIZE)); send_city_info(NULL, pcity); } if (tag > 0) { dsend_packet_edit_object_created(pc, tag, pcity->id); } conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Handle a request to change the internal state of a city. ****************************************************************************/ void handle_edit_city(struct connection *pc, struct packet_edit_city *packet) { struct tile *ptile; struct city *pcity, *oldcity; struct player *pplayer; char buf[1024]; int id; bool changed = FALSE; bool need_game_info = FALSE; bv_player need_player_info; pcity = game_find_city_by_number(packet->id); if (!pcity) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit city with invalid city ID %d."), packet->id); return; } pplayer = city_owner(pcity); ptile = city_tile(pcity); BV_CLR_ALL(need_player_info); /* Handle name change. */ if (0 != strcmp(pcity->name, packet->name)) { if (!is_allowed_city_name(pplayer, packet->name, buf, sizeof(buf))) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, _("Cannot edit city name: %s"), buf); } else { sz_strlcpy(pcity->name, packet->name); changed = TRUE; } } /* Handle size change. */ if (packet->size != pcity->size) { if (!(0 < packet->size && packet->size <= MAX_CITY_SIZE)) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, _("Invalid city size %d for city %s."), packet->size, city_link(pcity)); } else { /* FIXME: Slow and inefficient for large size changes. */ city_change_size(pcity, packet->size); changed = TRUE; } } /* Handle city improvement changes. */ improvement_iterate(pimprove) { oldcity = NULL; id = improvement_number(pimprove); if (is_special_improvement(pimprove)) { if (packet->built[id] >= 0) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, _("It is impossible for a city to have %s!"), improvement_name_translation(pimprove)); } continue; } /* FIXME: game.info.great_wonder_owners and pplayer->wonders * logic duplication with city_build_building. */ if (city_has_building(pcity, pimprove) && packet->built[id] < 0) { city_remove_improvement(pcity, pimprove); changed = TRUE; } else if (!city_has_building(pcity, pimprove) && packet->built[id] >= 0) { if (is_great_wonder(pimprove)) { oldcity = find_city_from_great_wonder(pimprove); if (oldcity != pcity) { BV_SET(need_player_info, player_index(pplayer)); } if (NULL != oldcity && city_owner(oldcity) != pplayer) { /* Great wonders make more changes. */ need_game_info = TRUE; BV_SET(need_player_info, player_index(city_owner(oldcity))); } } else if (is_small_wonder(pimprove)) { oldcity = find_city_from_small_wonder(pplayer, pimprove); if (oldcity != pcity) { BV_SET(need_player_info, player_index(pplayer)); } } if (oldcity) { city_remove_improvement(oldcity, pimprove); city_refresh_queue_add(oldcity); } city_add_improvement(pcity, pimprove); changed = TRUE; } } improvement_iterate_end; /* Handle food stock change. */ if (packet->food_stock != pcity->food_stock) { int max = city_granary_size(pcity->size); if (!(0 <= packet->food_stock && packet->food_stock <= max)) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, _("Invalid city food stock amount %d for city %s " "(allowed range is %d to %d)."), packet->food_stock, city_link(pcity), 0, max); } else { pcity->food_stock = packet->food_stock; changed = TRUE; } } /* Handle shield stock change. */ if (packet->shield_stock != pcity->shield_stock) { int max = USHRT_MAX; /* Limited to uint16 by city info packet. */ if (!(0 <= packet->shield_stock && packet->shield_stock <= max)) { notify_conn(pc->self, ptile, E_BAD_COMMAND, ftc_editor, _("Invalid city shield stock amount %d for city %s " "(allowed range is %d to %d)."), packet->shield_stock, city_link(pcity), 0, max); } else { pcity->shield_stock = packet->shield_stock; changed = TRUE; } } /* TODO: Handle more property edits. */ if (changed) { city_refresh_queue_add(pcity); conn_list_do_buffer(game.est_connections); city_refresh_queue_processing(); /* FIXME: city_refresh_queue_processing only sends to city owner? */ send_city_info(NULL, pcity); conn_list_do_unbuffer(game.est_connections); } /* Update wonder infos. */ if (need_game_info) { send_game_info(NULL); } if (BV_ISSET_ANY(need_player_info)) { players_iterate(aplayer) { if (BV_ISSET(need_player_info, player_index(aplayer))) { /* No need to send to detached connections. */ send_player_info(aplayer, NULL); } } players_iterate_end; } } /**************************************************************************** Handle a request to create a new player. ****************************************************************************/ void handle_edit_player_create(struct connection *pc, int tag) { struct player *pplayer; struct nation_type *pnation; if (player_count() >= player_slot_count()) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No more players can be added because the maximum " "number of players (%d) has been reached."), player_slot_count()); return; } if (player_count() >= nation_count() ) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No more players can be added because there are " "no available nations (%d used)."), nation_count()); return; } pnation = pick_a_nation(NULL, TRUE, TRUE, NOT_A_BARBARIAN); if (!pnation) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Player cannot be created because random nation " "selection failed.")); return; } pplayer = server_create_player(); if (!pplayer) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Player creation failed.")); return; } player_map_free(pplayer); server_player_init(pplayer, TRUE, TRUE); player_set_nation(pplayer, pnation); pick_random_player_name(pnation, pplayer->name); sz_strlcpy(pplayer->username, ANON_USER_NAME); pplayer->is_connected = FALSE; pplayer->government = pnation->init_government; pplayer->capital = FALSE; pplayer->economic.gold = 0; pplayer->economic = player_limit_to_max_rates(pplayer); init_tech(pplayer, TRUE); give_global_initial_techs(pplayer); give_nation_initial_techs(pplayer); send_player_info(pplayer, NULL); if (tag > 0) { dsend_packet_edit_object_created(pc, tag, player_number(pplayer)); } } /**************************************************************************** Handle a request to remove a player. ****************************************************************************/ void handle_edit_player_remove(struct connection *pc, int id) { struct player *pplayer; pplayer = valid_player_by_number(id); if (pplayer == NULL) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No such player (ID %d)."), id); return; } /* Don't use conn_list_iterate here because connection_detach() can be * recursive and free the next connection pointer. */ while (conn_list_size(pplayer->connections) > 0) { connection_detach(conn_list_get(pplayer->connections, 0)); } kill_player(pplayer); whole_map_iterate(ptile) { map_clear_known(ptile, pplayer); } whole_map_iterate_end; server_remove_player(pplayer); send_player_slot_info_c(pplayer, NULL); } /************************************************************************** Handle editing of any or all player properties. ***************************************************************************/ void handle_edit_player(struct connection *pc, struct packet_edit_player *packet) { struct player *pplayer; bool changed = FALSE, update_research = FALSE; struct nation_type *pnation; struct player_research *research; enum tech_state known; pplayer = valid_player_by_number(packet->id); if (!pplayer) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit player with invalid player ID %d."), packet->id); return; } research = get_player_research(pplayer); /* Handle player name change. */ if (0 != strcmp(packet->name, player_name(pplayer))) { if (packet->name[0] == '\0') { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot set empty name for player (%d) '%s'."), player_number(pplayer), player_name(pplayer)); } else { bool valid = TRUE; players_iterate(other_player) { if (other_player == pplayer) { continue; } if (0 != fc_strcasecmp(player_name(other_player), packet->name)) { continue; } notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot change name of player (%d) '%s' to '%s': " "another player (%d) already has that name."), player_number(pplayer), player_name(pplayer), packet->name, player_number(other_player)); valid = FALSE; break; } players_iterate_end; if (valid) { sz_strlcpy(pplayer->name, packet->name); changed = TRUE; } } } /* Handle nation change. */ pnation = nation_by_number(packet->nation); if (nation_of_player(pplayer) != pnation) { if (pnation == NULL) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot change nation for player %d (%s) " "because the given nation ID %d is invalid."), player_number(pplayer), player_name(pplayer), packet->nation); } else if (pnation->player != NULL) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot change nation for player %d (%s) " "to nation %d (%s) because that nation is " "already assigned to player %d (%s)."), player_number(pplayer), player_name(pplayer), packet->nation, nation_plural_translation(pnation), player_number(pnation->player), player_name(pnation->player)); } else { changed = player_set_nation(pplayer, pnation); } } /* Handle a change in known inventions. */ /* FIXME: Modifies struct player_research directly. */ advance_index_iterate(A_FIRST, tech) { known = player_invention_state(pplayer, tech); if ((packet->inventions[tech] && known == TECH_KNOWN) || (!packet->inventions[tech] && known != TECH_KNOWN)) { continue; } if (packet->inventions[tech]) { /* FIXME: Side-effect modifies game.info.global_advances. */ player_invention_set(pplayer, tech, TECH_KNOWN); research->techs_researched++; } else { player_invention_set(pplayer, tech, TECH_UNKNOWN); research->techs_researched--; } changed = TRUE; update_research = TRUE; } advance_index_iterate_end; /* Handle a change in the player's gold. */ if (packet->gold != pplayer->economic.gold) { if (!(0 <= packet->gold && packet->gold <= 1000000)) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot set gold for player %d (%s) because " "the value %d is outside the allowed range."), player_number(pplayer), player_name(pplayer), packet->gold); } else { pplayer->economic.gold = packet->gold; changed = TRUE; } } /* TODO: Handle more property edits. */ if (update_research) { Tech_type_id current, goal; player_research_update(pplayer); /* FIXME: Modifies struct player_research directly. */ current = research->researching; goal = research->tech_goal; if (current != A_UNSET) { known = player_invention_state(pplayer, current); if (known != TECH_PREREQS_KNOWN) { research->researching = A_UNSET; } } if (goal != A_UNSET) { known = player_invention_state(pplayer, goal); if (known == TECH_KNOWN) { research->tech_goal = A_UNSET; } } changed = TRUE; /* Inform everybody about global advances */ send_game_info(NULL); } if (changed) { send_player_info(pplayer, NULL); } } /**************************************************************************** Handles vision editing requests from client. ****************************************************************************/ void handle_edit_player_vision(struct connection *pc, int plr_no, int tile, bool known, int size) { struct player *pplayer; struct tile *ptile_center; ptile_center = index_to_tile(tile); if (!ptile_center) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot edit vision because %d is not a valid " "tile index on this map!"), tile); return; } pplayer = valid_player_by_number(plr_no); if (!pplayer) { notify_conn(pc->self, ptile_center, E_BAD_COMMAND, ftc_editor, /* TRANS: ..." at because"... */ _("Cannot edit vision for the tile at %s because " "given player id %d is invalid."), tile_link(ptile_center), plr_no); return; } conn_list_do_buffer(game.est_connections); square_iterate(ptile_center, size - 1, ptile) { if (known && map_is_known(ptile, pplayer)) { continue; } if (!known) { struct city *pcity = tile_city(ptile); bool cannot_make_unknown = FALSE; if (pcity && city_owner(pcity) == pplayer) { continue; } unit_list_iterate(ptile->units, punit) { if (unit_owner(punit) == pplayer || really_gives_vision(pplayer, unit_owner(punit))) { cannot_make_unknown = TRUE; break; } } unit_list_iterate_end; if (cannot_make_unknown) { continue; } /* The client expects tiles which become unseen to * contain no units (client/packhand.c +2368). * So here we tell it to remove units that do * not give it vision. */ unit_list_iterate(ptile->units, punit) { conn_list_iterate(pplayer->connections, pconn) { dsend_packet_unit_remove(pconn, punit->id); } conn_list_iterate_end; } unit_list_iterate_end; } if (known) { map_show_tile(pplayer, ptile); } else { map_hide_tile(pplayer, ptile); } } square_iterate_end; conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Client editor requests us to recalculate borders. Note that this does not necessarily extend borders to their maximum due to the way the borders code is written. This may be considered a feature or limitation. ****************************************************************************/ void handle_edit_recalculate_borders(struct connection *pc) { map_calculate_borders(); } /**************************************************************************** Remove any city at the given location. ****************************************************************************/ void handle_edit_city_remove(struct connection *pc, int id) { struct city *pcity; pcity = game_find_city_by_number(id); if (pcity == NULL) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No such city (ID %d)."), id); return; } remove_city(pcity); } /**************************************************************************** Run any pending tile checks. ****************************************************************************/ void handle_edit_check_tiles(struct connection *pc) { check_edited_tile_terrains(); } /**************************************************************************** Temporarily remove fog-of-war for the player with player number 'plr_no'. This will only stay in effect while the server is in edit mode and the connection is editing. Has no effect if fog-of-war is disabled globally. ****************************************************************************/ void handle_edit_toggle_fogofwar(struct connection *pc, int plr_no) { struct player *pplayer; if (!game.info.fogofwar) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot toggle fog-of-war when it is already " "disabled.")); return; } pplayer = valid_player_by_number(plr_no); if (!pplayer) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot toggle fog-of-war for invalid player ID %d."), plr_no); return; } conn_list_do_buffer(game.est_connections); if (unfogged_players[player_number(pplayer)]) { enable_fog_of_war_player(pplayer); unfogged_players[player_number(pplayer)] = FALSE; } else { disable_fog_of_war_player(pplayer); unfogged_players[player_number(pplayer)] = TRUE; } conn_list_do_unbuffer(game.est_connections); } /**************************************************************************** Set the given position to be the start position for the given nation. ****************************************************************************/ void handle_edit_startpos(struct connection *pc, int tile, Nation_type_id nation) { struct tile *ptile; const struct nation_type *pnation, *old; bool removed = FALSE; ptile = index_to_tile(tile); if (!ptile) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot place a start position %d is not a valid " "tile index on this map!"), tile); return; } old = map_get_startpos(ptile); if (nation == NATION_NONE) { if (map_has_startpos(ptile)) { map_clear_startpos(ptile); removed = TRUE; } pnation = NULL; } else if (nation == NATION_ANY) { map_set_startpos(ptile, NULL); pnation = NULL; } else { pnation = nation_by_number(nation); map_set_startpos(ptile, pnation); } if (old != pnation || removed) { send_tile_info(NULL, ptile, FALSE); } } /**************************************************************************** Handle edit requests to the main game data structure. ****************************************************************************/ void handle_edit_game(struct connection *pc, struct packet_edit_game *packet) { bool changed = FALSE; if (packet->year != game.info.year) { /* 'year' is stored in a signed short. */ const short min_year = -30000, max_year = 30000; if (!(min_year <= packet->year && packet->year <= max_year)) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Cannot set invalid game year %d. Valid year range " "is from %d to %d."), packet->year, min_year, max_year); } else { game.info.year = packet->year; changed = TRUE; } } if (packet->scenario != game.scenario.is_scenario) { game.scenario.is_scenario = packet->scenario; changed = TRUE; } if (0 != strncmp(packet->scenario_name, game.scenario.name, 256)) { sz_strlcpy(game.scenario.name, packet->scenario_name); changed = TRUE; } if (0 != strncmp(packet->scenario_desc, game.scenario.description, MAX_LEN_PACKET)) { sz_strlcpy(game.scenario.description, packet->scenario_desc); changed = TRUE; } if (packet->scenario_players != game.scenario.players) { game.scenario.players = packet->scenario_players; changed = TRUE; } if (changed) { send_scenario_info(NULL); send_game_info(NULL); } } /**************************************************************************** Make scenario file out of current game. ****************************************************************************/ void handle_save_scenario(struct connection *pc, char *name) { if (pc->access_level != ALLOW_HACK) { notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("No permissions to remotely save scenario.")); return; } if (!game.scenario.is_scenario) { /* Scenario information not available */ notify_conn(pc->self, NULL, E_BAD_COMMAND, ftc_editor, _("Scenario information not set. Cannot save scenario.")); return; } save_game(name, "Scenario", TRUE); } /**************************************************************************** Handle scenario information packet ****************************************************************************/ void handle_scenario_info(struct connection *pc, struct packet_scenario_info *packet) { game.scenario.is_scenario = packet->is_scenario; sz_strlcpy(game.scenario.name, packet->name); sz_strlcpy(game.scenario.description, packet->description); game.scenario.players = packet->players; /* Send new info to everybody. */ send_scenario_info(NULL); }