OpenTTD Source 20260218-master-g2123fca5ea
misc_gui.cpp
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <https://www.gnu.org/licenses/old-licenses/gpl-2.0>.
6 */
7
9
10#include "stdafx.h"
11#include "debug.h"
12#include "landscape.h"
13#include "error.h"
14#include "gui.h"
15#include "gfx_layout.h"
16#include "tilehighlight_func.h"
17#include "command_func.h"
18#include "company_func.h"
19#include "town.h"
20#include "string_func.h"
21#include "company_base.h"
22#include "station_base.h"
23#include "waypoint_base.h"
24#include "texteff.hpp"
25#include "strings_func.h"
26#include "window_func.h"
27#include "querystring_gui.h"
29#include "newgrf_debug.h"
30#include "zoom_func.h"
31#include "viewport_func.h"
32#include "landscape_cmd.h"
33#include "station_cmd.h"
34#include "waypoint_cmd.h"
35#include "rev.h"
36#include "timer/timer.h"
37#include "timer/timer_window.h"
39
40#include "widgets/misc_widget.h"
41
42#include "table/strings.h"
43
44#include "safeguards.h"
45
53
54
55static constexpr std::initializer_list<NWidgetPart> _nested_land_info_widgets = {
57 NWidget(WWT_CLOSEBOX, COLOUR_GREY),
58 NWidget(WWT_CAPTION, COLOUR_GREY), SetStringTip(STR_LAND_AREA_INFORMATION_CAPTION, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
59 NWidget(WWT_PUSHIMGBTN, COLOUR_GREY, WID_LI_LOCATION), SetAspect(WidgetDimensions::ASPECT_LOCATION), SetSpriteTip(SPR_GOTO_LOCATION, STR_LAND_AREA_INFORMATION_LOCATION_TOOLTIP),
60 NWidget(WWT_DEBUGBOX, COLOUR_GREY),
63};
64
65static WindowDesc _land_info_desc(
66 WDP_AUTO, {}, 0, 0,
68 {},
69 _nested_land_info_widgets
70);
71
72class LandInfoWindow : public Window {
74 std::string cargo_acceptance{};
75
76public:
78
79 void DrawWidget(const Rect &r, WidgetID widget) const override
80 {
81 if (widget != WID_LI_BACKGROUND) return;
82
83 Rect ir = r.Shrink(WidgetDimensions::scaled.frametext);
84 for (size_t i = 0; i < this->landinfo_data.size(); i++) {
85 DrawString(ir, this->landinfo_data[i], i == 0 ? TC_LIGHT_BLUE : TC_FROMSTRING, SA_HOR_CENTER);
86 ir.top += GetCharacterHeight(FS_NORMAL) + (i == 0 ? WidgetDimensions::scaled.vsep_wide : WidgetDimensions::scaled.vsep_normal);
87 }
88
89 if (!this->cargo_acceptance.empty()) {
90 DrawStringMultiLine(ir, GetString(STR_JUST_RAW_STRING, this->cargo_acceptance), TC_FROMSTRING, SA_CENTER);
91 }
92 }
93
94 void UpdateWidgetSize(WidgetID widget, Dimension &size, [[maybe_unused]] const Dimension &padding, [[maybe_unused]] Dimension &fill, [[maybe_unused]] Dimension &resize) override
95 {
96 if (widget != WID_LI_BACKGROUND) return;
97
98 size.height = WidgetDimensions::scaled.frametext.Vertical();
99 for (size_t i = 0; i < this->landinfo_data.size(); i++) {
100 uint width = GetStringBoundingBox(this->landinfo_data[i]).width + WidgetDimensions::scaled.frametext.Horizontal();
101 size.width = std::max(size.width, width);
102
103 size.height += GetCharacterHeight(FS_NORMAL) + (i == 0 ? WidgetDimensions::scaled.vsep_wide : WidgetDimensions::scaled.vsep_normal);
104 }
105
106 if (!this->cargo_acceptance.empty()) {
107 uint width = GetStringBoundingBox(this->cargo_acceptance).width + WidgetDimensions::scaled.frametext.Horizontal();
108 size.width = std::max(size.width, std::min(static_cast<uint>(ScaleGUITrad(300)), width));
109 size.height += GetStringHeight(GetString(STR_JUST_RAW_STRING, this->cargo_acceptance), size.width - WidgetDimensions::scaled.frametext.Horizontal());
110 }
111 }
112
113 LandInfoWindow(Tile tile) : Window(_land_info_desc), tile(tile)
114 {
115 this->InitNested();
116
117#if defined(_DEBUG)
118# define LANDINFOD_LEVEL 0
119#else
120# define LANDINFOD_LEVEL 1
121#endif
122 Debug(misc, LANDINFOD_LEVEL, "TILE: {0} (0x{0:x}) ({1},{2})", (TileIndex)tile, TileX(tile), TileY(tile));
123 Debug(misc, LANDINFOD_LEVEL, "type = 0x{:x}", tile.type());
124 Debug(misc, LANDINFOD_LEVEL, "height = 0x{:x}", tile.height());
125 Debug(misc, LANDINFOD_LEVEL, "m1 = 0x{:x}", tile.m1());
126 Debug(misc, LANDINFOD_LEVEL, "m2 = 0x{:x}", tile.m2());
127 Debug(misc, LANDINFOD_LEVEL, "m3 = 0x{:x}", tile.m3());
128 Debug(misc, LANDINFOD_LEVEL, "m4 = 0x{:x}", tile.m4());
129 Debug(misc, LANDINFOD_LEVEL, "m5 = 0x{:x}", tile.m5());
130 Debug(misc, LANDINFOD_LEVEL, "m6 = 0x{:x}", tile.m6());
131 Debug(misc, LANDINFOD_LEVEL, "m7 = 0x{:x}", tile.m7());
132 Debug(misc, LANDINFOD_LEVEL, "m8 = 0x{:x}", tile.m8());
133
134 PrintWaterRegionDebugInfo(tile);
135#undef LANDINFOD_LEVEL
136 }
137
138 void OnInit() override
139 {
140 Town *t = ClosestTownFromTile(tile, _settings_game.economy.dist_local_authority);
141
142 TileDesc td{};
143 td.owner_type[0] = STR_LAND_AREA_INFORMATION_OWNER; // At least one owner is displayed, though it might be "N/A".
144
145 CargoArray acceptance{};
146 CargoTypes always_accepted{};
147 AddAcceptedCargo(tile, acceptance, always_accepted);
148 GetTileDesc(tile, td);
149
150 this->landinfo_data.clear();
151
152 /* Tiletype */
153 this->landinfo_data.push_back(GetString(td.str, td.dparam));
154
155 /* Up to four owners */
156 for (uint i = 0; i < 4; i++) {
157 if (td.owner_type[i] == STR_NULL) continue;
158
159 if (td.owner[i] == OWNER_NONE || td.owner[i] == OWNER_WATER) {
160 this->landinfo_data.push_back(GetString(td.owner_type[i], STR_LAND_AREA_INFORMATION_OWNER_N_A, std::monostate{}));
161 } else {
162 auto params = GetParamsForOwnedBy(td.owner[i], tile);
163 this->landinfo_data.push_back(GetStringWithArgs(td.owner_type[i], params));
164 }
165 }
166
167 /* Cost to clear/revenue when cleared */
169 if (c != nullptr) {
171 CommandCost costclear = Command<Commands::LandscapeClear>::Do(DoCommandFlag::QueryCost, tile);
172 if (costclear.Succeeded()) {
173 Money cost = costclear.GetCost();
174 StringID str;
175 if (cost < 0) {
176 cost = -cost; // Negate negative cost to a positive revenue
177 str = STR_LAND_AREA_INFORMATION_REVENUE_WHEN_CLEARED;
178 } else {
179 str = STR_LAND_AREA_INFORMATION_COST_TO_CLEAR;
180 }
181 this->landinfo_data.push_back(GetString(str, cost));
182 } else {
183 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_COST_TO_CLEAR_N_A));
184 }
185 } else {
186 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_COST_TO_CLEAR_N_A));
187 }
188
189 /* Location */
190 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_LANDINFO_COORDS, TileX(tile), TileY(tile), GetTileZ(tile)));
191
192 /* Tile index */
193 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_LANDINFO_INDEX, tile, tile));
194
195 /* Local authority */
196 if (t == nullptr) {
197 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_LOCAL_AUTHORITY, STR_LAND_AREA_INFORMATION_LOCAL_AUTHORITY_NONE, std::monostate{}));
198 } else {
199 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_LOCAL_AUTHORITY, STR_TOWN_NAME, t->index));
200 }
201
202 /* Build date */
204 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_BUILD_DATE, td.build_date));
205 }
206
207 /* Station class */
208 if (td.station_class != STR_NULL) {
209 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_STATION_CLASS, td.station_class));
210 }
211
212 /* Station type name */
213 if (td.station_name != STR_NULL) {
214 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_STATION_TYPE, td.station_name));
215 }
216
217 /* Airport class */
218 if (td.airport_class != STR_NULL) {
219 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_AIRPORT_CLASS, td.airport_class));
220 }
221
222 /* Airport name */
223 if (td.airport_name != STR_NULL) {
224 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_AIRPORT_NAME, td.airport_name));
225 }
226
227 /* Airport tile name */
228 if (td.airport_tile_name != STR_NULL) {
229 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_AIRPORTTILE_NAME, td.airport_tile_name));
230 }
231
232 /* Rail type name */
233 if (td.railtype != STR_NULL) {
234 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_RAIL_TYPE, td.railtype));
235 }
236
237 /* Rail speed limit */
238 if (td.rail_speed != 0) {
239 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_RAIL_SPEED_LIMIT, PackVelocity(td.rail_speed, VEH_TRAIN)));
240 }
241
242 /* Road type name */
243 if (td.roadtype != STR_NULL) {
244 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_ROAD_TYPE, td.roadtype));
245 }
246
247 /* Road speed limit */
248 if (td.road_speed != 0) {
249 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_ROAD_SPEED_LIMIT, PackVelocity(td.road_speed, VEH_ROAD)));
250 }
251
252 /* Tram type name */
253 if (td.tramtype != STR_NULL) {
254 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_TRAM_TYPE, td.tramtype));
255 }
256
257 /* Tram speed limit */
258 if (td.tram_speed != 0) {
259 this->landinfo_data.push_back(GetString(STR_LANG_AREA_INFORMATION_TRAM_SPEED_LIMIT, PackVelocity(td.tram_speed, VEH_ROAD)));
260 }
261
262 /* Tile protection status */
263 if (td.town_can_upgrade.has_value()) {
264 this->landinfo_data.push_back(GetString(td.town_can_upgrade.value() ? STR_LAND_AREA_INFORMATION_TOWN_CAN_UPGRADE : STR_LAND_AREA_INFORMATION_TOWN_CANNOT_UPGRADE));
265 }
266
267 /* NewGRF name */
268 if (td.grf.has_value()) {
269 this->landinfo_data.push_back(GetString(STR_LAND_AREA_INFORMATION_NEWGRF_NAME, std::move(*td.grf)));
270 }
271
272 /* Cargo acceptance is displayed in a extra multiline */
273 auto line = BuildCargoAcceptanceString(acceptance, STR_LAND_AREA_INFORMATION_CARGO_ACCEPTED);
274 if (line.has_value()) {
275 this->cargo_acceptance = std::move(*line);
276 } else {
277 this->cargo_acceptance.clear();
278 }
279 }
280
281 bool IsNewGRFInspectable() const override
282 {
283 return ::IsNewGRFInspectable(GetGrfSpecFeature(this->tile), this->tile.base());
284 }
285
286 void ShowNewGRFInspectWindow() const override
287 {
288 ::ShowNewGRFInspectWindow(GetGrfSpecFeature(this->tile), this->tile.base());
289 }
290
291 void OnClick([[maybe_unused]] Point pt, WidgetID widget, [[maybe_unused]] int click_count) override
292 {
293 switch (widget) {
294 case WID_LI_LOCATION:
295 if (_ctrl_pressed) {
296 ShowExtraViewportWindow(this->tile);
297 } else {
298 ScrollMainWindowToTile(this->tile);
299 }
300 break;
301 }
302 }
303
309 void OnInvalidateData([[maybe_unused]] int data = 0, [[maybe_unused]] bool gui_scope = true) override
310 {
311 if (!gui_scope) return;
312
313 /* ReInit, "debug" sprite might have changed */
314 if (data == 1) this->ReInit();
315 }
316};
317
323{
325 new LandInfoWindow(tile);
326}
327
328static constexpr std::initializer_list<NWidgetPart> _nested_about_widgets = {
330 NWidget(WWT_CLOSEBOX, COLOUR_GREY),
331 NWidget(WWT_CAPTION, COLOUR_GREY), SetStringTip(STR_ABOUT_OPENTTD, STR_TOOLTIP_WINDOW_TITLE_DRAG_THIS),
332 EndContainer(),
333 NWidget(WWT_PANEL, COLOUR_GREY), SetPIP(4, 2, 4),
334 NWidget(WWT_LABEL, INVALID_COLOUR), SetStringTip(STR_ABOUT_ORIGINAL_COPYRIGHT),
335 NWidget(WWT_LABEL, INVALID_COLOUR), SetStringTip(STR_ABOUT_VERSION),
336 NWidget(WWT_FRAME, COLOUR_GREY), SetPadding(0, 5, 1, 5),
337 NWidget(WWT_EMPTY, INVALID_COLOUR, WID_A_SCROLLING_TEXT),
338 EndContainer(),
339 NWidget(WWT_LABEL, INVALID_COLOUR, WID_A_WEBSITE),
340 NWidget(WWT_LABEL, INVALID_COLOUR, WID_A_COPYRIGHT),
341 EndContainer(),
342};
343
344static WindowDesc _about_desc(
345 WDP_CENTER, {}, 0, 0,
347 {},
348 _nested_about_widgets
349);
350
351static const std::initializer_list<const std::string_view> _credits = {
352 "Original design by Chris Sawyer",
353 "Original graphics by Simon Foster",
354 "",
355 "The OpenTTD team (in alphabetical order):",
356 " Matthijs Kooijman (blathijs) - Pathfinder-guru, Debian port (since 0.3)",
357 " Christoph Elsenhans (frosch) - General coding (since 0.6)",
358 " Lo\u00efc Guilloux (glx) - General / Windows Expert (since 0.4.5)",
359 " Koen Bussemaker (Kuhnovic) - General / Ship pathfinder (since 14)",
360 " Charles Pigott (LordAro) - General / Correctness police (since 1.9)",
361 " Michael Lutz (michi_cc) - Path based signals (since 0.7)",
362 " Niels Martin Hansen (nielsm) - Music system, general coding (since 1.9)",
363 " Owen Rudge (orudge) - Forum host, OS/2 port (since 0.1)",
364 " Peter Nelson (peter1138) - Spiritual descendant from NewGRF gods (since 0.4.5)",
365 " Remko Bijker (Rubidium) - Coder and way more (since 0.4.5)",
366 " Patric Stout (TrueBrain) - NoProgrammer (since 0.3), sys op",
367 " Tyler Trahan (2TallTyler) - General / Time Lord (since 13)",
368 " Richard Wheeler (zephyris) - Precision pixel production (since 15)",
369 "",
370 "Inactive Developers:",
371 " Grzegorz Duczy\u0144ski (adf88) - General coding (1.7 - 1.8)",
372 " Albert Hofkamp (Alberth) - GUI expert (0.7 - 1.9)",
373 " Jean-Fran\u00e7ois Claeys (Belugas) - GUI, NewGRF and more (0.4.5 - 1.0)",
374 " Bjarni Corfitzen (Bjarni) - MacOSX port, coder and vehicles (0.3 - 0.7)",
375 " Victor Fischer (Celestar) - Programming everywhere you need him to (0.3 - 0.6)",
376 " Ulf Hermann (fonsinchen) - Cargo Distribution (1.3 - 1.6)",
377 " Jaroslav Mazanec (KUDr) - YAPG (Yet Another Pathfinder God) ;) (0.4.5 - 0.6)",
378 " Jonathan Coome (Maedhros) - High priest of the NewGRF Temple (0.5 - 0.6)",
379 " Attila B\u00e1n (MiHaMiX) - Developer WebTranslator 1 and 2 (0.3 - 0.5)",
380 " Ingo von Borstel (planetmaker) - General coding, Support (1.1 - 1.9)",
381 " Zden\u011bk Sojka (SmatZ) - Bug finder and fixer (0.6 - 1.3)",
382 " Jos\u00e9 Soler (Terkhen) - General coding (1.0 - 1.4)",
383 " Christoph Mallon (Tron) - Programmer, code correctness police (0.3 - 0.5)",
384 " Thijs Marinussen (Yexo) - AI Framework, General (0.6 - 1.3)",
385 " Leif Linse (Zuu) - AI/Game Script (1.2 - 1.6)",
386 "",
387 "Retired Developers:",
388 " Tam\u00e1s Farag\u00f3 (Darkvater) - Ex-Lead coder (0.3 - 0.5)",
389 " Dominik Scherer (dominik81) - Lead programmer, GUI expert (0.3 - 0.3)",
390 " Emil Djupfeld (egladil) - MacOSX (0.4.5 - 0.6)",
391 " Simon Sasburg (HackyKid) - Many bugfixes (0.4 - 0.4.5)",
392 " Ludvig Strigeus (ludde) - Original author of OpenTTD, main coder (0.1 - 0.3)",
393 " Cian Duffy (MYOB) - BeOS port / manual writing (0.1 - 0.3)",
394 " Petr Baudi\u0161 (pasky) - Many patches, NewGRF support (0.3 - 0.3)",
395 " Benedikt Br\u00fcggemeier (skidd13) - Bug fixer and code reworker (0.6 - 0.7)",
396 " Serge Paquet (vurlix) - 2nd contributor after ludde (0.1 - 0.3)",
397 "",
398 "Special thanks go out to:",
399 " Josef Drexler - For his great work on TTDPatch",
400 " Marcin Grzegorczyk - Track foundations and for describing TTD internals",
401 " Stefan Mei\u00dfner (sign_de) - For his work on the console",
402 " Mike Ragsdale - OpenTTD installer",
403 " Christian Rosentreter (tokai) - MorphOS / AmigaOS port",
404 " Richard Kempton (richK) - additional airports, initial TGP implementation",
405 " Alberto Demichelis - Squirrel scripting language \u00a9 2003-2008",
406 " L. Peter Deutsch - MD5 implementation \u00a9 1999, 2000, 2002",
407 " Michael Blunck - Pre-signals and semaphores \u00a9 2003",
408 " George - Canal/Lock graphics \u00a9 2003-2004",
409 " Andrew Parkhouse (andythenorth) - River graphics",
410 " David Dallaston (Pikka) - Tram tracks",
411 " All Translators - Who made OpenTTD a truly international game",
412 " Bug Reporters - Without whom OpenTTD would still be full of bugs!",
413 "",
414 "",
415 "And last but not least:",
416 " Chris Sawyer - For an amazing game!"
417};
418
419struct AboutWindow : public Window {
421 int line_height = 0;
422 static const int num_visible_lines = 19;
423
424 AboutWindow() : Window(_about_desc)
425 {
427
428 this->text_position = this->GetWidget<NWidgetBase>(WID_A_SCROLLING_TEXT)->pos_y + this->GetWidget<NWidgetBase>(WID_A_SCROLLING_TEXT)->current_y;
429 }
430
431 std::string GetWidgetString(WidgetID widget, StringID stringid) const override
432 {
433 if (widget == WID_A_WEBSITE) return "Website: https://www.openttd.org";
434 if (widget == WID_A_COPYRIGHT) return GetString(STR_ABOUT_COPYRIGHT_OPENTTD, _openttd_revision_year);
435 return this->Window::GetWidgetString(widget, stringid);
436 }
437
438 void UpdateWidgetSize(WidgetID widget, Dimension &size, [[maybe_unused]] const Dimension &padding, [[maybe_unused]] Dimension &fill, [[maybe_unused]] Dimension &resize) override
439 {
440 if (widget != WID_A_SCROLLING_TEXT) return;
441
442 this->line_height = GetCharacterHeight(FS_NORMAL);
443
444 Dimension d;
445 d.height = this->line_height * num_visible_lines;
446
447 d.width = 0;
448 for (const auto &str : _credits) {
449 d.width = std::max(d.width, GetStringBoundingBox(str).width);
450 }
451 size = maxdim(size, d);
452 }
453
454 void DrawWidget(const Rect &r, WidgetID widget) const override
455 {
456 if (widget != WID_A_SCROLLING_TEXT) return;
457
458 int y = this->text_position;
459
460 /* Show all scrolling _credits */
461 for (const auto &str : _credits) {
462 if (y >= r.top + 7 && y < r.bottom - this->line_height) {
463 DrawString(r.left, r.right, y, str, TC_BLACK, SA_LEFT | SA_FORCE);
464 }
465 y += this->line_height;
466 }
467 }
468
474 const IntervalTimer<TimerWindow> scroll_interval = {std::chrono::milliseconds(2100) / GetCharacterHeight(FS_NORMAL), [this](uint count) {
475 this->text_position -= count;
476 /* If the last text has scrolled start a new from the start */
477 if (this->text_position < (int)(this->GetWidget<NWidgetBase>(WID_A_SCROLLING_TEXT)->pos_y - std::size(_credits) * this->line_height)) {
478 this->text_position = this->GetWidget<NWidgetBase>(WID_A_SCROLLING_TEXT)->pos_y + this->GetWidget<NWidgetBase>(WID_A_SCROLLING_TEXT)->current_y;
479 }
481 }};
482};
483
484void ShowAboutWindow()
485{
487 new AboutWindow();
488}
489
496void ShowEstimatedCostOrIncome(Money cost, int x, int y)
497{
498 StringID msg = STR_MESSAGE_ESTIMATED_COST;
499
500 if (cost < 0) {
501 cost = -cost;
502 msg = STR_MESSAGE_ESTIMATED_INCOME;
503 }
504 ShowErrorMessage(GetEncodedString(msg, cost), {}, WL_INFO, x, y);
505}
506
514void ShowCostOrIncomeAnimation(int x, int y, int z, Money cost)
515{
516 if (cost == 0) {
517 return;
518 }
519 Point pt = RemapCoords(x, y, z);
520 StringID msg = STR_INCOME_FLOAT_COST;
521
522 if (cost < 0) {
523 cost = -cost;
524 msg = STR_INCOME_FLOAT_INCOME;
525 }
526 AddTextEffect(GetEncodedString(msg, cost), pt.x, pt.y, Ticks::DAY_TICKS, TE_RISING);
527}
528
537void ShowFeederIncomeAnimation(int x, int y, int z, Money transfer, Money income)
538{
539 Point pt = RemapCoords(x, y, z);
540
541 if (income == 0) {
542 AddTextEffect(GetEncodedString(STR_FEEDER, transfer), pt.x, pt.y, Ticks::DAY_TICKS, TE_RISING);
543 } else {
544 StringID msg = STR_FEEDER_COST;
545 if (income < 0) {
546 income = -income;
547 msg = STR_FEEDER_INCOME;
548 }
549 AddTextEffect(GetEncodedString(msg, transfer, income), pt.x, pt.y, Ticks::DAY_TICKS, TE_RISING);
550 }
551}
552
562TextEffectID ShowFillingPercent(int x, int y, int z, uint8_t percent, StringID string)
563{
564 Point pt = RemapCoords(x, y, z);
565
566 assert(string != STR_NULL);
567
568 return AddTextEffect(GetEncodedString(string, percent), pt.x, pt.y, 0, TE_STATIC);
569}
570
577void UpdateFillingPercent(TextEffectID te_id, uint8_t percent, StringID string)
578{
579 assert(string != STR_NULL);
580
581 UpdateTextEffect(te_id, GetEncodedString(string, percent));
582}
583
588void HideFillingPercent(TextEffectID *te_id)
589{
590 if (*te_id == INVALID_TE_ID) return;
591
592 RemoveTextEffect(*te_id);
593 *te_id = INVALID_TE_ID;
594}
595
596static constexpr std::initializer_list<NWidgetPart> _nested_tooltips_widgets = {
597 NWidget(WWT_EMPTY, INVALID_COLOUR, WID_TT_BACKGROUND),
598};
599
600static WindowDesc _tool_tips_desc(
601 WDP_MANUAL, {}, 0, 0, // Coordinates and sizes are not used,
604 _nested_tooltips_widgets
605);
606
608struct TooltipsWindow : public Window
609{
611 TooltipCloseCondition close_cond{};
612
613 TooltipsWindow(Window *parent, EncodedString &&text, TooltipCloseCondition close_tooltip) : Window(_tool_tips_desc), text(std::move(text))
614 {
615 this->parent = parent;
616 this->close_cond = close_tooltip;
617
618 this->InitNested();
619
621 }
622
623 Point OnInitialPosition([[maybe_unused]] int16_t sm_width, [[maybe_unused]] int16_t sm_height, [[maybe_unused]] int window_number) override
624 {
625 /* Find the free screen space between the main toolbar at the top, and the statusbar at the bottom.
626 * Add a fixed distance 2 so the tooltip floats free from both bars.
627 */
628 int scr_top = GetMainViewTop() + 2;
629 int scr_bot = GetMainViewBottom() - 2;
630
631 Point pt;
632
633 /* Correctly position the tooltip position, watch out for window and cursor size
634 * Clamp value to below main toolbar and above statusbar. If tooltip would
635 * go below window, flip it so it is shown above the cursor */
636 pt.y = SoftClamp(_cursor.pos.y + _cursor.total_size.y + _cursor.total_offs.y + 5, scr_top, scr_bot);
637 if (pt.y + sm_height > scr_bot) pt.y = std::min(_cursor.pos.y + _cursor.total_offs.y - 5, scr_bot) - sm_height;
638 pt.x = sm_width >= _screen.width ? 0 : SoftClamp(_cursor.pos.x - (sm_width >> 1), 0, _screen.width - sm_width);
639
640 return pt;
641 }
642
643 void UpdateWidgetSize(WidgetID widget, Dimension &size, [[maybe_unused]] const Dimension &padding, [[maybe_unused]] Dimension &fill, [[maybe_unused]] Dimension &resize) override
644 {
645 if (widget != WID_TT_BACKGROUND) return;
646
647 auto str = this->text.GetDecodedString();
648 size.width = std::min<uint>(GetStringBoundingBox(str).width, ScaleGUITrad(194));
649 size.height = GetStringHeight(str, size.width);
650
651 /* Increase slightly to have some space around the box. */
652 size.width += WidgetDimensions::scaled.framerect.Horizontal() + WidgetDimensions::scaled.fullbevel.Horizontal();
653 size.height += WidgetDimensions::scaled.framerect.Vertical() + WidgetDimensions::scaled.fullbevel.Vertical();
654 }
655
656 void DrawWidget(const Rect &r, WidgetID widget) const override
657 {
658 if (widget != WID_TT_BACKGROUND) return;
661
662 DrawStringMultiLine(r.Shrink(WidgetDimensions::scaled.framerect).Shrink(WidgetDimensions::scaled.fullbevel), this->text.GetDecodedString(), TC_BLACK, SA_CENTER);
663 }
664
665 void OnMouseLoop() override
666 {
667 /* Always close tooltips when the cursor is not in our window. */
668 if (!_cursor.in_window) {
669 this->Close();
670 return;
671 }
672
673 /* We can show tooltips while dragging tools. These are shown as long as
674 * we are dragging the tool. Normal tooltips work with hover or rmb. */
675 switch (this->close_cond) {
676 case TCC_RIGHT_CLICK: if (!_right_button_down) this->Close(); break;
677 case TCC_HOVER: if (!_mouse_hovering) this->Close(); break;
678 case TCC_NONE: break;
679
680 case TCC_EXIT_VIEWPORT: {
681 Window *w = FindWindowFromPt(_cursor.pos.x, _cursor.pos.y);
682 if (w == nullptr || IsPtInWindowViewport(w, _cursor.pos.x, _cursor.pos.y) == nullptr) this->Close();
683 break;
684 }
685 }
686 }
687};
688
695void GuiShowTooltips(Window *parent, EncodedString &&text, TooltipCloseCondition close_tooltip)
696{
698
699 if (text.empty() || !_cursor.in_window) return;
700
701 new TooltipsWindow(parent, std::move(text), close_tooltip);
702}
703
704void QueryString::HandleEditBox(Window *w, WidgetID wid)
705{
706 if (w->IsWidgetGloballyFocused(wid) && this->text.HandleCaret()) {
707 w->SetWidgetDirty(wid);
708
709 /* For the OSK also invalidate the parent window */
710 if (w->window_class == WC_OSK) w->InvalidateData();
711 }
712}
713
714static int GetCaretWidth()
715{
716 return GetCharacterWidth(FS_NORMAL, '_');
717}
718
726{
727 const int linewidth = tb.pixels + GetCaretWidth();
728 const int boxwidth = r.Width();
729 if (linewidth <= boxwidth) return r;
730
731 /* Extend to cover whole string. This is left-aligned, adjusted by caret position. */
732 r = r.WithWidth(linewidth, false);
733
734 /* Slide so that the caret is at the centre unless limited by bounds of the line, i.e. near either end. */
735 return r.Translate(-std::clamp(tb.caretxoffs - (boxwidth / 2), 0, linewidth - boxwidth), 0);
736}
737
738void QueryString::DrawEditBox(const Window *w, WidgetID wid) const
739{
740 const NWidgetLeaf *wi = w->GetWidget<NWidgetLeaf>(wid);
741
742 assert((wi->type & WWT_MASK) == WWT_EDITBOX);
743
744 bool rtl = _current_text_dir == TD_RTL;
745 Dimension sprite_size = GetScaledSpriteSize(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT);
746 int clearbtn_width = sprite_size.width + WidgetDimensions::scaled.imgbtn.Horizontal();
747
748 Rect r = wi->GetCurrentRect();
749 Rect cr = r.WithWidth(clearbtn_width, !rtl);
750 Rect fr = r.Indent(clearbtn_width, !rtl);
751
752 DrawFrameRect(cr, wi->colour, wi->IsLowered() ? FrameFlag::Lowered : FrameFlags{});
753 DrawSpriteIgnorePadding(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT, PAL_NONE, cr, SA_CENTER);
754 if (this->text.GetText().empty()) GfxFillRect(cr.Shrink(WidgetDimensions::scaled.bevel), GetColourGradient(wi->colour, SHADE_DARKER), FILLRECT_CHECKER);
755
756 DrawFrameRect(fr, wi->colour, {FrameFlag::Lowered, FrameFlag::Darkened});
758
759 fr = fr.Shrink(WidgetDimensions::scaled.framerect);
760 /* Limit the drawing of the string inside the widget boundaries */
761 DrawPixelInfo dpi;
762 if (!FillDrawPixelInfo(&dpi, fr)) return;
763 /* Keep coordinates relative to the window. */
764 dpi.left += fr.left;
765 dpi.top += fr.top;
766
767 AutoRestoreBackup dpi_backup(_cur_dpi, &dpi);
768
769 /* We will take the current widget length as maximum width, with a small
770 * space reserved at the end for the caret to show */
771 const Textbuf *tb = &this->text;
772 fr = ScrollEditBoxTextRect(fr, *tb);
773
774 /* If we have a marked area, draw a background highlight. */
775 if (tb->marklength != 0) GfxFillRect(fr.left + tb->markxoffs, fr.top, fr.left + tb->markxoffs + tb->marklength - 1, fr.bottom, PC_GREY);
776
777 DrawString(fr.left, fr.right, CentreBounds(fr.top, fr.bottom, GetCharacterHeight(FS_NORMAL)), tb->GetText(), TC_YELLOW);
778 bool focussed = w->IsWidgetGloballyFocused(wid) || IsOSKOpenedFor(w, wid);
779 if (focussed && tb->caret) {
780 int caret_width = GetCaretWidth();
781 if (rtl) {
782 DrawString(fr.right - tb->pixels + tb->caretxoffs - caret_width, fr.right - tb->pixels + tb->caretxoffs, CentreBounds(fr.top, fr.bottom, GetCharacterHeight(FS_NORMAL)), "_", TC_WHITE);
783 } else {
784 DrawString(fr.left + tb->caretxoffs, fr.left + tb->caretxoffs + caret_width, CentreBounds(fr.top, fr.bottom, GetCharacterHeight(FS_NORMAL)), "_", TC_WHITE);
785 }
786 }
787}
788
796{
797 const NWidgetLeaf *wi = w->GetWidget<NWidgetLeaf>(wid);
798
799 assert((wi->type & WWT_MASK) == WWT_EDITBOX);
800
801 bool rtl = _current_text_dir == TD_RTL;
802 Dimension sprite_size = GetScaledSpriteSize(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT);
803 int clearbtn_width = sprite_size.width + WidgetDimensions::scaled.imgbtn.Horizontal();
804
805 Rect r = wi->GetCurrentRect().Indent(clearbtn_width, !rtl).Shrink(WidgetDimensions::scaled.framerect);
806
807 /* Clamp caret position to be inside out current width. */
808 const Textbuf *tb = &this->text;
809 r = ScrollEditBoxTextRect(r, *tb);
810
811 Point pt = {r.left + tb->caretxoffs, r.top};
812 return pt;
813}
814
823Rect QueryString::GetBoundingRect(const Window *w, WidgetID wid, size_t from, size_t to) const
824{
825 const NWidgetLeaf *wi = w->GetWidget<NWidgetLeaf>(wid);
826
827 assert((wi->type & WWT_MASK) == WWT_EDITBOX);
828
829 bool rtl = _current_text_dir == TD_RTL;
830 Dimension sprite_size = GetScaledSpriteSize(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT);
831 int clearbtn_width = sprite_size.width + WidgetDimensions::scaled.imgbtn.Horizontal();
832
833 Rect r = wi->GetCurrentRect().Indent(clearbtn_width, !rtl).Shrink(WidgetDimensions::scaled.framerect);
834
835 /* Clamp caret position to be inside our current width. */
836 const Textbuf *tb = &this->text;
837 r = ScrollEditBoxTextRect(r, *tb);
838
839 /* Get location of first and last character. */
840 const auto p1 = GetCharPosInString(tb->GetText(), from, FS_NORMAL);
841 const auto p2 = from != to ? GetCharPosInString(tb->GetText(), to, FS_NORMAL) : p1;
842
843 return r.WithX(Clamp(r.left + p1.left, r.left, r.right), Clamp(r.left + p2.right, r.left, r.right));
844}
845
853ptrdiff_t QueryString::GetCharAtPosition(const Window *w, WidgetID wid, const Point &pt) const
854{
855 const NWidgetLeaf *wi = w->GetWidget<NWidgetLeaf>(wid);
856
857 assert((wi->type & WWT_MASK) == WWT_EDITBOX);
858
859 bool rtl = _current_text_dir == TD_RTL;
860 Dimension sprite_size = GetScaledSpriteSize(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT);
861 int clearbtn_width = sprite_size.width + WidgetDimensions::scaled.imgbtn.Horizontal();
862
863 Rect r = wi->GetCurrentRect().Indent(clearbtn_width, !rtl).Shrink(WidgetDimensions::scaled.framerect);
864
865 if (!IsInsideMM(pt.y, r.top, r.bottom)) return -1;
866
867 /* Clamp caret position to be inside our current width. */
868 const Textbuf *tb = &this->text;
869 r = ScrollEditBoxTextRect(r, *tb);
870
871 return ::GetCharAtPosition(tb->GetText(), pt.x - r.left);
872}
873
874void QueryString::ClickEditBox(Window *w, Point pt, WidgetID wid, int click_count, bool focus_changed)
875{
876 const NWidgetLeaf *wi = w->GetWidget<NWidgetLeaf>(wid);
877
878 assert((wi->type & WWT_MASK) == WWT_EDITBOX);
879
880 bool rtl = _current_text_dir == TD_RTL;
881 Dimension sprite_size = GetScaledSpriteSize(rtl ? SPR_IMG_DELETE_RIGHT : SPR_IMG_DELETE_LEFT);
882 int clearbtn_width = sprite_size.width + WidgetDimensions::scaled.imgbtn.Horizontal();
883
884 Rect cr = wi->GetCurrentRect().WithWidth(clearbtn_width, !rtl);
885
886 if (IsInsideMM(pt.x, cr.left, cr.right)) {
887 if (!this->text.GetText().empty()) {
888 this->text.DeleteAll();
889 w->HandleButtonClick(wid);
890 w->OnEditboxChanged(wid);
891 }
892 return;
893 }
894
896 (!focus_changed || _settings_client.gui.osk_activation == OSKA_IMMEDIATELY) &&
897 (click_count == 2 || _settings_client.gui.osk_activation != OSKA_DOUBLE_CLICK)) {
898 /* Open the OSK window */
899 ShowOnScreenKeyboard(w, wid);
900 }
901}
902
904struct QueryStringWindow : public Window
905{
907 QueryStringFlags flags{};
908
910
911 QueryStringWindow(std::string_view str, StringID caption, uint max_bytes, uint max_chars, WindowDesc &desc, Window *parent, CharSetFilter afilter, QueryStringFlags flags) :
912 Window(desc), editbox(max_bytes, max_chars)
913 {
914 this->editbox.text.Assign(str);
915
916 if (!flags.Test(QueryStringFlag::AcceptUnchanged)) this->editbox.orig = this->editbox.text.GetText();
917
918 this->querystrings[WID_QS_TEXT] = &this->editbox;
919 this->editbox.caption = caption;
920 this->editbox.cancel_button = WID_QS_CANCEL;
921 this->editbox.ok_button = WID_QS_OK;
922 this->editbox.text.afilter = afilter;
923 this->flags = flags;
924
925 this->CreateNestedTree();
927 this->GetWidget<NWidgetStacked>(WID_QS_MOVE_SEL)->SetDisplayedPlane((this->flags.Test(QueryStringFlag::EnableMove)) ? 0 : SZSP_NONE);
928 this->FinishInitNested(WN_QUERY_STRING);
929
930 this->parent = parent;
931
932 this->SetFocusedWidget(WID_QS_TEXT);
933 }
934
935 std::string GetWidgetString(WidgetID widget, StringID stringid) const override
936 {
937 if (widget == WID_QS_CAPTION) return GetString(this->editbox.caption);
938
939 return this->Window::GetWidgetString(widget, stringid);
940 }
941
942 void OnOk()
943 {
944 if (!this->editbox.orig.has_value() || this->editbox.text.GetText() != this->editbox.orig) {
945 assert(this->parent != nullptr);
946
947 this->parent->OnQueryTextFinished(std::string{this->editbox.text.GetText()});
948 this->editbox.handled = true;
949 }
950 }
951
952 void OnClick([[maybe_unused]] Point pt, WidgetID widget, [[maybe_unused]] int click_count) override
953 {
954 switch (widget) {
955 case WID_QS_DEFAULT:
956 this->editbox.text.DeleteAll();
957 [[fallthrough]];
958
959 case WID_QS_OK:
960 this->OnOk();
961 [[fallthrough]];
962
963 case WID_QS_CANCEL:
964 this->Close();
965 break;
966
967 case WID_QS_MOVE:
968 this->last_user_action = widget;
969
970 if (Station::IsExpected(Station::Get(this->parent->window_number))) {
971 /* this is a station */
972 SetViewportStationRect(Station::Get(this->parent->window_number), !this->IsWidgetLowered(WID_QS_MOVE));
973 } else {
974 /* this is a waypoint */
975 SetViewportWaypointRect(Waypoint::Get(this->parent->window_number), !this->IsWidgetLowered(WID_QS_MOVE));
976 }
977
978 HandlePlacePushButton(this, WID_QS_MOVE, SPR_CURSOR_SIGN, HT_RECT);
979 break;
980 }
981 }
982
983 void OnPlaceObject([[maybe_unused]] Point pt, TileIndex tile) override
984 {
985 switch (this->last_user_action) {
986 case WID_QS_MOVE: // Move name button
987 if (Station::IsExpected(Station::Get(this->parent->window_number))) {
988 /* this is a station */
989 Command<Commands::MoveStationName>::Post(STR_ERROR_CAN_T_MOVE_STATION_NAME, CcMoveStationName, this->parent->window_number, tile);
990 } else {
991 /* this is a waypoint */
992 Command<Commands::MoveWaypointNAme>::Post(STR_ERROR_CAN_T_MOVE_WAYPOINT_NAME, CcMoveWaypointName, this->parent->window_number, tile);
993 }
994 break;
995
996 default: NOT_REACHED();
997 }
998 }
999
1000 void OnPlaceObjectAbort() override
1001 {
1002 if (Station::IsExpected(Station::Get(this->parent->window_number))) {
1003 /* this is a station */
1004 SetViewportStationRect(Station::Get(this->parent->window_number), false);
1005 } else {
1006 /* this is a waypoint */
1007 SetViewportWaypointRect(Waypoint::Get(this->parent->window_number), false);
1008 }
1009
1010 this->RaiseButtons();
1011 }
1012
1013 void Close([[maybe_unused]] int data = 0) override
1014 {
1015 if (this->parent != nullptr) {
1016 if (this->parent->window_class == WC_STATION_VIEW) SetViewportStationRect(Station::Get(this->parent->window_number), false);
1017 if (this->parent->window_class == WC_WAYPOINT_VIEW) SetViewportWaypointRect(Waypoint::Get(this->parent->window_number), false);
1018
1019 if (!this->editbox.handled) {
1020 Window *parent = this->parent;
1021 this->parent = nullptr; // so parent doesn't try to close us again
1022 parent->OnQueryTextFinished(std::nullopt);
1023 }
1024 }
1025
1026 this->Window::Close();
1027 }
1028};
1029
1030static constexpr std::initializer_list<NWidgetPart> _nested_query_string_widgets = {
1032 NWidget(WWT_CLOSEBOX, COLOUR_GREY),
1033 NWidget(WWT_CAPTION, COLOUR_GREY, WID_QS_CAPTION), SetTextStyle(TC_WHITE),
1034 EndContainer(),
1035 NWidget(WWT_PANEL, COLOUR_GREY),
1036 NWidget(WWT_EDITBOX, COLOUR_GREY, WID_QS_TEXT), SetMinimalSize(256, 0), SetFill(1, 0), SetPadding(2, 2, 2, 2),
1037 EndContainer(),
1039 NWidget(NWID_SELECTION, INVALID_COLOUR, WID_QS_DEFAULT_SEL),
1040 NWidget(WWT_TEXTBTN, COLOUR_GREY, WID_QS_DEFAULT), SetMinimalSize(65, 12), SetFill(1, 1), SetStringTip(STR_BUTTON_DEFAULT),
1041 EndContainer(),
1042 NWidget(WWT_TEXTBTN, COLOUR_GREY, WID_QS_CANCEL), SetMinimalSize(65, 12), SetFill(1, 1), SetStringTip(STR_BUTTON_CANCEL),
1043 NWidget(WWT_TEXTBTN, COLOUR_GREY, WID_QS_OK), SetMinimalSize(65, 12), SetFill(1, 1), SetStringTip(STR_BUTTON_OK),
1044 NWidget(NWID_SELECTION, INVALID_COLOUR, WID_QS_MOVE_SEL),
1045 NWidget(WWT_TEXTBTN, COLOUR_GREY, WID_QS_MOVE), SetMinimalSize(65, 12), SetFill(1, 1), SetStringTip(STR_BUTTON_MOVE),
1046 EndContainer(),
1047 EndContainer(),
1048};
1049
1050static WindowDesc _query_string_desc(
1051 WDP_CENTER, {}, 0, 0,
1053 {},
1054 _nested_query_string_widgets
1055);
1056
1066void ShowQueryString(std::string_view str, StringID caption, uint maxsize, Window *parent, CharSetFilter afilter, QueryStringFlags flags)
1067{
1068 assert(parent != nullptr);
1069
1071 new QueryStringWindow(str, caption, (flags.Test(QueryStringFlag::LengthIsInChars) ? MAX_CHAR_LENGTH : 1) * maxsize, maxsize, _query_string_desc, parent, afilter, flags);
1072}
1073
1078void UpdateQueryStringDefault(std::string_view str)
1079{
1081 if (w != nullptr) w->editbox.orig = str;
1082}
1083
1087struct QueryWindow : public Window {
1091
1093 : Window(desc), proc(callback), caption(std::move(caption)), message(std::move(message))
1094 {
1095 this->parent = parent;
1096
1097 this->CreateNestedTree();
1099 }
1100
1101 void Close([[maybe_unused]] int data = 0) override
1102 {
1103 if (this->proc != nullptr) this->proc(this->parent, false);
1104 this->Window::Close();
1105 }
1106
1107 void FindWindowPlacementAndResize(int, int, bool) override
1108 {
1109 /* Position query window over the calling window, ensuring it's within screen bounds. */
1110 this->left = SoftClamp(parent->left + (parent->width / 2) - (this->width / 2), 0, _screen.width - this->width);
1111 this->top = SoftClamp(parent->top + (parent->height / 2) - (this->height / 2), 0, _screen.height - this->height);
1112 this->SetDirty();
1113 }
1114
1115 std::string GetWidgetString(WidgetID widget, StringID stringid) const override
1116 {
1117 switch (widget) {
1118 case WID_Q_CAPTION:
1119 return this->caption.GetDecodedString();
1120
1121 default:
1122 return this->Window::GetWidgetString(widget, stringid);
1123 }
1124 }
1125
1126 void UpdateWidgetSize(WidgetID widget, Dimension &size, [[maybe_unused]] const Dimension &padding, [[maybe_unused]] Dimension &fill, [[maybe_unused]] Dimension &resize) override
1127 {
1128 if (widget != WID_Q_TEXT) return;
1129
1130 size = GetStringMultiLineBoundingBox(this->message.GetDecodedString(), size);
1131 }
1132
1133 void DrawWidget(const Rect &r, WidgetID widget) const override
1134 {
1135 if (widget != WID_Q_TEXT) return;
1136
1137 DrawStringMultiLine(r, this->message.GetDecodedString(), TC_FROMSTRING, SA_CENTER);
1138 }
1139
1140 void OnClick([[maybe_unused]] Point pt, WidgetID widget, [[maybe_unused]] int click_count) override
1141 {
1142 switch (widget) {
1143 case WID_Q_YES: {
1144 /* in the Generate New World window, clicking 'Yes' causes
1145 * CloseNonVitalWindows() to be called - we shouldn't be in a window then */
1146 QueryCallbackProc *proc = this->proc;
1147 Window *parent = this->parent;
1148 /* Prevent the destructor calling the callback function */
1149 this->proc = nullptr;
1150 this->Close();
1151 if (proc != nullptr) {
1152 proc(parent, true);
1153 proc = nullptr;
1154 }
1155 break;
1156 }
1157 case WID_Q_NO:
1158 this->Close();
1159 break;
1160 }
1161 }
1162
1163 EventState OnKeyPress([[maybe_unused]] char32_t key, uint16_t keycode) override
1164 {
1165 /* ESC closes the window, Enter confirms the action */
1166 switch (keycode) {
1167 case WKC_RETURN:
1168 case WKC_NUM_ENTER:
1169 if (this->proc != nullptr) {
1170 this->proc(this->parent, true);
1171 this->proc = nullptr;
1172 }
1173 [[fallthrough]];
1174
1175 case WKC_ESC:
1176 this->Close();
1177 return ES_HANDLED;
1178 }
1179 return ES_NOT_HANDLED;
1180 }
1181};
1182
1183static constexpr std::initializer_list<NWidgetPart> _nested_query_widgets = {
1185 NWidget(WWT_CLOSEBOX, COLOUR_RED),
1186 NWidget(WWT_CAPTION, COLOUR_RED, WID_Q_CAPTION),
1187 EndContainer(),
1188 NWidget(WWT_PANEL, COLOUR_RED),
1190 NWidget(WWT_TEXT, INVALID_COLOUR, WID_Q_TEXT), SetMinimalSize(200, 12),
1192 NWidget(WWT_PUSHTXTBTN, COLOUR_YELLOW, WID_Q_NO), SetMinimalSize(71, 12), SetFill(1, 1), SetStringTip(STR_QUIT_NO),
1193 NWidget(WWT_PUSHTXTBTN, COLOUR_YELLOW, WID_Q_YES), SetMinimalSize(71, 12), SetFill(1, 1), SetStringTip(STR_QUIT_YES),
1194 EndContainer(),
1195 EndContainer(),
1196 EndContainer(),
1197};
1198
1199static WindowDesc _query_desc(
1200 WDP_CENTER, {}, 0, 0,
1203 _nested_query_widgets
1204);
1205
1216void ShowQuery(EncodedString &&caption, EncodedString &&message, Window *parent, QueryCallbackProc *callback, bool focus)
1217{
1218 if (parent == nullptr) parent = GetMainWindow();
1219
1220 for (Window *w : Window::Iterate()) {
1221 if (w->window_class != WC_CONFIRM_POPUP_QUERY) continue;
1222
1223 QueryWindow *qw = dynamic_cast<QueryWindow *>(w);
1224 assert(qw != nullptr);
1225 if (qw->parent != parent || qw->proc != callback) continue;
1226
1227 qw->Close();
1228 break;
1229 }
1230
1231 QueryWindow *q = new QueryWindow(_query_desc, std::move(caption), std::move(message), parent, callback);
1232 if (focus) SetFocusedWindow(q);
1233}
std::optional< std::string > BuildCargoAcceptanceString(const CargoArray &acceptance, StringID label)
Build comma-separated cargo acceptance string.
constexpr bool Test(Tvalue_type value) const
Test if the value-th bit is set.
constexpr Timpl & Reset()
Reset all bits.
Common return value for all commands.
bool Succeeded() const
Did this command succeed?
Money GetCost() const
The costs as made up to this moment.
Container for an encoded string, created by GetEncodedString.
std::string GetDecodedString() const
Decode the encoded string.
Definition strings.cpp:207
An interval timer will fire every interval, and will continue to fire until it is deleted.
Definition timer.h:76
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
Definition misc_gui.cpp:79
void OnInvalidateData(int data=0, bool gui_scope=true) override
Some data on this window has become invalid.
Definition misc_gui.cpp:309
StringList landinfo_data
Info lines to show.
Definition misc_gui.cpp:73
std::string cargo_acceptance
Centered multi-line string for cargo acceptance.
Definition misc_gui.cpp:74
bool IsNewGRFInspectable() const override
Is the data related to this window NewGRF inspectable?
Definition misc_gui.cpp:281
void OnInit() override
Notification that the nested widget tree gets initialized.
Definition misc_gui.cpp:138
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
Definition misc_gui.cpp:291
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
Definition misc_gui.cpp:94
void ShowNewGRFInspectWindow() const override
Show the NewGRF inspection window.
Definition misc_gui.cpp:286
WidgetType type
Type of the widget / nested widget.
Colours colour
Colour of this widget.
bool IsLowered() const
Return whether the widget is lowered.
Leaf widget.
Stacked widgets, widgets all occupying the same space in the window.
bool SetDisplayedPlane(int plane)
Select which plane to show (for NWID_SELECTION only).
Definition widget.cpp:1423
static constexpr TimerGameTick::Ticks DAY_TICKS
1 day is 74 ticks; TimerGameCalendar::date_fract used to be uint16_t and incremented by 885.
Wrapper class to abstract away the way the tiles are stored.
Definition map_func.h:25
static constexpr TimerGame< struct Calendar >::Date INVALID_DATE
static WidgetDimensions scaled
Widget dimensions scaled for current zoom level.
Definition window_gui.h:30
RectPadding imgbtn
Padding around image button image.
Definition window_gui.h:34
static const WidgetDimensions unscaled
Unscaled widget dimensions.
Definition window_gui.h:93
RectPadding bevel
Bevel thickness, affected by "scaled bevels" game option.
Definition window_gui.h:38
Functions related to commands.
@ QueryCost
query cost only, don't build.
Definition of stuff that is very close to a company, like the company struct itself.
std::array< StringParameter, 2 > GetParamsForOwnedBy(Owner owner, TileIndex tile)
Get the right StringParameters for STR_ERROR_OWNED_BY.
CompanyID _local_company
Company controlled by the human player at this client. Can also be COMPANY_SPECTATOR.
CompanyID _current_company
Company currently doing an action.
Functions related to companies.
static constexpr Owner OWNER_NONE
The tile has no ownership.
static constexpr Owner OWNER_WATER
The tile/execution is done by "water".
Functions related to debugging.
#define Debug(category, level, format_string,...)
Output a line of debugging information.
Definition debug.h:37
Functions related to errors.
@ WL_INFO
Used for DoCommand-like (and some non-fatal AI GUI) errors/information.
Definition error.h:24
void ShowErrorMessage(EncodedString &&summary_msg, int x, int y, CommandCost &cc)
Display an error message in a window.
int GetCharacterHeight(FontSize size)
Get height of a character for a given font size.
Definition fontcache.cpp:87
Dimension maxdim(const Dimension &d1, const Dimension &d2)
Compute bounding box of both dimensions.
Geometry functions.
int CentreBounds(int min, int max, int size)
Determine where to position a centred object.
int GetStringHeight(std::string_view str, int maxw, FontSize fontsize)
Calculates height of string (in pixels).
Definition gfx.cpp:717
Dimension GetStringBoundingBox(std::string_view str, FontSize start_fontsize)
Return the string dimension in pixels.
Definition gfx.cpp:900
int DrawString(int left, int right, int top, std::string_view str, TextColour colour, StringAlignment align, bool underline, FontSize fontsize)
Draw string, possibly truncated to make it fit in its allocated space.
Definition gfx.cpp:669
bool _ctrl_pressed
Is Ctrl pressed?
Definition gfx.cpp:39
uint8_t GetCharacterWidth(FontSize size, char32_t key)
Return width of character glyph.
Definition gfx.cpp:1278
Dimension GetStringMultiLineBoundingBox(StringID str, const Dimension &suggestion)
Calculate string bounding box for multi-line strings.
Definition gfx.cpp:753
int DrawStringMultiLine(int left, int right, int top, int bottom, std::string_view str, TextColour colour, StringAlignment align, bool underline, FontSize fontsize)
Draw string, possibly over multiple lines.
Definition gfx.cpp:788
void GfxFillRect(int left, int top, int right, int bottom, const std::variant< PixelColour, PaletteID > &colour, FillRectMode mode)
Applies a certain FillRectMode-operation to a rectangle [left, right] x [top, bottom] on the screen.
Definition gfx.cpp:116
bool FillDrawPixelInfo(DrawPixelInfo *n, int left, int top, int width, int height)
Set up a clipping area for only drawing into a certain area.
Definition gfx.cpp:1573
bool _right_button_down
Is right mouse button pressed?
Definition gfx.cpp:44
Dimension GetScaledSpriteSize(SpriteID sprid)
Scale sprite size for GUI.
Definition widget.cpp:68
ParagraphLayouter::Position GetCharPosInString(std::string_view str, size_t pos, FontSize start_fontsize)
Get the leading corner of a character in a single-line string relative to the start of the string.
Functions related to laying out the texts.
@ FS_NORMAL
Index of the normal font in the font tables.
Definition gfx_type.h:249
@ SA_LEFT
Left align the text.
Definition gfx_type.h:388
@ SA_HOR_CENTER
Horizontally center the text.
Definition gfx_type.h:389
@ SA_FORCE
Force the alignment, i.e. don't swap for RTL languages.
Definition gfx_type.h:400
@ SA_CENTER
Center both horizontally and vertically.
Definition gfx_type.h:398
@ FILLRECT_CHECKER
Draw only every second pixel, used for greying-out.
Definition gfx_type.h:346
constexpr NWidgetPart SetFill(uint16_t fill_x, uint16_t fill_y)
Widget part function for setting filling.
constexpr NWidgetPart SetSpriteTip(SpriteID sprite, StringID tip={})
Widget part function for setting the sprite and tooltip.
constexpr NWidgetPart SetPIP(uint8_t pre, uint8_t inter, uint8_t post)
Widget part function for setting a pre/inter/post spaces.
constexpr NWidgetPart SetPadding(uint8_t top, uint8_t right, uint8_t bottom, uint8_t left)
Widget part function for setting additional space around a widget.
constexpr NWidgetPart SetStringTip(StringID string, StringID tip={})
Widget part function for setting the string and tooltip.
constexpr NWidgetPart SetAspect(float ratio, AspectFlags flags=AspectFlag::ResizeX)
Widget part function for setting the aspect ratio.
constexpr NWidgetPart SetTextStyle(TextColour colour, FontSize size=FS_NORMAL)
Widget part function for setting the text style.
constexpr NWidgetPart SetMinimalSize(int16_t x, int16_t y)
Widget part function for setting the minimal size.
constexpr NWidgetPart EndContainer()
Widget part function for denoting the end of a container (horizontal, vertical, WWT_FRAME,...
constexpr NWidgetPart NWidget(WidgetType tp, Colours col, WidgetID idx=INVALID_WIDGET)
Widget part function for starting a new 'real' widget.
void SetDirty() const
Mark entire window as dirty (in need of re-paint).
Definition window.cpp:967
GUI functions that shouldn't be here.
void ShowExtraViewportWindow(TileIndex tile=INVALID_TILE)
Show a new Extra Viewport window.
Functions related to OTTD's landscape.
Point RemapCoords(int x, int y, int z)
Map 3D world or tile coordinate to equivalent 2D coordinate as used in the viewports and smallmap.
Definition landscape.h:81
Command definitions related to landscape (slopes etc.).
#define Rect
Macro that prevents name conflicts between included headers.
#define Point
Macro that prevents name conflicts between included headers.
bool HandlePlacePushButton(Window *w, WidgetID widget, CursorID cursor, HighLightStyle mode)
This code is shared for the majority of the pushbuttons.
Definition main_gui.cpp:63
static uint TileY(TileIndex tile)
Get the Y component of a tile.
Definition map_func.h:430
static uint TileX(TileIndex tile)
Get the X component of a tile.
Definition map_func.h:420
constexpr bool IsInsideMM(const size_t x, const size_t min, const size_t max) noexcept
Checks if a value is in an interval.
constexpr T SoftClamp(const T a, const T min, const T max)
Clamp a value between an interval.
constexpr T Clamp(const T a, const T min, const T max)
Clamp a value between an interval.
Definition math_func.hpp:79
void GuiShowTooltips(Window *parent, EncodedString &&text, TooltipCloseCondition close_tooltip)
Shows a tooltip.
Definition misc_gui.cpp:695
void ShowQuery(EncodedString &&caption, EncodedString &&message, Window *parent, QueryCallbackProc *callback, bool focus)
Show a confirmation window with standard 'yes' and 'no' buttons The window is aligned to the centre o...
void HideFillingPercent(TextEffectID *te_id)
Hide vehicle loading indicators.
Definition misc_gui.cpp:588
void ShowCostOrIncomeAnimation(int x, int y, int z, Money cost)
Display animated income or costs on the map.
Definition misc_gui.cpp:514
void ShowEstimatedCostOrIncome(Money cost, int x, int y)
Display estimated costs.
Definition misc_gui.cpp:496
OskActivation
Method to open the OSK.
Definition misc_gui.cpp:47
@ OSKA_SINGLE_CLICK
Single click after focus click opens OSK.
Definition misc_gui.cpp:50
@ OSKA_IMMEDIATELY
Focusing click already opens OSK.
Definition misc_gui.cpp:51
@ OSKA_DISABLED
The OSK shall not be activated at all.
Definition misc_gui.cpp:48
@ OSKA_DOUBLE_CLICK
Double click on the edit box opens OSK.
Definition misc_gui.cpp:49
void ShowLandInfo(TileIndex tile)
Show land information window.
Definition misc_gui.cpp:322
void ShowFeederIncomeAnimation(int x, int y, int z, Money transfer, Money income)
Display animated feeder income.
Definition misc_gui.cpp:537
void ShowQueryString(std::string_view str, StringID caption, uint maxsize, Window *parent, CharSetFilter afilter, QueryStringFlags flags)
Show a query popup window with a textbox in it.
void UpdateFillingPercent(TextEffectID te_id, uint8_t percent, StringID string)
Update vehicle loading indicators.
Definition misc_gui.cpp:577
TextEffectID ShowFillingPercent(int x, int y, int z, uint8_t percent, StringID string)
Display vehicle loading indicators.
Definition misc_gui.cpp:562
void UpdateQueryStringDefault(std::string_view str)
Updates default text value of query strign window.
static Rect ScrollEditBoxTextRect(Rect r, const Textbuf &tb)
Reposition edit text box rect based on textbuf length can caret position.
Definition misc_gui.cpp:725
Types related to the misc widgets.
@ WID_TT_BACKGROUND
Background of the window.
Definition misc_widget.h:21
@ WID_QS_MOVE_SEL
Container for move button, which can be hidden.
Definition misc_widget.h:40
@ WID_QS_DEFAULT
Default button.
Definition misc_widget.h:35
@ WID_QS_CAPTION
Caption of the window.
Definition misc_widget.h:33
@ WID_QS_TEXT
Text of the query.
Definition misc_widget.h:34
@ WID_QS_MOVE
Move button.
Definition misc_widget.h:39
@ WID_QS_CANCEL
Cancel button.
Definition misc_widget.h:37
@ WID_QS_DEFAULT_SEL
Container for default button, which can be hidden.
Definition misc_widget.h:36
@ WID_QS_OK
OK button.
Definition misc_widget.h:38
@ WID_A_WEBSITE
URL of OpenTTD website.
Definition misc_widget.h:27
@ WID_A_COPYRIGHT
Copyright string.
Definition misc_widget.h:28
@ WID_A_SCROLLING_TEXT
The actually scrolling text.
Definition misc_widget.h:26
@ WID_Q_NO
Yes button.
Definition misc_widget.h:47
@ WID_Q_YES
No button.
Definition misc_widget.h:48
@ WID_Q_CAPTION
Caption of the window.
Definition misc_widget.h:45
@ WID_Q_TEXT
Text of the query.
Definition misc_widget.h:46
@ WID_LI_BACKGROUND
Background of the window.
Definition misc_widget.h:16
@ WID_LI_LOCATION
Scroll to location.
Definition misc_widget.h:15
Functions/types related to NewGRF debugging.
GrfSpecFeature GetGrfSpecFeature(TileIndex tile)
Get the GrfSpecFeature associated with the tile.
PixelColour GetColourGradient(Colours colour, ColourShade shade)
Get colour gradient palette index.
Definition palette.cpp:388
static constexpr PixelColour PC_GREY
Grey palette colour.
static constexpr PixelColour PC_BLACK
Black palette colour.
static constexpr PixelColour PC_LIGHT_YELLOW
Light yellow palette colour.
Base for the GUIs that have an edit box in them.
void ShowOnScreenKeyboard(Window *parent, WidgetID button)
Show the on-screen keyboard (osk) associated with a given textbox.
Definition osk_gui.cpp:391
bool IsOSKOpenedFor(const Window *w, WidgetID button)
Check whether the OSK is opened for a specific editbox.
Definition osk_gui.cpp:422
Declaration of OTTD revision dependent variables.
A number of safeguards to prevent using unsafe methods.
GameSettings _settings_game
Game settings of a running game or the scenario editor.
Definition settings.cpp:61
ClientSettings _settings_client
The current settings for this game.
Definition settings.cpp:60
Base classes/functions for stations.
void CcMoveStationName(Commands, const CommandCost &result, StationID station_id)
Callback function that is called after a name is moved.
Command definitions related to stations.
Definition of base types and functions in a cross-platform compatible way.
Functions related to low-level strings.
CharSetFilter
Valid filter types for IsValidChar.
Definition string_type.h:24
std::vector< std::string > StringList
Type for a list of strings.
Definition string_type.h:60
void GetStringWithArgs(StringBuilder &builder, StringID string, StringParameters &args, uint case_index, bool game_script)
Get a parsed string with most special stringcodes replaced by the string parameters.
Definition strings.cpp:336
EncodedString GetEncodedString(StringID str)
Encode a string with no parameters into an encoded string.
Definition strings.cpp:90
std::string GetString(StringID string)
Resolve the given StringID into a std::string with formatting but no parameters.
Definition strings.cpp:424
TextDirection _current_text_dir
Text direction of the currently selected language.
Definition strings.cpp:56
Functions related to OTTD's strings.
int64_t PackVelocity(uint speed, VehicleType type)
Pack velocity and vehicle type for use with SCC_VELOCITY string parameter.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
@ TD_RTL
Text is written right-to-left by default.
static const int MAX_CHAR_LENGTH
Max. length of UTF-8 encoded unicode character.
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
Definition misc_gui.cpp:454
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
Definition misc_gui.cpp:438
const IntervalTimer< TimerWindow > scroll_interval
Scroll the text in the about window slow.
Definition misc_gui.cpp:474
std::string GetWidgetString(WidgetID widget, StringID stringid) const override
Get the raw string for a widget.
Definition misc_gui.cpp:431
int text_position
The top of the scrolling text.
Definition misc_gui.cpp:420
static const int num_visible_lines
The number of lines visible simultaneously.
Definition misc_gui.cpp:422
int line_height
The height of a single line.
Definition misc_gui.cpp:421
Class for storing amounts of cargo.
Definition cargo_type.h:115
GUISettings gui
settings related to the GUI
T y
Y coordinate.
T x
X coordinate.
Dimensions (a width and height) of a rectangle in 2D.
uint8_t osk_activation
Mouse gesture to trigger the OSK.
static Company * GetIfValid(auto index)
Class for the string query window.
Definition misc_gui.cpp:905
void OnPlaceObjectAbort() override
The user cancelled a tile highlight mode that has been set.
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
Definition misc_gui.cpp:952
void OnPlaceObject(Point pt, TileIndex tile) override
The user clicked some place on the map when a tile highlight mode has been set.
Definition misc_gui.cpp:983
void Close(int data=0) override
Hide the window and all its child windows, and mark them for a later deletion.
QueryString editbox
Editbox.
Definition misc_gui.cpp:906
std::string GetWidgetString(WidgetID widget, StringID stringid) const override
Get the raw string for a widget.
Definition misc_gui.cpp:935
QueryStringFlags flags
Flags controlling behaviour of the window.
Definition misc_gui.cpp:907
WidgetID last_user_action
Last started user action.
Definition misc_gui.cpp:909
Data stored about a string that can be modified in the GUI.
ptrdiff_t GetCharAtPosition(const Window *w, WidgetID wid, const Point &pt) const
Get the character that is rendered at a position.
Definition misc_gui.cpp:853
Point GetCaretPosition(const Window *w, WidgetID wid) const
Get the current caret position.
Definition misc_gui.cpp:795
Rect GetBoundingRect(const Window *w, WidgetID wid, size_t from, size_t to) const
Get the bounding rectangle for a range of the query string.
Definition misc_gui.cpp:823
Window used for asking the user a YES/NO question.
void Close(int data=0) override
Hide the window and all its child windows, and mark them for a later deletion.
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
QueryCallbackProc * proc
callback function executed on closing of popup. Window* points to parent, bool is true if 'yes' click...
EventState OnKeyPress(char32_t key, uint16_t keycode) override
A key has been pressed.
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
std::string GetWidgetString(WidgetID widget, StringID stringid) const override
Get the raw string for a widget.
EncodedString caption
caption for query window.
EncodedString message
message for query window.
void FindWindowPlacementAndResize(int, int, bool) override
Resize window towards the default size.
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
constexpr uint Horizontal() const
Get total horizontal padding of RectPadding.
Specification of a rectangle with absolute coordinates of all edges.
Rect WithWidth(int width, bool end) const
Copy Rect and set its width.
int Width() const
Get width of Rect.
Rect Shrink(int s) const
Copy and shrink Rect by s pixels.
Rect Indent(int indent, bool end) const
Copy Rect and indent it from its position.
Rect WithX(int new_left, int new_right) const
Create a new Rect, replacing the left and right coordiates.
Rect Translate(int x, int y) const
Copy and translate Rect by x,y pixels.
static bool IsExpected(const BaseStation *st)
static Station * Get(auto index)
Helper/buffer for input fields.
uint16_t pixels
the current size of the string in pixels
uint16_t markxoffs
the start position of the marked area in pixels
void DeleteAll()
Delete every character in the textbuffer.
Definition textbuf.cpp:112
std::string_view GetText() const
Get the current text.
Definition textbuf.cpp:284
uint16_t caretxoffs
the current position of the caret in pixels
uint16_t marklength
the length of the marked area in pixels
void Assign(std::string_view text)
Copy a string into the textbuffer.
Definition textbuf.cpp:420
bool caret
is the caret ("_") visible or not
Tile description for the 'land area information' tool.
Definition tile_cmd.h:38
uint16_t rail_speed
Speed limit of rail (bridges and track).
Definition tile_cmd.h:51
std::optional< std::string > grf
newGRF used for the tile contents
Definition tile_cmd.h:49
StringID station_name
Type of station within the class.
Definition tile_cmd.h:45
StringID str
Description of the tile.
Definition tile_cmd.h:39
TimerGameCalendar::Date build_date
Date of construction of tile contents.
Definition tile_cmd.h:43
std::array< Owner, 4 > owner
Name of the owner(s).
Definition tile_cmd.h:41
uint64_t dparam
Parameter of the str string.
Definition tile_cmd.h:40
StringID airport_class
Name of the airport class.
Definition tile_cmd.h:46
StringID airport_name
Name of the airport.
Definition tile_cmd.h:47
uint16_t tram_speed
Speed limit of tram (bridges and track).
Definition tile_cmd.h:55
StringID roadtype
Type of road on the tile.
Definition tile_cmd.h:52
StringID tramtype
Type of tram on the tile.
Definition tile_cmd.h:54
StringID railtype
Type of rail on the tile.
Definition tile_cmd.h:50
uint16_t road_speed
Speed limit of road (bridges and track).
Definition tile_cmd.h:53
std::array< StringID, 4 > owner_type
Type of each owner.
Definition tile_cmd.h:42
std::optional< bool > town_can_upgrade
Whether the town can upgrade this house during town growth.
Definition tile_cmd.h:56
StringID airport_tile_name
Name of the airport tile.
Definition tile_cmd.h:48
StringID station_class
Class of station.
Definition tile_cmd.h:44
Window for displaying a tooltip.
Definition misc_gui.cpp:609
TooltipCloseCondition close_cond
Condition for closing the window.
Definition misc_gui.cpp:611
EncodedString text
String to display as tooltip.
Definition misc_gui.cpp:610
void OnMouseLoop() override
Called for every mouse loop run, which is at least once per (game) tick.
Definition misc_gui.cpp:665
Point OnInitialPosition(int16_t sm_width, int16_t sm_height, int window_number) override
Compute the initial position of the window.
Definition misc_gui.cpp:623
void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override
Update size and resize step of a widget in the window.
Definition misc_gui.cpp:643
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
Definition misc_gui.cpp:656
Town data structure.
Definition town.h:63
High level window description.
Definition window_gui.h:168
Data structure for an opened window.
Definition window_gui.h:274
void ReInit(int rx=0, int ry=0, bool reposition=false)
Re-initialize a window, and optionally change its size.
Definition window.cpp:979
virtual void Close(int data=0)
Hide the window and all its child windows, and mark them for a later deletion.
Definition window.cpp:1104
bool IsWidgetGloballyFocused(WidgetID widget_index) const
Check if given widget has user input focus.
Definition window_gui.h:432
void FinishInitNested(WindowNumber window_number=0)
Perform the second part of the initialization of a nested widget tree.
Definition window.cpp:1809
void InvalidateData(int data=0, bool gui_scope=true)
Mark this window's data as invalid (in need of re-computing).
Definition window.cpp:3247
Window * parent
Parent window.
Definition window_gui.h:329
void SetWidgetDirty(WidgetID widget_index) const
Invalidate a widget, i.e.
Definition window.cpp:557
virtual std::string GetWidgetString(WidgetID widget, StringID stringid) const
Get the raw string for a widget.
Definition window.cpp:505
ResizeInfo resize
Resize information.
Definition window_gui.h:315
void CreateNestedTree()
Perform the first part of the initialization of a nested widget tree.
Definition window.cpp:1799
WindowClass window_class
Window class.
Definition window_gui.h:302
virtual void OnQueryTextFinished(std::optional< std::string > str)
The query window opened from this window has closed.
Definition window_gui.h:788
void RaiseButtons(bool autoraise=false)
Raise the buttons of the window.
Definition window.cpp:531
int left
x position of left edge of the window
Definition window_gui.h:310
virtual void OnEditboxChanged(WidgetID widget)
The text in an editbox has been edited.
Definition window_gui.h:780
int top
y position of top edge of the window
Definition window_gui.h:311
Window(WindowDesc &desc)
Empty constructor, initialization has been moved to InitNested() called from the constructor of the d...
Definition window.cpp:1832
const NWID * GetWidget(WidgetID widnum) const
Get the nested widget with number widnum from the nested widget tree.
Definition window_gui.h:986
void HandleButtonClick(WidgetID widget)
Do all things to make a button look clicked and mark it to be unclicked in a few ticks.
Definition window.cpp:596
void InitNested(WindowNumber number=0)
Perform complete initialization of the Window with nested widgets, to allow use.
Definition window.cpp:1822
WindowFlags flags
Window flags.
Definition window_gui.h:301
AllWindows< false > Iterate
Iterate all windows in whatever order is easiest.
Definition window_gui.h:936
int width
width of the window (number of pixels to the right in x direction)
Definition window_gui.h:312
WindowNumber window_number
Window number within the window class.
Definition window_gui.h:303
@ EnableMove
enable the 'Move' button
Definition textbuf_gui.h:22
@ AcceptUnchanged
return success even when the text didn't change
Definition textbuf_gui.h:19
@ EnableDefault
enable the 'Default' button ("\0" is returned)
Definition textbuf_gui.h:20
@ LengthIsInChars
the length of the string is counted in characters
Definition textbuf_gui.h:21
void QueryCallbackProc(Window *, bool)
Callback procedure for the ShowQuery method.
Definition textbuf_gui.h:28
Functions related to text effects.
@ TE_STATIC
Keep the text effect static.
Definition texteff.hpp:23
@ TE_RISING
Make the text effect slowly go upwards.
Definition texteff.hpp:22
void AddAcceptedCargo(TileIndex tile, CargoArray &acceptance, CargoTypes &always_accepted)
Obtain cargo acceptance of a tile.
Definition tile_cmd.h:185
int GetTileZ(TileIndex tile)
Get bottom height of the tile.
Definition tile_map.cpp:115
StrongType::Typedef< uint32_t, struct TileIndexTag, StrongType::Compare, StrongType::Integer, StrongType::Compatible< int32_t >, StrongType::Compatible< int64_t > > TileIndex
The index/ID of a Tile.
Definition tile_type.h:92
constexpr TileIndex INVALID_TILE
The very nice invalid tile marker.
Definition tile_type.h:100
Functions related to tile highlights.
@ HT_RECT
rectangle (stations, depots, ...)
Definition of Interval and OneShot timers.
Definition of the Window system.
Base of the town class.
Town * ClosestTownFromTile(TileIndex tile, uint threshold)
Return the town closest (in distance or ownership) to a given tile, within a given threshold.
@ VEH_ROAD
Road vehicle type.
@ VEH_TRAIN
Train vehicle type.
bool ScrollMainWindowToTile(TileIndex tile, bool instant)
Scrolls the viewport of the main window to a given location.
void SetViewportStationRect(const Station *st, bool sel)
Select or deselect station for rectangle area highlight.
Viewport * IsPtInWindowViewport(const Window *w, int x, int y)
Is a xy position inside the viewport of the window?
Definition viewport.cpp:408
void SetViewportWaypointRect(const Waypoint *wp, bool sel)
Select or deselect waypoint for rectangle area highlight.
Functions related to (drawing on) viewports.
Handles dividing the water in the map into regions to assist pathfinding.
Base of waypoints.
void CcMoveWaypointName(Commands, const CommandCost &result, StationID waypoint_id)
Callback function that is called after a name is moved.
Command definitions related to waypoints.
void DrawFrameRect(int left, int top, int right, int bottom, Colours colour, FrameFlags flags)
Draw frame rectangle.
Definition widget.cpp:289
static RectPadding ScaleGUITrad(const RectPadding &r)
Scale a RectPadding to GUI zoom level.
Definition widget.cpp:49
@ WWT_PUSHTXTBTN
Normal push-button (no toggle button) with text caption.
@ WWT_PUSHIMGBTN
Normal push-button (no toggle button) with image caption.
@ WWT_LABEL
Centered label.
Definition widget_type.h:48
@ WWT_EDITBOX
a textbox for typing
Definition widget_type.h:62
@ NWID_HORIZONTAL
Horizontal container.
Definition widget_type.h:66
@ WWT_TEXTBTN
(Toggle) Button with text
Definition widget_type.h:44
@ WWT_PANEL
Simple depressed panel.
Definition widget_type.h:39
@ WWT_CAPTION
Window caption (window title between closebox and stickybox).
Definition widget_type.h:52
@ NWID_VERTICAL
Vertical container.
Definition widget_type.h:68
@ WWT_CLOSEBOX
Close box (at top-left of a window).
Definition widget_type.h:60
@ WWT_FRAME
Frame.
Definition widget_type.h:51
@ WWT_EMPTY
Empty widget, place holder to reserve space in widget tree.
Definition widget_type.h:37
@ WWT_TEXT
Pure simple text.
Definition widget_type.h:49
@ WWT_DEBUGBOX
NewGRF debug box (at top-right of a window, between WWT_CAPTION and WWT_SHADEBOX).
Definition widget_type.h:54
@ NWID_SELECTION
Stacked widgets, only one visible at a time (eg in a panel with tabs).
Definition widget_type.h:71
@ SZSP_NONE
Display plane with zero size in both directions (none filling and resizing).
@ EqualSize
Containers should keep all their (resizing) children equally large.
void CloseWindowById(WindowClass cls, WindowNumber number, bool force, int data)
Close a window by its class and window number (if it is open).
Definition window.cpp:1196
Window * GetMainWindow()
Get the main window, i.e.
Definition window.cpp:1182
void SetFocusedWindow(Window *w)
Set the window that has the focus.
Definition window.cpp:422
Window * FindWindowByClass(WindowClass cls)
Find any window by its class.
Definition window.cpp:1168
Window * FindWindowFromPt(int x, int y)
Do a search for a window at specific coordinates.
Definition window.cpp:1844
int GetMainViewTop()
Return the top of the main view available for general use.
Definition window.cpp:2139
void CloseWindowByClass(WindowClass cls, int data)
Close all windows of a given class.
Definition window.cpp:1209
int GetMainViewBottom()
Return the bottom of the main view available for general use.
Definition window.cpp:2150
bool _mouse_hovering
The mouse is hovering over the same point.
Definition window.cpp:94
Window functions not directly related to making/drawing windows.
@ NoClose
This window can't be interactively closed.
Definition window_gui.h:156
@ NoFocus
This window won't get focus/make any other window lose focus when click.
Definition window_gui.h:155
@ Modal
The window is a modal child of some other window, meaning the parent is 'inactive'.
Definition window_gui.h:154
@ Lowered
If set the frame is lowered and the background colour brighter (ie. buttons when pressed).
Definition window_gui.h:27
@ WhiteBorder
Window white border counter bit mask.
Definition window_gui.h:235
@ WDP_CENTER
Center the window.
Definition window_gui.h:145
@ WDP_AUTO
Find a place automatically.
Definition window_gui.h:144
@ WDP_MANUAL
Manually align the window (so no automatic location finding).
Definition window_gui.h:143
int WidgetID
Widget ID.
Definition window_type.h:20
@ WN_QUERY_STRING
Query string.
Definition window_type.h:34
@ WN_GAME_OPTIONS_ABOUT
About window.
Definition window_type.h:29
@ WN_CONFIRM_POPUP_QUERY
Query popup confirm.
Definition window_type.h:37
EventState
State of handling an event.
@ ES_HANDLED
The passed event is handled.
@ ES_NOT_HANDLED
The passed event is not handled.
static constexpr WidgetID INVALID_WIDGET
An invalid widget index.
Definition window_type.h:23
@ WC_LAND_INFO
Land info window; Window numbers:
@ WC_OSK
On Screen Keyboard; Window numbers:
@ WC_WAYPOINT_VIEW
Waypoint view; Window numbers:
@ WC_NONE
No window, redirects to WC_MAIN_WINDOW.
Definition window_type.h:50
@ WC_STATION_VIEW
Station view; Window numbers:
@ WC_CONFIRM_POPUP_QUERY
Popup with confirm question; Window numbers:
@ WC_GAME_OPTIONS
Game options window; Window numbers:
@ WC_QUERY_STRING
Query string window; Window numbers:
@ WC_TOOLTIPS
Tooltip window; Window numbers:
Functions related to zooming.