The Battle for Wesnoth  1.19.10+dev
game_display.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 /**
17  * @file
18  * During a game, show map & info-panels at top+right.
19  */
20 
21 #include "game_display.hpp"
22 
23 #include <utility>
24 
25 
26 #include "cursor.hpp"
27 #include "display_chat_manager.hpp"
28 #include "fake_unit_manager.hpp"
29 #include "floating_label.hpp"
30 #include "game_board.hpp"
32 #include "log.hpp"
33 #include "map/map.hpp"
34 #include "font/standard_colors.hpp"
35 #include "reports.hpp"
36 #include "resources.hpp"
37 #include "tod_manager.hpp"
38 #include "color.hpp"
39 #include "synced_context.hpp"
40 #include "units/unit.hpp"
41 #include "units/drawer.hpp"
42 #include "utils/general.hpp"
43 #include "whiteboard/manager.hpp"
44 #include "overlay.hpp"
45 #include "draw.hpp"
46 
47 static lg::log_domain log_display("display");
48 #define ERR_DP LOG_STREAM(err, log_display)
49 #define LOG_DP LOG_STREAM(info, log_display)
50 #define DBG_DP LOG_STREAM(debug, log_display)
51 
52 static lg::log_domain log_engine("engine");
53 #define ERR_NG LOG_STREAM(err, log_engine)
54 
55 
57  std::weak_ptr<wb::manager> wb,
58  reports& reports_object,
59  const std::string& theme_id,
60  const config& level)
61  : display(&board, std::move(wb), reports_object, theme_id, level)
62  , overlay_map_()
63  , attack_indicator_src_()
64  , attack_indicator_dst_()
65  , route_()
66  , displayedUnitHex_()
67  , first_turn_(true)
68  , in_game_(false)
69  , chat_man_(new display_chat_manager(*this))
70  , mode_(RUNNING)
71  , needs_rebuild_(false)
72 {
73 }
74 
76 {
77  try {
78  chat_man_->prune_chat_messages(true);
79  } catch(...) {
80  DBG_DP << "Caught exception in game_display destructor: " << utils::get_unknown_exception_type();
81  }
82 }
83 
85 {
86  if(!first_turn_) {
89 
90  if(old_tod.image_mask != tod.image_mask) {
91  fade_tod_mask(old_tod.image_mask, tod.image_mask);
92  }
93  }
94 
95  first_turn_ = false;
96 
97  update_tod();
98 }
99 
101 {
102  if(hex.valid() && fogged(hex)) {
103  return;
104  }
105  display::select_hex(hex);
106 
107  display_unit_hex(hex);
108 }
109 
111 {
112  wb::future_map_if future(!synced_context::is_synced()); /**< Lasts for whole method. */
113 
115  if (u) {
116  displayedUnitHex_ = hex;
117  invalidate_unit();
118  } else {
120  if (u) {
121  // mouse moved from unit hex to non-unit hex
122  if (context().units().count(selectedHex_)) {
124  invalidate_unit();
125  }
126  }
127  }
128 
131 }
132 
133 
135 {
136  if (!hex.valid())
137  return;
138 
139  wb::future_map_if future(!synced_context::is_synced()); /**< Lasts for whole method. */
140 
142  if (u) {
143  displayedUnitHex_ = hex;
144  invalidate_unit();
145  }
146 }
147 
149 {
150  if (src == displayedUnitHex_) {
152  invalidate_unit();
153  }
154 }
155 
156 void game_display::scroll_to_leader(int side, SCROLL_TYPE scroll_type,bool force)
157 {
159 
160  if(leader.valid() && leader->is_visible_to_team(viewing_team(), false)) {
161  scroll_to_tile(leader->get_location(), scroll_type, true, force);
162  }
163 }
164 
166 {
167  display::update();
168 
169  if (std::shared_ptr<wb::manager> w = wb_.lock()) {
170  w->pre_draw();
171  }
173  /**
174  * @todo FIXME: must modify changed, but best to do it at the
175  * floating_label level
176  */
177  chat_man_->prune_chat_messages();
178 }
179 
180 
182 {
183  display::render();
184 
185  if (std::shared_ptr<wb::manager> w = wb_.lock()) {
186  w->post_draw();
187  }
188 }
189 
191 {
193  if (fake_unit_man_->empty()) {
194  return;
195  }
196  unit_drawer drawer = unit_drawer(*this);
197 
198  for(const unit* temp_unit : *fake_unit_man_) {
199  const map_location& loc = temp_unit->get_location();
200  if(utils::contains(invalidated_, loc) && unit_can_draw_here(loc, *temp_unit)) {
201  drawer.redraw_unit(*temp_unit);
202  }
203  }
204 }
205 
206 namespace
207 {
208 const std::string mouseover_normal_top = "misc/hover-hex-top.png~RC(magenta>gold)";
209 const std::string mouseover_normal_bot = "misc/hover-hex-bottom.png~RC(magenta>gold)";
210 
211 const std::string mouseover_enemy_top = "misc/hover-hex-enemy-top.png~RC(magenta>red)";
212 const std::string mouseover_enemy_bot = "misc/hover-hex-enemy-bottom.png~RC(magenta>red)";
213 
214 const std::string mouseover_self_top = "misc/hover-hex-top.png~RC(magenta>green)";
215 const std::string mouseover_self_bot = "misc/hover-hex-bottom.png~RC(magenta>green)";
216 
217 const std::string mouseover_ally_top = "misc/hover-hex-top.png~RC(magenta>lightblue)";
218 const std::string mouseover_ally_bot = "misc/hover-hex-bottom.png~RC(magenta>lightblue)";
219 
220 /**
221  * Function to return 2 half-hex footsteps images for the given location.
222  * Only loc is on the current route set by set_route.
223  */
224 std::vector<texture> footsteps_images(const map_location& loc, const pathfind::marked_route& route, const display_context* dc)
225 {
226  std::vector<texture> res;
227 
228  if (route.steps.size() < 2) {
229  return res; // no real "route"
230  }
231 
232  std::vector<map_location>::const_iterator i =
233  std::find(route.steps.begin(),route.steps.end(),loc);
234 
235  if( i == route.steps.end()) {
236  return res; // not on the route
237  }
238 
239  // Check which footsteps images of game_config we will use
240  int move_cost = 1;
241  const unit_map::const_iterator u = dc->units().find(route.steps.front());
242  if(u != dc->units().end()) {
243  move_cost = u->movement_cost(dc->map().get_terrain(loc));
244  }
245  int image_number = std::min<int>(move_cost, game_config::foot_speed_prefix.size());
246  if (image_number < 1) {
247  return res; // Invalid movement cost or no images
248  }
249  const std::string foot_speed_prefix = game_config::foot_speed_prefix[image_number-1];
250 
251  texture teleport;
252 
253  // We draw 2 half-hex (with possibly different directions),
254  // but skip the first for the first step.
255  const int first_half = (i == route.steps.begin()) ? 1 : 0;
256  // and the second for the last step
257  const int second_half = (i+1 == route.steps.end()) ? 0 : 1;
258 
259  for (int h = first_half; h <= second_half; ++h) {
260  const std::string sense( h==0 ? "-in" : "-out" );
261 
262  if (!tiles_adjacent(*(i+(h-1)), *(i+h))) {
263  std::string teleport_image =
265  teleport = image::get_texture(teleport_image, image::HEXED);
266  continue;
267  }
268 
269  // In function of the half, use the incoming or outgoing direction
270  map_location::direction dir = (i+(h-1))->get_relative_dir(*(i+h));
271 
272  std::string rotate;
274  // No image, take the opposite direction and do a 180 rotation
275  dir = i->get_opposite_direction(dir);
276  rotate = "~FL(horiz)~FL(vert)";
277  }
278 
279  const std::string image = foot_speed_prefix
280  + sense + "-" + i->write_direction(dir)
281  + ".png" + rotate;
282 
283  res.push_back(image::get_texture(image, image::HEXED));
284  }
285 
286  // we draw teleport image (if any) in last
287  if (teleport != nullptr) res.push_back(teleport);
288 
289  return res;
290 }
291 } //anonymous namespace
292 
294 {
295  const bool on_map = context().map().on_board(loc);
296  const bool is_shrouded = shrouded(loc);
297 
299 
300  if(cursor::get() == cursor::WAIT) {
301  // Interaction is disabled, so we don't need anything else
302  return;
303  }
304 
305  if(on_map && loc == mouseoverHex_ && !map_screenshot_) {
307  const unit* u = context().get_visible_unit(loc, viewing_team());
308  if(u != nullptr) {
309  hex_top_layer = drawing_layer::mouseover_top;
310  }
311 
312  const std::string* mo_top_path;
313  const std::string* mo_bot_path;
314 
315  if(u == nullptr) {
316  mo_top_path = &mouseover_normal_top;
317  mo_bot_path = &mouseover_normal_bot;
318  } else if(viewing_team().is_enemy(u->side())) {
319  mo_top_path = &mouseover_enemy_top;
320  mo_bot_path = &mouseover_enemy_bot;
321  } else if(viewing_team().side() == u->side()) {
322  mo_top_path = &mouseover_self_top;
323  mo_bot_path = &mouseover_self_bot;
324  } else {
325  mo_top_path = &mouseover_ally_top;
326  mo_bot_path = &mouseover_ally_bot;
327  }
328 
329  drawing_buffer_add(hex_top_layer, loc,
330  [tex = image::get_texture(*mo_top_path, image::HEXED)](const rect& dest) { draw::blit(tex, dest); });
331 
333  [tex = image::get_texture(*mo_bot_path, image::HEXED)](const rect& dest) { draw::blit(tex, dest); });
334  }
335 
336  // Draw reach_map information.
337  if(!is_shrouded && !reach_map_.empty() && reach_map_.find(loc) != reach_map_.end()) {
338  // draw the reachmap tint below units and high terrain graphics
339  std::string color = prefs::get().reach_map_color();
340  std::string tint_opacity = std::to_string(prefs::get().reach_map_tint_opacity());
341 
342  drawing_buffer_add(drawing_layer::reachmap_highlight, loc, [tex = image::get_texture(game_config::reach_map_prefix + ".png~RC(magenta>"+color+")~O("+tint_opacity+"%)", image::HEXED)](const rect& dest) {
343  draw::blit(tex, dest);
344  });
345  // We remove the reachmap border mask of the hovered hex to avoid weird interactions with other visual objects.
346  if(loc != mouseoverHex_) {
347  // draw the highlight borders on top of units and terrain
349  for(const texture& t : images) {
350  draw::blit(t, dest);
351  }
352  });
353  }
354  }
355 
356  if(std::shared_ptr<wb::manager> w = wb_.lock()) {
357  w->draw_hex(loc);
358 
359  if(!(w->is_active() && w->has_temp_move())) {
360  std::vector<texture> footstepImages = footsteps_images(loc, route_, dc_);
361  if(!footstepImages.empty()) {
362  drawing_buffer_add(drawing_layer::footsteps, loc, [images = std::move(footstepImages)](const rect& dest) {
363  for(const texture& t : images) {
364  draw::blit(t, dest);
365  }
366  });
367  }
368  }
369  }
370 
371  // Draw the attack direction indicator
372  if(on_map && loc == attack_indicator_src_) {
374  [tex = image::get_texture("misc/attack-indicator-src-" + attack_indicator_direction() + ".png", image::HEXED)](const rect& dest)
375  { draw::blit(tex, dest); }
376  );
377  } else if(on_map && loc == attack_indicator_dst_) {
379  [tex = image::get_texture("misc/attack-indicator-dst-" + attack_indicator_direction() + ".png", image::HEXED)](const rect& dest)
380  { draw::blit(tex, dest); }
381  );
382  }
383 
384  // Linger overlay unconditionally otherwise it might give glitches
385  // so it's drawn over the shroud and fog.
386  if(mode_ != RUNNING) {
389  [tex = image::get_texture(linger, image::TOD_COLORED)](const rect& dest) { draw::blit(tex, dest); });
390  }
391 
392  if(on_map && loc == selectedHex_ && !game_config::images::selected.empty()) {
395  [tex = image::get_texture(selected, image::HEXED)](const rect& dest) { draw::blit(tex, dest); });
396  }
397 
398  // Show def% and turn to reach info
399  if(!is_shrouded && on_map) {
401  }
402 }
403 
405 {
407 }
408 
410 {
412 }
413 
415 {
416  display::layout();
417 
418  // We need teams for the reports below
419  if(context().teams().empty()) {
420  return;
421  }
422 
423  refresh_report("report_clock");
424  refresh_report("report_battery");
425  refresh_report("report_countdown");
426 
428  {
429  wb::future_map future; // start planned unit map scope
430 
431  // We display the unit the mouse is over if it is over a unit,
432  // otherwise we display the unit that is selected.
433  for (const std::string &name : reports_object_->report_list()) {
434  refresh_report(name);
435  }
436  invalidateGameStatus_ = false;
437  }
438 }
439 
440 
442 {
443  if(mode != mode_) {
444  mode_ = mode;
445  invalidate_all();
446  }
447 }
448 
450 {
451  // Search if there is a mark here
453 
454  std::shared_ptr<wb::manager> wb = wb_.lock();
455 
456  // Don't use empty route or the first step (the unit will be there)
457  if(w != route_.marks.end()
458  && !route_.steps.empty() && route_.steps.front() != loc) {
459  const unit_map::const_iterator un =
460  (wb && wb->get_temp_move_unit().valid()) ?
461  wb->get_temp_move_unit() : context().units().find(route_.steps.front());
462  if(un != context().units().end()) {
463  // Display the def% of this terrain
464  int move_cost = un->movement_cost(context().map().get_terrain(loc));
465  int def = (move_cost == movetype::UNREACHABLE ?
466  0 : 100 - un->defense_modifier(context().map().get_terrain(loc)));
467  std::stringstream def_text;
468  def_text << def << "%";
469 
470  color_t color = game_config::red_to_green(def, false);
471 
472  // simple mark (no turn point) use smaller font
473  int def_font = w->second.turns > 0 ? 18 : 16;
474  draw_text_in_hex(loc, drawing_layer::move_info, def_text.str(), def_font, color);
475 
477  [inv = w->second.invisible, zoc = w->second.zoc, cap = w->second.capture](const rect& dest) {
478  if(inv) {
479  draw::blit(image::get_texture(image::locator{"misc/hidden.png"}, image::HEXED), dest);
480  }
481 
482  if(zoc) {
483  draw::blit(image::get_texture(image::locator{"misc/zoc.png"}, image::HEXED), dest);
484  }
485 
486  if(cap) {
487  draw::blit(image::get_texture(image::locator{"misc/capture.png"}, image::HEXED), dest);
488  }
489  });
490 
491  //we display turn info only if different from a simple last "1"
492  if (w->second.turns > 1 || (w->second.turns == 1 && loc != route_.steps.back())) {
493  std::stringstream turns_text;
494  turns_text << w->second.turns;
495  draw_text_in_hex(loc, drawing_layer::move_info, turns_text.str(), 17, font::NORMAL_COLOR, 0.5,0.8);
496  }
497 
498  // The hex is full now, so skip the "show enemy moves"
499  return;
500  }
501  }
502  // When out-of-turn, it's still interesting to check out the terrain defs of the selected unit
503  else if (selectedHex_.valid() && loc == mouseoverHex_)
504  {
505  const unit_map::const_iterator selectedUnit = resources::gameboard->find_visible_unit(selectedHex_,viewing_team());
506  const unit_map::const_iterator mouseoveredUnit = resources::gameboard->find_visible_unit(mouseoverHex_,viewing_team());
507  if(selectedUnit != context().units().end() && mouseoveredUnit == context().units().end()) {
508  // Display the def% of this terrain
509  int move_cost = selectedUnit->movement_cost(context().map().get_terrain(loc));
510  int def = (move_cost == movetype::UNREACHABLE ?
511  0 : 100 - selectedUnit->defense_modifier(context().map().get_terrain(loc)));
512  std::stringstream def_text;
513  def_text << def << "%";
514 
515  color_t color = game_config::red_to_green(def, false);
516 
517  // use small font
518  int def_font = 16;
519  draw_text_in_hex(loc, drawing_layer::move_info, def_text.str(), def_font, color);
520  }
521  }
522 
523  if (!reach_map_.empty()) {
524  reach_map::iterator reach = reach_map_.find(loc);
525  if (reach != reach_map_.end() && reach->second > 1) {
526  const std::string num = std::to_string(reach->second);
527  draw_text_in_hex(loc, drawing_layer::move_info, num, 16, font::YELLOW_COLOR);
528  }
529  }
530 }
531 
533 {
535  highlight_another_reach(paths_list);
536 }
537 
539  const map_location& goal)
540 {
541  // Fold endpoints of routes into reachability map.
542  for (const pathfind::paths::step &dest : paths_list.destinations) {
543  reach_map_[dest.curr]++;
544  }
545  reach_map_changed_ = true;
546 
547  if(goal != map_location::null_location() && paths_list.destinations.contains(goal)) {
548  const auto& path_to_goal = paths_list.destinations.get_path(paths_list.destinations.find(goal));
549  const map_location enemy_unit_location = path_to_goal[0];
550  units_that_can_reach_goal_.insert(enemy_unit_location);
551  }
552 }
553 
555 {
557  if(!reach_map_.empty()) {
558  reach_map_.clear();
559  reach_map_changed_ = true;
560  return true;
561  } else {
562  return false;
563  }
564 }
565 
567 {
568  for(std::vector<map_location>::const_iterator i = route_.steps.begin();
569  i != route_.steps.end(); ++i) {
570  invalidate(*i);
571  }
572 }
573 
575 {
577 
578  if(route != nullptr) {
579  route_ = *route;
580  } else {
581  route_.steps.clear();
582  route_.marks.clear();
583  }
584 
586 }
587 
588 void game_display::float_label(const map_location& loc, const std::string& text, const color_t& color)
589 {
590  if(prefs::get().floating_labels() == false || fogged(loc)) {
591  return;
592  }
593 
594  rect loc_rect = get_location_rect(loc);
595 
596  font::floating_label flabel(text);
598  flabel.set_color(color);
599  flabel.set_position(loc_rect.center().x, loc_rect.y); // middle of top edge
600  flabel.set_move(0, -0.1 * turbo_speed() * get_zoom_factor());
601  flabel.set_lifetime(std::chrono::milliseconds{static_cast<int>(1000 / turbo_speed())});
603 
604  font::add_floating_label(flabel);
605 }
606 
608 {
612 
615 
618  }
619 }
620 
622 {
624 }
625 
627 {
628  in_game_ = true;
629  create_buttons();
630  invalidate_all();
631 }
632 
634  if (b) {
635  needs_rebuild_ = true;
636  }
637 }
638 
640  if (needs_rebuild_) {
641  needs_rebuild_ = false;
643  invalidate_all();
644  rebuild_all();
645  return true;
646  }
647  return false;
648 }
649 
651 {
652  return overlay_map_;
653 }
654 
655 std::vector<texture> game_display::get_reachmap_images(const map_location& loc) const
656 {
657  // Use get_direction() from display.cpp namespace display_direction
658  using namespace display_direction;
659  std::vector<std::string> names;
660  const auto adjacent = get_adjacent_tiles(loc);
661 
662  enum visibility { REACH = 0, ENEMY = 1, CLEAR = 2 };
663  std::array<visibility, 6> tiles;
664 
665  for(int i = 0; i < 6; ++i) {
666  // look for units adjacent to loc
667  std::string test_location = std::to_string(adjacent[i].x) + "," + std::to_string(adjacent[i].y);
668  const unit *u = context().get_visible_unit(adjacent[i], viewing_team());
669  if(reach_map_.find(adjacent[i]) != reach_map_.end()) {
670  DBG_DP << test_location << " is REACHABLE";
671  tiles[i] = REACH;
672  }
673  /**
674  * Make sure there is a unit selected or displayed when drawing the reachmap with enemy detection.
675  * Enemy detection needs to be "disabled" when the reach_map_team_index_ failsafes to viewing_team_index.
676  */
678  DBG_DP << test_location << " is NOT REACHABLE";
679  tiles[i] = CLEAR;
680  }
681  // Grab the reachmap-context team index updated in "display::process_reachmap_changes()" and test for adjacent enemy units
682  else if(u != nullptr && resources::gameboard->get_team(display::reach_map_team_index_).is_enemy(u->side())) {
683  DBG_DP << test_location << " has an ENEMY";
684  tiles[i] = ENEMY;
685  } else {
686  DBG_DP << test_location << " is NOT REACHABLE";
687  tiles[i] = CLEAR;
688  }
689  }
690 
691  // Find out if location is in the inner part of reachmap (surrounded by reach)
692  int s;
693  for(s = 0; s != 6; ++s) {
694  if(tiles[s] != REACH) {
695  break;
696  }
697  }
698 
699  if(s == 6) {
700  // Completely surrounded by reach. This may have a special graphic.
701  DBG_DP << "Tried completely surrounding";
702  std::string name = game_config::reach_map_prefix + "-all.png";
703  if(image::exists(name)) {
704  names.push_back(std::move(name));
705  }
706  }
707 
708  // Find all the directions overlap occurs from
709  for(int i = 0, cap1 = 0; cap1 != 6; ++cap1) {
710  if(tiles[i] != REACH) {
711  std::ostringstream stream;
712  std::string suffix;
713  std::string name;
715 
716  std::string color = prefs::get().reach_map_color();
717  std::string enemy_color = prefs::get().reach_map_enemy_color();
718  std::string border_opacity = std::to_string(prefs::get().reach_map_border_opacity());
719 
720  if(tiles[i] == ENEMY) {
721  suffix = ".png~RC(magenta>"+enemy_color+")~O("+border_opacity+"%)";
722  } else {
723  suffix = ".png~RC(magenta>"+color+")~O("+border_opacity+"%)";
724  }
725 
726  for(int cap2 = 0; tiles[i] != REACH && cap2 != 6; i = (i + 1) % 6, ++cap2) {
727  stream << get_direction(i);
728  if(!image::exists(stream.str() + ".png")) {
729  DBG_DP << "Image does not exist: " << stream.str() + ".png on " << loc;
730  // If we don't have any surface at all,
731  // then move onto the next overlapped area
732  if(name.empty()) {
733  i = (i + 1) % 6;
734  }
735  break;
736  } else {
737  name = stream.str();
738  }
739  }
740 
741  if(!name.empty()) {
742  names.push_back(name + suffix);
743  }
744  } else {
745  i = (i + 1) % 6;
746  }
747  }
748 
749  // now get the textures
750  std::vector<texture> res;
751 
752  for(const std::string& name : names) {
753  DBG_DP << "Pushing: " << name;
754  if(texture tex = image::get_texture(name, image::HEXED)) {
755  res.push_back(std::move(tex));
756  }
757  }
758 
759  return res;
760 }
map_location loc
Definition: move.cpp:172
const std::vector< map_location > & route_
Definition: move.cpp:386
double t
Definition: astarsearch.cpp:63
std::vector< std::string > names
Definition: build_info.cpp:67
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
Abstract class for exposing game data that doesn't depend on the GUI, however which for historical re...
const unit * get_visible_unit(const map_location &loc, const team &current_team, bool see_all=false) const
virtual const gamemap & map() const =0
virtual const unit_map & units() const =0
Sort-of-Singleton that many classes, both GUI and non-GUI, use to access the game data.
Definition: display.hpp:96
const team & viewing_team() const
Definition: display.cpp:335
bool map_screenshot_
Used to indicate to drawing functions that we are doing a map screenshot.
Definition: display.hpp:871
void draw_text_in_hex(const map_location &loc, const drawing_layer layer, const std::string &text, std::size_t font_size, color_t color, double x_in_hex=0.5, double y_in_hex=0.5)
Draw text on a hex.
Definition: display.cpp:1367
std::size_t viewing_team_index_
Definition: display.hpp:722
std::map< map_location, std::vector< overlay > > overlay_map
Definition: display.hpp:893
map_location selectedHex_
Definition: display.hpp:770
void recalculate_minimap()
Schedule the minimap for recalculation.
Definition: display.cpp:1474
bool unit_can_draw_here(const map_location &loc, const unit &unit) const
Returns true if there is no exclusive draw request for loc, or if there is, that it's for unit.
Definition: display.cpp:379
virtual void render() override
Update offscreen render buffers.
Definition: display.cpp:2303
void fade_tod_mask(const std::string &old, const std::string &new_)
ToD mask smooth fade.
Definition: display.cpp:2098
bool invalidate(const map_location &loc)
Function to invalidate a specific tile for redrawing.
Definition: display.cpp:2979
double turbo_speed() const
Definition: display.cpp:2023
bool reach_map_changed_
Definition: display.hpp:888
static double get_zoom_factor()
Returns the current zoom factor.
Definition: display.hpp:269
void invalidate_game_status()
Function to invalidate the game status displayed on the sidebar.
Definition: display.hpp:315
void rebuild_all()
Rebuild all dynamic terrain.
Definition: display.cpp:426
virtual void layout() override
Finalize screen layout.
Definition: display.cpp:2272
virtual void highlight_hex(map_location hex)
Definition: display.cpp:1398
void update_tod(const time_of_day *tod_override=nullptr)
Applies r,g,b coloring to the map.
Definition: display.cpp:385
void process_reachmap_changes()
Definition: display.cpp:3151
void scroll_to_tile(const map_location &loc, SCROLL_TYPE scroll_type=ONSCREEN, bool check_fogged=true, bool force=true)
Scroll such that location loc is on-screen.
Definition: display.cpp:1873
map_location mouseoverHex_
Definition: display.hpp:771
bool fogged(const map_location &loc) const
Returns true if location (x,y) is covered in fog.
Definition: display.cpp:667
void invalidate_all()
Function to invalidate all tiles.
Definition: display.cpp:2972
void drawing_buffer_add(const drawing_layer layer, const map_location &loc, decltype(draw_helper::do_draw) draw_func)
Add an item to the drawing buffer.
Definition: display.cpp:1250
std::size_t reach_map_team_index_
Definition: display.hpp:890
const display_context & context() const
Definition: display.hpp:192
rect get_location_rect(const map_location &loc) const
Returns the on-screen rect corresponding to a loc.
Definition: display.cpp:680
std::set< map_location > invalidated_
Definition: display.hpp:760
void create_buttons()
Definition: display.cpp:838
bool invalidateGameStatus_
Definition: display.hpp:748
virtual void draw_invalidated()
Only called when there's actual redrawing to do.
Definition: display.cpp:2459
reports * reports_object_
Definition: display.hpp:750
virtual void update() override
Update animations and internal state.
Definition: display.cpp:2255
virtual void draw_hex(const map_location &loc)
Redraws a single gamemap location.
Definition: display.cpp:2495
const std::unique_ptr< fake_unit_manager > fake_unit_man_
Definition: display.hpp:741
const display_context * dc_
Definition: display.hpp:679
bool shrouded(const map_location &loc) const
Returns true if location (x,y) is covered in shroud.
Definition: display.cpp:662
bool dont_show_all_
Definition: display.hpp:723
void refresh_report(const std::string &report_name, const config *new_cfg=nullptr)
Update the given report.
Definition: display.cpp:2728
std::weak_ptr< wb::manager > wb_
Definition: display.hpp:681
virtual void select_hex(map_location hex)
Definition: display.cpp:1390
reach_map reach_map_
Definition: display.hpp:886
void set_move(double xmove, double ymove)
void set_position(double xpos, double ypos)
void set_lifetime(const std::chrono::milliseconds &lifetime, const std::chrono::milliseconds &fadeout=std::chrono::milliseconds{100})
void set_color(const color_t &color)
void set_scroll_mode(LABEL_SCROLL_MODE scroll)
void set_font_size(int font_size)
Game board class.
Definition: game_board.hpp:47
unit_map::iterator find_visible_unit(const map_location &loc, const team &current_team, bool see_all=false)
Definition: game_board.cpp:185
virtual void draw_invalidated() override
Only called when there's actual redrawing to do.
void draw_movement_info(const map_location &loc)
Draws the movement info (turns available) for a given location.
void display_unit_hex(map_location hex)
Change the unit to be displayed in the sidebar.
void invalidate_unit_after_move(const map_location &src, const map_location &dst)
Same as invalidate_unit() if moving the displayed unit.
std::string attack_indicator_direction() const
Function to get attack direction suffix.
game_mode mode_
virtual void highlight_hex(map_location hex) override
Function to highlight a location.
map_location displayedUnitHex_
void invalidate_route()
std::vector< texture > get_reachmap_images(const map_location &loc) const
void scroll_to_leader(int side, SCROLL_TYPE scroll_type=ONSCREEN, bool force=true)
Scrolls to the leader of a certain side.
void set_game_mode(const game_mode mode)
virtual void draw_hex(const map_location &loc) override
Redraws a single gamemap location.
void set_attack_indicator(const map_location &src, const map_location &dst)
Set the attack direction indicator.
const std::unique_ptr< display_chat_manager > chat_man_
pathfind::marked_route route_
virtual void render() override
TLD render() override.
void set_route(const pathfind::marked_route *route)
Sets the route along which footsteps are drawn to show movement of a unit.
virtual overlay_map & get_overlays() override
Inherited from display.
std::set< map_location > units_that_can_reach_goal_
void invalidate_unit()
Function to invalidate that unit status displayed on the sidebar.
virtual void select_hex(map_location hex) override
Function to display a location as selected.
void highlight_reach(const pathfind::paths &paths_list)
Sets the paths that are currently displayed as available for the unit to move along.
void new_turn()
Update lighting settings.
void clear_attack_indicator()
virtual const time_of_day & get_time_of_day(const map_location &loc) const override
void highlight_another_reach(const pathfind::paths &paths_list, const map_location &goal=map_location::null_location())
Add more paths to highlight.
map_location attack_indicator_dst_
game_mode
Sets the linger mode for the display.
@ RUNNING
no linger overlay, show fog and shroud.
bool unhighlight_reach()
Reset highlighting of paths.
void needs_rebuild(bool b)
Sets whether the screen (map visuals) needs to be rebuilt.
overlay_map overlay_map_
bool maybe_rebuild()
Rebuilds the screen if needs_rebuild(true) was previously called, and resets the flag.
virtual void layout() override
TLD layout() override.
virtual bool has_time_area() const override
game_display(game_board &board, std::weak_ptr< wb::manager > wb, reports &reports_object, const std::string &theme_id, const config &level)
virtual void update() override
TLD update() override.
map_location attack_indicator_src_
void float_label(const map_location &loc, const std::string &text, const color_t &color)
Function to float a label above a tile.
terrain_code get_terrain(const map_location &loc) const
Looks up terrain at a particular location.
Definition: map.cpp:301
bool on_board(const map_location &loc) const
Tell if a location is on the map.
Definition: map.cpp:384
Generic locator abstracting the location of an image.
Definition: picture.hpp:59
static const int UNREACHABLE
Magic value that signifies a hex is unreachable.
Definition: movetype.hpp:174
static prefs & get()
std::string reach_map_color()
std::string reach_map_enemy_color()
const std::set< std::string > & report_list()
Definition: reports.cpp:1841
static bool is_synced()
Wrapper class to encapsulate creation and management of an SDL_Texture.
Definition: texture.hpp:33
bool has_time_area() const
const time_of_day & get_time_of_day(int for_turn=0) const
Returns global time of day for the passed turn.
Definition: tod_manager.hpp:56
const time_of_day & get_previous_time_of_day() const
void redraw_unit(const unit &u) const
draw a unit.
Definition: drawer.cpp:202
unit_iterator end()
Definition: map.hpp:428
unit_iterator find(std::size_t id)
Definition: map.cpp:302
unit_iterator find_leader(int side)
Definition: map.cpp:320
This class represents a single unit of a specific type.
Definition: unit.hpp:133
Drawing functions, for drawing things on the screen.
drawing_layer
@ linger_overlay
The overlay used for the linger mode.
@ reachmap_highlight
Overlay on reachable hexes.
@ footsteps
Footsteps showing path from unit to mouse.
@ attack_indicator
Layer which holds the attack indicator.
@ reachmap_border
Overlay border of reachable hexes.
@ mouseover_top
Top half of image following the mouse.
@ mouseover_bottom
Bottom half of image following the mouse.
@ move_info
Movement info (defense%, etc...)
@ selected_hex
Image on the selected unit.
std::size_t i
Definition: function.cpp:1022
int w
static lg::log_domain log_engine("engine")
static lg::log_domain log_display("display")
#define DBG_DP
int side() const
The side this unit belongs to.
Definition: unit.hpp:343
T end(const std::pair< T, T > &p)
void get_adjacent_tiles(const map_location &a, map_location *res)
Function which, given a location, will place all adjacent locations in res.
Definition: location.cpp:479
bool tiles_adjacent(const map_location &a, const map_location &b)
Function which tells if two locations are adjacent.
Definition: location.cpp:507
Standard logging facilities (interface).
bool is_shrouded(const display *disp, const map_location &loc)
Our definition of map labels being obscured is if the tile is obscured, or the tile below is obscured...
Definition: label.cpp:32
CURSOR_TYPE get()
Definition: cursor.cpp:216
@ WAIT
Definition: cursor.hpp:28
const std::string & get_direction(std::size_t n)
Definition: display.cpp:830
void blit(const texture &tex, const SDL_Rect &dst)
Draws a texture, or part of a texture, at the given location.
Definition: draw.cpp:381
const int SIZE_FLOAT_LABEL
Definition: constants.cpp:32
const color_t YELLOW_COLOR
int add_floating_label(const floating_label &flabel)
add a label floating on the screen above everything else.
const color_t NORMAL_COLOR
@ ANCHOR_LABEL_MAP
std::string selected
std::string foot_teleport_enter
std::vector< std::string > foot_speed_prefix
std::string foot_teleport_exit
int reach_map_border_opacity
std::string reach_map_prefix
Definition: game_config.cpp:58
int reach_map_tint_opacity
color_t red_to_green(double val, bool for_text)
Return a color corresponding to the value val red for val=0.0 to green for val=100....
Functions to load and save images from/to disk.
bool exists(const image::locator &i_locator)
Returns true if the given image actually exists, without loading it.
Definition: picture.cpp:820
@ HEXED
Standard hexagonal tile mask applied, removing portions that don't fit.
Definition: picture.hpp:166
@ TOD_COLORED
Same as HEXED, but with Time of Day color tint applied.
Definition: picture.hpp:168
texture get_texture(const image::locator &i_locator, TYPE type, bool skip_cache)
Returns an image texture suitable for hardware-accelerated rendering.
Definition: picture.cpp:922
Unit and team statistics.
::tod_manager * tod_manager
Definition: resources.cpp:29
game_board * gameboard
Definition: resources.cpp:20
std::size_t size(std::string_view str)
Length in characters of a UTF-8 string.
Definition: unicode.cpp:85
bool contains(const Container &container, const Value &value)
Returns true iff value is found in container.
Definition: general.hpp:86
std::string get_unknown_exception_type()
Utility function for finding the type of thing caught with catch(...).
Definition: general.cpp:23
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
Definition: display.hpp:45
std::string::const_iterator iterator
Definition: tokenizer.hpp:25
rect dst
Location on the final composed sheet.
rect src
Non-transparent portion of the surface to compose.
The basic class for representing 8-bit RGB or RGBA colour values.
Definition: color.hpp:59
Encapsulates the map of the game.
Definition: location.hpp:45
bool valid() const
Definition: location.hpp:110
direction
Valid directions which can be moved in our hexagonal world.
Definition: location.hpp:47
static const map_location & null_location()
Definition: location.hpp:102
Structure which holds a single route and marks for special events.
Definition: pathfind.hpp:142
std::vector< map_location > & steps
Definition: pathfind.hpp:187
std::vector< map_location > get_path(const const_iterator &) const
Returns the path going from the source point (included) to the destination point j (excluded).
Definition: pathfind.cpp:497
bool contains(const map_location &) const
Definition: pathfind.cpp:514
const_iterator find(const map_location &) const
Definition: pathfind.cpp:478
map_location curr
Definition: pathfind.hpp:89
Object which contains all the possible locations a unit can move to, with associated best routes to t...
Definition: pathfind.hpp:73
dest_vect destinations
Definition: pathfind.hpp:101
An abstract description of a rectangle with integer coordinates.
Definition: rect.hpp:49
constexpr point center() const
The center point of the rectangle, accounting for origin.
Definition: rect.hpp:106
Object which defines a time of day with associated bonuses, image, sounds etc.
Definition: time_of_day.hpp:57
std::string image_mask
The image that is to be laid over all images while this time of day lasts.
Definition: time_of_day.hpp:96
bool valid() const
Definition: map.hpp:273
Applies the planned unit map for the duration of the struct's life.
Definition: manager.hpp:253
static map_location::direction s
#define h
#define b