The Battle for Wesnoth  1.19.10+dev
gui.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2008 - 2025
3  by Mark de Wever <koraq@xs4all.nl>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #define GETTEXT_DOMAIN "wesnoth-lib"
17 
18 #include "gui/gui.hpp"
19 
20 #include "config_cache.hpp"
21 #include "filesystem.hpp"
22 #include "gettext.hpp"
23 #include "gui/core/log.hpp"
25 #include "gui/widgets/settings.hpp"
26 #include "serialization/parser.hpp"
29 #include "wml_exception.hpp"
30 
31 namespace gui2
32 {
33 namespace
34 {
35 config read_and_validate(const std::string& path)
36 try {
37  preproc_map defines;
39  return io::read(*preprocess_file(path, &defines), &validator);
40 
41 } catch(const utils::bad_optional_access&) {
42  FAIL("GUI2: schema/gui.cfg not found.");
43 
44 } catch(const abstract_validator::error& e) {
45  FAIL("GUI2: could not read schema file: " + e.message);
46 
47 } catch(const config::error& e) {
48  ERR_GUI_P << "Could not read gui file: " << path;
49  ERR_GUI_P << e.what();
50  return {};
51 }
52 
53 /**
54  * Adds a theme definition object to the global registry.
55  *
56  * @param def A valid gui_definition config.
57  *
58  * @returns An optional iterator to the newly-constructed definition.
59  * If gui_definition throws wml_exception or a theme with the
60  * given ID already exists, returns nullopt.
61  */
62 auto register_theme(const config& def) -> utils::optional<gui_theme_map_t::iterator>
63 try {
64  auto [iter, is_unique] = guis.try_emplace(def["id"], def);
65  if(is_unique) return iter;
66 
67  ERR_GUI_P << "UI Theme '" << def["id"] << "' already exists.";
68  return utils::nullopt;
69 
70 } catch(const wml_exception& e) {
71  ERR_GUI_P << "Invalid UI theme: " << def["id"];
72  ERR_GUI_P << e.user_message;
73  return utils::nullopt;
74 }
75 
76 /**
77  * Parses any GUI2 theme definitions at the file path specified.
78  *
79  * @param full_path Path to file containing one or more [gui] tags.
80  * @param is_core If true, look for the default theme here.
81  */
82 void parse(const std::string& full_path, bool is_core)
83 {
84 #if __cpp_range_based_for >= 202211L // lifetime extension of temporaries
85  for(const config& def : read_and_validate(full_path).child_range("gui")) {
86 #else
87  config cfg = read_and_validate(full_path);
88  for(const config& def : cfg.child_range("gui")) {
89 #endif
90  const bool is_default = def["id"] == "default";
91 
92  if(is_default && !is_core) {
93  ERR_GUI_P << "UI theme id 'default' is reserved for core themes.";
94  continue;
95  }
96 
97  const auto iter = register_theme(def);
98  if(!iter) continue;
99 
100  if(is_default && is_core) {
101  default_gui = *iter;
103  }
104  }
105 }
106 
107 } // namespace
108 
109 void init()
110 {
111  LOG_GUI_G << "Initializing UI subststem.";
112 
113  // Reset the registry in case we're re-initializing
114  guis.clear();
115 
116  // Save current screen size.
118 
119  //
120  // Parse GUI definitions from mainline
121  //
122 
123  try {
124  parse(filesystem::get_wml_location("gui/_main.cfg").value(), true);
125  } catch(const utils::bad_optional_access&) {
126  FAIL("GUI2: gui/_main.cfg not found.");
127  }
128 
129  // The default GUI must be in mainline
130  VALIDATE(default_gui != guis.end(), _("No default gui defined."));
131 
132  //
133  // Parse GUI definitions from addons
134  //
135 
136  std::vector<std::string> addon_dirs;
137  const std::string umc_dir = filesystem::get_addons_dir();
139 
140  // Search for all $user_campaign_dir/*/gui-theme.cfg files
141  for(const std::string& umc : addon_dirs) {
142  const std::string gui_file = umc + "/gui-theme.cfg";
143 
144  if(filesystem::file_exists(gui_file)) {
145  parse(gui_file, false);
146  }
147  }
148 }
149 
150 void switch_theme(const std::string& theme_id)
151 {
152  if(theme_id.empty() || theme_id == "default") {
154  } else {
155  current_gui = std::find_if(guis.begin(), guis.end(),
156  [&](const auto& theme) { return theme.first == theme_id; });
157 
158  if(current_gui == guis.end()) {
159  ERR_GUI_P << "Missing [gui] definition for '" << theme_id << "'";
161  }
162  }
163 
164  current_gui->second.activate();
165 }
166 
167 } // namespace gui2
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
child_itors child_range(config_key_type key)
Definition: config.cpp:268
Realization of serialization/validator.hpp abstract validator.
Definition: theme.hpp:44
Declarations for File-IO.
static std::string _(const char *str)
Definition: gettext.hpp:103
Define the common log macros for the gui toolkit.
#define LOG_GUI_G
Definition: log.hpp:42
#define ERR_GUI_P
Definition: log.hpp:69
void get_files_in_dir(const std::string &dir, std::vector< std::string > *files, std::vector< std::string > *dirs, name_mode mode, filter_mode filter, reorder_mode reorder, file_tree_checksum *checksum)
Get a list of all files and/or directories in a given directory.
Definition: filesystem.cpp:450
static bool file_exists(const bfs::path &fpath)
Definition: filesystem.cpp:328
utils::optional< std::string > get_wml_location(const std::string &path, const utils::optional< std::string > &current_dir)
Returns a translated path to the actual file or directory, if it exists.
std::string get_addons_dir()
std::string path
Definition: filesystem.cpp:93
void update_screen_size_variables()
Update the size of the screen variables in settings.
Definition: settings.cpp:49
Generic file dialog.
void init()
Initializes the GUI subsystems.
Definition: gui.cpp:109
gui_theme_map_t guis
Map of all known GUIs.
void switch_theme(const std::string &theme_id)
Set and activate the given gui2 theme.
Definition: gui.cpp:150
gui_theme_map_t::iterator current_gui
Iterator pointing to the current GUI.
gui_theme_map_t::iterator default_gui
Iterator pointing to the default GUI.
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
filesystem::scoped_istream preprocess_file(const std::string &fname, preproc_map *defines)
Function to use the WML preprocessor on a file.
std::map< std::string, struct preproc_define > preproc_map
One of the realizations of serialization/validator.hpp abstract validator.
This file contains the settings handling of the widget library.
Used to manage with not initialized validators Supposed to be thrown from the constructor.
Definition: validator.hpp:97
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 FAIL(message)
#define VALIDATE(cond, message)
The macro to use for the validation of WML.
#define e