The Battle for Wesnoth  1.19.0-dev
singleplayer.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2008 - 2024
3  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
4 
5  This program is free software; you can redistribute it and/or modify
6  it under the terms of the GNU General Public License as published by
7  the Free Software Foundation; either version 2 of the License, or
8  (at your option) any later version.
9  This program is distributed in the hope that it will be useful,
10  but WITHOUT ANY WARRANTY.
11 
12  See the COPYING file for more details.
13 */
14 
16 
17 #include "config.hpp"
21 #include "game_launcher.hpp"
23 #include "gui/dialogs/message.hpp"
25 #include "gui/widgets/retval.hpp"
26 #include "wml_exception.hpp"
27 
28 static lg::log_domain log_engine("engine");
29 #define ERR_NG LOG_STREAM(err, log_engine)
30 
31 namespace sp
32 {
33 bool select_campaign(saved_game& state, jump_to_campaign_info jump_to_campaign)
34 {
35  while(true) {
36  ng::create_engine create_eng(state);
37  create_eng.set_current_level_type(level_type::type::sp_campaign);
38 
39  const std::vector<ng::create_engine::level_ptr> campaigns =
40  create_eng.get_levels_by_type_unfiltered(level_type::type::sp_campaign);
41 
42  if(campaigns.empty()) {
43  gui2::show_error_message(_("No campaigns are available."));
44  return false;
45  }
46 
47  std::string random_mode = "", difficulty = "";
48 
49  // No campaign selected from command line
50  if(jump_to_campaign.campaign_id.empty()) {
51  gui2::dialogs::campaign_selection dlg(create_eng);
52 
53  try {
54  dlg.show();
55  } catch(const wml_exception& e) {
56  e.show();
57  return false;
58  }
59 
60  if(dlg.get_retval() != gui2::retval::OK) {
61  return false;
62  }
63 
64  switch(dlg.get_rng_mode()) {
66  random_mode = "";
67  break;
69  random_mode = "deterministic";
70  break;
72  random_mode = "biased";
73  break;
74  }
75 
76  difficulty = dlg.get_difficulty();
77  } else {
78  // Don't reset the campaign_id_ so we can know
79  // if we should quit the game or return to the main menu
80 
81  // Checking for valid campaign name
82  const auto campaign = std::find_if(campaigns.begin(), campaigns.end(), [&jump_to_campaign](ng::create_engine::level_ptr level) {
83  return level->data()["id"] == jump_to_campaign.campaign_id;
84  });
85 
86  // Didn't find a campaign with that id
87  if(campaign == campaigns.end()) {
88  ERR_NG << "No such campaign id to jump to: [" << jump_to_campaign.campaign_id << "]";
89  return false;
90  }
91 
92  create_eng.set_current_level(std::distance(campaigns.begin(), campaign));
93  }
94 
95  state.classification().random_mode = random_mode;
96 
97  const auto selected_difficulty = difficulty;
98 
99  if(selected_difficulty == "FAIL") return false;
100  if(selected_difficulty == "CANCEL") {
101  if(!jump_to_campaign.campaign_id.empty()) {
102  jump_to_campaign.campaign_id = "";
103  }
104 
105  // Canceled difficulty dialog, relaunch the campaign selection dialog
106  return select_campaign(state, jump_to_campaign);
107  }
108 
109  create_eng.prepare_for_era_and_mods();
110  create_eng.prepare_for_campaign(selected_difficulty);
111 
112  if(!jump_to_campaign.scenario_id.empty()) {
114  config {"next_scenario", jump_to_campaign.scenario_id}
115  );
116  }
117 
118  if(!state.valid()) {
119  ERR_NG << "Cannot load scenario with id=" << state.get_scenario_id();
120  return false;
121  }
122 
123  if(configure_campaign(state, create_eng)) {
124  break;
125  }
126  }
127 
128  return true;
129 }
130 
132 {
133  // We create the config engine here in order to ensure values like use_map_settings are set correctly
134  // TODO: should this be passed to this function instead of created here?
135  ng::configure_engine config_eng(create_eng.get_state());
136 
137  // TODO: needed?
138  config_eng.update_initial_cfg(create_eng.current_level().data());
139 
140  if(!gui2::dialogs::sp_options_configure::execute(create_eng, config_eng)) {
141  return false;
142  }
143 
144  create_eng.get_parameters();
145  create_eng.prepare_for_new_level();
146 
147  ng::connect_engine connect_eng(state, true, nullptr);
148  connect_eng.start_game();
149 
150  return true;
151 }
152 
153 } // end namespace sp
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:159
This shows the dialog which allows the user to choose which campaign to play.
const std::string & get_difficulty() const
bool show(const unsigned auto_close_time=0)
Shows the window.
int get_retval() const
Returns the cached window exit code.
static bool execute(ng::create_engine &create_engine, ng::configure_engine &config_engine)
Execute function.
void update_initial_cfg(const config &cfg)
std::shared_ptr< level > level_ptr
void set_current_level_type(const level_type::type type)
saved_game & get_state()
void prepare_for_campaign(const std::string &difficulty="")
std::vector< level_ptr > get_levels_by_type_unfiltered(level_type::type type) const
const mp_game_settings & get_parameters()
void set_current_level(const std::size_t index)
void prepare_for_era_and_mods()
level & current_level() const
const config & data() const
game_classification & classification()
Definition: saved_game.hpp:56
void set_carryover_sides_start(config carryover_sides_start)
Definition: saved_game.cpp:163
std::string get_scenario_id() const
Definition: saved_game.cpp:678
bool valid() const
Definition: saved_game.cpp:582
static std::string _(const char *str)
Definition: gettext.hpp:93
void show_error_message(const std::string &msg, bool message_use_markup)
Shows an error message to the user.
Definition: message.cpp:203
@ OK
Dialog was closed with the OK button.
Definition: retval.hpp:35
bool select_campaign(saved_game &state, jump_to_campaign_info jump_to_campaign)
bool configure_campaign(saved_game &state, ng::create_engine &create_eng)
static lg::log_domain log_engine("engine")
#define ERR_NG
std::string campaign_id
The ID of the campaign to launch.
std::string scenario_id
The ID of the scenario within the campaign to jump to.
Helper class, don't construct this directly.
Add a special kind of assert to validate whether the input from WML doesn't contain any problems that...
#define e