The Battle for Wesnoth  1.19.7+dev
playsingle_controller.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2006 - 2024
3  by Joerg Hinrichs <joerg.hinrichs@alice-dsl.de>
4  Copyright (C) 2003 by David White <dave@whitevine.net>
5  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
6 
7  This program is free software; you can redistribute it and/or modify
8  it under the terms of the GNU General Public License as published by
9  the Free Software Foundation; either version 2 of the License, or
10  (at your option) any later version.
11  This program is distributed in the hope that it will be useful,
12  but WITHOUT ANY WARRANTY.
13 
14  See the COPYING file for more details.
15 */
16 
17 /**
18  * @file
19  * Logic for single-player game.
20  */
21 
23 
24 #include "actions/undo.hpp"
25 #include "ai/manager.hpp"
26 #include "ai/testing.hpp"
27 #include "display_chat_manager.hpp"
28 #include "formula/string_utils.hpp"
29 #include "game_end_exceptions.hpp"
30 #include "game_events/pump.hpp"
31 #include "gettext.hpp"
35 #include "log.hpp"
36 #include "map/label.hpp"
37 #include "map/map.hpp"
39 #include "replay_controller.hpp"
40 #include "replay_helper.hpp"
41 #include "resources.hpp"
42 #include "saved_game.hpp"
43 #include "savegame.hpp"
45 #include "sound.hpp"
46 #include "soundsource.hpp"
47 #include "synced_context.hpp"
48 #include "video.hpp"
50 #include "whiteboard/manager.hpp"
51 
52 #include <thread>
53 
54 static lg::log_domain log_aitesting("ai/testing");
55 #define LOG_AIT LOG_STREAM(info, log_aitesting)
56 // If necessary, this define can be replaced with `#define LOG_AIT std::cout` to restore previous behavior
57 
58 static lg::log_domain log_engine("engine");
59 #define ERR_NG LOG_STREAM(err, log_engine)
60 #define LOG_NG LOG_STREAM(info, log_engine)
61 #define DBG_NG LOG_STREAM(debug, log_engine)
62 
63 static lg::log_domain log_enginerefac("enginerefac");
64 #define LOG_RG LOG_STREAM(info, log_enginerefac)
65 
67  : play_controller(level, state_of_game)
68  , cursor_setter_(cursor::NORMAL)
69  , end_turn_requested_(false)
70  , ai_fallback_(false)
71  , replay_controller_()
72 {
73  // upgrade hotkey handler to the sp (whiteboard enabled) version
74  hotkey_handler_ = std::make_unique<hotkey_handler>(*this, saved_game_);
75 
76 
77  plugins_context_->set_accessor_string("level_result", std::bind(&playsingle_controller::describe_result, this));
78  plugins_context_->set_accessor_int("turn", std::bind(&play_controller::turn, this));
79 }
80 
81 ///Defined here to reduce file includes.
83 
84 
86 {
87  if(!is_regular_game_end()) {
88  return "NONE";
89  } else if(get_end_level_data().is_victory) {
90  return "VICTORY";
91  } else {
92  return "DEFEAT";
93  }
94 }
95 
97 {
98  LOG_NG << "Initializing GUI... " << timer();
99  // If we are retarting replay from linger mode.
102 
103  // Scroll to the starting position of the first team. If there is a
104  // human team, use that team; otherwise use team 1. If the map defines
105  // a starting position for the selected team, scroll to that tile. Note
106  // this often does not matter since many scenario start with messages,
107  // which will usually scroll to the speaker. Also note that the map
108  // does not necessarily define the starting positions. While usually
109  // best to use the map, the scenarion may explicitly set the positions,
110  // overriding those found in the map (if any).
111  if(map_start_.valid()) {
112  gui_->scroll_to_tile(map_start_, game_display::WARP, false);
113  LOG_NG << "Found good stored ui location " << map_start_;
114  } else {
115 
116  int scroll_team = find_viewing_side();
117  if(scroll_team == 0) {
118  scroll_team = 1;
119  }
120 
121  map_location loc(get_map().starting_position(scroll_team));
122  if((loc.x >= 0) && (loc.y >= 0)) {
123  gui_->scroll_to_tile(loc, game_display::WARP);
124  LOG_NG << "Found bad stored ui location " << map_start_ << " using side starting location " << loc;
125  } else {
126  LOG_NG << "Found bad stored ui location";
127  }
128  }
129 
130  // Fade in
131  gui_->set_prevent_draw(false);
132  gui_->queue_repaint();
133  if(!video::headless() && !video::testing()) {
134  gui_->fade_to({0,0,0,0}, std::chrono::milliseconds{500});
135  } else {
136  gui_->set_fade({0,0,0,0});
137  }
138 
140 }
141 
143 {
144  gui_->labels().read(level);
146 
147  // Read sound sources
148  assert(soundsources_manager_ != nullptr);
149  for(const config& s : level.child_range("sound_source")) {
150  try {
152  soundsources_manager_->add(spec);
153  } catch(const bad_lexical_cast&) {
154  ERR_NG << "Error when parsing sound_source config: bad lexical cast.";
155  ERR_NG << "sound_source config was: " << s.debug();
156  ERR_NG << "Skipping this sound source...";
157  }
158  }
159 
160  // At the beginning of the scenario, save a snapshot as replay_start
161  if(saved_game_.replay_start().empty()) {
163  }
164 
165  fire_preload();
167 
168  start_game();
169  gamestate_->player_number_ = skip_empty_sides(gamestate_->player_number_).side_num;
170 
171  if(!get_teams().empty()) {
172  init_side_begin();
173  if(gamestate().in_phase(game_data::TURN_PLAYING)) {
174  init_side_end();
175  }
176  }
177 
179  // This won't cause errors later but we should notify the user about it in case he didn't knew it.
181  // TODO: find a better title
182  _("Game Error"),
183  _("This multiplayer game uses an alternative random mode, if you don’t know what this message means, then "
184  "most likely someone is cheating or someone reloaded a corrupt game."));
185  }
186 }
187 
189 {
190  const int sides = static_cast<int>(get_teams().size());
191  const int max = side_num + sides;
192 
193  for (; side_num != max; ++side_num) {
194  int side_num_mod = modulo(side_num, sides, 1);
195  if(!gamestate().board_.get_team(side_num_mod).is_empty()) {
196  return { side_num_mod, side_num_mod != side_num };
197  }
198  }
199  return { side_num, true };
200 }
201 
203 {
204  //TODO: Its still unclear to me when end_turn_requested_ should be reset, i guess the idea is
205  // in particular that in rare cases when the player looses control at the same time
206  // as he presses "end turn" and then regains control back, the "end turn" should be discarded?
207  //One of the main reasonsy why this is here is probably also that play_controller has no access to it.
209 
211 
213  play_side();
214  assert(is_regular_game_end() || gamestate().in_phase(game_data::TURN_ENDED));
215  }
216 
217  if (!is_regular_game_end() && gamestate().in_phase(game_data::TURN_ENDED)) {
219  }
220 
221  if (is_regular_game_end() && !gamestate().in_phase(game_data::GAME_ENDED)) {
223  do_end_level();
225  }
226 
227  if (gamestate().in_phase(game_data::GAME_ENDED)) {
229  maybe_linger();
230  }
231 }
232 
234 {
235  do {
236  if(std::find_if(get_teams().begin(), get_teams().end(), [](const team& t) { return !t.is_empty(); }) == get_teams().end()){
237  throw game::game_error("The scenario has no (non-empty) sides defined");
238  }
240 
242  if(is_regular_game_end()) {
243  return;
244  }
245  // This flag can be set by derived classes (in overridden functions).
246  player_type_changed_ = false;
247 
248 
249  play_side_impl();
250 
251  if(is_regular_game_end()) {
252  return;
253  }
254  } while(player_type_changed_);
255 
256  // Keep looping if the type of a team (human/ai/networked) has changed mid-turn
257  sync_end_turn();
258 }
259 
261 {
262  if(is_regular_game_end()) {
263  return;
264  }
265 
266  /// Make a copy, since the [end_turn] was already sent to to server any changes to
267  // next_player_number by wml would cause OOS otherwise.
268  int next_player_number_temp = gamestate_->next_player_number_;
269  whiteboard_manager_->on_finish_side_turn(current_side());
270 
272  if(is_regular_game_end()) {
273  return;
274  }
275 
276  auto [next_player_number, new_turn] = skip_empty_sides(next_player_number_temp);
277 
278  if(new_turn) {
279  finish_turn();
280  if(is_regular_game_end()) {
281  return;
282  }
283  // Time has run out
284  check_time_over();
285  if(is_regular_game_end()) {
286  return;
287  }
288  did_tod_sound_this_turn_ = false;
289  }
290 
291  gamestate_->player_number_ = next_player_number;
292  if(current_team().is_empty()) {
293  // We don't support this case (turn end events emptying the next sides controller) since the server cannot handle it.
294  throw game::game_error("Empty side after new turn events");
295  }
296 
297  if(new_turn) {
298  whiteboard_manager_->on_gamestate_change();
299  gui_->new_turn();
300  gui_->invalidate_game_status();
301  }
304  did_autosave_this_turn_ = false;
305  end_turn_requested_ = false;
306  init_side_begin();
307 }
308 
310 {
311  LOG_NG << "starting main loop\n" << timer();
312 
314  while(!(gamestate().in_phase(game_data::GAME_ENDED) && end_turn_requested_ )) {
315  try {
316  play_some();
317  } catch(const reset_gamestate_exception& ex) {
318  boost::dynamic_bitset<> local_players;
319  local_players.resize(get_teams().size(), true);
320  // Preserve side controllers, because we won't get the side controoller updates again when replaying.
321  for(std::size_t i = 0; i < local_players.size(); ++i) {
322  local_players[i] = get_teams()[i].is_local();
323  }
324 
325  if(ex.stats_) {
326  // "Back to turn"
328  } else {
329  // "Reset Replay To start"
331  }
332 
333  reset_gamestate(*ex.level, (*ex.level)["replay_pos"].to_int());
334 
335  for(std::size_t i = 0; i < local_players.size(); ++i) {
336  resources::gameboard->teams()[i].set_local(local_players[i]);
337  }
338 
339  // TODO: we currently don't set the music to the initial playlist, should we?
340 
342 
343  if(replay_controller_ == nullptr) {
344  replay_controller_ = std::make_unique<replay_controller>(*this, false, ex.level, [this]() { on_replay_end(false); });
345  }
346 
347  if(ex.start_replay) {
348  replay_controller_->play_replay();
349  }
350  }
351  } // end for loop
352 }
353 
355 {
357  exit(0);
358  }
359  const bool is_victory = get_end_level_data().is_victory;
360 
362 
363  const end_level_data& end_level = get_end_level_data();
364 
365  if(get_teams().empty()) {
366  // this is probably only a story scenario, i.e. has its endlevel in the prestart event
367  return;
368  }
369 
370 
371  pump().fire(is_victory ? "local_victory" : "local_defeat");
372 
373  { // Block for set_scontext_synced_base
375  pump().fire(end_level.proceed_to_next_level ? level_result::victory : level_result::defeat);
376  pump().fire("scenario_end");
377  }
378 
379  if(end_level.proceed_to_next_level) {
381  }
382 
383  if(is_observer()) {
384  gui2::show_transient_message(_("Game Over"), _("The game is over."));
385  }
386 
387  // If we're a player, and the result is victory/defeat, then send
388  // a message to notify the server of the reason for the game ending.
390  "info", config {
391  "type", "termination",
392  "condition", "game over",
393  "result", is_victory ? level_result::victory : level_result::defeat,
394  },
395  });
396 
397  // Play victory music once all victory events
398  // are finished, if we aren't observers and the
399  // carryover dialog isn't disabled.
400  //
401  // Some scenario authors may use 'continue'
402  // result for something that is not story-wise
403  // a victory, so let them use [music] tags
404  // instead should they want special music.
405  const std::string& end_music = select_music(is_victory);
406  if((!is_victory || end_level.transient.carryover_report) && !end_music.empty()) {
408  sound::play_music_once(end_music);
409  }
410 
412 }
413 
415 {
416  LOG_NG << "in playsingle_controller::play_scenario()...";
417 
418  // Start music.
419  for(const config& m : level.child_range("music")) {
420  sound::play_music_config(m, true);
421  }
422 
424 
426  // Combine all the [story] tags into a single config. Handle this here since
427  // storyscreen::controller doesn't have a default constructor.
428  config cfg;
429  for(const auto& iter : level.child_range("story")) {
430  cfg.append_children(iter);
431  }
432 
433  if(!cfg.empty()) {
435  }
436  }
437 
438 
439  try {
441  // clears level config (the intention was probably just to save some ram),
442  // Note: this might clear 'level', so don't use level after this.
444 
446 
447  // TODO: would it be better if the is_networked_mp() check was done in is_observer() ?
448  if(is_networked_mp() && is_observer()) {
449  return level_result::type::observer_end;
450  }
451  return level_result::get_enum(get_end_level_data().test_result).value_or(get_end_level_data().is_victory ? level_result::type::victory : level_result::type::defeat);
452  } catch(const savegame::load_game_exception&) {
453  // Loading a new game is effectively a quit.
454  saved_game_.clear();
455  throw;
456  } catch(const wesnothd_error& e) {
457  scoped_savegame_snapshot snapshot(*this);
458  savegame::ingame_savegame save(saved_game_, prefs::get().save_compression_format());
459  if(e.message == "") {
461  _("A network disconnection has occurred, and the game cannot continue. Do you want to save the game?"),
463  } else {
465  _("This game has been ended.\nReason: ") + e.message + _("\nDo you want to save the game?"),
467  }
468 
469  if(dynamic_cast<const ingame_wesnothd_error*>(&e) || dynamic_cast<const leavegame_wesnothd_error*>(&e)) {
470  return level_result::type::quit;
471  } else {
472  throw;
473  }
474  }
475 }
476 
478 {
479  while(!should_return_to_play_side()) {
481  using namespace std::chrono_literals;
482  std::this_thread::sleep_for(10ms);
483  }
484 }
485 
487 {
488  if(replay_controller_.get() != nullptr) {
489  replay_controller_->play_side_impl();
490 
492  replay_controller_.reset();
493  }
494  } else if((current_team().is_local_human() && current_team().is_proxy_human())) {
495  LOG_NG << "is human...";
496  // If a side is dead end the turn, but play at least side=1's
497  // turn in case all sides are dead
498  if(gamestate().board_.side_units(current_side()) == 0 && !(get_units().empty() && current_side() == 1)) {
500  }
501 
502 
503  if(!end_turn_requested_) {
505  play_human_turn();
506  }
507 
510  }
511 
512  LOG_NG << "human finished turn...";
513  } else if(current_team().is_local_ai() || (current_team().is_local_human() && current_team().is_droid())) {
514  play_ai_turn();
515  } else if(current_team().is_network()) {
517  } else if(current_team().is_local_human() && current_team().is_idle()) {
518  end_turn_enable(false);
521 
523  play_idle_loop();
524  }
525  } else {
526  // we should have skipped over empty controllers before so this shouldn't be possible
527  ERR_NG << "Found invalid side controller " << side_controller::get_string(current_team().controller()) << " ("
528  << side_proxy_controller::get_string(current_team().proxy_controller()) << ") for side " << current_team().side();
529  }
530 }
531 
533 {
534  log_scope("player turn");
535  assert(!is_linger_mode());
537  return;
538  }
539 
540  if(!did_autosave_this_turn_ && !game_config::disable_autosave && prefs::get().auto_save_max() > 0) {
542  scoped_savegame_snapshot snapshot(*this);
543  savegame::autosave_savegame save(saved_game_, prefs::get().save_compression_format());
545  }
546 
547  if(prefs::get().turn_bell()) {
549  }
550 }
551 
553 {
554  if(prefs::get().turn_dialog() && !is_regular_game_end()) {
555  blindfold b(*gui_, true); // apply a blindfold for the duration of this dialog
556  gui_->queue_rerender();
557  std::string message = _("It is now $name|’s turn");
558  utils::string_map symbols;
559  symbols["name"] = gamestate().board_.get_team(current_side()).side_name();
560  message = utils::interpolate_variables_into_string(message, &symbols);
561  gui2::show_transient_message("", message);
562  }
563 }
564 
566 {
568  return;
569  }
570 
571  try {
573  } catch(const return_to_play_side_exception&) {
574  }
575 }
576 
578 {
580 
581  if(!prefs::get().disable_auto_moves()) {
582  execute_gotos();
583  }
584 
585  end_turn_enable(true);
586 
590  }
591 }
592 
594 {
595  if(is_linger_mode()) {
596  // If we need to set the status depending on the completion state
597  // the key to it is here.
598  gui_->set_game_mode(game_display::LINGER);
599  // change the end-turn button text from "End Turn" to "End Scenario"
600  gui_->get_theme().refresh_title2("button-endturn", "title2");
601 
602  if(get_end_level_data().transient.reveal_map) {
603  // Change the view of all players and observers
604  // to see the whole map regardless of shroud and fog.
605  update_gui_to_player(gui_->viewing_team_index(), true);
606  }
607  } else {
608  gui_->set_game_mode(game_display::RUNNING);
609  // change the end-turn button text from "End Scenario" to "End Turn"
610  gui_->get_theme().refresh_title2("button-endturn", "title");
611  }
612  // Also checcks whether the button can be pressed.
613  gui_->queue_rerender();
614 }
615 
617 {
618  LOG_NG << "beginning end-of-scenario linger";
619 
620  // Make all of the able-to-move units' orbs consistently red
622 
624 
625  if(replay_controller_.get() != nullptr) {
626  replay_controller_->play_side_impl();
628  replay_controller_.reset();
629  }
630  }
631  while(!end_turn_requested_) {
632  play_slice();
633  }
634 
635  LOG_NG << "ending end-of-scenario linger";
636 }
637 
639 {
640  // TODO: this is really only needed to refresh the visual state of the buttons on each turn.
641  // It looks like we can put it in play_side_impl, but it definitely should be removed from
642  // here since other code already takes care of actually enabling/disabling the end turn button.
644 }
645 
647 {
648  // Clear moves from the GUI.
649  gui_->set_route(nullptr);
650  gui_->unhighlight_reach();
651 }
652 
654 {
655  LOG_NG << "is ai...";
656 
657  end_turn_enable(false);
658 
659  const cursor::setter cursor_setter(cursor::WAIT);
660 
661  // Correct an oddball case where a human could have left delayed shroud
662  // updates on before giving control to the AI. (The AI does not bother
663  // with the undo stack, so it cannot delay shroud updates.)
664  team& cur_team = current_team();
665  if(!cur_team.auto_shroud_updates()) {
666  // We just took control, so the undo stack is empty. We still need
667  // to record this change for the replay though.
669  }
670 
671  undo_stack().clear();
672 
673  try {
674  try {
677  }
678  } catch(const return_to_play_side_exception&) {
679  } catch(const fallback_ai_to_human_exception&) {
681  player_type_changed_ = true;
682  ai_fallback_ = true;
683  }
684  } catch(...) {
685  DBG_NG << "Caught exception playing ai turn: " << utils::get_unknown_exception_type();
686  throw;
687  }
688 
691  }
692 }
693 
694 /**
695  * Will handle sending a networked notification in descendent classes.
696  */
698 {
699  gui_->get_chat_manager().add_chat_message(std::time(nullptr), "Wesnoth", 0,
700  "This side is in an idle state. To proceed with the game, the host must assign it to another controller.",
702 }
703 
704 /**
705  * Will handle networked turns in descendent classes.
706  */
708 {
709  // There should be no networked sides in single-player.
710  ERR_NG << "Networked team encountered by playsingle_controller.";
711 }
712 
713 void playsingle_controller::handle_generic_event(const std::string& name)
714 {
715  if(name == "ai_user_interact") {
716  play_slice();
717  }
718 }
719 
721 {
722  if(is_linger_mode()) {
723  end_turn_requested_ = true;
724  } else if(!is_browsing() && menu_handler_.end_turn(current_side())) {
726  }
727 }
728 
730 {
732  end_turn_requested_ = true;
733 }
734 
736 {
737  end_turn_requested_ = true;
738 }
739 
741 {
742  if(!get_teams().empty()) {
743  const team& t = gui_->viewing_team();
744 
745  if(!is_regular_game_end() && !is_browsing() && t.objectives_changed()) {
746  show_objectives();
747  }
748  }
749 }
750 
752 {
753  // mouse_handler expects at least one team for linger mode to work.
754  assert(is_regular_game_end());
755  linger();
756  end_turn_requested_ = true;
757 }
758 
760 {
761  // We cannot add [end_turn] to the recorder while executing another action.
763 
764  if(!gamestate().in_phase(game_data::TURN_ENDED)) {
765  assert(end_turn_requested_);
766  assert(current_team().is_local());
767  assert(gamestate().in_phase(game_data::TURN_PLAYING));
768  // TODO: we should also send this immediately.
769  resources::recorder->end_turn(gamestate_->next_player_number_);
771 
772  }
773 
774 
775  assert(gamestate().in_phase(game_data::TURN_ENDED));
776 
777  if(ai_fallback_) {
778  current_team().make_ai();
779  ai_fallback_ = false;
780  }
781 }
782 
783 bool playsingle_controller::is_team_visible(int team_num, bool observer) const
784 {
785  const team& t = gamestate().board_.get_team(team_num);
786  if(observer) {
787  return !t.get_disallow_observers() && !t.is_empty();
788  } else {
789  return t.is_local_human() && !t.is_idle();
790  }
791 }
792 
794 {
795  const int num_teams = get_teams().size();
796  const bool observer = is_observer();
797 
798  for(int i = 0; i < num_teams; i++) {
799  const int team_num = modulo(current_side() + i, num_teams, 1);
800  if(is_team_visible(team_num, observer)) {
801  return team_num;
802  }
803  }
804 
805  return 0;
806 }
807 
809 {
810  if(replay_controller_ && replay_controller_->is_controlling_view()) {
811  replay_controller_->update_viewing_player();
812  } else if(int side_num = find_viewing_side()) {
813  if(side_num != gui_->viewing_team().side() || gui_->show_everything()) {
814  update_gui_to_player(side_num - 1);
815  }
816  }
817 }
818 
820 {
821  if(replay_controller_ && replay_controller_->allow_reset_replay()) {
822  replay_controller_->stop_replay();
823  throw reset_gamestate_exception(replay_controller_->get_reset_state(), {}, false);
824  } else {
825  ERR_NG << "received invalid reset replay";
826  }
827 }
828 
830 {
831  replay_controller_ = std::make_unique<replay_controller>(
832  *this,
833  true,
834  std::make_shared<config>(saved_game_.get_replay_starting_point()),
835  std::bind(&playsingle_controller::on_replay_end, this, is_unit_test)
836  );
837 
838  if(is_unit_test) {
839  replay_controller_->play_replay();
840  }
841 }
842 
844 {
846  return true;
847  } else if(gamestate().in_phase(game_data::TURN_ENDED)) {
848  return true;
849  } else if((gamestate().in_phase(game_data::TURN_STARTING_WAITING) || end_turn_requested_) && replay_controller_.get() == nullptr && current_team().is_local() && !current_team().is_idle()) {
850  // When we are a locally controlled side and havent done init_side yet also return to play_side
851  return true;
852  } else {
853  return false;
854  }
855 }
856 
858 {
859  if(is_networked_mp()) {
860  // we are using the "Back to turn (replay)" feature
861  // And have reached the current gamestate: end the replay and continue normally.
863  } else if(is_unit_test) {
864  replay_controller_->return_to_play_side();
865  if(!is_regular_game_end()) {
867  e.proceed_to_next_level = false;
868  e.is_victory = false;
870  }
871  } else {
872  replay_controller_->stop_replay();
873  }
874 }
map_location loc
Definition: move.cpp:172
Managing the AIs lifecycle - headers TODO: Refactor history handling and internal commands.
double t
Definition: astarsearch.cpp:63
void clear()
Clears the stack of undoable (and redoable) actions.
Definition: undo.cpp:131
static manager & get_singleton()
Definition: manager.hpp:140
void play_turn(side_number side)
Plays a turn for the specified side using its active AI.
Definition: manager.cpp:731
static void log_game_end()
Definition: testing.cpp:101
static void log_game_start()
Definition: testing.cpp:90
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:172
void append_children(const config &cfg)
Adds children from cfg.
Definition: config.cpp:167
bool empty() const
Definition: config.cpp:849
virtual void play_slice()
void execute_gotos(mouse_handler &mousehandler, int side_num)
bool end_turn(int side_num)
virtual const std::vector< team > & teams() const override
Definition: game_board.hpp:80
void heal_all_survivors()
Definition: game_board.cpp:95
team & get_team(int i)
Definition: game_board.hpp:92
void set_all_units_user_end_turn()
Definition: game_board.cpp:88
void set_end_turn_forced(bool v)
Definition: game_data.hpp:145
void set_phase(PHASE phase)
Definition: game_data.hpp:106
@ GAME_ENDED
The game has ended and the user is observing the final state "lingering" The game can be saved here.
Definition: game_data.hpp:102
@ TURN_PLAYING
The User is controlling the game and invoking actions The game can be saved here.
Definition: game_data.hpp:93
@ TURN_ENDED
The turn_end, side_turn_end etc [events] are fired next phase: TURN_STARTING_WAITING (default),...
Definition: game_data.hpp:96
@ TURN_STARTING_WAITING
we are waiting for the turn to start.
Definition: game_data.hpp:86
@ GAME_ENDING
The victory etc.
Definition: game_data.hpp:99
static PHASE read_phase(const config &cfg)
Definition: game_data.cpp:177
bool end_turn_forced() const
Definition: game_data.hpp:146
@ RUNNING
no linger overlay, show fog and shroud.
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.
Definition: pump.cpp:399
game_board board_
Definition: game_state.hpp:44
game_data gamedata_
Definition: game_state.hpp:43
static void display(const std::string &scenario_name, const config &story)
const std::string & select_music(bool victory) const
config to_config() const
Builds the snapshot config from members and their respective configs.
std::vector< team > & get_teams()
std::unique_ptr< hotkey_handler > hotkey_handler_
virtual void init_gui()
std::unique_ptr< game_state > gamestate_
void show_objectives() const
events::menu_handler menu_handler_
void fire_preload()
preload events cannot be synced
bool is_linger_mode() const
actions::undo_list & undo_stack()
const unit_map & get_units() const
void set_end_level_data(const end_level_data &data)
bool is_observer() const
void reset_gamestate(const config &level, int replay_pos)
bool is_skipping_story() const
bool is_regular_game_end() const
saved_game & get_saved_game()
saved_game & saved_game_
game_state & gamestate()
hotkey::command_executor * get_hotkey_command_executor() override
Optionally get a command executor to handle context menu events.
std::unique_ptr< game_display > gui_
void maybe_do_init_side()
Called by turn_info::process_network_data() or init_side() to call do_init_side() if necessary.
bool is_browsing() const override
virtual void send_to_wesnothd(const config &, const std::string &="unknown") const
std::unique_ptr< soundsource::manager > soundsources_manager_
const end_level_data & get_end_level_data() const
int current_side() const
Returns the number of the side whose turn it is.
virtual bool is_networked_mp() const
bool is_skipping_replay() const
const gamemap & get_map() const
bool did_autosave_this_turn_
Whether we did init sides in this session (false = we did init sides before we reloaded the game).
bool player_type_changed_
true when the controller of the currently playing side has changed.
std::size_t turn() const
map_location map_start_
void update_gui_to_player(const int team_index, const bool observe=false)
Changes the UI for this client to the passed side index.
events::mouse_handler mouse_handler_
const auto & timer() const
std::unique_ptr< plugins_context > plugins_context_
game_events::wml_event_pump & pump()
void finish_side_turn_events()
persist_manager persist_
std::shared_ptr< wb::manager > whiteboard_manager_
virtual void check_time_over()
t_string get_scenario_name() const
void end_turn_enable(bool enable)
virtual void play_network_turn()
Will handle networked turns in descendent classes.
virtual void init_gui() override
void on_replay_end(bool is_unit_test)
std::string describe_result() const
std::unique_ptr< replay_controller > replay_controller_
non-null when replay mode in active, is used in singleplayer and for the "back to turn" feature in mu...
bool ai_fallback_
true when the current side is actually an ai side but was taken over by a human (usually for debuggin...
virtual void check_objectives() override
ses_result skip_empty_sides(int side_num)
Calculates the current side, starting at side_num that is non-empty.
bool is_team_visible(int team_num, bool observer) const
level_result::type play_scenario(const config &level)
virtual bool should_return_to_play_side() const override
~playsingle_controller()
Defined here to reduce file includes.
void update_viewing_player() override
playsingle_controller(const config &level, saved_game &state_of_game)
virtual void handle_generic_event(const std::string &name) override
virtual void do_idle_notification()
Will handle sending a networked notification in descendent classes.
bool end_turn_requested_
true iff the user has pressed the end turn button this turn.
int find_viewing_side() const override
returns 0 if no such team was found.
void play_scenario_init(const config &level)
void enable_replay(bool is_unit_test=false)
static prefs & get()
static config get_auto_shroud(bool turned_on)
Records that the player has toggled automatic shroud updates.
void end_turn(int next_player_number)
Definition: replay.cpp:290
Exception used to escape form the ai or ui code to playsingle_controller::play_side.
game_classification & classification()
Definition: saved_game.hpp:56
void clear()
Definition: saved_game.cpp:814
config & replay_start()
Definition: saved_game.hpp:128
void remove_snapshot()
Definition: saved_game.cpp:606
const config & get_replay_starting_point()
Definition: saved_game.cpp:617
statistics_record::campaign_stats_t & statistics()
Definition: saved_game.hpp:143
Class for autosaves.
Definition: savegame.hpp:281
void autosave(const bool disable_autosave, const int autosave_max, const int infinite_autosaves)
Definition: savegame.cpp:601
Class for "normal" midgame saves.
Definition: savegame.hpp:254
Exception used to signal that the user has decided to abortt a game, and to load another game instead...
Definition: savegame.hpp:85
bool save_game_interactive(const std::string &message, DIALOG_TYPE dialog_type)
Save a game interactively through the savegame dialog.
Definition: savegame.cpp:378
Sound source info class.
static bool run_and_store(const std::string &commandname, const config &data, action_spectator &spectator=get_default_spectator())
This class stores all the data for a single 'side' (in game nomenclature).
Definition: team.hpp:75
const std::string & side_name() const
Definition: team.hpp:293
int side() const
Definition: team.hpp:175
bool auto_shroud_updates() const
Definition: team.hpp:324
void make_ai()
Definition: team.hpp:260
void make_human()
Definition: team.hpp:259
std::size_t i
Definition: function.cpp:1029
Contains the exception interfaces used to signal completion of a scenario, campaign or turn.
static std::string _(const char *str)
Definition: gettext.hpp:93
An extension of play_controller::hotkey_handler, which has support for SP wesnoth features like white...
Standard logging facilities (interface).
#define log_scope(description)
Definition: log.hpp:276
constexpr T modulo(T num, int mod, T min=0)
Definition: math.hpp:62
@ WAIT
Definition: cursor.hpp:28
@ NORMAL
Definition: cursor.hpp:28
std::string observer
std::string turn_bell
bool disable_autosave
Definition: game_config.cpp:91
bool exit_at_end
Definition: game_config.cpp:90
void show_transient_message(const std::string &title, const std::string &message, const std::string &image, const bool message_use_markup, const bool title_use_markup)
Shows a transient message to the user.
const int INFINITE_AUTO_SAVES
Definition: preferences.hpp:59
game_board * gameboard
Definition: resources.cpp:20
replay * recorder
Definition: resources.cpp:28
void empty_playlist()
Definition: sound.cpp:613
void play_music_config(const config &music_node, bool allow_interrupt_current_track, int i)
Definition: sound.cpp:716
void play_music_once(const std::string &file)
Definition: sound.cpp:604
void commit_music_changes()
Definition: sound.cpp:843
void play_bell(const std::string &files)
Definition: sound.cpp:1062
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
Definition: unicode.cpp:85
std::string interpolate_variables_into_string(const std::string &str, const string_map *const symbols)
Function which will interpolate variables, starting with '$' in the string 'str' with the equivalent ...
std::string get_unknown_exception_type()
Utility function for finding the type of thing caught with catch(...).
Definition: general.cpp:23
std::map< std::string, t_string > string_map
bool headless()
The game is running headless.
Definition: video.cpp:139
bool testing()
The game is running unit tests.
Definition: video.cpp:144
static lg::log_domain log_engine("engine")
#define ERR_NG
static lg::log_domain log_aitesting("ai/testing")
static lg::log_domain log_enginerefac("enginerefac")
#define DBG_NG
#define LOG_NG
Define the game's event mechanism.
Thrown when a lexical_cast fails.
Additional information on the game outcome which can be provided by WML.
bool proceed_to_next_level
whether to proceed to the next scenario, equals is_victory in sp.
transient_end_level transient
Error used for any general game error, e.g.
Definition: game_errors.hpp:47
We received invalid data from wesnothd during a game This means we cannot continue with the game but ...
Encapsulates the map of the game.
Definition: location.hpp:45
bool valid() const
Definition: location.hpp:110
std::shared_ptr< config > stats_
std::shared_ptr< config > level
void read(const config &cfg, bool append=false)
void clear_current_scenario()
Delete the current scenario from the stats.
static std::string get_string(enum_type key)
Converts a enum to its string equivalent.
Definition: enum_base.hpp:46
static constexpr utils::optional< enum_type > get_enum(const std::string_view value)
Converts a string into its enum equivalent.
Definition: enum_base.hpp:57
bool carryover_report
Should a summary of the scenario outcome be displayed?
bool linger_mode
Should linger mode be invoked?
An error occurred during when trying to communicate with the wesnothd server.
static map_location::direction s
Gather statistics important for AI testing and output them.
Various functions that implement the undoing (and redoing) of in-game commands.
#define e
#define b