The Battle for Wesnoth  1.19.5+dev
addon_list.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2016 - 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 
15 #define GETTEXT_DOMAIN "wesnoth-lib"
16 
18 
19 #include "addon/client.hpp"
20 #include "color.hpp"
21 #include "formatter.hpp"
22 #include "gettext.hpp"
25 #include "gui/widgets/button.hpp"
27 #include "gui/widgets/label.hpp"
28 #include "gui/widgets/listbox.hpp"
31 #include "gui/widgets/window.hpp"
32 #include "serialization/markup.hpp"
33 #include "wml_exception.hpp"
34 
35 #include <algorithm>
36 
37 namespace gui2
38 {
39 namespace
40 {
41 const color_t color_outdated {255, 127, 0};
42 
43 const unsigned CONTROL_STACK_LAYER_INSTALL = 0;
44 const unsigned CONTROL_STACK_LAYER_UPDATE = 1;
45 const unsigned CONTROL_STACK_LAYER_PUBLISH = 2;
46 
47 } // end anon namespace
48 
49 REGISTER_WIDGET(addon_list)
50 
51 addon_list::addon_list(const implementation::builder_addon_list& builder)
52  : container_base(builder, type())
53  , addon_vector_()
54  , install_status_visibility_(visibility::visible)
55  , install_buttons_visibility_(visibility::invisible)
56  , install_function_()
57  , uninstall_function_()
58  , publish_function_()
59  , delete_function_()
60 {
61 }
62 
63 std::string addon_list::colorize_addon_state_string(const std::string& str, ADDON_STATUS state, bool verbose)
64 {
65  color_t colorname = font::NORMAL_COLOR;
66 
67  switch(state) {
68  case ADDON_NONE:
69  if(!verbose) {
70  return str;
71  }
72  colorname = font::weapon_details_color;
73  break;
74  case ADDON_INSTALLED:
76  case ADDON_NOT_TRACKED:
77  colorname = font::GOOD_COLOR;
78  break;
80  colorname = font::YELLOW_COLOR;
81  break;
83  colorname = color_outdated;
84  break;
86  colorname = font::BAD_COLOR;
87  break;
88  default:
89  colorname = font::GRAY_COLOR;
90  break;
91  }
92 
93  return markup::span_color(colorname, str);
94 }
95 
97 {
98  std::string tx;
99 
100  switch(info.state) {
101  case ADDON_NONE:
102  tx = info.can_publish ? _("addon_state^Published, not installed") : _("addon_state^Not installed");
103  break;
104 
105  case ADDON_INSTALLED:
106  case ADDON_NOT_TRACKED:
107  // Consider add-ons without version information as installed
108  // for the main display. Their Description info should elaborate
109  // on their status.
110  tx = info.can_publish ? _("addon_state^Published") : _("addon_state^Installed");
111  break;
113  tx = info.can_publish ? _("addon_state^Ready to publish") : _("addon_state^Installed, not ready to publish");
114  break;
115 
117  tx = info.can_publish ? _("addon_state^Published, upgradable") : _("addon_state^Installed, upgradable");
118  break;
119 
121  tx = info.can_publish ? _("addon_state^Published, outdated on server") : _("addon_state^Installed, outdated on server");
122  break;
123 
125  tx = info.can_publish ? _("addon_state^Published, broken") : _("addon_state^Installed, broken");
126  break;
127 
128  default:
129  tx = _("addon_state^Unknown");
130  }
131 
132  return colorize_addon_state_string(tx, info.state, true);
133 }
134 
135 void addon_list::addon_action_wrapper(addon_op_func_t& func, const addon_info& addon, bool& handled, bool& halt)
136 {
137  try {
138  func(addon);
139 
140  handled = halt = true;
141  } catch(const addons_client::user_exit&) {
142  // User canceled the op.
143  }
144 }
145 
146 const std::string addon_list::display_title_full_shift(const addon_info& addon) const
147 {
148  const std::string& local_title = addon.display_title_translated();
149  const std::string& display_title = addon.display_title();
150  if(local_title.empty()) {
151  return display_title;
152  }
153  return local_title + "\n" + markup::tag("small", "(", display_title, ")");
154 }
155 
157 {
158  listbox& list = get_listbox();
159  list.clear();
160 
161  addon_vector_.clear();
162 
163  for(const auto& a : addons) {
164  const addon_info& addon = a.second;
165  addon_tracking_info tracking_info = get_addon_tracking_info(addon);
166 
167  addon_vector_.push_back(&addon);
168 
170  widget_item item;
171 
172  if(!tracking_info.can_publish) {
173  item["label"] = addon.display_icon();
174  data.emplace("icon", item);
175 
176  item["label"] = display_title_full_shift(addon);
177  data.emplace("name", item);
178  } else {
179  item["label"] = addon.display_icon() + "~SCALE(72,72)~BLIT(icons/icon-addon-publish.png,8,8)";
180  data.emplace("icon", item);
181 
182  const std::string publish_name = markup::span_color(font::GOOD_COLOR, display_title_full_shift(addon));
183 
184  item["label"] = publish_name;
185  data.emplace("name", item);
186  }
187 
188  item["label"] = describe_status(tracking_info);
189  data.emplace("installation_status", item);
190 
191  // If the addon is upgradable or ourdated on server, we display the two relevant
192  // versions directly in the list for convenience.
193  const bool special_version_display =
194  tracking_info.state == ADDON_INSTALLED_UPGRADABLE ||
195  tracking_info.state == ADDON_INSTALLED_OUTDATED;
196 
197  std::ostringstream ss;
198  if(special_version_display) {
199  ss << tracking_info.installed_version.str() << "\n";
200  }
201 
202  ss << (*addon.versions.begin()).str();
203 
204  if(special_version_display) {
205  ss.str(colorize_addon_state_string(ss.str(), tracking_info.state, false));
206  }
207 
208  item["label"] = ss.str();
209  data.emplace("version", item);
210 
211  item["label"] = addon.author;
212  data.emplace("author", item);
213 
214  item["label"] = size_display_string(addon.size);
215  data.emplace("size", item);
216 
217  item["label"] = std::to_string(addon.downloads);
218  data.emplace("downloads", item);
219 
220  item["label"] = addon.display_type();
221  data.emplace("type", item);
222 
223  grid* row_grid = &list.add_row(data);
224 
225  // Set special retval for the toggle panels
226  row_grid->find_widget<toggle_panel>("list_panel").set_retval(DEFAULT_ACTION_RETVAL);
227 
228  // The control button grid is excluded on lower resolutions.
229  grid* control_grid = row_grid->find_widget<grid>("single_install_buttons", false, false);
230  if(!control_grid) {
231  continue;
232  }
233 
234  //
235  // Set up the inline control buttons.
236  //
237  stacked_widget& install_update_stack = control_grid->find_widget<stacked_widget>("install_update_stack");
238 
239  // These three buttons are in the install_update_stack. Only one is shown depending on the addon's state.
240  button& install_button = control_grid->find_widget<button>("single_install");
241  button& update_button = control_grid->find_widget<button>("single_update");
242  button& publish_button = control_grid->find_widget<button>("single_publish");
243 
244  // This button is always shown.
245  button& uninstall_button = control_grid->find_widget<button>("single_uninstall");
246 
247  const bool is_installed = is_installed_addon_status(tracking_info.state);
248  const bool is_local = tracking_info.state == ADDON_INSTALLED_LOCAL_ONLY;
249 
250  // Select the right button layer and set its callback.
251  if(tracking_info.can_publish) {
252  install_update_stack.select_layer(CONTROL_STACK_LAYER_PUBLISH);
253 
254  publish_button.set_active(true);
255 
256  if(publish_function_ != nullptr) {
257  connect_signal_mouse_left_click(publish_button,
258  std::bind(&addon_list::addon_action_wrapper, this, publish_function_, std::ref(addon), std::placeholders::_3, std::placeholders::_4));
259 
260  install_button.set_tooltip(_("Publish add-on"));
261  }
262  } else if(tracking_info.state == ADDON_INSTALLED_UPGRADABLE) {
263  install_update_stack.select_layer(CONTROL_STACK_LAYER_UPDATE);
264 
265  update_button.set_active(true);
266 
267  if(update_function_ != nullptr) {
268  connect_signal_mouse_left_click(update_button,
269  std::bind(&addon_list::addon_action_wrapper, this, update_function_, std::ref(addon), std::placeholders::_3, std::placeholders::_4));
270  }
271  } else {
272  install_update_stack.select_layer(CONTROL_STACK_LAYER_INSTALL);
273 
274  install_button.set_active(!is_installed);
275 
276  if(install_function_ != nullptr) {
277  connect_signal_mouse_left_click(install_button,
278  std::bind(&addon_list::addon_action_wrapper, this, install_function_, std::ref(addon), std::placeholders::_3, std::placeholders::_4));
279  }
280  }
281 
282  // Set up the Uninstall button.
283  if(tracking_info.can_publish) {
284  // Use the uninstall button as a delete-from-server button if the addon's already been published...
285  uninstall_button.set_active(!is_local);
286 
287  if(!is_local && delete_function_ != nullptr) {
288  connect_signal_mouse_left_click(uninstall_button,
289  std::bind(&addon_list::addon_action_wrapper, this, delete_function_, std::ref(addon), std::placeholders::_3, std::placeholders::_4));
290 
291  uninstall_button.set_tooltip(_("Delete add-on from server"));
292  }
293  } else {
294  // ... else it functions as normal.
295  uninstall_button.set_active(is_installed);
296 
297  if(is_installed && uninstall_function_ != nullptr) {
298  connect_signal_mouse_left_click(uninstall_button,
299  std::bind(&addon_list::addon_action_wrapper, this, uninstall_function_, std::ref(addon), std::placeholders::_3, std::placeholders::_4));
300  }
301  }
302 
304  row_grid->find_widget<label>("installation_status").set_visible(install_status_visibility_);
305  }
306 
308 }
309 
311 {
312  const listbox& list = get_grid().find_widget<const listbox>("addons");
313 
314  try {
315  return addon_vector_.at(list.get_selected_row());
316  } catch(const std::out_of_range&) {
317  return nullptr;
318  }
319 }
320 
322 {
323  const addon_info* addon = get_selected_addon();
324  if(addon == nullptr || !get_addon_tracking_info(*addon).can_publish) {
325  return "";
326  } else {
327  return addon->id;
328  }
329 }
330 
331 void addon_list::select_addon(const std::string& id)
332 {
333  listbox& list = get_listbox();
334 
335  auto iter = std::find_if(addon_vector_.begin(), addon_vector_.end(),
336  [&id](const addon_info* a) { return a->id == id; }
337  );
338 
339  // Corner case: if you publish an addon with an out-of-folder .pbl file and
340  // delete it locally before deleting it from the server, the game will try
341  // to reselect an addon that now doesn't exist.
342  //
343  // I don't anticipate this check will match very often, but in the case that
344  // some other weird corner case crops up, it's probably better to just exit
345  // silently than asserting, as was the pervious behavior.
346  if(iter == addon_vector_.end()) {
347  return;
348  }
349 
350  const addon_info& info = **iter;
351 
352  for(unsigned int i = 0u; i < list.get_item_count(); ++i) {
353  grid* row = list.get_row_grid(i);
354 
355  const label& name_label = row->find_widget<label>("name");
356  if(name_label.get_label().base_str() == display_title_full_shift(info)) {
357  list.select_row(i);
358  }
359  }
360 }
361 
363 {
364  return get_grid().find_widget<listbox>("addons");
365 }
366 
368 {
369  if(window* window = get_window()) {
371  }
372 }
373 
375 {
376  listbox& list = get_listbox();
377 
378  list.register_translatable_sorting_option(0, [this](const int i) { return addon_vector_[i]->display_title_full(); });
379  list.register_sorting_option(1, [this](const int i) { return addon_vector_[i]->author; });
380  list.register_sorting_option(2, [this](const int i) { return addon_vector_[i]->size; });
381  list.register_sorting_option(3, [this](const int i) { return addon_vector_[i]->downloads; });
382  list.register_translatable_sorting_option(4, [this](const int i) { return addon_vector_[i]->display_type(); });
383 
384  auto order = std::pair(0, sort_order::type::ascending);
385  list.set_active_sorting_option(order);
386 }
387 
389 {
390  listbox& list = get_listbox();
391 
392  generator_base::order_func generator_func = [this, func](unsigned a, unsigned b)
393  {
394  return func(*addon_vector_[a], *addon_vector_[b]);
395  };
396 
397  list.mark_as_unsorted();
398  list.order_by(generator_func);
399 }
400 
402 {
403  if(addon_vector_.empty()) {
404  // Happens in the dialog unit test.
405  return;
406  }
408 }
409 
412 {
413  DBG_GUI_P << "Parsing add-on list " << id;
414 
415  load_resolutions<resolution>(cfg);
416 }
417 
419  : resolution_definition(cfg), grid(nullptr)
420 {
421  // Add a dummy state since every widget needs a state.
422  static config dummy("draw");
423  state.emplace_back(dummy);
424 
425  auto child = cfg.optional_child("grid");
426  VALIDATE(child, _("No grid defined."));
427 
428  grid = std::make_shared<builder_grid>(*child);
429 }
430 
431 namespace implementation
432 {
433 
434 static widget::visibility parse_visibility(const std::string& str)
435 {
436  if(str == "visible") {
438  } else if(str == "hidden") {
440  } else if(str == "invisible") {
442  } else {
443  FAIL("Invalid visibility value");
444  }
445 }
446 
447 builder_addon_list::builder_addon_list(const config& cfg)
448  : builder_styled_widget(cfg)
449  , install_status_visibility_(widget::visibility::visible)
450  , install_buttons_visibility_(widget::visibility::invisible)
451 {
452  if(cfg.has_attribute("install_status_visibility")) {
453  install_status_visibility_ = parse_visibility(cfg["install_status_visibility"]);
454  }
455 
456  if(cfg.has_attribute("install_buttons_visibility")) {
457  install_buttons_visibility_ = parse_visibility(cfg["install_buttons_visibility"]);
458  }
459 }
460 
461 std::unique_ptr<widget> builder_addon_list::build() const
462 {
463  auto widget = std::make_unique<addon_list>(*this);
464 
465  DBG_GUI_G << "Window builder: placed add-on list '" << id <<
466  "' with definition '" << definition << "'.";
467 
468  const auto conf = widget->cast_config_to<addon_list_definition>();
469  assert(conf != nullptr);
470 
471  widget->init_grid(*conf->grid);
472 
473  widget->set_install_status_visibility(install_status_visibility_);
474  widget->set_install_buttons_visibility(install_buttons_visibility_);
475 
476  widget->finalize_setup();
477 
478  return widget;
479 }
480 
481 } // end namespace implementation
482 
483 } // end namespace gui2
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:172
bool has_attribute(config_key_type key) const
Definition: config.cpp:157
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.
Definition: config.cpp:384
addon_op_func_t uninstall_function_
Definition: addon_list.hpp:165
std::function< bool(const addon_info &, const addon_info &)> addon_sort_func
Definition: addon_list.hpp:40
void add_list_to_keyboard_chain()
Adds the internal listbox to the keyboard event chain.
Definition: addon_list.cpp:367
void addon_action_wrapper(addon_op_func_t &func, const addon_info &addon, bool &handled, bool &halt)
Helper to wrap the execution of any of the addon operation functions.
Definition: addon_list.cpp:135
const addon_info * get_selected_addon() const
Returns the selected add-on.
Definition: addon_list.cpp:310
visibility install_buttons_visibility_
Definition: addon_list.hpp:162
listbox & get_listbox()
Returns the underlying list box.
Definition: addon_list.cpp:362
std::string get_remote_addon_id()
Returns the selected add-on id, for use with remote publish/delete ops.
Definition: addon_list.cpp:321
void set_addon_order(addon_sort_func func)
Definition: addon_list.cpp:388
const std::string display_title_full_shift(const addon_info &addon) const
Definition: addon_list.cpp:146
static const int DEFAULT_ACTION_RETVAL
Special retval for the toggle panels in the addons list.
Definition: addon_list.hpp:45
static std::string colorize_addon_state_string(const std::string &str, ADDON_STATUS state, bool verbose=false)
Changes the color of an add-on state string (installed, outdated, etc.) according to the state itself...
Definition: addon_list.cpp:63
static std::string describe_status(const addon_tracking_info &info)
Definition: addon_list.cpp:96
std::vector< const addon_info * > addon_vector_
Definition: addon_list.hpp:159
addon_op_func_t update_function_
Definition: addon_list.hpp:166
void select_first_addon()
Choose the item at the top of the list (taking account of sort order).
Definition: addon_list.cpp:401
addon_op_func_t delete_function_
Definition: addon_list.hpp:169
addon_op_func_t publish_function_
Definition: addon_list.hpp:168
visibility install_status_visibility_
Definition: addon_list.hpp:161
void select_addon(const std::string &id)
Selects the add-on with the given ID.
Definition: addon_list.cpp:331
std::function< void(const addon_info &)> addon_op_func_t
Definition: addon_list.hpp:67
void set_addons(const addons_list &addons)
Sets the add-ons to show.
Definition: addon_list.cpp:156
addon_op_func_t install_function_
Definition: addon_list.hpp:164
Simple push button.
Definition: button.hpp:36
A generic container base class.
const grid & get_grid() const
std::function< bool(unsigned, unsigned)> order_func
Definition: generator.hpp:248
Base container class.
Definition: grid.hpp:32
The listbox class.
Definition: listbox.hpp:43
void mark_as_unsorted()
Deactivates all sorting toggle buttons at the top, making the list look like it's not sorted.
Definition: listbox.cpp:654
void set_active_sorting_option(const order_pair &sort_by, const bool select_first=false)
Sorts the listbox by a pre-set sorting option.
Definition: listbox.cpp:620
void order_by(const generator_base::order_func &func)
Definition: listbox.cpp:596
grid & add_row(const widget_item &item, const int index=-1)
When an item in the list is selected by the user we need to update the state.
Definition: listbox.cpp:58
const grid * get_row_grid(const unsigned row) const
Returns the grid of the wanted row.
Definition: listbox.cpp:229
void register_translatable_sorting_option(const int col, translatable_sorter_func_t f)
Registers a special sorting function specifically for translatable values.
Definition: listbox.cpp:612
bool select_row(const unsigned row, const bool select=true)
Selects a row.
Definition: listbox.cpp:242
void register_sorting_option(const int col, const Func &f)
Definition: listbox.hpp:260
void clear()
Removes all the rows in the listbox, clearing it.
Definition: listbox.cpp:117
int get_selected_row() const
Returns the first selected row.
Definition: listbox.cpp:267
unsigned get_item_count() const
Returns the number of items in the listbox.
Definition: listbox.cpp:123
bool select_row_at(const unsigned row, const bool select=true)
Selects a row at the given position, regardless of sorting order.
Definition: listbox.cpp:255
void select_layer(const int layer)
Selects and displays a particular layer.
const t_string & get_label() const
Base class for all widgets.
Definition: widget.hpp:55
NOT_DANGLING T * find_widget(const std::string &id, const bool must_be_active, const bool must_exist)
Gets a widget with the wanted id.
Definition: widget.hpp:742
void set_visible(const visibility visible)
Definition: widget.cpp:479
window * get_window()
Get the parent window.
Definition: widget.cpp:117
visibility
Visibility settings done by the user.
Definition: widget.hpp:65
@ visible
The user sets the widget visible, that means:
@ invisible
The user set the widget invisible, that means:
@ hidden
The user sets the widget hidden, that means:
base class of top level items, the only item which needs to store the final canvases to draw on.
Definition: window.hpp:61
void add_to_keyboard_chain(widget *widget)
Adds the widget to the keyboard chain.
Definition: window.cpp:1213
std::string base_str() const
Definition: tstring.hpp:202
std::string str() const
Serializes the version number into string form.
Networked add-ons (campaignd) client interface.
std::size_t i
Definition: function.cpp:1028
static auto & dummy
static std::string _(const char *str)
Definition: gettext.hpp:93
#define DBG_GUI_G
Definition: log.hpp:41
#define DBG_GUI_P
Definition: log.hpp:66
This file contains the window object, this object is a top level container which has the event manage...
std::string size_display_string(double size)
Get a human-readable representation of the specified byte count.
Definition: info.cpp:311
std::map< std::string, addon_info > addons_list
Definition: info.hpp:27
const color_t YELLOW_COLOR
const color_t GOOD_COLOR
const color_t BAD_COLOR
const color_t GRAY_COLOR
const color_t weapon_details_color
const color_t NORMAL_COLOR
void connect_signal_mouse_left_click(dispatcher &dispatcher, const signal &signal)
Connects a signal handler for a left mouse button click.
Definition: dispatcher.cpp:177
static widget::visibility parse_visibility(const std::string &str)
Definition: addon_list.cpp:434
Generic file dialog.
std::map< std::string, widget_item > widget_data
Definition: widget.hpp:36
std::map< std::string, t_string > widget_item
Definition: widget.hpp:33
Contains the implementation details for lexical_cast and shouldn't be used directly.
logger & info()
Definition: log.cpp:319
std::string tag(const std::string &tag_name, Args &&... contents)
Definition: markup.hpp:45
std::string span_color(const color_t &color, Args &&... data)
Definition: markup.hpp:68
std::string_view data
Definition: picture.cpp:178
#define REGISTER_WIDGET(id)
Wrapper for REGISTER_WIDGET3.
addon_tracking_info get_addon_tracking_info(const addon_info &addon)
Get information about an add-on comparing its local state with the add-ons server entry.
Definition: state.cpp:25
bool is_installed_addon_status(ADDON_STATUS s)
Definition: state.hpp:40
ADDON_STATUS
Defines various add-on installation statuses.
Definition: state.hpp:22
@ ADDON_NOT_TRACKED
No tracking information available.
Definition: state.hpp:37
@ ADDON_INSTALLED_OUTDATED
Version in the server is older than local installation.
Definition: state.hpp:30
@ ADDON_NONE
Add-on is not installed.
Definition: state.hpp:24
@ ADDON_INSTALLED_UPGRADABLE
Version in the server is newer than local installation.
Definition: state.hpp:28
@ ADDON_INSTALLED
Version in the server matches local installation.
Definition: state.hpp:26
@ ADDON_INSTALLED_LOCAL_ONLY
No version in the server.
Definition: state.hpp:32
@ ADDON_INSTALLED_BROKEN
Dependencies not satisfied.
Definition: state.hpp:35
std::string display_type() const
Get an add-on type identifier for display in the user's language.
Definition: info.cpp:249
int downloads
Definition: info.hpp:87
std::string display_title_translated() const
Definition: info.cpp:196
std::string display_icon() const
Get an icon path fixed for display (e.g.
Definition: info.cpp:232
std::string display_title() const
Get a title or automatic title for display.
Definition: info.cpp:155
std::string author
Definition: info.hpp:84
std::set< version_info, std::greater< version_info > > versions
Definition: info.hpp:82
int size
Definition: info.hpp:86
std::string id
Definition: info.hpp:75
Stores additional status information about add-ons.
Definition: state.hpp:47
ADDON_STATUS state
Definition: state.hpp:57
version_info installed_version
Definition: state.hpp:60
The basic class for representing 8-bit RGB or RGBA colour values.
Definition: color.hpp:59
addon_list_definition(const config &cfg)
Definition: addon_list.cpp:410
widget::visibility install_buttons_visibility_
Definition: addon_list.hpp:222
virtual std::unique_ptr< widget > build() const override
Definition: addon_list.cpp:461
std::string definition
Parameters for the styled_widget.
std::vector< state_definition > state
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 b