The Battle for Wesnoth  1.19.10+dev
help_impl.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2003 - 2025
3  by David White <dave@whitevine.net>
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 #include "help/help_impl.hpp"
17 
18 #include "actions/attack.hpp" // for time_of_day bonus
19 #include "color.hpp"
20 #include "formula/string_utils.hpp" // for VGETTEXT
21 #include "font/standard_colors.hpp" // for NORMAL_COLOR
22 #include "game_config.hpp" // for debug, menu_contract, etc
23 #include "game_config_manager.hpp" // for game_config_manager
24 #include "gettext.hpp" // for _, gettext, N_
26 #include "hotkey/hotkey_command.hpp" // for is_scope_active, etc
27 #include "log.hpp" // for LOG_STREAM, logger, etc
28 #include "map/map.hpp" // for gamemap
29 #include "picture.hpp" // for get_image, locator
30 #include "preferences/preferences.hpp" // for encountered_terrains, etc
31 #include "resources.hpp" // for tod_manager, config_manager
32 #include "serialization/markup.hpp" // for markup utility functions
33 #include "serialization/parser.hpp"
34 #include "serialization/string_utils.hpp" // for split, quoted_split, etc
35 #include "serialization/unicode.hpp" // for iterator
36 #include "serialization/utf8_exception.hpp" // for char_t, etc
37 #include "terrain/terrain.hpp" // for terrain_type
38 #include "terrain/translation.hpp" // for operator==, ter_list, etc
39 #include "terrain/type_data.hpp" // for terrain_type_data, etc
40 #include "time_of_day.hpp" // for time_of_day
41 #include "tod_manager.hpp" // for tod_manager
42 #include "tstring.hpp" // for t_string, operator<<
43 #include "units/race.hpp" // for unit_race, etc
44 #include "units/types.hpp" // for unit_type, unit_type_data, etc
45 #include "utils/general.hpp" // for contains
46 
47 #include <algorithm> // for sort, find, transform, etc
48 #include <boost/algorithm/string.hpp>
49 #include <cassert> // for assert
50 #include <iterator> // for back_insert_iterator, etc
51 #include <map> // for map, etc
52 #include <set>
53 #include <utility>
54 
55 static lg::log_domain log_help("help");
56 #define WRN_HP LOG_STREAM(warn, log_help)
57 #define DBG_HP LOG_STREAM(debug, log_help)
58 
59 namespace help {
60 
61 const game_config_view *game_cfg = nullptr;
62 // The default toplevel.
64 // All sections and topics not referenced from the default toplevel.
66 
69 boost::tribool last_debug_state = boost::indeterminate;
70 
71 std::vector<std::string> empty_string_vector;
72 const int max_section_level = 15;
75 const int box_width = 2;
77 const unsigned max_history = 100;
78 const std::string topic_img = "help/topic.png";
79 const std::string closed_section_img = "help/closed_section.png";
80 const std::string open_section_img = "help/open_section.png";
81 // The topic to open by default when opening the help dialog.
82 const std::string default_show_topic = "..introduction";
83 const std::string unknown_unit_topic = ".unknown_unit";
84 const std::string unit_prefix = "unit_";
85 const std::string terrain_prefix = "terrain_";
86 const std::string race_prefix = "race_";
87 const std::string faction_prefix = "faction_";
88 const std::string era_prefix = "era_";
89 const std::string variation_prefix = "variation_";
90 const std::string ability_prefix = "ability_";
91 
92 static bool is_cjk_char(const char32_t ch)
93 {
94  /**
95  * You can check these range at http://unicode.org/charts/
96  * see the "East Asian Scripts" part.
97  * Notice that not all characters in that part is still in use today, so don't list them all here.
98  * Below are characters that I guess may be used in wesnoth translations.
99  */
100 
101  //FIXME add range from Japanese-specific and Korean-specific section if you know the characters are used today.
102 
103  if (ch < 0x2e80) return false; // shortcut for common non-CJK
104 
105  return
106  //Han Ideographs: all except Supplement
107  (ch >= 0x4e00 && ch < 0x9fcf) ||
108  (ch >= 0x3400 && ch < 0x4dbf) ||
109  (ch >= 0x20000 && ch < 0x2a6df) ||
110  (ch >= 0xf900 && ch < 0xfaff) ||
111  (ch >= 0x3190 && ch < 0x319f) ||
112 
113  //Radicals: all except Ideographic Description
114  (ch >= 0x2e80 && ch < 0x2eff) ||
115  (ch >= 0x2f00 && ch < 0x2fdf) ||
116  (ch >= 0x31c0 && ch < 0x31ef) ||
117 
118  //Chinese-specific: Bopomofo and Bopomofo Extended
119  (ch >= 0x3104 && ch < 0x312e) ||
120  (ch >= 0x31a0 && ch < 0x31bb) ||
121 
122  //Yi-specific: Yi Radicals, Yi Syllables
123  (ch >= 0xa490 && ch < 0xa4c7) ||
124  (ch >= 0xa000 && ch < 0xa48d) ||
125 
126  //Japanese-specific: Hiragana, Katakana, Kana Supplement
127  (ch >= 0x3040 && ch <= 0x309f) ||
128  (ch >= 0x30a0 && ch <= 0x30ff) ||
129  (ch >= 0x1b000 && ch <= 0x1b001) ||
130 
131  //Ainu-specific: Katakana Phonetic Extensions
132  (ch >= 0x31f0 && ch <= 0x31ff) ||
133 
134  //Korean-specific: Hangul Syllables, Hangul Jamo, Hangul Jamo Extended-A, Hangul Jamo Extended-B
135  (ch >= 0xac00 && ch < 0xd7af) ||
136  (ch >= 0x1100 && ch <= 0x11ff) ||
137  (ch >= 0xa960 && ch <= 0xa97c) ||
138  (ch >= 0xd7b0 && ch <= 0xd7fb) ||
139 
140  //CJK Symbols and Punctuation
141  (ch >= 0x3000 && ch < 0x303f) ||
142 
143  //Halfwidth and Fullwidth Forms
144  (ch >= 0xff00 && ch < 0xffef);
145 }
146 
147 bool section_is_referenced(const std::string &section_id, const config &cfg)
148 {
149  if (auto toplevel = cfg.optional_child("toplevel"))
150  {
151  const std::vector<std::string> toplevel_refs
152  = utils::quoted_split(toplevel["sections"]);
153  if (std::find(toplevel_refs.begin(), toplevel_refs.end(), section_id)
154  != toplevel_refs.end()) {
155  return true;
156  }
157  }
158 
159  for (const config &section : cfg.child_range("section"))
160  {
161  const std::vector<std::string> sections_refd
162  = utils::quoted_split(section["sections"]);
163  if (std::find(sections_refd.begin(), sections_refd.end(), section_id)
164  != sections_refd.end()) {
165  return true;
166  }
167  }
168  return false;
169 }
170 
171 bool topic_is_referenced(const std::string &topic_id, const config &cfg)
172 {
173  if (auto toplevel = cfg.optional_child("toplevel"))
174  {
175  const std::vector<std::string> toplevel_refs
176  = utils::quoted_split(toplevel["topics"]);
177  if (std::find(toplevel_refs.begin(), toplevel_refs.end(), topic_id)
178  != toplevel_refs.end()) {
179  return true;
180  }
181  }
182 
183  for (const config &section : cfg.child_range("section"))
184  {
185  const std::vector<std::string> topics_refd
186  = utils::quoted_split(section["topics"]);
187  if (std::find(topics_refd.begin(), topics_refd.end(), topic_id)
188  != topics_refd.end()) {
189  return true;
190  }
191  }
192  return false;
193 }
194 
195 void parse_config_internal(const config *help_cfg, const config *section_cfg,
196  section &sec, int level)
197 {
198  if (level > max_section_level) {
199  PLAIN_LOG << "Maximum section depth has been reached. Maybe circular dependency?";
200  }
201  else if (section_cfg != nullptr) {
202  const std::vector<std::string> sections = utils::quoted_split((*section_cfg)["sections"]);
203  std::string id = level == 0 ? "toplevel" : (*section_cfg)["id"].str();
204  if (level != 0) {
205  if (!is_valid_id(id)) {
206  std::stringstream ss;
207  ss << "Invalid ID, used for internal purpose: '" << id << "'";
208  throw parse_error(ss.str());
209  }
210  }
211  std::string title = level == 0 ? "" : (*section_cfg)["title"].str();
212  sec.id = id;
213  sec.title = title;
214  std::vector<std::string>::const_iterator it;
215  // Find all child sections.
216  for (it = sections.begin(); it != sections.end(); ++it) {
217  if (auto child_cfg = help_cfg->find_child("section", "id", *it))
218  {
219  section child_section;
220  parse_config_internal(help_cfg, child_cfg.ptr(), child_section, level + 1);
221  sec.add_section(child_section);
222  }
223  else {
224  std::stringstream ss;
225  ss << "Help-section '" << *it << "' referenced from '"
226  << id << "' but could not be found.";
227  throw parse_error(ss.str());
228  }
229  }
230 
231  generate_sections(help_cfg, (*section_cfg)["sections_generator"], sec, level);
232  if ((*section_cfg)["sort_sections"] == "yes") {
233  sec.sections.sort(section_less());
234  }
235 
236  bool sort_topics = false;
237  bool sort_generated = true;
238 
239  if ((*section_cfg)["sort_topics"] == "yes") {
240  sort_topics = true;
241  sort_generated = false;
242  } else if ((*section_cfg)["sort_topics"] == "no") {
243  sort_topics = false;
244  sort_generated = false;
245  } else if ((*section_cfg)["sort_topics"] == "generated") {
246  sort_topics = false;
247  sort_generated = true;
248  } else if (!(*section_cfg)["sort_topics"].empty()) {
249  std::stringstream ss;
250  ss << "Invalid sort option: '" << (*section_cfg)["sort_topics"] << "'";
251  throw parse_error(ss.str());
252  }
253 
254  std::vector<topic> generated_topics = generate_topics(sort_generated,(*section_cfg)["generator"]);
255 
256  const std::vector<std::string> topics_id = utils::quoted_split((*section_cfg)["topics"]);
257  std::vector<topic> topics;
258 
259  // Find all topics in this section.
260  for (it = topics_id.begin(); it != topics_id.end(); ++it) {
261  if (auto topic_cfg = help_cfg->find_child("topic", "id", *it))
262  {
263  std::string text = topic_cfg["text"];
264  text += generate_topic_text(topic_cfg["generator"], help_cfg, sec);
265  topic child_topic(topic_cfg["title"], topic_cfg["id"], text);
266  if (!is_valid_id(child_topic.id)) {
267  std::stringstream ss;
268  ss << "Invalid ID, used for internal purpose: '" << id << "'";
269  throw parse_error(ss.str());
270  }
271  topics.push_back(child_topic);
272  }
273  else {
274  std::stringstream ss;
275  ss << "Help-topic '" << *it << "' referenced from '" << id
276  << "' but could not be found." << std::endl;
277  throw parse_error(ss.str());
278  }
279  }
280 
281  if (sort_topics) {
282  std::sort(topics.begin(),topics.end(), title_less());
283  std::sort(generated_topics.begin(),
284  generated_topics.end(), title_less());
285  std::merge(generated_topics.begin(),
286  generated_topics.end(),topics.begin(),topics.end()
287  ,std::back_inserter(sec.topics),title_less());
288  }
289  else {
290  sec.topics.insert(sec.topics.end(),
291  topics.begin(), topics.end());
292  sec.topics.insert(sec.topics.end(),
293  generated_topics.begin(), generated_topics.end());
294  }
295  }
296 }
297 
299 {
300  section sec;
301  if (cfg != nullptr) {
302  auto toplevel_cfg = cfg->optional_child("toplevel");
303  parse_config_internal(cfg, toplevel_cfg.ptr(), sec);
304  }
305  return sec;
306 }
307 
308 std::vector<topic> generate_topics(const bool sort_generated,const std::string &generator)
309 {
310  std::vector<topic> res;
311  if (generator.empty()) {
312  return res;
313  }
314 
315  if (generator == "abilities") {
316  res = generate_ability_topics(sort_generated);
317  } else if (generator == "weapon_specials") {
318  res = generate_weapon_special_topics(sort_generated);
319  } else if (generator == "time_of_days") {
320  res = generate_time_of_day_topics(sort_generated);
321  } else if (generator == "traits") {
322  res = generate_trait_topics(sort_generated);
323  } else {
324  std::vector<std::string> parts = utils::split(generator, ':', utils::STRIP_SPACES);
325  if (parts.size() > 1 && parts[0] == "units") {
326  res = generate_unit_topics(sort_generated, parts[1]);
327  } else if (parts[0] == "era" && parts.size()>1) {
328  res = generate_era_topics(sort_generated, parts[1]);
329  } else {
330  WRN_HP << "Found a topic generator that I didn't recognize: " << generator;
331  }
332  }
333 
334  return res;
335 }
336 
337 void generate_sections(const config *help_cfg, const std::string &generator, section &sec, int level)
338 {
339  if (generator == "races") {
340  generate_races_sections(help_cfg, sec, level);
341  } else if (generator == "terrains") {
343  } else if (generator == "eras") {
344  DBG_HP << "Generating eras...";
345  generate_era_sections(help_cfg, sec, level);
346  } else {
347  std::vector<std::string> parts = utils::split(generator, ':', utils::STRIP_SPACES);
348  if (parts.size() > 1 && parts[0] == "units") {
349  generate_unit_sections(help_cfg, sec, level, true, parts[1]);
350  } else if (generator.size() > 0) {
351  WRN_HP << "Found a section generator that I didn't recognize: " << generator;
352  }
353  }
354 }
355 
356 std::string generate_topic_text(const std::string &generator, const config *help_cfg, const section &sec)
357 {
358  std::string empty_string = "";
359  if (generator.empty()) {
360  return empty_string;
361  } else {
362  std::vector<std::string> parts = utils::split(generator, ':');
363  if (parts.size() > 1 && parts[0] == "contents") {
364  if (parts[1] == "generated") {
365  return generate_contents_links(sec);
366  } else {
367  return generate_contents_links(parts[1], help_cfg);
368  }
369  }
370  }
371  return empty_string;
372 }
373 
374 topic_text& topic_text::operator=(std::shared_ptr<topic_generator> g)
375 {
376  generator_ = std::move(g);
377  return *this;
378 }
379 
381 {
382  if (generator_) {
384  // This caches the result, so doesn't need the generator any more
385  generator_.reset();
386  }
387  return parsed_text_;
388 }
389 
390 static std::string time_of_day_bonus_colored(const int time_of_day_bonus)
391 {
392  return markup::span_color((time_of_day_bonus > 0 ? "green" : (time_of_day_bonus < 0 ? "red" : "white")), time_of_day_bonus);
393 }
394 
395 std::vector<topic> generate_time_of_day_topics(const bool /*sort_generated*/)
396 {
397  std::vector<topic> topics;
398  std::stringstream toplevel;
399 
400  if (!resources::tod_manager) {
401  toplevel << _("Only available during a scenario.");
402  topics.emplace_back(_("Time of Day Schedule"), "..schedule", toplevel.str());
403  return topics;
404  }
405 
406  const std::vector<time_of_day>& times = resources::tod_manager->times();
407  for (const time_of_day& time : times)
408  {
409  const std::string id = "time_of_day_" + time.id;
410  const std::string image = markup::img(time.image);
411  const std::string image_lawful = markup::img("icons/alignments/alignment_lawful_30.png");
412  const std::string image_neutral = markup::img("icons/alignments/alignment_neutral_30.png");
413  const std::string image_chaotic = markup::img("icons/alignments/alignment_chaotic_30.png");
414  const std::string image_liminal = markup::img("icons/alignments/alignment_liminal_30.png");
415  std::stringstream text, row_ss;
416 
417  const int lawful_bonus = generic_combat_modifier(time.lawful_bonus, unit_alignments::type::lawful, false, resources::tod_manager->get_max_liminal_bonus());
418  const int neutral_bonus = generic_combat_modifier(time.lawful_bonus, unit_alignments::type::neutral, false, resources::tod_manager->get_max_liminal_bonus());
419  const int chaotic_bonus = generic_combat_modifier(time.lawful_bonus, unit_alignments::type::chaotic, false, resources::tod_manager->get_max_liminal_bonus());
420  const int liminal_bonus = generic_combat_modifier(time.lawful_bonus, unit_alignments::type::liminal, false, resources::tod_manager->get_max_liminal_bonus());
421 
422  row_ss << markup::tag("col", markup::make_link(time.name.str(), id))
423  << markup::tag("col", image)
424  << markup::tag("col", image_lawful, time_of_day_bonus_colored(lawful_bonus))
425  << markup::tag("col", image_neutral, time_of_day_bonus_colored(neutral_bonus))
426  << markup::tag("col", image_chaotic, time_of_day_bonus_colored(chaotic_bonus))
427  << markup::tag("col", image_liminal, time_of_day_bonus_colored(liminal_bonus));
428  toplevel << markup::tag("row", row_ss.str());
429 
430  text << image << '\n'
431  << time.description.str() << '\n'
432  << image_lawful << _("Lawful Bonus:") << ' ' << time_of_day_bonus_colored(lawful_bonus) << '\n'
433  << image_neutral << _("Neutral Bonus:") << ' ' << time_of_day_bonus_colored(neutral_bonus) << '\n'
434  << image_chaotic << _("Chaotic Bonus:") << ' ' << time_of_day_bonus_colored(chaotic_bonus) << '\n'
435  << image_liminal << _("Liminal Bonus:") << ' ' << time_of_day_bonus_colored(liminal_bonus) << '\n' << '\n'
436  << markup::make_link(_("Schedule"), "..schedule");
437 
438  topics.emplace_back(time.name.str(), id, text.str());
439  }
440 
441  topics.emplace_back(_("Time of Day Schedule"), "..schedule", markup::tag("table", toplevel.str()));
442  return topics;
443 }
444 
445 std::vector<topic> generate_weapon_special_topics(const bool sort_generated)
446 {
447  std::vector<topic> topics;
448 
449  std::map<t_string, std::string> special_description;
450  std::map<t_string, std::set<std::string, string_less>> special_units;
451 
452  for (const unit_type_data::unit_type_map::value_type &type_mapping : unit_types.types())
453  {
454  const unit_type &type = type_mapping.second;
455  // Only show the weapon special if we find it on a unit that
456  // detailed description should be shown about.
458  continue;
459 
460  for (const attack_type& atk : type.attacks()) {
461 
462  std::vector<std::pair<t_string, t_string>> specials = atk.special_tooltips();
463  for ( std::size_t i = 0; i != specials.size(); ++i )
464  {
465  special_description.emplace(specials[i].first, specials[i].second);
466 
467  if (!type.hide_help()) {
468  //add a link in the list of units having this special
469  std::string type_name = type.type_name();
470  //check for variations (walking corpse/soulless etc)
471  const std::string section_prefix = type.show_variations_in_help() ? ".." : "";
472  std::string ref_id = section_prefix + unit_prefix + type.id();
473  //we put the translated name at the beginning of the hyperlink,
474  //so the automatic alphabetic sorting of std::set can use it
475  std::string link = markup::make_link(type_name, ref_id);
476  special_units[specials[i].first].insert(link);
477  }
478  }
479  }
480 
481  for(config adv : type.modification_advancements()) {
482  for(config effect : adv.child_range("effect")) {
483  if(effect["apply_to"] == "new_attack" && effect.has_child("specials")) {
484  for(const auto [key, special] : effect.mandatory_child("specials").all_children_view()) {
485  if(!special["name"].empty()) {
486  special_description.emplace(special["name"].t_str(), special["description"].t_str());
487  if(!type.hide_help()) {
488  //add a link in the list of units having this special
489  std::string type_name = type.type_name();
490  //check for variations (walking corpse/soulless etc)
491  const std::string section_prefix = type.show_variations_in_help() ? ".." : "";
492  std::string ref_id = section_prefix + unit_prefix + type.id();
493  //we put the translated name at the beginning of the hyperlink,
494  //so the automatic alphabetic sorting of std::set can use it
495  std::string link = markup::make_link(type_name, ref_id);
496  special_units[special["name"]].insert(link);
497  }
498  }
499  }
500  } else if(effect["apply_to"] == "attack" && effect.has_child("set_specials")) {
501  for(const auto [key, special] : effect.mandatory_child("set_specials").all_children_view()) {
502  if(!special["name"].empty()) {
503  special_description.emplace(special["name"].t_str(), special["description"].t_str());
504  if(!type.hide_help()) {
505  //add a link in the list of units having this special
506  std::string type_name = type.type_name();
507  //check for variations (walking corpse/soulless etc)
508  const std::string section_prefix = type.show_variations_in_help() ? ".." : "";
509  std::string ref_id = section_prefix + unit_prefix + type.id();
510  //we put the translated name at the beginning of the hyperlink,
511  //so the automatic alphabetic sorting of std::set can use it
512  std::string link = markup::make_link(type_name, ref_id);
513  special_units[special["name"]].insert(link);
514  }
515  }
516  }
517  }
518  }
519  }
520  }
521 
522  for (std::map<t_string, std::string>::iterator s = special_description.begin();
523  s != special_description.end(); ++s) {
524  // use untranslated name to have universal topic id
525  std::string id = "weaponspecial_" + s->first.base_str();
526  std::stringstream text;
527  text << s->second;
528  text << "\n\n" << _("<header>text='Units with this special attack'</header>") << "\n";
529  std::set<std::string, string_less> &units = special_units[s->first];
530  for (std::set<std::string, string_less>::iterator u = units.begin(); u != units.end(); ++u) {
531  text << font::unicode_bullet << " " << (*u) << "\n";
532  }
533 
534  topics.emplace_back(s->first, id, text.str());
535  }
536 
537  if (sort_generated)
538  std::sort(topics.begin(), topics.end(), title_less());
539  return topics;
540 }
541 
542 std::vector<topic> generate_ability_topics(const bool sort_generated)
543 {
544  std::vector<topic> topics;
545 
546  std::map<std::string, const unit_type::ability_metadata*> ability_topic_data;
547  std::map<std::string, std::set<std::string, string_less>> ability_units;
548 
549  const auto parse = [&](const unit_type& type, const unit_type::ability_metadata& ability) {
550  // NOTE: neither ability names nor ability ids are necessarily unique. Creating
551  // topics for either each unique name or each unique id means certain abilities
552  // will be excluded from help. So... the ability topic ref id is a combination
553  // of id and (untranslated) name. It's rather ugly, but it works.
554  const std::string topic_ref = ability.id + ability.name.base_str();
555 
556  ability_topic_data.emplace(topic_ref, &ability);
557 
558  if(!type.hide_help()) {
559  // Add a link in the list of units with this ability
560  // We put the translated name at the beginning of the hyperlink,
561  // so the automatic alphabetic sorting of std::set can use it
562  const std::string link = markup::make_link(type.type_name(), unit_prefix + type.id());
563  ability_units[topic_ref].insert(link);
564  }
565  };
566 
567  // Look through all the unit types. If a unit of that type would have a full
568  // description, add its abilities to the potential topic list. We don't want
569  // to show abilities that the user has not encountered yet.
570  for(const auto& type_mapping : unit_types.types()) {
571  const unit_type& type = type_mapping.second;
572 
574  continue;
575  }
576 
577  for(const unit_type::ability_metadata& ability : type.abilities_metadata()) {
578  parse(type, ability);
579  }
580 
581  for(const unit_type::ability_metadata& ability : type.adv_abilities_metadata()) {
582  parse(type, ability);
583  }
584  }
585 
586  for(const auto& a : ability_topic_data) {
587  if (a.second->name.empty()) {
588  continue;
589  }
590  std::ostringstream text;
591  text << a.second->description;
592  text << "\n\n" << _("<header>text='Units with this ability'</header>") << "\n";
593 
594  for(const auto& u : ability_units[a.first]) { // first is the topic ref id
595  text << font::unicode_bullet << " " << u << "\n";
596  }
597 
598  topics.emplace_back(a.second->name, ability_prefix + a.first, text.str());
599  }
600 
601  if(sort_generated) {
602  std::sort(topics.begin(), topics.end(), title_less());
603  }
604 
605  return topics;
606 }
607 
608 std::vector<topic> generate_era_topics(const bool sort_generated, const std::string & era_id)
609 {
610  std::vector<topic> topics;
611 
612  auto era = game_cfg->find_child("era","id", era_id);
613  if(era && !era["hide_help"].to_bool()) {
614  topics = generate_faction_topics(*era, sort_generated);
615 
616  std::vector<std::string> faction_links;
617  for (const topic & t : topics) {
618  faction_links.push_back(markup::make_link(t.title, t.id));
619  }
620 
621  std::stringstream text;
622  const config::attribute_value& description = era["description"];
623  if (!description.empty()) {
624  text << description.t_str() << "\n";
625  text << "\n";
626  }
627 
628  text << markup::tag("header", _("Factions")) << "\n";
629 
630  std::sort(faction_links.begin(), faction_links.end());
631  for (const std::string &link : faction_links) {
632  text << font::unicode_bullet << " " << link << "\n";
633  }
634 
635  topic era_topic(era["name"], ".." + era_prefix + era["id"].str(), text.str());
636 
637  topics.push_back( era_topic );
638  }
639  return topics;
640 }
641 
642 std::vector<topic> generate_faction_topics(const config & era, const bool sort_generated)
643 {
644  std::vector<topic> topics;
645  for (const config &f : era.child_range("multiplayer_side")) {
646  const std::string& id = f["id"];
647  if (id == "Random")
648  continue;
649 
650  std::stringstream text;
651 
652  const config::attribute_value& description = f["description"];
653  if (!description.empty()) {
654  text << description.t_str() << "\n";
655  text << "\n";
656  }
657 
658  const std::vector<std::string> recruit_ids = utils::split(f["recruit"]);
659  std::set<std::string> races;
660  std::set<std::string> alignments;
661 
662  for (const std::string & u_id : recruit_ids) {
663  if (const unit_type * t = unit_types.find(u_id, unit_type::HELP_INDEXED)) {
664  assert(t);
665  const unit_type & type = *t;
666 
667  if (const unit_race *r = unit_types.find_race(type.race_id())) {
668  races.insert(markup::make_link(r->plural_name(), std::string("..") + race_prefix + r->id()));
669  }
670  alignments.insert(markup::make_link(type.alignment_description(type.alignment(), type.genders().front()), "time_of_day"));
671  }
672  }
673 
674  if (!races.empty()) {
675  std::set<std::string>::iterator it = races.begin();
676  text << _("Races: ") << *(it++);
677  while(it != races.end()) {
678  text << ", " << *(it++);
679  }
680  text << "\n\n";
681  }
682 
683  if (!alignments.empty()) {
684  std::set<std::string>::iterator it = alignments.begin();
685  text << _("Alignments: ") << *(it++);
686  while(it != alignments.end()) {
687  text << ", " << *(it++);
688  }
689  text << "\n\n";
690  }
691 
692  text << markup::tag("header", _("Leaders")) << "\n";
693  const std::vector<std::string> leaders =
694  make_unit_links_list( utils::split(f["leader"]), true );
695  for (const std::string &link : leaders) {
696  text << font::unicode_bullet << " " << link << "\n";
697  }
698 
699  text << "\n";
700 
701  text << markup::tag("header", _("Recruits")) << "\n";
702  const std::vector<std::string> recruit_links =
703  make_unit_links_list( recruit_ids, true );
704  for (const std::string &link : recruit_links) {
705  text << font::unicode_bullet << " " << link << "\n";
706  }
707 
708  const std::string name = f["name"];
709  const std::string ref_id = faction_prefix + era["id"] + "_" + id;
710  topics.emplace_back(name, ref_id, text.str());
711  }
712  if (sort_generated)
713  std::sort(topics.begin(), topics.end(), title_less());
714  return topics;
715 }
716 
717 std::vector<topic> generate_trait_topics(const bool sort_generated)
718 {
719  // All traits that could be assigned to at least one discovered or HIDDEN_BUT_SHOW_MACROS unit.
720  // This is collected from the [units][trait], [race][traits], and [unit_type][traits] tags. If
721  // there are duplicates with the same id, it takes the first one encountered.
722  std::map<std::string, const config> trait_list;
723 
724  // The global traits that are direct children of a [units] tag
725  for (const config & trait : unit_types.traits()) {
726  trait_list.emplace(trait["id"], trait);
727  }
728 
729  // Search for discovered unit types
730  std::set<std::string> races;
731  for(const auto& i : unit_types.types()) {
732  const unit_type& type = i.second;
734 
735  // Remember which races have been discovered.
736  //
737  // For unit types, unit_type::possible_traits() usually includes racial traits; however it's
738  // possible that all discovered units of a race have ignore_race_traits=yes, and so we still
739  // need to loop over the [race] tags looking for more traits.
740  if(desc_type == FULL_DESCRIPTION) {
741  races.insert(type.race_id());
742  }
743 
744  // Handle [unit_type][trait]s.
745  //
746  // It would be better if we only looked at the traits that are specific to the unit_type,
747  // but that unmerged unit_type_data.traits() isn't available. We're forced to use
748  // possible_traits() instead which returns all of the traits, including the ones that units
749  // with ignore_race_traits=no have inherited from their [race] tag.
750  if (desc_type == FULL_DESCRIPTION || desc_type == HIDDEN_BUT_SHOW_MACROS) {
751  for (const config& trait : type.possible_traits()) {
752  trait_list.emplace(trait["id"], trait);
753  }
754  }
755  }
756 
757  // Race traits, even those that duplicate a global trait (which will be dropped by emplace()).
758  //
759  // For traits, assume we don't discover additional races via the [race]help_taxonomy= links. The
760  // traits themselves don't propagate down those links, so if the trait is interesting w.r.t. the
761  // discovered units then their own race will already include it.
762  for(const auto& race_id : races) {
763  if(const unit_race *r = unit_types.find_race(race_id)) {
764  for(const config & trait : r->additional_traits()) {
765  trait_list.emplace(trait["id"], trait);
766  }
767  }
768  }
769 
770  std::vector<topic> topics;
771  for(auto& a : trait_list) {
772  std::string id = "traits_" + a.first;
773  const config& trait = a.second;
774 
775  std::string name = trait["male_name"].str();
776  if (name.empty()) name = trait["female_name"].str();
777  if (name.empty()) name = trait["name"].str();
778  if (name.empty()) continue; // Hidden trait
779 
780  std::stringstream text;
781  if (!trait["help_text"].empty()) {
782  text << trait["help_text"];
783  } else if (!trait["description"].empty()) {
784  text << trait["description"];
785  } else {
786  text << _("No description available.");
787  }
788  text << "\n\n";
789 
790  topics.emplace_back(name, id, text.str());
791  }
792 
793  if (sort_generated)
794  std::sort(topics.begin(), topics.end(), title_less());
795  return topics;
796 }
797 
798 
799 std::string make_unit_link(const std::string& type_id)
800 {
801  std::string link;
802 
804  if (!type) {
805  PLAIN_LOG << "Unknown unit type : " << type_id;
806  // don't return an hyperlink (no page)
807  // instead show the id (as hint)
808  link = type_id;
809  } else if (!type->hide_help()) {
810  std::string name = type->type_name();
811  std::string ref_id;
813  const std::string section_prefix = type->show_variations_in_help() ? ".." : "";
814  ref_id = section_prefix + unit_prefix + type->id();
815  } else {
816  ref_id = unknown_unit_topic;
817  name += " (?)";
818  }
819  link = markup::make_link(name, ref_id);
820  } // if hide_help then link is an empty string
821 
822  return link;
823 }
824 
825 std::vector<std::string> make_unit_links_list(const std::vector<std::string>& type_id_list, bool ordered)
826 {
827  std::vector<std::string> links_list;
828  for (const std::string &type_id : type_id_list) {
829  std::string unit_link = make_unit_link(type_id);
830  if (!unit_link.empty())
831  links_list.push_back(unit_link);
832  }
833 
834  if (ordered)
835  std::sort(links_list.begin(), links_list.end());
836 
837  return links_list;
838 }
839 
840 void generate_races_sections(const config* help_cfg, section& sec, int level)
841 {
842  std::set<std::string, string_less> races;
843  std::set<std::string, string_less> visible_races;
844 
845  // Calculate which races have been discovered, from the list of discovered unit types.
846  for(const auto& i : unit_types.types()) {
847  const unit_type& type = i.second;
849  if(desc_type == FULL_DESCRIPTION) {
850  races.insert(type.race_id());
851  if(!type.hide_help())
852  visible_races.insert(type.race_id());
853  }
854  }
855 
856  // Propagate visibility up the help_taxonomy tree.
857  std::set<std::string, string_less> last_sweep = visible_races;
858  while(!last_sweep.empty()) {
859  std::set<std::string, string_less> current_sweep;
860  for(const auto& race_id : last_sweep) {
861  if(const unit_race* r = unit_types.find_race(race_id)) {
862  const auto& help_taxonomy = r->help_taxonomy();
863  if(!help_taxonomy.empty() && !visible_races.count(help_taxonomy) && unit_types.find_race(help_taxonomy)) {
864  current_sweep.insert(help_taxonomy);
865  races.insert(help_taxonomy);
866  visible_races.insert(help_taxonomy);
867  }
868  }
869  }
870  last_sweep = std::move(current_sweep);
871  }
872 
873  struct taxonomy_queue_type
874  {
875  std::string parent_id;
876  section content;
877  };
878  std::vector<taxonomy_queue_type> taxonomy_queue;
879 
880  // Add all races without a [race]help_taxonomy= to the documentation section, and queue the others.
881  // This avoids a race condition dependency on the order that races are encountered in help_cfg.
882  for(const auto& race_id : races) {
883  section race_section;
884  config section_cfg;
885 
886  bool hidden = (visible_races.count(race_id) == 0);
887 
888  section_cfg["id"] = hidden_symbol(hidden) + race_prefix + race_id;
889 
890  std::string title;
891  std::string help_taxonomy;
892  if(const unit_race* r = unit_types.find_race(race_id)) {
893  title = r->plural_name();
894  help_taxonomy = r->help_taxonomy();
895  } else {
896  title = _("race^Miscellaneous");
897  // leave help_taxonomy empty
898  }
899  section_cfg["title"] = title;
900 
901  section_cfg["sections_generator"] = "units:" + race_id;
902  section_cfg["generator"] = "units:" + race_id;
903 
904  parse_config_internal(help_cfg, &section_cfg, race_section, level + 1);
905 
906  if(help_taxonomy.empty()) {
907  sec.add_section(race_section);
908  } else {
909  bool parent_hidden = (visible_races.count(help_taxonomy) == 0);
910  auto parent_id = hidden_symbol(parent_hidden) + race_prefix + help_taxonomy;
911  taxonomy_queue.push_back({std::move(parent_id), std::move(race_section)});
912  }
913  }
914 
915  // Each run through this loop handles one level of nesting of [race]help_taxonomy=
916  bool process_queue_again = true;
917  while(process_queue_again && !taxonomy_queue.empty()) {
918  process_queue_again = false;
919  auto to_process = std::exchange(taxonomy_queue, {});
920 
921  for(auto& x : to_process) {
922  auto parent = find_section(sec, x.parent_id);
923  if(parent) {
924  parent->add_section(x.content);
925  process_queue_again = true;
926  } else {
927  taxonomy_queue.push_back(std::move(x));
928  }
929  }
930  }
931 
932  // Fallback to adding the new race at the top level, as if it had help_taxonomy.empty().
933  for(auto& x : taxonomy_queue) {
934  sec.add_section(x.content);
935  }
936 }
937 
938 void generate_era_sections(const config* help_cfg, section & sec, int level)
939 {
940  for (const config & era : game_cfg->child_range("era")) {
941  if (era["hide_help"].to_bool()) {
942  continue;
943  }
944 
945  DBG_HP << "Adding help section: " << era["id"].str();
946 
947  section era_section;
948  config section_cfg;
949  section_cfg["id"] = era_prefix + era["id"].str();
950  section_cfg["title"] = era["name"];
951 
952  section_cfg["generator"] = "era:" + era["id"].str();
953 
954  DBG_HP << section_cfg.debug();
955 
956  parse_config_internal(help_cfg, &section_cfg, era_section, level+1);
957  sec.add_section(era_section);
958  }
959 }
960 
961 void generate_terrain_sections(section& sec, int /*level*/)
962 {
963  std::shared_ptr<terrain_type_data> tdata = load_terrain_types_data();
964 
965  if (!tdata) {
966  WRN_HP << "When building terrain help sections, couldn't acquire terrain types data, aborting.";
967  return;
968  }
969 
970  std::map<std::string, section> base_map;
971 
972  const t_translation::ter_list& t_listi = tdata->list();
973 
974  for (const t_translation::terrain_code& t : t_listi) {
975 
976  const terrain_type& info = tdata->get_terrain_info(t);
977 
978  bool hidden = info.hide_help();
979 
980  if (prefs::get().encountered_terrains().find(t)
981  == prefs::get().encountered_terrains().end() && !info.is_overlay())
982  hidden = true;
983 
984  topic terrain_topic;
985  terrain_topic.title = info.editor_name();
986  terrain_topic.id = hidden_symbol(hidden) + terrain_prefix + info.id();
987  terrain_topic.text = std::make_shared<terrain_topic_generator>(info);
988 
989  t_translation::ter_list base_terrains = tdata->underlying_union_terrain(t);
990  if (info.has_default_base()) {
991  for (const auto base : tdata->underlying_union_terrain(info.default_base())) {
992  if (!utils::contains(base_terrains, base)) {
993  base_terrains.emplace_back(base);
994  }
995  }
996  }
997  for (const t_translation::terrain_code& base : base_terrains) {
998 
999  const terrain_type& base_info = tdata->get_terrain_info(base);
1000 
1001  if (!base_info.is_nonnull() || base_info.hide_help())
1002  continue;
1003 
1004  section& base_section = base_map[base_info.id()];
1005 
1006  base_section.id = terrain_prefix + base_info.id();
1007  base_section.title = base_info.editor_name();
1008 
1009  if (base_info.id() == info.id())
1010  terrain_topic.id = ".." + terrain_prefix + info.id();
1011  base_section.topics.push_back(terrain_topic);
1012  }
1013  }
1014 
1015  for (const auto& base : base_map) {
1016  sec.add_section(base.second);
1017  }
1018 }
1019 
1020 void generate_unit_sections(const config* /*help_cfg*/, section& sec, int /*level*/, const bool /*sort_generated*/, const std::string& race)
1021 {
1022  for (const unit_type_data::unit_type_map::value_type &i : unit_types.types()) {
1023  const unit_type &type = i.second;
1024 
1025  if (type.race_id() != race)
1026  continue;
1027 
1028  if (!type.show_variations_in_help())
1029  continue;
1030 
1031  section base_unit;
1032  for (const std::string &variation_id : type.variations()) {
1033  // TODO: Do we apply encountered stuff to variations?
1034  const unit_type &var_type = type.get_variation(variation_id);
1035  const std::string topic_name = var_type.variation_name();
1036  const std::string var_ref = hidden_symbol(var_type.hide_help()) + variation_prefix + var_type.id() + "_" + variation_id;
1037 
1038  topic var_topic(topic_name, var_ref, "");
1039  var_topic.text = std::make_shared<unit_topic_generator>(var_type, variation_id);
1040  base_unit.topics.push_back(var_topic);
1041  }
1042 
1043  const std::string type_name = type.type_name();
1044  const std::string ref_id = hidden_symbol(type.hide_help()) + unit_prefix + type.id();
1045 
1046  base_unit.id = ref_id;
1047  base_unit.title = type_name;
1048 
1049  sec.add_section(base_unit);
1050  }
1051 }
1052 
1053 std::vector<topic> generate_unit_topics(const bool sort_generated, const std::string& race)
1054 {
1055  std::vector<topic> topics;
1056  std::set<std::string, string_less> race_units;
1057  std::set<std::string, string_less> race_topics;
1058  std::set<std::string> alignments;
1059 
1060  for (const unit_type_data::unit_type_map::value_type &i : unit_types.types())
1061  {
1062  const unit_type &type = i.second;
1063 
1064  if (type.race_id() != race)
1065  continue;
1066 
1068  if (desc_type != FULL_DESCRIPTION)
1069  continue;
1070 
1071  const std::string debug_suffix = (game_config::debug ? " (" + type.id() + ")" : "");
1072  const std::string type_name = type.type_name() + (type.id() == type.type_name().str() ? "" : debug_suffix);
1073  const std::string real_prefix = type.show_variations_in_help() ? ".." : "";
1074  const std::string ref_id = hidden_symbol(type.hide_help()) + real_prefix + unit_prefix + type.id();
1075  topic unit_topic(type_name, ref_id, "");
1076  unit_topic.text = std::make_shared<unit_topic_generator>(type);
1077  topics.push_back(unit_topic);
1078 
1079  if (!type.hide_help()) {
1080  // we also record an hyperlink of this unit
1081  // in the list used for the race topic
1082  std::string link = markup::make_link(type_name, ref_id);
1083  race_units.insert(link);
1084 
1085  alignments.insert(markup::make_link(type.alignment_description(type.alignment(), type.genders().front()), "time_of_day"));
1086  }
1087  }
1088 
1089  //generate the hidden race description topic
1090  std::string race_id = "..race_"+race;
1091  std::string race_name;
1092  std::string race_description;
1093  std::string race_help_taxonomy;
1094  if (const unit_race *r = unit_types.find_race(race)) {
1095  race_name = r->plural_name();
1096  race_description = r->description();
1097  race_help_taxonomy = r->help_taxonomy();
1098  // if (description.empty()) description = _("No description Available");
1099  for (const config &additional_topic : r->additional_topics())
1100  {
1101  std::string id = additional_topic["id"];
1102  std::string title = additional_topic["title"];
1103  std::string text = additional_topic["text"];
1104  //topic additional_topic(title, id, text);
1105  topics.emplace_back(title,id,text);
1106  std::string link = markup::make_link(title, id);
1107  race_topics.insert(link);
1108  }
1109  } else {
1110  race_name = _ ("race^Miscellaneous");
1111  // description = _("Here put the description of the Miscellaneous race");
1112  }
1113 
1114  // Find any other races whose [race]help_taxonomy points to the current race
1115  std::map<std::string, t_string> subgroups;
1116  for (const auto &r : unit_types.races()) {
1117  if (r.second.help_taxonomy() == race) {
1118  if (!r.second.plural_name().empty())
1119  subgroups[r.first] = r.second.plural_name();
1120  else
1121  subgroups[r.first] = r.first;
1122  }
1123  }
1124 
1125  std::stringstream text;
1126 
1127  if (!race_description.empty()) {
1128  text << race_description << "\n\n";
1129  }
1130 
1131  if (!alignments.empty()) {
1132  std::set<std::string>::iterator it = alignments.begin();
1133  text << (alignments.size() > 1 ? _("Alignments: ") : _("Alignment: ")) << *(it++);
1134  while(it != alignments.end()) {
1135  text << ", " << *(it++);
1136  }
1137  text << "\n\n";
1138  }
1139 
1140  if (!race_help_taxonomy.empty()) {
1141  utils::string_map symbols;
1142  symbols["topic_id"] = "..race_"+race_help_taxonomy;
1143  if (const unit_race *r = unit_types.find_race(race_help_taxonomy)) {
1144  symbols["help_taxonomy"] = r->plural_name();
1145  } else {
1146  // Fall back to using showing the race id for the race that we couldn't find.
1147  // Not great, but probably useful if UMC has a broken link.
1148  symbols["help_taxonomy"] = race_help_taxonomy;
1149  }
1150  // TRANSLATORS: this is expected to say "[Dunefolk are] a group of units, all of whom are Humans",
1151  // or "[Quenoth Elves are] a group of units, all of whom are Elves".
1152  text << VGETTEXT("This is a group of units, all of whom are <ref>dst='$topic_id' text='$help_taxonomy'</ref>.", symbols) << "\n\n";
1153  }
1154 
1155  if (!subgroups.empty()) {
1156  if (!race_help_taxonomy.empty()) {
1157  text << _("<header>text='Subgroups of units within this group'</header>") << "\n";
1158  } else {
1159  text << _("<header>text='Groups of units within this race'</header>") << "\n";
1160  }
1161  for (const auto &sg : subgroups) {
1162  text << font::unicode_bullet << " " << markup::make_link(sg.second, "..race_" + sg.first) << "\n";
1163  }
1164  text << "\n";
1165  }
1166 
1167  if (!race_help_taxonomy.empty()) {
1168  text << _("<header>text='Units of this group'</header>") << "\n";
1169  } else {
1170  text << _("<header>text='Units of this race'</header>") << "\n";
1171  }
1172  for (const auto &u : race_units) {
1173  text << font::unicode_bullet << " " << u << "\n";
1174  }
1175 
1176  topics.emplace_back(race_name, race_id, text.str());
1177 
1178  if (sort_generated)
1179  std::sort(topics.begin(), topics.end(), title_less());
1180 
1181  return topics;
1182 }
1183 
1185 {
1186  if (game_config::debug || prefs::get().show_all_units_in_help() ||
1188  return FULL_DESCRIPTION;
1189  }
1190 
1191  const std::set<std::string> &encountered_units = prefs::get().encountered_units();
1192  if (encountered_units.find(type.id()) != encountered_units.end()) {
1193  return FULL_DESCRIPTION;
1194  }
1195 
1196  // See the docs of HIDDEN_BUT_SHOW_MACROS
1197  if (type.id() == "Fog Clearer") {
1198  return HIDDEN_BUT_SHOW_MACROS;
1199  }
1200 
1201  return NO_DESCRIPTION;
1202 }
1203 
1204 std::string generate_contents_links(const std::string& section_name, config const *help_cfg)
1205 {
1206  auto section_cfg = help_cfg->find_child("section", "id", section_name);
1207  if (!section_cfg) {
1208  return std::string();
1209  }
1210 
1211  std::ostringstream res;
1212 
1213  std::vector<std::string> topics = utils::quoted_split(section_cfg["topics"]);
1214 
1215  // we use an intermediate structure to allow a conditional sorting
1216  typedef std::pair<std::string,std::string> link;
1217  std::vector<link> topics_links;
1218 
1220  // Find all topics in this section.
1221  for (t = topics.begin(); t != topics.end(); ++t) {
1222  if (auto topic_cfg = help_cfg->find_child("topic", "id", *t)) {
1223  std::string id = topic_cfg["id"];
1224  if (is_visible_id(id))
1225  topics_links.emplace_back(topic_cfg["title"], id);
1226  }
1227  }
1228 
1229  if (section_cfg["sort_topics"] == "yes") {
1230  std::sort(topics_links.begin(),topics_links.end());
1231  }
1232 
1234  for (l = topics_links.begin(); l != topics_links.end(); ++l) {
1235  std::string link = markup::make_link(l->first, l->second);
1236  res << font::unicode_bullet << " " << link << "\n";
1237  }
1238 
1239  return res.str();
1240 }
1241 
1242 std::string generate_contents_links(const section &sec)
1243 {
1244  std::stringstream res;
1245 
1246  for (auto &s : sec.sections) {
1247  if (is_visible_id(s.id)) {
1248  std::string link = markup::make_link(s.title, ".."+s.id);
1249  res << font::unicode_bullet << " " << link << "\n";
1250  }
1251  }
1252 
1253  for(const topic& t : sec.topics) {
1254  if (is_visible_id(t.id)) {
1255  std::string link = markup::make_link(t.title, t.id);
1256  res << font::unicode_bullet << " " << link << "\n";
1257  }
1258  }
1259  return res.str();
1260 }
1261 
1262 bool topic::operator==(const topic &t) const
1263 {
1264  return t.id == id;
1265 }
1266 
1267 bool topic::operator<(const topic &t) const
1268 {
1269  return id < t.id;
1270 }
1271 
1272 bool section::operator==(const section &sec) const
1273 {
1274  return sec.id == id;
1275 }
1276 
1277 bool section::operator<(const section &sec) const
1278 {
1279  return id < sec.id;
1280 }
1281 
1283 {
1284  sections.emplace_back(s);
1285 }
1286 
1288 {
1289  topics.clear();
1290  sections.clear();
1291 }
1292 
1293 const topic *find_topic(const section &sec, const std::string &id)
1294 {
1295  topic_list::const_iterator tit =
1296  std::find_if(sec.topics.begin(), sec.topics.end(), has_id(id));
1297  if (tit != sec.topics.end()) {
1298  return &(*tit);
1299  }
1300  for (const auto &s : sec.sections) {
1301  const auto t = find_topic(s, id);
1302  if (t != nullptr) {
1303  return t;
1304  }
1305  }
1306  return nullptr;
1307 }
1308 
1309 const section *find_section(const section &sec, const std::string &id)
1310 {
1311  const auto sit =
1312  std::find_if(sec.sections.begin(), sec.sections.end(), has_id(id));
1313  if (sit != sec.sections.end()) {
1314  return &*sit;
1315  }
1316  for (const auto &subsection : sec.sections) {
1317  const auto s = find_section(subsection, id);
1318  if (s != nullptr) {
1319  return s;
1320  }
1321  }
1322  return nullptr;
1323 }
1324 
1325 section *find_section(section &sec, const std::string &id)
1326 {
1327  return const_cast<section *>(find_section(const_cast<const section &>(sec), id));
1328 }
1329 
1330 std::string remove_first_space(const std::string& text)
1331 {
1332  if (text.length() > 0 && text[0] == ' ') {
1333  return text.substr(1);
1334  }
1335  return text;
1336 }
1337 
1338 std::string get_first_word(const std::string &s)
1339 {
1340  std::size_t first_word_start = s.find_first_not_of(' ');
1341  if (first_word_start == std::string::npos) {
1342  return s;
1343  }
1344  std::size_t first_word_end = s.find_first_of(" \n", first_word_start);
1345  if( first_word_end == first_word_start ) {
1346  // This word is '\n'.
1347  first_word_end = first_word_start+1;
1348  }
1349 
1350  //if no gap(' ' or '\n') found, test if it is CJK character
1351  std::string re = s.substr(0, first_word_end);
1352 
1353  utf8::iterator ch(re);
1354  if (ch == utf8::iterator::end(re))
1355  return re;
1356 
1357  char32_t firstchar = *ch;
1358  if (is_cjk_char(firstchar)) {
1359  re = unicode_cast<std::string>(firstchar);
1360  }
1361  return re;
1362 }
1363 
1365 {
1368  if (game_cfg != nullptr) {
1369  const config *help_config = &game_cfg->child_or_empty("help");
1370  try {
1371  default_toplevel = parse_config(help_config);
1372  // Create a config object that contains everything that is
1373  // not referenced from the toplevel element. Read this
1374  // config and save these sections and topics so that they
1375  // can be referenced later on when showing help about
1376  // specified things, but that should not be shown when
1377  // opening the help browser in the default manner.
1378  config hidden_toplevel;
1379  std::stringstream ss;
1380  for (const config &section : help_config->child_range("section"))
1381  {
1382  const std::string id = section["id"];
1383  if (find_section(default_toplevel, id) == nullptr) {
1384  // This section does not exist referenced from the
1385  // toplevel. Hence, add it to the hidden ones if it
1386  // is not referenced from another section.
1387  if (!section_is_referenced(id, *help_config)) {
1388  if (!ss.str().empty()) {
1389  ss << ",";
1390  }
1391  ss << id;
1392  }
1393  }
1394  }
1395  hidden_toplevel["sections"] = ss.str();
1396  ss.str("");
1397  for (const config &topic : help_config->child_range("topic"))
1398  {
1399  const std::string id = topic["id"];
1400  if (find_topic(default_toplevel, id) == nullptr) {
1401  if (!topic_is_referenced(id, *help_config)) {
1402  if (!ss.str().empty()) {
1403  ss << ",";
1404  }
1405  ss << id;
1406  }
1407  }
1408  }
1409  hidden_toplevel["topics"] = ss.str();
1410  config hidden_cfg = *help_config;
1411  // Change the toplevel to our new, custom built one.
1412  hidden_cfg.clear_children("toplevel");
1413  hidden_cfg.add_child("toplevel", std::move(hidden_toplevel));
1414  hidden_sections = parse_config(&hidden_cfg);
1415  }
1416  catch (parse_error& e) {
1417  std::stringstream msg;
1418  msg << "Parse error when parsing help text: '" << e.message << "'";
1419  PLAIN_LOG << msg.str();
1420  }
1421  }
1422 }
1423 
1424 // id starting with '.' are hidden
1425 std::string hidden_symbol(bool hidden) {
1426  return (hidden ? "." : "");
1427 }
1428 
1429 bool is_visible_id(const std::string &id) {
1430  return (id.empty() || id[0] != '.');
1431 }
1432 
1433 /**
1434  * Return true if the id is valid for user defined topics and
1435  * sections. Some IDs are special, such as toplevel and may not be
1436  * be defined in the config.
1437  */
1438 bool is_valid_id(const std::string &id) {
1439  if (id == "toplevel") {
1440  return false;
1441  }
1442  if (id.compare(0, unit_prefix.length(), unit_prefix) == 0 || id.compare(hidden_symbol().length(), unit_prefix.length(), unit_prefix) == 0) {
1443  return false;
1444  }
1445  if (id.compare(0, 8, "ability_") == 0) {
1446  return false;
1447  }
1448  if (id.compare(0, 14, "weaponspecial_") == 0) {
1449  return false;
1450  }
1451  if (id == "hidden") {
1452  return false;
1453  }
1454  return true;
1455 }
1456 
1457 /** Load the appropriate terrain types data to use */
1458 std::shared_ptr<terrain_type_data> load_terrain_types_data()
1459 {
1460  if (game_config_manager::get()){
1462  } else {
1463  return {};
1464  }
1465 }
1466 
1467 
1468 } // end namespace help
int generic_combat_modifier(int lawful_bonus, unit_alignments::type alignment, bool is_fearless, int max_liminal_bonus)
Returns the amount that a unit's damage should be multiplied by due to a given lawful_bonus.
Definition: attack.cpp:1620
Various functions that implement attacks and attack calculations.
double t
Definition: astarsearch.cpp:63
double g
Definition: astarsearch.cpp:63
Variant for storing WML attributes.
bool empty() const
Tests for an attribute that either was never set or was set to "".
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
optional_config_impl< config > find_child(config_key_type key, const std::string &name, const std::string &value)
Returns the first child of tag key with a name attribute containing value.
Definition: config.cpp:780
void clear_children(T... keys)
Definition: config.hpp:602
child_itors child_range(config_key_type key)
Definition: config.cpp:268
std::string debug() const
Definition: config.cpp:1236
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:380
config & add_child(config_key_type key)
Definition: config.cpp:436
static game_config_manager * get()
const std::shared_ptr< terrain_type_data > & terrain_types() const
A class grating read only view to a vector of config objects, viewed as one config with all children ...
const config & child_or_empty(config_key_type key) const
optional_const_config find_child(config_key_type key, const std::string &name, const std::string &value) const
config_array_view child_range(config_key_type key) const
To be used as a function object to locate sections and topics with a specified ID.
Definition: help_impl.hpp:176
To be used as a function object when sorting section lists on the title.
Definition: help_impl.hpp:198
To be used as a function object when sorting topic lists on the title.
Definition: help_impl.hpp:188
The text displayed in a topic.
Definition: help_impl.hpp:85
std::shared_ptr< topic_generator > generator_
Definition: help_impl.hpp:87
topic_text & operator=(topic_text &&t)=default
const config & parsed_text() const
Definition: help_impl.cpp:380
config parsed_text_
Definition: help_impl.hpp:86
std::set< std::string > & encountered_units()
static prefs & get()
bool hide_help() const
For instances created from a [terrain_type] tag, the value in the tag (with default false).
Definition: terrain.hpp:61
bool is_nonnull() const
True if this object represents some sentinel values.
Definition: terrain.hpp:129
const std::string & id() const
Definition: terrain.hpp:52
const t_string & editor_name() const
Definition: terrain.hpp:49
int get_max_liminal_bonus() const
const std::vector< time_of_day > & times(const map_location &loc=map_location::null_location()) const
static iterator_base end(const string_type &str)
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.
Definition: types.cpp:1265
const race_map & races() const
Definition: types.hpp:406
const unit_race * find_race(const std::string &) const
Definition: types.cpp:1369
const unit_type_map & types() const
Definition: types.hpp:396
config_array_view traits() const
Definition: types.hpp:408
A single unit type that the player may recruit.
Definition: types.hpp:43
const std::string & id() const
The id for this unit_type.
Definition: types.hpp:141
@ HELP_INDEXED
Definition: types.hpp:74
bool hide_help() const
Definition: types.cpp:626
const t_string & variation_name() const
Definition: types.hpp:174
#define VGETTEXT(msgid,...)
Handy wrappers around interpolate_variables_into_string and gettext.
std::size_t i
Definition: function.cpp:1022
static std::string _(const char *str)
Definition: gettext.hpp:103
std::string id
Text to match against addon_info.tags()
Definition: manager.cpp:199
#define DBG_HP
Definition: help_impl.cpp:57
#define WRN_HP
Definition: help_impl.cpp:56
static lg::log_domain log_help("help")
Standard logging facilities (interface).
#define PLAIN_LOG
Definition: log.hpp:296
const int SIZE_LARGE
Definition: constants.cpp:30
const int SIZE_PLUS
Definition: constants.cpp:29
const std::string unicode_bullet
Definition: constants.cpp:47
const int SIZE_NORMAL
Definition: constants.cpp:20
const bool & debug
Definition: game_config.cpp:95
std::string get_first_word(const std::string &s)
Return the first word in s, not removing any spaces in the start of it.
Definition: help_impl.cpp:1338
std::vector< topic > generate_unit_topics(const bool sort_generated, const std::string &race)
Definition: help_impl.cpp:1053
std::string hidden_symbol(bool hidden)
Definition: help_impl.cpp:1425
void generate_unit_sections(const config *, section &sec, int, const bool, const std::string &race)
Definition: help_impl.cpp:1020
UNIT_DESCRIPTION_TYPE
Definition: help_impl.hpp:244
@ FULL_DESCRIPTION
Definition: help_impl.hpp:245
@ HIDDEN_BUT_SHOW_MACROS
Although the unit itself is hidden, traits reachable via this unit are not hidden.
Definition: help_impl.hpp:256
@ NO_DESCRIPTION
Ignore this unit for documentation purposes.
Definition: help_impl.hpp:247
void generate_races_sections(const config *help_cfg, section &sec, int level)
Definition: help_impl.cpp:840
bool topic_is_referenced(const std::string &topic_id, const config &cfg)
Return true if the topic with id topic_id is referenced from another section in the config,...
Definition: help_impl.cpp:171
std::vector< topic > generate_time_of_day_topics(const bool)
Definition: help_impl.cpp:395
int last_num_encountered_units
Definition: help_impl.cpp:67
void generate_terrain_sections(section &sec, int)
Definition: help_impl.cpp:961
std::string make_unit_link(const std::string &type_id)
return a hyperlink with the unit's name and pointing to the unit page return empty string if this uni...
Definition: help_impl.cpp:799
const std::string open_section_img
Definition: help_impl.cpp:80
const std::string unit_prefix
Definition: help_impl.cpp:84
const std::string variation_prefix
Definition: help_impl.cpp:89
void parse_config_internal(const config *help_cfg, const config *section_cfg, section &sec, int level)
Recursive function used by parse_config.
Definition: help_impl.cpp:195
bool is_visible_id(const std::string &id)
Definition: help_impl.cpp:1429
const std::string closed_section_img
Definition: help_impl.cpp:79
std::vector< topic > generate_faction_topics(const config &era, const bool sort_generated)
Definition: help_impl.cpp:642
const int box_width
Definition: help_impl.cpp:75
void generate_sections(const config *help_cfg, const std::string &generator, section &sec, int level)
Dispatch generators to their appropriate functions.
Definition: help_impl.cpp:337
const std::string topic_img
Definition: help_impl.cpp:78
UNIT_DESCRIPTION_TYPE description_type(const unit_type &type)
Return the type of description that should be shown for a unit of the given kind.
Definition: help_impl.cpp:1184
const std::string race_prefix
Definition: help_impl.cpp:86
const std::string ability_prefix
Definition: help_impl.cpp:90
std::vector< std::string > make_unit_links_list(const std::vector< std::string > &type_id_list, bool ordered)
return a list of hyperlinks to unit's pages (ordered or not)
Definition: help_impl.cpp:825
std::vector< topic > generate_topics(const bool sort_generated, const std::string &generator)
Definition: help_impl.cpp:308
const section * find_section(const section &sec, const std::string &id)
Search for the section with the specified identifier in the section and its subsections.
Definition: help_impl.cpp:1309
const int title2_size
Definition: help_impl.cpp:74
static std::string time_of_day_bonus_colored(const int time_of_day_bonus)
Definition: help_impl.cpp:390
const int normal_font_size
Definition: help_impl.cpp:76
void generate_contents()
Generate the help contents from the configurations given to the manager.
Definition: help_impl.cpp:1364
const std::string terrain_prefix
Definition: help_impl.cpp:85
std::vector< topic > generate_weapon_special_topics(const bool sort_generated)
Definition: help_impl.cpp:445
std::vector< topic > generate_ability_topics(const bool sort_generated)
Definition: help_impl.cpp:542
std::string remove_first_space(const std::string &text)
Definition: help_impl.cpp:1330
const std::string unknown_unit_topic
Definition: help_impl.cpp:83
const int max_section_level
Definition: help_impl.cpp:72
const unsigned max_history
Definition: help_impl.cpp:77
std::vector< std::string > empty_string_vector
Definition: help_impl.cpp:71
bool section_is_referenced(const std::string &section_id, const config &cfg)
Return true if the section with id section_id is referenced from another section in the config,...
Definition: help_impl.cpp:147
std::shared_ptr< terrain_type_data > load_terrain_types_data()
Load the appropriate terrain types data to use.
Definition: help_impl.cpp:1458
bool is_valid_id(const std::string &id)
Return true if the id is valid for user defined topics and sections.
Definition: help_impl.cpp:1438
boost::tribool last_debug_state
Definition: help_impl.cpp:69
std::string generate_contents_links(const std::string &section_name, config const *help_cfg)
Definition: help_impl.cpp:1204
std::string generate_topic_text(const std::string &generator, const config *help_cfg, const section &sec)
Definition: help_impl.cpp:356
help::section default_toplevel
Definition: help_impl.cpp:63
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
section parse_config(const config *cfg)
Parse a help config, return the top level section.
Definition: help_impl.cpp:298
std::vector< topic > generate_era_topics(const bool sort_generated, const std::string &era_id)
Definition: help_impl.cpp:608
const game_config_view * game_cfg
Definition: help_impl.cpp:61
int last_num_encountered_terrains
Definition: help_impl.cpp:68
help::section hidden_sections
Definition: help_impl.cpp:65
std::vector< topic > generate_trait_topics(const bool sort_generated)
Definition: help_impl.cpp:717
const std::string default_show_topic
Definition: help_impl.cpp:82
void generate_era_sections(const config *help_cfg, section &sec, int level)
Definition: help_impl.cpp:938
static bool is_cjk_char(const char32_t ch)
Definition: help_impl.cpp:92
const std::string era_prefix
Definition: help_impl.cpp:88
const int title_size
Definition: help_impl.cpp:73
const std::string faction_prefix
Definition: help_impl.cpp:87
bool is_scope_active(scope s)
Functions to load and save images from/to disk.
logger & info()
Definition: log.cpp:318
std::string img(const std::string &src, const std::string &align, bool floating)
Generates a Help markup tag corresponding to an image.
Definition: markup.cpp:31
std::string make_link(const std::string &text, const std::string &dst)
Generates a Help markup tag corresponding to a reference or link.
Definition: markup.cpp:25
std::string span_color(const color_t &color, Args &&... data)
Applies Pango markup to the input specifying its display color.
Definition: markup.hpp:116
std::string tag(std::string_view tag, Args &&... data)
Wraps the given data in the specified tag.
Definition: markup.hpp:51
config parse_text(const std::string &text)
Parse a xml style marked up text string.
Definition: markup.cpp:403
rng * generator
This generator is automatically synced during synced context.
Definition: random.cpp:60
::tod_manager * tod_manager
Definition: resources.cpp:29
std::vector< terrain_code > ter_list
Definition: translation.hpp:77
int compare(const std::string &s1, const std::string &s2)
Case-sensitive lexicographical comparison.
Definition: gettext.cpp:502
@ STRIP_SPACES
REMOVE_EMPTY: remove empty elements.
std::vector< std::string > quoted_split(const std::string &val, char c, int flags, char quote)
This function is identical to split(), except it does not split when it otherwise would if the previo...
bool contains(const Container &container, const Value &value)
Returns true iff value is found in container.
Definition: general.hpp:86
std::map< std::string, t_string > string_map
std::vector< std::string > split(const config_attribute_value &val)
auto * find(Container &container, const Value &value)
Convenience wrapper for using find on a container without needing to comare to end()
Definition: general.hpp:140
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
static void msg(const char *act, debug_info &i, const char *to="", const char *result="")
Definition: debugger.cpp:109
Thrown when the help system fails to parse something.
Definition: help_impl.hpp:227
A section contains topics and sections along with title and ID.
Definition: help_impl.hpp:148
section_list sections
Definition: help_impl.hpp:168
bool operator<(const section &) const
Comparison on the ID.
Definition: help_impl.cpp:1277
void add_section(const section &s)
Allocate memory for and add the section.
Definition: help_impl.cpp:1282
std::string id
Definition: help_impl.hpp:166
std::string title
Definition: help_impl.hpp:166
bool operator==(const section &) const
Two sections are equal if their IDs are equal.
Definition: help_impl.cpp:1272
topic_list topics
Definition: help_impl.hpp:167
A topic contains a title, an id and some text.
Definition: help_impl.hpp:115
bool operator==(const topic &) const
Two topics are equal if their IDs are equal.
Definition: help_impl.cpp:1262
std::string id
Definition: help_impl.hpp:139
topic_text text
Definition: help_impl.hpp:140
bool operator<(const topic &) const
Comparison on the ID.
Definition: help_impl.cpp:1267
std::string title
Definition: help_impl.hpp:139
A terrain string which is converted to a terrain is a string with 1 or 2 layers the layers are separa...
Definition: translation.hpp:49
Object which defines a time of day with associated bonuses, image, sounds etc.
Definition: time_of_day.hpp:57
static map_location::direction s
unit_type_data unit_types
Definition: types.cpp:1504
#define e
#define f