OpenTTD Source 20260218-master-g2123fca5ea
picker_gui.h
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#ifndef PICKER_GUI_H
11#define PICKER_GUI_H
12
13#include "newgrf_badge.h"
14#include "newgrf_badge_gui.h"
15#include "querystring_gui.h"
16#include "sortlist_type.h"
17#include "stringfilter_type.h"
18#include "strings_type.h"
19#include "timer/timer.h"
21#include "timer/timer_window.h"
22#include "window_gui.h"
23#include "window_type.h"
24
25struct PickerItem {
26 uint32_t grfid;
27 uint16_t local_id;
28 int class_index;
29 int index;
30
31 inline auto operator<=>(const PickerItem &other) const
32 {
33 if (auto cmp = this->grfid <=> other.grfid; cmp != 0) return cmp;
34 return this->local_id <=> other.local_id;
35 }
36};
37
39class PickerCallbacks {
40public:
41 explicit PickerCallbacks(const std::string &ini_group);
42 virtual ~PickerCallbacks();
43
45 virtual void Close([[maybe_unused]] int data) { }
46
51 virtual GrfSpecFeature GetFeature() const = 0;
56 virtual bool IsActive() const = 0;
61 virtual bool HasClassChoice() const = 0;
62
63 /* Class callbacks */
68 virtual StringID GetClassTooltip() const = 0;
74 virtual int GetClassCount() const = 0;
79 virtual int GetSelectedClass() const = 0;
84 virtual void SetSelectedClass(int id) const = 0;
90 virtual StringID GetClassName(int id) const = 0;
91
92 /* Type callbacks */
97 virtual StringID GetTypeTooltip() const = 0;
104 virtual int GetTypeCount(int cls_id) const = 0;
109 virtual int GetSelectedType() const = 0;
114 virtual void SetSelectedType(int id) const = 0;
121 virtual PickerItem GetPickerItem(int cls_id, int id) const = 0;
128 virtual StringID GetTypeName(int cls_id, int id) const = 0;
135 virtual std::span<const BadgeID> GetTypeBadges(int cls_id, int id) const = 0;
142 virtual bool IsTypeAvailable(int cls_id, int id) const = 0;
150 virtual void DrawType(int x, int y, int cls_id, int id) const = 0;
151
152 /* Collection Callbacks */
157 virtual StringID GetCollectionTooltip() const = 0;
158
163 virtual void FillUsedItems(std::set<PickerItem> &items) = 0;
169 virtual std::map<std::string, std::set<PickerItem>> UpdateSavedItems(const std::map<std::string, std::set<PickerItem>> &src) = 0;
175 inline std::set<std::string> InitializeInactiveCollections(const std::map<std::string, std::set<PickerItem>> collections)
176 {
177 std::set<std::string> inactive;
178
179 for (const auto &collection : collections) {
180 if ((collection.second.size() == 1 && collection.second.contains({})) || collection.first == "") continue;
181 for (const PickerItem &item : collection.second) {
182 if (item.class_index == -1 || item.index == -1) {
183 inactive.emplace(collection.first);
184 break;
185 }
186 if (GetTypeName(item.class_index, item.index) == INVALID_STRING_ID) {
187 inactive.emplace(collection.first);
188 break;
189 }
190 }
191 }
192 return inactive;
193 }
194
197
198 Listing type_last_sorting = { false, 0 };
200
202
203 const std::string ini_group;
204 uint8_t mode = 0;
205 bool rename_collection = false;
206 std::string sel_collection;
207 std::string edit_collection;
208 std::set<std::string> rm_collections;
209
211
212 std::set<PickerItem> used;
213 std::map<std::string, std::set<PickerItem>> saved;
214};
215
217template <typename T>
218class PickerCallbacksNewGRFClass : public PickerCallbacks {
219public:
220 explicit PickerCallbacksNewGRFClass(const std::string &ini_group) : PickerCallbacks(ini_group) {}
221
222 inline typename T::index_type GetClassIndex(int cls_id) const { return static_cast<typename T::index_type>(cls_id); }
223 inline const T *GetClass(int cls_id) const { return T::Get(this->GetClassIndex(cls_id)); }
224 inline const typename T::spec_type *GetSpec(int cls_id, int id) const { return this->GetClass(cls_id)->GetSpec(id); }
225
226 bool HasClassChoice() const override { return T::GetUIClassCount() > 1; }
227
228 int GetClassCount() const override { return T::GetClassCount(); }
229 int GetTypeCount(int cls_id) const override { return this->GetClass(cls_id)->GetSpecCount(); }
230
231 PickerItem GetPickerItem(const typename T::spec_type *spec, int cls_id = -1, int id = -1) const
232 {
233 if (spec == nullptr) return {0, 0, cls_id, id};
234 return {spec->grf_prop.grfid, spec->grf_prop.local_id, spec->class_index, spec->index};
235 }
236
237 PickerItem GetPickerItem(int cls_id, int id) const override
238 {
239 return GetPickerItem(GetClass(cls_id)->GetSpec(id), cls_id, id);
240 }
241
242 std::map<std::string, std::set<PickerItem>> UpdateSavedItems(const std::map<std::string, std::set<PickerItem>> &src) override
243 {
244 if (src.empty()) return {};
245
246 std::map<std::string, std::set<PickerItem>> dst;
247 for (auto it = src.begin(); it != src.end(); it++) {
248 if (it->second.empty() || (it->second.size() == 1 && it->second.contains({}))) {
249 dst[it->first];
250 continue;
251 }
252
253 for (const auto &item : it->second) {
254 const auto *spec = T::GetByGrf(item.grfid, item.local_id);
255 if (spec == nullptr) {
256 dst[it->first].emplace(item.grfid, item.local_id, -1, -1);
257 } else {
258 dst[it->first].emplace(GetPickerItem(spec));
259 }
260 }
261 }
262 return dst;
263 }
264};
265
268 std::optional<BadgeTextFilter> btf;
269 std::optional<BadgeDropdownFilter> bdf;
270};
271
275
276class PickerWindow : public PickerWindowBase {
277public:
278 enum PickerFilterModes : uint8_t {
282 };
283
293 using PickerInvalidations = EnumBitSet<PickerInvalidation, uint8_t>;
294
295 static constexpr PickerInvalidations PICKER_INVALIDATION_ALL{PickerInvalidation::Class, PickerInvalidation::Type, PickerInvalidation::Position, PickerInvalidation::Validate};
296
297 static constexpr int PREVIEW_WIDTH = 64;
298 static constexpr int PREVIEW_HEIGHT = 48;
299 static constexpr int PREVIEW_LEFT = 31;
300 static constexpr int PREVIEW_BOTTOM = 31;
301
302 static constexpr int STEP_PREVIEW_HEIGHT = 16;
303 static constexpr int MAX_PREVIEW_HEIGHT = PREVIEW_HEIGHT * 3;
304
305 static constexpr uint EDITBOX_MAX_SIZE = 16;
306
307 bool has_class_picker = false;
308 bool has_type_picker = false;
311 std::set<std::string> inactive;
312
313 PickerWindow(WindowDesc &desc, Window *parent, int window_number, PickerCallbacks &callbacks);
314 void OnInit() override;
315 void Close(int data = 0) override;
316 void UpdateWidgetSize(WidgetID widget, Dimension &size, const Dimension &padding, Dimension &fill, Dimension &resize) override;
317 std::string GetWidgetString(WidgetID widget, StringID stringid) const override;
318 DropDownList BuildCollectionDropDownList();
319 void DrawWidget(const Rect &r, WidgetID widget) const override;
320 void OnDropdownSelect(WidgetID widget, int index, int click_result) override;
321 void OnResize() override;
322 void static DeletePickerCollectionCallback(Window *win, bool confirmed);
323 void OnClick(Point pt, WidgetID widget, int click_count) override;
324 void OnQueryTextFinished(std::optional<std::string> str) override;
325 void OnInvalidateData(int data = 0, bool gui_scope = true) override;
326 EventState OnHotkey(int hotkey) override;
327 void OnEditboxChanged(WidgetID wid) override;
328
333
334 void InvalidateData(PickerInvalidations data) { this->Window::InvalidateData(data.base()); }
335
336protected:
337 void ConstructWindow();
338
339 PickerCallbacks &callbacks;
340
341private:
343 PickerFilterData class_string_filter;
345
347 void EnsureSelectedClassIsValid();
348 void EnsureSelectedClassIsVisible();
349
351 PickerFilterData type_string_filter;
353
354 void RefreshUsedTypeList();
355 void BuildPickerTypeList();
356 void EnsureSelectedTypeIsValid();
357 void EnsureSelectedTypeIsVisible();
358
360
362
363 GUIBadgeClasses badge_classes;
364 std::pair<WidgetID, WidgetID> badge_filters{};
365 BadgeFilterChoices badge_filter_choices{};
366
367 const IntervalTimer<TimerGameCalendar> yearly_interval = {{TimerGameCalendar::Trigger::Year, TimerGameCalendar::Priority::None}, [this](auto) {
368 this->SetDirty();
369 }};
370
371 const IntervalTimer<TimerWindow> refresh_interval = {std::chrono::seconds(3), [this](auto) {
372 RefreshUsedTypeList();
373 }};
374};
375
376class NWidgetBase;
377std::unique_ptr<NWidgetBase> MakePickerClassWidgets();
378std::unique_ptr<NWidgetBase> MakePickerTypeWidgets();
379
380#endif /* PICKER_GUI_H */
Enum-as-bit-set wrapper.
List template of 'things' T to sort in a GUI.
Baseclass for nested widgets.
std::map< std::string, std::set< PickerItem > > UpdateSavedItems(const std::map< std::string, std::set< PickerItem > > &src) override
Update link between grfid/localidx and class_index/index in saved items.
Definition picker_gui.h:242
PickerItem GetPickerItem(int cls_id, int id) const override
Get data about an item.
Definition picker_gui.h:237
int GetClassCount() const override
Get the number of classes.
Definition picker_gui.h:228
int GetTypeCount(int cls_id) const override
Get the number of types in a class.
Definition picker_gui.h:229
bool HasClassChoice() const override
Are there multiple classes to chose from?
Definition picker_gui.h:226
Class for PickerClassWindow to collect information and retain state.
Definition picker_gui.h:39
std::set< std::string > InitializeInactiveCollections(const std::map< std::string, std::set< PickerItem > > collections)
Initialize the list of active collections for sorting purposes.
Definition picker_gui.h:175
virtual int GetSelectedClass() const =0
Get the index of the selected class.
std::string edit_collection
Collection to rename or delete.
Definition picker_gui.h:207
Filtering type_last_filtering
Default filtering of PickerTypeList.
Definition picker_gui.h:199
virtual PickerItem GetPickerItem(int cls_id, int id) const =0
Get data about an item.
const std::string ini_group
Ini Group for saving favourites.
Definition picker_gui.h:203
bool rename_collection
Are we renaming a collection?
Definition picker_gui.h:205
Listing collection_last_sorting
Default sorting of PickerCollectionList.
Definition picker_gui.h:201
virtual void SetSelectedClass(int id) const =0
Set the selected class.
std::string sel_collection
Currently selected collection of saved items.
Definition picker_gui.h:206
virtual bool IsActive() const =0
Should picker class/type selection be enabled?
virtual StringID GetTypeName(int cls_id, int id) const =0
Get the item name of a type.
virtual bool IsTypeAvailable(int cls_id, int id) const =0
Test if an item is currently buildable.
virtual void FillUsedItems(std::set< PickerItem > &items)=0
Fill a set with all items that are used by the current player.
virtual StringID GetTypeTooltip() const =0
Get the tooltip string for the type grid.
virtual bool HasClassChoice() const =0
Are there multiple classes to chose from?
Listing type_last_sorting
Default sorting of PickerTypeList.
Definition picker_gui.h:198
virtual StringID GetCollectionTooltip() const =0
Get the tooltip string for the collection list.
std::set< PickerItem > used
Set of items used in the current game by the current company.
Definition picker_gui.h:212
virtual StringID GetClassTooltip() const =0
Get the tooltip string for the class list.
virtual int GetTypeCount(int cls_id) const =0
Get the number of types in a class.
virtual int GetClassCount() const =0
Get the number of classes.
virtual GrfSpecFeature GetFeature() const =0
NewGRF feature this picker is for.
virtual void SetSelectedType(int id) const =0
Set the selected type.
virtual StringID GetClassName(int id) const =0
Get the name of a class.
int preview_height
Previously adjusted height.
Definition picker_gui.h:210
virtual void DrawType(int x, int y, int cls_id, int id) const =0
Draw preview image of an item.
virtual std::span< const BadgeID > GetTypeBadges(int cls_id, int id) const =0
Get the item's badges of a type.
virtual std::map< std::string, std::set< PickerItem > > UpdateSavedItems(const std::map< std::string, std::set< PickerItem > > &src)=0
Update link between grfid/localidx and class_index/index in saved items.
Filtering class_last_filtering
Default filtering of PickerClassList.
Definition picker_gui.h:196
virtual void Close(int data)
Hide the window and all its child windows, and mark them for a later deletion.
Definition picker_gui.h:45
std::set< std::string > rm_collections
Set of removed or renamed collections for updating ini file.
Definition picker_gui.h:208
std::map< std::string, std::set< PickerItem > > saved
Set of saved collections of items.
Definition picker_gui.h:213
virtual int GetSelectedType() const =0
Get the selected type.
uint8_t mode
Bitmask of PickerFilterModes.
Definition picker_gui.h:204
Listing class_last_sorting
Default sorting of PickerClassList.
Definition picker_gui.h:195
static constexpr int MAX_PREVIEW_HEIGHT
Maximum height of each preview button.
Definition picker_gui.h:303
static constexpr int PREVIEW_WIDTH
Width of each preview button.
Definition picker_gui.h:297
void Close(int data=0) override
Hide the window and all its child windows, and mark them for a later deletion.
int preview_height
Height of preview images.
Definition picker_gui.h:310
bool has_class_picker
Set if this window has a class picker 'component'.
Definition picker_gui.h:307
static constexpr int PREVIEW_HEIGHT
Height of each preview button.
Definition picker_gui.h:298
static constexpr uint EDITBOX_MAX_SIZE
The maximum number of characters for the filter edit box.
Definition picker_gui.h:305
void DrawWidget(const Rect &r, WidgetID widget) const override
Draw the contents of a nested widget.
PickerInvalidation
The things of a picker that can be invalidated.
Definition picker_gui.h:285
@ Position
Update scroll positions.
Definition picker_gui.h:289
@ Class
Refresh the class list.
Definition picker_gui.h:286
@ Type
Refresh the type list.
Definition picker_gui.h:287
@ Validate
Validate selected item.
Definition picker_gui.h:290
@ Collection
Refresh the collection list.
Definition picker_gui.h:288
@ Filter
Update filter state.
Definition picker_gui.h:291
bool has_collection_picker
Set if this window has a collection picker 'component'.
Definition picker_gui.h:309
static constexpr int PREVIEW_LEFT
Offset from left edge to draw preview.
Definition picker_gui.h:299
PickerTypeList types
List of types.
Definition picker_gui.h:350
std::set< std::string > inactive
Set of collections with inactive items.
Definition picker_gui.h:311
QueryString class_editbox
Filter editbox.
Definition picker_gui.h:344
bool has_type_picker
Set if this window has a type picker 'component'.
Definition picker_gui.h:308
void BuildPickerClassList()
Builds the filter list of classes.
@ PFM_USED
Show used types.
Definition picker_gui.h:280
@ PFM_ALL
Show all classes.
Definition picker_gui.h:279
@ PFM_SAVED
Show saved types.
Definition picker_gui.h:281
void BuildPickerCollectionList()
Builds the filter list of collections.
void OnEditboxChanged(WidgetID wid) override
The text in an editbox has been edited.
void OnQueryTextFinished(std::optional< std::string > str) override
The query window opened from this window has closed.
void BuildPickerTypeList()
Builds the filter list of types.
void OnDropdownSelect(WidgetID widget, int index, int click_result) override
A dropdown option associated to this window has been selected.
void OnResize() override
Called after the window got resized.
static constexpr int PREVIEW_BOTTOM
Offset from bottom edge to draw preview.
Definition picker_gui.h:300
PickerClassWindowHotkeys
Enum referring to the Hotkeys in the picker window.
Definition picker_gui.h:330
@ PCWHK_FOCUS_FILTER_BOX
Focus the edit box for editing the filter string.
Definition picker_gui.h:331
std::string GetWidgetString(WidgetID widget, StringID stringid) const override
Get the raw string for a widget.
PickerCollectionList collections
List of collections.
Definition picker_gui.h:359
void OnClick(Point pt, WidgetID widget, int click_count) override
A click with the left mouse button has been made on the window.
void OnInvalidateData(int data=0, bool gui_scope=true) override
Some data on this window has become invalid.
QueryString type_editbox
Filter editbox.
Definition picker_gui.h:352
EventState OnHotkey(int hotkey) override
A hotkey has been pressed.
void OnInit() override
Notification that the nested widget tree gets initialized.
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.
PickerClassList classes
List of classes.
Definition picker_gui.h:342
static constexpr int STEP_PREVIEW_HEIGHT
Step for decreasing or increase preview button height.
Definition picker_gui.h:302
std::vector< std::unique_ptr< const DropDownListItem > > DropDownList
A drop down list is a collection of drop down list items.
void SetDirty() const
Mark entire window as dirty (in need of re-paint).
Definition window.cpp:967
#define Rect
Macro that prevents name conflicts between included headers.
#define Point
Macro that prevents name conflicts between included headers.
GrfSpecFeature
Definition newgrf.h:71
Functions related to NewGRF badges.
GUI functions related to NewGRF badges.
GUIList< PickerItem, std::nullptr_t, PickerFilterData & > PickerTypeList
GUIList holding classes/types to display.
Definition picker_gui.h:273
GUIList< int, std::nullptr_t, PickerFilterData & > PickerClassList
GUIList holding classes to display.
Definition picker_gui.h:272
std::unique_ptr< NWidgetBase > MakePickerClassWidgets()
Create nested widgets for the class picker widgets.
std::unique_ptr< NWidgetBase > MakePickerTypeWidgets()
Create nested widgets for the type picker widgets.
GUIList< std::string, std::nullptr_t, PickerFilterData & > PickerCollectionList
GUIList holding collections to display.
Definition picker_gui.h:274
Base for the GUIs that have an edit box in them.
Base types for having sorted lists in GUIs.
Searching and filtering using a stringterm.
Types related to strings.
uint32_t StringID
Numeric value that represents a string, independent of the selected language.
static const StringID INVALID_STRING_ID
Constant representing an invalid string (16bit in case it is used in savegames).
Dimensions (a width and height) of a rectangle in 2D.
Data structure describing what to show in the list (filter criteria).
Data structure describing how to show the list (what sort direction and criteria).
const PickerCallbacks * callbacks
Callbacks for filter functions to access to callbacks.
Definition picker_gui.h:267
Data stored about a string that can be modified in the GUI.
StringFilter(const bool *case_sensitive=nullptr, bool locale_aware=true)
Constructor for filter.
High level window description.
Definition window_gui.h:168
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
ResizeInfo resize
Resize information.
Definition window_gui.h:315
Window(WindowDesc &desc)
Empty constructor, initialization has been moved to InitNested() called from the constructor of the d...
Definition window.cpp:1832
WindowNumber window_number
Window number within the window class.
Definition window_gui.h:303
Definition of Interval and OneShot timers.
Definition of the game-calendar-timer.
Definition of the Window system.
Functions, definitions and such used only by the GUI.
Types related to windows.
int WidgetID
Widget ID.
Definition window_type.h:20
EventState
State of handling an event.