44 #define DBG_REPLAY LOG_STREAM(debug, log_replay)
45 #define LOG_REPLAY LOG_STREAM(info, log_replay)
46 #define WRN_REPLAY LOG_STREAM(warn, log_replay)
47 #define ERR_REPLAY LOG_STREAM(err, log_replay)
63 static map* instance =
new map();
76 if ( !from.
valid() ) {
80 spectator.error(
"Missing leader location for recruitment.\n");
84 std::stringstream errbuf;
85 errbuf <<
"Recruiting leader not found at " << from <<
".\n";
86 spectator.error(errbuf.str());
90 std::string type_id = child[
"type"];
91 if ( type_id.empty() ) {
92 spectator.error(
"Recruitment is missing a unit type.");
98 std::stringstream errbuf;
99 errbuf <<
"Recruiting illegal unit: '" << type_id <<
"'.\n";
100 spectator.
error(errbuf.str());
107 std::stringstream errbuf;
108 errbuf <<
"cannot recruit unit: " << res <<
"\n";
109 spectator.error(errbuf.str());
113 const int beginning_gold = current_team.
gold();
117 if ( u_type->
cost() > beginning_gold ) {
118 std::stringstream errbuf;
119 errbuf <<
"unit '" << type_id <<
"' is too expensive to recruit: "
120 << u_type->
cost() <<
"/" << beginning_gold <<
"\n";
121 spectator.error(errbuf.str());
126 LOG_REPLAY <<
"recruit: team=" << current_team_num <<
" '" << type_id <<
"' at (" <<
loc
127 <<
") cost=" << u_type->
cost() <<
" from gold=" << beginning_gold <<
' '
128 <<
"-> " << current_team.
gold();
137 const std::string& unit_id = child[
"value"];
142 spectator.error(
"illegal recall: unit_id '" + unit_id +
"' could not be found within the recall list.\n");
151 const auto destination = child.optional_child(
"destination");
152 const auto source = child.optional_child(
"source");
156 spectator.error(
"no destination found in attack\n");
161 spectator.error(
"no source found in attack \n");
170 int weapon_num = child[
"weapon"].to_int();
176 int def_weapon_num = child[
"defender_weapon"].to_int(-2);
177 if (def_weapon_num == -2) {
179 LOG_REPLAY <<
"Old data, having to guess weapon";
185 spectator.error(
"unfound location for source of attack\n");
189 if (child.has_attribute(
"attacker_type")) {
190 const std::string &att_type_id = child[
"attacker_type"];
191 if (u->type_id() != att_type_id) {
192 WRN_REPLAY <<
"unexpected attacker type: " << att_type_id <<
"(game state gives: " << u->type_id() <<
")";
196 if (
static_cast<unsigned>(weapon_num) >= u->attacks().size()) {
197 spectator.error(
"illegal weapon type in attack\n");
204 std::stringstream errbuf;
205 errbuf <<
"unfound defender for attack: " <<
src <<
" -> " <<
dst <<
'\n';
206 spectator.error(errbuf.str());
210 if (child.has_attribute(
"defender_type")) {
211 const std::string &def_type_id = child[
"defender_type"];
212 if (tgt->type_id() != def_type_id) {
213 WRN_REPLAY <<
"unexpected defender type: " << def_type_id <<
"(game state gives: " << tgt->type_id() <<
")";
217 if (def_weapon_num >=
static_cast<int>(tgt->attacks().size())) {
219 spectator.error(
"illegal defender weapon type in attack\n");
223 DBG_REPLAY <<
"Attacker XP (before attack): " << u->experience();
236 const std::string& unit_id = child[
"value"];
241 if (!dismissed_unit) {
242 spectator.error(
"illegal disband\n");
258 std::vector<map_location> steps;
262 }
catch (
const std::invalid_argument&) {
263 WRN_REPLAY <<
"Warning: Path data contained something which could not be parsed to a sequence of locations:" <<
"\n config = " << child.debug();
269 WRN_REPLAY <<
"Warning: Missing path data found in [move]";
277 WRN_REPLAY <<
"Warning: Move with identical source and destination. Skipping...";
284 WRN_REPLAY <<
"Warning: Move destination " <<
dst <<
" appears occupied.";
292 std::stringstream errbuf;
293 errbuf <<
"unfound location for source of movement: "
294 <<
src <<
" -> " <<
dst <<
'\n';
295 spectator.error(errbuf.str());
299 bool skip_sighted = child[
"skip_sighted"] ==
"all";
300 bool skip_ally_sighted = child[
"skip_sighted"] ==
"only_ally";
309 if(
const auto last_select = child.optional_child(
"last_select"))
314 const std::string &event_name = child[
"raise"];
315 if (
const auto source = child.optional_child(
"source")) {
336 bool active = child[
"active"].to_bool();
337 bool block_undo = child[
"block_undo"].to_bool();
357 spectator.error(
"Team has DSU disabled but we found an explicit shroud update");
368 void debug_notification(
const std::string& text,
bool message_is_command =
false)
371 auto& current_team =
controller.current_team();
372 static bool ignore =
false;
373 bool show_long_message =
controller.is_replay() || !current_team.is_local();
378 if(i18n_vars[
"player"].empty()) {
379 i18n_vars[
"player"] =
_(
"(unknown player)");
382 if(message_is_command) {
383 i18n_vars[
"command"] = text;
384 message =
VGETTEXT(
"The :$command debug command was used during $player’s turn", i18n_vars);
389 if(show_long_message && !ignore) {
391 std::stringstream sbuilder;
392 sbuilder <<
_(
"A player used a debug command during the game. If this is unexpected, it is possible the player in question is cheating.")
394 <<
_(
"Details:") <<
"\n"
397 <<
_(
"Do you wish to save the game before continuing?");
399 save.set_title(
_(
"Debug Command Used"));
408 void debug_cmd_notification(
const std::string& command)
410 debug_notification(command,
true);
417 debug_cmd_notification(
"terrain");
420 const std::string&
terrain_type = child[
"terrain_type"];
421 const std::string& mode_str = child[
"mode_str"];
435 debug_cmd_notification(
"unit");
437 const std::string name = child[
"name"];
438 const std::string value = child[
"value"];
444 if (name ==
"advances" ) {
448 }
catch (
const std::invalid_argument&) {
449 WRN_REPLAY <<
"Warning: Invalid unit advancement argument: " << value;
452 for (
int levels=0; levels<int_value; levels++) {
453 i->set_experience(
i->max_experience());
461 }
else if (name ==
"status" ) {
464 if (status.length() >= 1 && status[0] ==
'-') {
466 status = status.substr(1);
468 if (status.empty()) {
471 i->set_state(status, add);
481 new_u->set_location(
loc);
492 if (name ==
"fail") {
505 debug_notification(
N_(
"A unit was created using debug mode during $player’s turn"));
508 const std::string& variation = child[
"variation"].str();
512 spectator.
error(
"Invalid unit type");
541 if (unit_it.
valid() )
550 debug_cmd_notification(
"lua");
560 debug_cmd_notification(
"teleport");
562 const map_location teleport_from(child[
"teleport_from_x"].to_int(), child[
"teleport_from_y"].to_int(),
wml_loc());
563 const map_location teleport_to(child[
"teleport_to_x"].to_int(), child[
"teleport_to_y"].to_int(),
wml_loc());
567 if(unit_iter.
valid()) {
579 debug_cmd_notification(
"kill");
584 const int dying_side =
i->side();
607 debug_cmd_notification(
"next_level");
609 std::string next_level = child[
"next_level"];
610 if (!next_level.empty())
613 e.transient.carryover_report =
false;
614 e.prescenario_save =
true;
615 e.transient.linger_mode =
false;
616 e.proceed_to_next_level =
true;
629 debug_cmd_notification(
"turn_limit");
640 debug_cmd_notification(
"turn");
654 debug_cmd_notification(
"set_var");
670 debug_cmd_notification(
"gold");
682 debug_cmd_notification(
"throw");
695 debug_cmd_notification(
"fog");
712 debug_cmd_notification(
"shroud");
void attack_unit_and_advance(const map_location &attacker, const map_location &defender, int attack_with, int defend_with, bool update_display)
Performs an attack, and advanced the units afterwards.
Various functions that implement attacks and attack calculations.
Various functions related to moving units.
void advance_unit_at(const advance_unit_params ¶ms)
Various functions that implement advancements of units.
Class to encapsulate fog/shroud clearing and the resultant sighted events.
game_events::pump_result_t fire_events()
Fires the sighted events that were earlier recorded by fog/shroud clearing.
bool clear_unit(const map_location &view_loc, team &view_team, std::size_t viewer_id, int sight_range, bool slowed, const movetype::terrain_costs &costs, const map_location &real_loc, const std::set< map_location > *known_units=nullptr, std::size_t *enemy_count=nullptr, std::size_t *friend_count=nullptr, move_unit_spectator *spectator=nullptr, bool instant=true)
Clears shroud (and fog) around the provided location for view_team based on sight_range,...
void add_dismissal(const unit_const_ptr &u)
Adds a dismissal to the undo stack.
bool commit_vision()
Updates fog/shroud based on the undo stack, then updates stack as needed.
void add_auto_shroud(bool turned_on)
Adds an auto-shroud toggle to the undo stack.
A config object defines a single node in a WML file, with access to child nodes.
void recalculate_minimap()
Schedule the minimap for recalculation.
void redraw_minimap()
Schedule the minimap to be redrawn.
bool invalidate(const map_location &loc)
Function to invalidate a specific tile for redrawing.
void announce(const std::string &msg, const color_t &color=font::GOOD_COLOR, const announce_options &options=announce_options())
Announce a message prominently.
void queue_rerender()
Marks everything for rendering including all tiles and sidebar.
static display * get_singleton()
Returns the display object if a display object exists.
bool change_terrain(const map_location &loc, const std::string &t, const std::string &mode, bool replace_if_failed)
unit_map::iterator find_visible_unit(const map_location &loc, const team ¤t_team, bool see_all=false)
virtual const unit_map & units() const override
void set_next_scenario(const std::string &next_scenario)
void set_variable(const std::string &varname, const config::attribute_value &value)
does nothing if varname is no valid variable name.
void invalidate_unit()
Function to invalidate that unit status displayed on the sidebar.
static game_display * get_singleton()
void new_turn()
Update lighting settings.
void needs_rebuild(bool b)
Sets whether the screen (map visuals) needs to be rebuilt.
bool maybe_rebuild()
Rebuilds the screen if needs_rebuild(true) was previously called, and resets the flag.
game_events::wml_event_pump & pump()
void flush_messages()
Flushes WML messages and errors.
pump_result_t fire(const std::string &event, const entity_location &loc1=entity_location::null_entity, const entity_location &loc2=entity_location::null_entity, const config &data=config())
Function to fire an event.
void custom_command(const std::string &, const config &)
void run(char const *prog, const std::string &name, int nArgs=0)
Runs a plain script.
void set_end_level_data(const end_level_data &data)
int current_side() const
Returns the number of the side whose turn it is.
bool is_skipping_replay() const
virtual void force_end_turn()=0
game_events::wml_event_pump & pump()
std::size_t size() const
Get the number of units on the list.
unit_ptr find_if_matches_id(const std::string &unit_id)
Find a unit by id.
void erase_if_matches_id(const std::string &unit_id)
Erase any unit with this id.
synced_command(const std::string &tag, handler function)
static map & registry()
using static function variable instead of static member variable to prevent static initialization fia...
std::map< std::string, handler > map
static void block_undo(bool do_block=true, bool clear_undo=true)
set this to false to prevent clearing the undo stack, this is important when we cannot change the gam...
This class stores all the data for a single 'side' (in game nomenclature).
void set_shroud(bool shroud)
bool auto_shroud_updates() const
void set_auto_shroud_updates(bool value)
void spend_gold(const int amount)
recall_list_manager & recall_list()
void set_turn(const int num, game_data *vars=nullptr, const bool increase_limit_if_needed=true)
Dynamically change the current turn number.
void set_number_of_turns(int num)
unit_iterator find(std::size_t id)
umap_retval_pair_t replace(const map_location &l, const unit_ptr &p)
Works like unit_map::add; but l is emptied first, if needed.
std::size_t erase(const map_location &l)
Erases the unit at location l, if any.
umap_retval_pair_t insert(const unit_ptr &p)
Inserts the unit pointed to by p into the map.
const unit_type * find(const std::string &key, unit_type::BUILD_STATUS status=unit_type::FULL) const
Finds a unit_type by its id() and makes sure it is built to the specified level.
A single unit type that the player may recruit.
static unit_ptr create(const config &cfg, bool use_traits=false, const vconfig *vcfg=nullptr)
Initializes a unit from a config.
Various functions related to the creation of units (recruits, recalls, and placed units).
static std::string _(const char *str)
std::vector< map_location > read_locations(const config &cfg)
Parse x,y keys of a config into a vector of locations.
Standard logging facilities (interface).
std::string find_recruit_location(const int side, map_location &recruit_location, map_location &recruited_from, const std::string &unit_type)
Finds a location on which to place a unit.
void teleport_unit_from_replay(const std::vector< map_location > &steps, bool continued_move, bool skip_ally_sighted, bool show_move)
Teleports a unit across the board.
bool clear_shroud(int side, bool reset_fog, bool fire_events)
Function that will clear shroud (and fog) based on current unit positions.
bool recall_unit(const std::string &id, team ¤t_team, const map_location &loc, const map_location &from, map_location::direction facing)
Recalls the unit with the indicated ID for the provided team.
game_events::pump_result_t actor_sighted(const unit &target, const std::vector< int > *cache)
Fires sighted events for the sides that can see target.
void recruit_unit(const unit_type &u_type, int side_num, const map_location &loc, const map_location &from)
Recruits a unit of the given type for the given side.
game_events::pump_result_t get_village(const map_location &loc, int side, bool *action_timebonus, bool fire_event)
Makes it so the village at the given location is owned by the given side.
void recalculate_fog(int side)
Function that recalculates the fog of war.
void execute_move_unit(const std::vector< map_location > &steps, bool continued_move, bool skip_ally_sighted, move_unit_spectator *move_spectator)
Moves a unit across the board.
void pump()
Process all events currently in the queue.
const color_t NORMAL_COLOR
void show(const gui2::tracked_drawable &target)
Displays the fps report popup for the given tracked_drawable.
bool fire_event(const ui_event event, const std::vector< std::pair< widget *, ui_event >> &event_chain, widget *dispatcher, widget *w, F &&... params)
Helper function for fire_event.
std::string tag(std::string_view tag, Args &&... data)
Wraps the given data in the specified tag.
::tod_manager * tod_manager
game_events::manager * game_events
actions::undo_list * undo_stack
game_lua_kernel * lua_kernel
play_controller * controller
std::shared_ptr< wb::manager > whiteboard
void unit_die(const map_location &loc, unit &loser, const const_attack_ptr &attack, const const_attack_ptr &secondary_attack, const map_location &winner_loc, const unit_ptr &winner)
Show a unit fading out.
void unit_recruited(const map_location &loc, const map_location &leader_loc)
int stoi(std::string_view str)
Same interface as std::stoi and meant as a drop in replacement, except:
std::map< std::string, t_string > string_map
std::vector< std::string > split(const config_attribute_value &val)
auto * find(Container &container, const Value &value)
Convenience wrapper for using find on a container without needing to comare to end()
std::shared_ptr< unit > unit_ptr
Define the game's event mechanism.
unit_race::GENDER string_gender(const std::string &str, unit_race::GENDER def)
rect dst
Location on the final composed sheet.
rect src
Non-transparent portion of the surface to compose.
advances the unit at loc if it has enough experience, maximum 20 times.
Holds options for calls to function 'announce' (announce).
Additional information on the game outcome which can be provided by WML.
Encapsulates the map of the game.
static lg::log_domain log_replay("replay")
SYNCED_COMMAND_HANDLER_FUNCTION(recruit, child, spectator)
unit_type_data unit_types
Display units performing various actions: moving, attacking, and dying.
Various functions that implement the undoing (and redoing) of in-game commands.