15 #define GETTEXT_DOMAIN "wesnoth-lib"
49 #define DBG_MP LOG_STREAM(debug, log_mp_connect_engine)
50 #define LOG_MP LOG_STREAM(info, log_mp_connect_engine)
51 #define WRN_MP LOG_STREAM(warn, log_mp_connect_engine)
52 #define ERR_MP LOG_STREAM(err, log_mp_connect_engine)
63 , network_connection_(connection)
65 , first_scenario_(first_scenario)
66 , observe_game_(observe_game)
67 , stop_updates_(false)
68 , player_list_(
nullptr)
69 , flg_dialog_(
nullptr)
71 set_show_even_without_video(
true);
72 set_allow_plugin_skip(
false);
93 bool has_scenario_and_controllers =
false;
94 while(!has_scenario_and_controllers) {
105 }
else if(revc.
has_child(
"leave_game")) {
112 has_scenario_and_controllers =
true;
113 }
else if(
auto controllers = revc.
optional_child(
"controllers")) {
122 has_scenario_and_controllers =
true;
126 if(
level_[
"started"].to_bool()) {
139 if(scenario_id ==
"") {
141 for(
const auto& content : addon.child_range(
"content")) {
142 if(content[
"type"] ==
"scenario") {
143 scenario_id = content[
"id"].str();
161 const config* side_choice =
nullptr;
163 int side_num_choice = 1, side_num_counter = 1;
167 if(side[
"controller"] == side_controller::reserved && side[
"current_player"] ==
prefs::get().login()) {
169 side_num_choice = side_num_counter;
173 if(side[
"controller"] == side_controller::human && side[
"player_id"].empty()) {
176 side_num_choice = side_num_counter;
179 if(side[
"current_player"] ==
prefs::get().login()) {
181 side_num_choice = side_num_counter;
186 if(side[
"player_id"] ==
prefs::get().login()) {
200 if((*side_choice)[
"allow_changes"].to_bool(
true)) {
216 const std::string controller_type = side[
"controller"].str();
217 const std::string reservation = side[
"current_player"].str();
218 const std::string owner = side[
"player_id"].str();
220 if(controller_type == side_controller::ai) {
221 return _(
"Computer Player");
222 }
else if(controller_type == side_controller::none) {
223 return _(
"Empty slot");
224 }
else if(controller_type == side_controller::reserved) {
225 return VGETTEXT(
"Reserved for $playername", {{
"playername", reservation}});
226 }
else if(owner.empty()) {
227 return _(
"Vacant slot");
228 }
else if(controller_type == side_controller::human) {
243 label& title = find_widget<label>(
"title");
255 chatbox& chat = find_widget<chatbox>(
"chat");
283 if(
auto side_choice =
get_scenario().optional_child(
"side", side_num - 1)) {
284 if(!side_choice[
"allow_changes"].to_bool(
true)) {
290 ERR_MP <<
"no era information";
295 if(possible_sides.empty()) {
296 WRN_MP <<
"no [multiplayer_side] found in era '" << era[
"id"] <<
"'.";
300 const std::string color = side_choice[
"color"].str();
302 std::vector<const config*> era_factions;
305 for(
const config& side : possible_sides) {
307 era_factions.push_back(&side_new);
311 const bool lock_settings =
get_scenario()[
"force_lock_settings"].to_bool(!is_mp);
312 const bool use_map_settings =
level_.
mandatory_child(
"multiplayer")[
"mp_use_map_settings"].to_bool();
315 ng::flg_manager flg(era_factions, *side_choice, lock_settings, use_map_settings,
saved_game == saved_game_mode::type::midgame);
321 if(!flg_dialog.
show() && !first_time) {
328 change[
"change_faction"] =
true;
334 change[
"side_num"] = side_num;
348 tree_view& tree = find_widget<tree_view>(
"side_list");
355 for(
const auto& side :
get_scenario().child_range(
"side")) {
357 if(!side[
"allow_player"].to_bool(
true)) {
367 data.emplace(
"tree_view_node_label", item);
378 const std::string color_str = !side[
"color"].empty() ? side[
"color"] : side[
"side"].str();
384 item[
"label"] = side[
"side"];
386 data.emplace(
"side_number", item);
391 std::string leader_type = leader[
"type"];
392 std::string leader_gender = leader[
"gender"];
393 std::string leader_name;
398 if(side_unit[
"canrecruit"].to_bool()) {
399 leader_type = side_unit[
"type"].str();
400 leader_gender = side_unit[
"gender"].str();
406 const unit_type&
type = ut->get_gender_unit_type(leader_gender);
408 leader_image =
formatter() <<
type.image() <<
"~RC(" <<
type.flag_rgb() <<
">" << color_str <<
")";
409 leader_name =
type.type_name();
412 item[
"label"] = leader_image;
413 data.emplace(
"leader_image", item);
416 if(!leader_name.empty()) {
417 description +=
formatter() <<
" (<i>" << leader_name <<
"</i>)";
420 item[
"label"] = description;
421 data.emplace(
"leader_type", item);
423 item[
"label"] = side[
"faction_name"];
424 data.emplace(
"leader_faction", item);
426 std::string gender_icon =
"icons/icon-random.png";
427 if(leader_gender !=
"null") {
428 gender_icon =
formatter() <<
"icons/icon-" << leader_gender <<
".png";
429 item[
"tooltip"] = leader_gender;
432 item[
"label"] = gender_icon;
433 data.emplace(
"leader_gender", item);
439 if(side[
"allow_changes"].to_bool()) {
440 item[
"label"] = side[
"gold"].str() +
" " +
_(
"Gold");
441 data.emplace(
"side_gold", item);
444 const int income_amt = side[
"income"].to_int();
445 if(income_amt != 0) {
446 const std::string income_string =
formatter() << (income_amt > 0 ?
"+" :
"") << income_amt <<
" " <<
_(
"Income");
448 item[
"label"] = income_string;
449 data.emplace(
"side_income", item);
456 auto* select_leader_button = &row_grid.
find_widget<
button>(
"select_leader",
false);
457 if(select_leader_button) {
458 if(side[
"player_id"] ==
prefs::get().
login() && side[
"allow_changes"].to_bool(
true)) {
465 const auto handler = [
this, side_num](
bool& handled,
bool& halt) {
473 handled = halt =
true;
482 if(income_amt == 0) {
499 if(
level_[
"started"].to_bool()) {
510 find_widget<chatbox>(
"chat").process_network_data(
data);
512 if(!
data[
"message"].empty()) {
516 if(
data[
"failed"].to_bool()) {
520 }
else if(
data.has_child(
"start_game")) {
525 }
else if(
data.has_child(
"leave_game")) {
531 if(
data.has_child(
"stop_updates")) {
533 }
else if(
auto c =
data.optional_child(
"scenario_diff")) {
538 }
else if(
auto change =
data.optional_child(
"change_controller")) {
539 if(
auto side_to_change =
get_scenario().find_child(
"side",
"side", change[
"side"])) {
540 side_to_change->merge_with(*change);
546 }
else if(
data.has_child(
"scenario") ||
data.has_child(
"snapshot") ||
data.has_child(
"next_scenario")) {
552 if(
data.has_child(
"turn")) {
553 ERR_MP <<
"received replay data\n" <<
data <<
"\n in mp join";
557 if(
data.has_child(
"user")) {
std::map< std::string, chatroom_log > default_chat_log
Variant for storing WML attributes.
A config object defines a single node in a WML file, with access to child nodes.
const config & child_or_empty(config_key_type key) const
Returns the first child with the given key, or an empty config if there is none.
config & mandatory_child(config_key_type key, int n=0)
Returns the nth child with the given key, or throws an error if there is none.
bool has_child(config_key_type key) const
Determine whether a config has a child or not.
bool has_attribute(config_key_type key) const
child_itors child_range(config_key_type key)
void apply_diff(const config &diff, bool track=false)
A function to apply a diff config onto this config object.
boost::iterator_range< const_child_iterator > const_child_itors
const attribute_value * get(config_key_type key) const
Returns a pointer to the attribute with the given key or nullptr if it does not exist.
optional_config_impl< config > optional_child(config_key_type key, int n=0)
Equivalent to mandatory_child, but returns an empty optional if the nth child was not found.
config & add_child(config_key_type key)
bool is_normal_mp_game() const
static game_config_manager * get()
void load_game_config_for_game(const game_classification &classification, const std::string &scenario_id)
static game_config_view wrap(const config &cfg)
lobby_chat_window * room_window_open(const std::string &room, const bool open_new, const bool allow_close=true)
Check if a room window for "room" is open, if open_new is true then it will be created if not found.
virtual void send_chat_message(const std::string &message, bool allies_only) override
Inherited form chat_handler.
void load_log(std::map< std::string, chatroom_log > &log, bool show_lobby)
void active_window_changed()
static void progress(loading_stage stage=loading_stage::none)
Report what is being loaded to the loading screen.
static void display(const std::function< void()> &f)
Abstract base class for all modal dialogs.
bool show(const unsigned auto_close_time=0)
Shows the window.
void generate_side_list()
std::unique_ptr< player_list_helper > player_list_
std::size_t update_timer_
virtual void pre_show() override
Actions to be taken before showing the window.
const bool first_scenario_
faction_select * flg_dialog_
std::map< std::string, tree_view_node * > team_tree_map_
virtual void post_show() override
Actions to be taken after the window has been shown.
bool show_flg_select(int side_num, bool first_time=false)
void close_faction_select_dialog_if_open()
Will close the Faction Select dialog if it's open.
wesnothd_connection & network_connection_
std::unique_ptr< plugins_context > plugins_context_
tree_view_node & add_sibling(const std::string &id, const widget_data &data)
Adds a sibling for a node at the end of the list.
tree_view_node & add_node(const std::string &id, const widget_data &data, const int index=-1)
void set_enter_disabled(const bool enter_disabled)
Disable the enter key.
void set_retval(const int retval, const bool close_window=true)
Sets there return value of the window.
static const std::string & type()
Static type getter that does not rely on the widget being constructed.
void set_escape_disabled(const bool escape_disabled)
Disable the escape key.
FLG stands for faction, leader and gender.
const std::string & current_gender() const
const config & current_faction() const
const std::string & current_leader() const
game_classification & classification()
std::string get_scenario_id() const
static t_string from_serialized(const std::string &string)
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.
Data-based RAII scope guard.
A class that represents a TCP/IP connection to the wesnothd server.
bool receive_data(config &result)
Receives the next pending data pack from the server, if available.
bool wait_and_receive_data(config &data)
Unlike receive_data, waits until data is available instead of returning immediately.
static std::string _(const char *str)
Standard logging facilities (interface).
static lg::log_domain log_mp_connect_engine("mp/connect/engine")
const std::string unicode_em_dash
static void add_color_info(const game_config_view &v, bool build_defaults)
std::chrono::milliseconds lobby_network_timer
std::map< std::string, std::vector< color_t >, std::less<> > team_rgb_colors
static std::string generate_user_description(const config &side)
REGISTER_DIALOG(editor_edit_unit)
void connect_signal_mouse_left_click(dispatcher &dispatcher, const signal &signal)
Connects a signal handler for a left mouse button click.
std::size_t add_timer(const std::chrono::milliseconds &interval, const std::function< void(std::size_t id)> &callback, const bool repeat)
Adds a new timer.
std::map< std::string, widget_item > widget_data
std::map< std::string, t_string > widget_item
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.
bool remove_timer(const std::size_t id)
Removes a timer.
@ OK
Dialog was closed with the OK button.
@ CANCEL
Dialog was closed with the CANCEL button.
Functions to load and save images from/to disk.
std::string span_color(const color_t &color, Args &&... data)
Applies Pango markup to the input specifying its display color.
void level_to_gamestate(const config &level, saved_game &state)
void send_to_server(const config &data)
Attempts to send given data to server if a connection is open.
const std::string random_enemy_picture("units/random-dice.png")
std::size_t index(std::string_view str, const std::size_t index)
Codepoint index corresponding to the nth character in a UTF-8 string.
static constexpr utils::optional< enum_type > get_enum(const std::string_view value)
Converts a string into its enum equivalent.
An error occurred during when trying to communicate with the wesnothd server.
Contains the gui2 timer routines.
unit_type_data unit_types