The Battle for Wesnoth  1.19.10+dev
help_browser.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2017 - 2025
3  by Charles Dang <exodia339@gmail.com>
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 
19 
20 #include "gui/widgets/button.hpp"
21 #include "gui/widgets/label.hpp"
25 #include "gui/widgets/text_box.hpp"
29 #include "gui/widgets/window.hpp"
30 #include "help/help.hpp"
32 #include "utils/ci_searcher.hpp"
33 #include "video.hpp"
34 
35 static lg::log_domain log_help("help");
36 #define ERR_HP LOG_STREAM(err, log_help)
37 #define WRN_HP LOG_STREAM(warn, log_help)
38 #define DBG_HP LOG_STREAM(debug, log_help)
39 
40 namespace gui2::dialogs
41 {
42 
43 REGISTER_DIALOG(help_browser)
44 
45 help_browser::help_browser(const help::section& toplevel, const std::string& initial)
46  : modal_dialog(window_id())
47  , initial_topic_(initial.empty() ? help::default_show_topic : initial)
48  , toplevel_(toplevel)
49  , history_()
50  , history_pos_(0)
51 {
52  if(initial_topic_.compare(0, 2, "..") == 0) {
53  initial_topic_.replace(0, 2, "+");
54  } else {
55  initial_topic_.insert(0, "-");
56  }
58 }
59 
61 {
62  tree_view& topic_tree = find_widget<tree_view>("topic_tree");
63 
64  button& back_button = find_widget<button>("back");
65  button& next_button = find_widget<button>("next");
66 
67  rich_label& topic_text = find_widget<rich_label>("topic_text");
68 
69  next_button.set_active(false);
70  back_button.set_active(false);
71  connect_signal_mouse_left_click(back_button, std::bind(&help_browser::on_history_navigate, this, true));
72  connect_signal_mouse_left_click(next_button, std::bind(&help_browser::on_history_navigate, this, false));
73 
74  toggle_button& contents = find_widget<toggle_button>("contents");
75 
76  if (video::window_size().x <= 800) {
77  contents.set_value(false);
78  connect_signal_mouse_left_click(contents, [&](auto&&...) {
79  topic_tree.set_visible(topic_tree.get_visible());
81  });
83  } else {
84  contents.set_value(true);
86  }
87 
88  text_box& filter = find_widget<text_box>("filter_box");
90  filter.on_modified([this](const auto& box) { update_list(box.text()); });
91 
92  topic_text.register_link_callback(std::bind(&help_browser::show_topic, this, std::placeholders::_1, true));
93 
95 
96  keyboard_capture(&topic_tree);
97 
99 
100  tree_view_node& initial_node = topic_tree.find_widget<tree_view_node>(initial_topic_);
101  initial_node.select_node(true);
102 
103  on_topic_select();
104 }
105 
106 void help_browser::update_list(const std::string& filter_text) {
107  tree_view& topic_tree = find_widget<tree_view>("topic_tree");
108  topic_tree.clear();
109  if(!add_topics_for_section(toplevel_, topic_tree.get_root_node(), filter_text)) {
110  // Add everything if nothing matches
112  }
113 }
114 
115 bool help_browser::add_topics_for_section(const help::section& parent_section, tree_view_node& parent_node, const std::string& filter_text)
116 {
117  bool topics_added = false;
118  const auto match = translation::make_ci_matcher(filter_text);
119 
120  for(const help::section& section : parent_section.sections) {
121  tree_view_node& section_node = add_topic(section.id, section.title, true, parent_node);
122  bool subtopics_added = add_topics_for_section(section, section_node, filter_text);
123 
124  if (subtopics_added || (match(section.id) || match(section.title))) {
125  if (!filter_text.empty()) {
126  section_node.unfold();
127  }
128  topics_added = true;
129  } else {
130  find_widget<tree_view>("topic_tree").remove_node(&section_node);
131  }
132  }
133 
134  for(const help::topic& topic : parent_section.topics) {
135  if ((match(topic.id) || match(topic.title)) && (topic.id.compare(0, 2, "..") != 0)) {
136  add_topic(topic.id, topic.title, false, parent_node);
137  topics_added = true;
138  }
139  }
140 
141  return topics_added;
142 }
143 
144 tree_view_node& help_browser::add_topic(const std::string& topic_id, const std::string& topic_title,
145  bool expands, tree_view_node& parent)
146 {
148  widget_item item;
149 
150  item["label"] = topic_title;
151  data.emplace("topic_name", item);
152 
153  tree_view_node& new_node = parent.add_child(expands ? "section" : "topic", data);
154  new_node.set_id(std::string(expands ? "+" : "-") + topic_id);
155 
156  return new_node;
157 }
158 
159 void help_browser::show_topic(std::string topic_id, bool add_to_history)
160 {
161  if(topic_id.empty()) {
162  return;
163  }
164 
165  if(topic_id[0] == '+') {
166  topic_id.replace(topic_id.begin(), topic_id.begin() + 1, 2, '.');
167  }
168 
169  if(topic_id[0] == '-') {
170  topic_id.erase(topic_id.begin(), topic_id.begin() + 1);
171  }
172 
173  auto iter = parsed_pages_.find(topic_id);
174  if(iter == parsed_pages_.end()) {
175  const help::topic* topic = help::find_topic(toplevel_, topic_id);
176  if(!topic) {
177  ERR_HP << "Help browser tried to show topic with id '" << topic_id
178  << "' but that topic could not be found." << std::endl;
179  return;
180  }
181 
182  DBG_HP << "Showing topic: " << topic->id << ": " << topic->title;
183 
185  widget_item item;
186 
187  item["label"] = topic->title;
188  data.emplace("topic_title", item);
189 
190  find_widget<label>("topic_title").set_label(topic->title);
191  find_widget<rich_label>("topic_text").set_dom(topic->text.parsed_text());
192 
194  scrollbar_panel& scroll = find_widget<scrollbar_panel>("topic_scroll_panel");
196  }
197 
198  if (add_to_history) {
199  // history pos is 0 initially, so it's already at first entry
200  // no need to increment first time
201  if (!history_.empty()) {
202  // don't add duplicate entries back-to-back
203  if (history_.back() == topic_id) {
204  return;
205  }
206  history_pos_++;
207  }
208  history_.push_back(topic_id);
209 
210  find_widget<button>("back").set_active(history_pos_ != 0);
211  }
212 }
213 
215 {
216  tree_view& topic_tree = find_widget<tree_view>("topic_tree");
217 
218  if(topic_tree.empty()) {
219  return;
220  }
221 
222  tree_view_node* selected = topic_tree.selected_item();
223  assert(selected);
224 
225  if (selected->id()[0] != '+' && video::window_size().x <= 800) {
226  find_widget<toggle_button>("contents").set_value(false);
228  }
229 
230  show_topic(selected->id());
231 }
232 
234 {
235  if(backwards) {
236  history_pos_--;
237  } else {
238  history_pos_++;
239  }
240  find_widget<button>("back").set_active(!history_.empty() && history_pos_ != 0);
241  find_widget<button>("next").set_active(!history_.empty() && history_pos_ != (history_.size()-1));
242 
243  show_topic(history_.at(history_pos_), false);
244 }
245 
246 } // namespace dialogs
Simple push button.
Definition: button.hpp:36
virtual void set_active(const bool active) override
See styled_widget::set_active.
Definition: button.cpp:64
Help browser dialog.
const help::section & toplevel_
void update_list(const std::string &)
bool add_topics_for_section(const help::section &parent_section, tree_view_node &parent_node, const std::string &filter_text="")
void show_topic(std::string topic_id, bool add_to_history=true)
std::map< std::string, int > parsed_pages_
virtual void pre_show() override
Actions to be taken before showing the window.
void on_history_navigate(bool backwards)
tree_view_node & add_topic(const std::string &topic_id, const std::string &topic_title, bool expands, tree_view_node &parent)
std::vector< std::string > history_
Abstract base class for all modal dialogs.
A rich_label takes marked up text and shows it correctly formatted and wrapped but no scrollbars are ...
Definition: rich_label.hpp:38
void register_link_callback(std::function< void(std::string)> link_handler)
Definition: rich_label.cpp:911
@ BEGIN
Go to begin position.
Definition: scrollbar.hpp:53
void scroll_vertical_scrollbar(const scrollbar_base::scroll_mode scroll)
Scrolls the vertical scrollbar.
A widget that allows the user to input text in single line.
Definition: text_box.hpp:125
virtual void set_value(unsigned selected, bool fire_event=false) override
Inherited from selectable_item.
void select_node(bool expand_parents=false)
void unfold(const bool recursive=false)
const tree_view_node & get_root_node() const
Definition: tree_view.hpp:53
bool empty() const
Definition: tree_view.cpp:99
tree_view_node * selected_item()
Definition: tree_view.hpp:98
void set_visible(const visibility visible)
Definition: widget.cpp:479
void set_id(const std::string &id)
Definition: widget.cpp:98
visibility get_visible() const
Definition: widget.cpp:506
@ invisible
The user set the widget invisible, that means:
T * find_widget(const std::string_view id, const bool must_be_active, const bool must_exist)
Gets a widget with the wanted id.
Definition: widget.hpp:747
widget * parent()
Definition: widget.cpp:170
void keyboard_capture(widget *widget)
Definition: window.cpp:1200
void invalidate_layout()
Updates the size of the window.
Definition: window.cpp:759
void add_to_keyboard_chain(widget *widget)
Adds the widget to the keyboard chain.
Definition: window.cpp:1206
const config & parsed_text() const
Definition: help_impl.cpp:380
This file contains the window object, this object is a top level container which has the event manage...
#define DBG_HP
#define ERR_HP
static lg::log_domain log_help("help")
std::string selected
REGISTER_DIALOG(editor_edit_unit)
void connect_signal_notify_modified(dispatcher &dispatcher, const signal_notification &signal)
Connects a signal handler for getting a notification upon modification.
Definition: dispatcher.cpp:189
void connect_signal_mouse_left_click(dispatcher &dispatcher, const signal &signal)
Connects a signal handler for a left mouse button click.
Definition: dispatcher.cpp:163
std::map< std::string, widget_item > widget_data
Definition: widget.hpp:36
std::map< std::string, t_string > widget_item
Definition: widget.hpp:33
void init_help()
Definition: help.cpp:178
const topic * find_topic(const section &sec, const std::string &id)
Search for the topic with the specified identifier in the section and its subsections.
Definition: help_impl.cpp:1293
const std::string default_show_topic
Definition: help_impl.cpp:82
auto make_ci_matcher(std::string_view filter_text)
Returns a function which performs locale-aware case-insensitive search.
Definition: ci_searcher.hpp:24
constexpr auto filter
Definition: ranges.hpp:38
point window_size()
Returns the size of the window in display units / screen coordinates.
Definition: video.cpp:418
std::string_view data
Definition: picture.cpp:178
A section contains topics and sections along with title and ID.
Definition: help_impl.hpp:148
section_list sections
Definition: help_impl.hpp:168
std::string id
Definition: help_impl.hpp:166
std::string title
Definition: help_impl.hpp:166
topic_list topics
Definition: help_impl.hpp:167
A topic contains a title, an id and some text.
Definition: help_impl.hpp:115
std::string id
Definition: help_impl.hpp:139
topic_text text
Definition: help_impl.hpp:140
std::string title
Definition: help_impl.hpp:139