Plutonium framework API 1.0.0
Easy-to-use, SDL2-based UI framework for Nintendo Switch homebrew
Loading...
Searching...
No Matches
elm_Menu.hpp
Go to the documentation of this file.
1/**
2 * Plutonium library
3 * @file elm_Menu.hpp
4 * @brief Contains classes for creating UI menus.
5 * @author XorTroll
6 * @copyright XorTroll
7 */
8
9#pragma once
10#include <pu/ui/elm/elm_Element.hpp>
11#include <chrono>
12#include <functional>
13
14namespace pu::ui::elm {
15
16 /**
17 * @brief Represents a menu item.
18 */
19 class MenuItem {
20 public:
21 /**
22 * @brief Represents a function called when the specific key is pressed over a the MenuItem.
23 */
24 using OnKeyCallback = std::function<void()>;
25
26 // Self-explanatory constants
27
28 static constexpr Color DefaultColor = { 10, 10, 10, 0xFF };
29
30 private:
31 std::string name;
32 Color clr;
33 sdl2::TextureHandle::Ref icon;
34 std::vector<OnKeyCallback> on_key_cbs;
35 std::vector<u64> on_key_cb_keys;
36
37 public:
38 /**
39 * @brief Creates a new instance of a MenuItem.
40 * @param name Name of the MenuItem.
41 */
42 MenuItem(const std::string &name) : name(name), clr(DefaultColor) {}
44
45 /**
46 * @brief Gets the name of the MenuItem.
47 * @return Name of the MenuItem.
48 */
49 inline std::string GetName() {
50 return this->name;
51 }
52
53 /**
54 * @brief Sets the name of the MenuItem.
55 * @param name New name of the MenuItem.
56 */
57 inline void SetName(const std::string &name) {
58 this->name = name;
59 }
60
61 PU_CLASS_POD_GETSET(Color, clr, Color)
62
63 /**
64 * @brief Adds a new OnKeyCallback to the MenuItem.
65 * @param on_key_cb Function to call when the key is pressed.
66 * @param key Key to call the function on. Default is A.
67 */
68 void AddOnKey(OnKeyCallback on_key_cb, const u64 key = HidNpadButton_A);
69
70 /**
71 * @brief Gets the amount of OnKeyCallbacks in the MenuItem.
72 * @return Amount of OnKeyCallbacks in the MenuItem.
73 */
74 inline u32 GetOnKeyCallbackCount() {
75 return this->on_key_cbs.size();
76 }
77
78 /**
79 * @brief Gets the OnKeyCallback at the specified index.
80 * @param idx Index of the OnKeyCallback.
81 * @return OnKeyCallback at the specified index.
82 */
83 inline OnKeyCallback GetOnKeyCallback(const u32 idx) {
84 if(idx < this->on_key_cbs.size()) {
85 return this->on_key_cbs.at(idx);
86 }
87 else {
88 return {};
89 }
90 }
91
92 /**
93 * @brief Gets the key of the OnKeyCallback at the specified index.
94 * @param idx Index of the OnKeyCallback.
95 * @return Key of the OnKeyCallback at the specified index.
96 */
97 inline u64 GetOnKeyCallbackKey(const u32 idx) {
98 if(idx < this->on_key_cb_keys.size()) {
99 return this->on_key_cb_keys.at(idx);
100 }
101 else {
102 return {};
103 }
104 }
105
106 /**
107 * @brief Gets the icon of the MenuItem.
108 * @return Icon of the MenuItem.
109 */
111 return this->icon;
112 }
113
114 /**
115 * @brief Sets the icon of the MenuItem.
116 * @param icon Icon to set.
117 */
118 void SetIcon(sdl2::TextureHandle::Ref icon);
119
120 /**
121 * @brief Gets whether the MenuItem has an icon.
122 * @return Whether the MenuItem has an icon.
123 */
124 inline bool HasIcon() {
125 return this->icon != nullptr;
126 }
127 };
128
129 /**
130 * @brief Represents a menu.
131 */
132 class Menu : public Element {
133 public:
134 // Self-explanatory constants
135
136 static constexpr Color DefaultScrollbarColor = { 110, 110, 110, 0xFF };
137
138 static constexpr u8 DefaultItemAlphaIncrementSteps = 15;
139
140 static constexpr float DefaultIconItemSizesFactor = 0.8f;
141
142 static constexpr u32 DefaultIconMargin = 37;
143 static constexpr u32 DefaultTextMargin = 37;
144
145 static constexpr u8 DefaultLightScrollbarColorFactor = 30;
146
147 static constexpr u32 DefaultScrollbarWidth = 30;
148
149 static constexpr u32 DefaultShadowHeight = 7;
150 static constexpr u8 DefaultShadowBaseAlpha = 160;
151
152 static constexpr s64 DefaultMoveWaitTimeMs = 150;
153
154 /**
155 * @brief Represents a function called when the selection of the Menu changes.
156 */
157 using OnSelectionChangedCallback = std::function<void()>;
158
159 private:
160 enum class MoveStatus : u8 {
161 None = 0,
162 WaitingUp = 1,
163 WaitingDown = 2
164 };
165
166 i32 x;
167 i32 y;
168 i32 w;
169 i32 items_h;
170 u32 items_to_show;
171 u32 selected_item_idx;
172 i32 selected_item_alpha;
173 SigmoidIncrementer<i32> selected_item_alpha_incr;
174 i32 prev_selected_item_idx;
175 i32 prev_selected_item_alpha;
176 SigmoidIncrementer<i32> prev_selected_item_alpha_incr;
177 u32 advanced_item_count;
178 Color scrollbar_clr;
179 Color items_clr;
180 Color items_focus_clr;
181 bool cooldown_enabled;
182 bool item_touched;
183 MoveStatus move_status;
184 std::chrono::time_point<std::chrono::steady_clock> move_start_time;
185 OnSelectionChangedCallback on_selection_changed_cb;
186 std::vector<MenuItem::Ref> items;
187 std::string font_name;
188 std::vector<sdl2::Texture> loaded_name_texs;
189 u8 item_alpha_incr_steps;
190 float icon_item_sizes_factor;
191 u32 icon_margin;
192 u32 text_margin;
193 u8 light_scrollbar_color_factor;
194 u32 scrollbar_width;
195 u32 shadow_height;
196 u8 shadow_base_alpha;
197 s64 move_wait_time_ms;
198
199 void ReloadItemRenders();
200 void MoveUp();
201 void MoveDown();
202
203 inline Color MakeItemsFocusColor(const u8 alpha) {
204 return this->items_focus_clr.WithAlpha(alpha);
205 }
206
207 inline constexpr Color MakeLighterScrollbarColor() {
208 i32 base_r = this->scrollbar_clr.r - this->light_scrollbar_color_factor;
209 if(base_r < 0) {
210 base_r = 0;
211 }
212 i32 base_g = this->scrollbar_clr.g - this->light_scrollbar_color_factor;
213 if(base_g < 0) {
214 base_g = 0;
215 }
216 i32 base_b = this->scrollbar_clr.b - this->light_scrollbar_color_factor;
217 if(base_b < 0) {
218 base_b = 0;
219 }
220
221 return { static_cast<u8>(base_r), static_cast<u8>(base_g), static_cast<u8>(base_b), this->scrollbar_clr.a };
222 }
223
224 inline void HandleOnSelectionChanged() {
225 if(this->on_selection_changed_cb) {
226 (this->on_selection_changed_cb)();
227 }
228 }
229
230 inline void RunSelectedItemCallback(const u64 keys) {
231 auto item = this->items.at(this->selected_item_idx);
232 const auto cb_count = item->GetOnKeyCallbackCount();
233 for(u32 i = 0; i < cb_count; i++) {
234 if(keys & item->GetOnKeyCallbackKey(i)) {
235 if(!this->cooldown_enabled) {
236 auto cb = item->GetOnKeyCallback(i);
237 if(cb) {
238 cb();
239 }
240 }
241 }
242 }
243 this->cooldown_enabled = false;
244 }
245
246 inline u32 GetItemCount() {
247 auto item_count = this->items_to_show;
248 if(item_count > this->items.size()) {
249 item_count = this->items.size();
250 }
251 if((item_count + this->advanced_item_count) > this->items.size()) {
252 item_count = this->items.size() - this->advanced_item_count;
253 }
254 return item_count;
255 }
256
257 public:
258 /**
259 * @brief Creates a new instance of a Menu.
260 * @param x X position of the Menu.
261 * @param y Y position of the Menu.
262 * @param width Width of the Menu.
263 * @param items_clr Color of the items in the Menu.
264 * @param items_focus_clr Color of the focused item in the Menu.
265 * @param items_height Height of each shown item in the Menu.
266 * @param items_to_show Number of items to show in the Menu.
267 */
268 Menu(const i32 x, const i32 y, const i32 width, const Color items_clr, const Color items_focus_clr, const i32 items_height, const u32 items_to_show);
270
271 inline i32 GetX() override {
272 return this->x;
273 }
274
275 /**
276 * @brief Sets the X position of the Menu.
277 * @param x New X position.
278 */
279 inline void SetX(const i32 x) {
280 this->x = x;
281 }
282
283 inline i32 GetY() override {
284 return this->y;
285 }
286
287 /**
288 * @brief Sets the Y position of the Menu.
289 * @param y New Y position.
290 */
291 inline void SetY(const i32 y) {
292 this->y = y;
293 }
294
295 inline i32 GetWidth() override {
296 return this->w;
297 }
298
299 /**
300 * @brief Sets the width of the Menu.
301 * @param width New width.
302 */
303 inline void SetWidth(const i32 width) {
304 this->w = width;
305 }
306
307 inline i32 GetHeight() override {
308 return this->items_h * this->items_to_show;
309 }
310
311 PU_CLASS_POD_GETSET(ItemsHeight, items_h, i32)
312 PU_CLASS_POD_GETSET(NumberOfItemsToShow, items_to_show, i32)
313 PU_CLASS_POD_GETSET(ItemsFocusColor, items_focus_clr, Color)
314 PU_CLASS_POD_GETSET(ItemsColor, items_clr, Color)
315 PU_CLASS_POD_GETSET(ScrollbarColor, scrollbar_clr, Color)
316 PU_CLASS_POD_GETSET(ItemAlphaIncrementSteps, item_alpha_incr_steps, u8)
317 PU_CLASS_POD_GETSET(IconItemSizesFactor, icon_item_sizes_factor, float)
318 PU_CLASS_POD_GETSET(IconMargin, icon_margin, u32)
319 PU_CLASS_POD_GETSET(TextMargin, text_margin, u32)
320 PU_CLASS_POD_GETSET(LightScrollbarColorFactor, light_scrollbar_color_factor, u8)
321 PU_CLASS_POD_GETSET(ScrollbarWidth, scrollbar_width, u32)
322 PU_CLASS_POD_GETSET(ShadowHeight, shadow_height, u32)
323 PU_CLASS_POD_GETSET(ShadowBaseAlpha, shadow_base_alpha, u8)
324 PU_CLASS_POD_GETSET(MoveWaitTimeMs, move_wait_time_ms, s64)
325
326 /**
327 * @brief Sets the font name to use for the Menu.
328 * @param font_name Name of the font to use.
329 */
330 inline void SetOnSelectionChanged(OnSelectionChangedCallback on_selection_changed_cb) {
331 this->on_selection_changed_cb = on_selection_changed_cb;
332 }
333
334 /**
335 * @brief Adds a new item to the Menu.
336 * @param item Item to add.
337 */
338 inline void AddItem(MenuItem::Ref &item) {
339 this->items.push_back(item);
340 }
341
342 /**
343 * @brief Clears all items from the Menu.
344 */
346
347 /**
348 * @brief Forces the Menu to reload all items.
349 */
350 inline void ForceReloadItems() {
351 this->ReloadItemRenders();
352 }
353
354 PU_CLASS_POD_SET(CooldownEnabled, cooldown_enabled, bool)
355
356 /**
357 * @brief Gets the selected item of the Menu.
358 * @return Selected item of the Menu.
359 */
360 inline MenuItem::Ref &GetSelectedItem() {
361 return this->items.at(this->selected_item_idx);
362 }
363
364 /**
365 * @brief Gets the items of the Menu.
366 * @return Items of the Menu.
367 */
369 return this->items;
370 }
371
372 PU_CLASS_POD_GET(SelectedIndex, selected_item_idx, i32)
373
374 /**
375 * @brief Sets the selected index of the Menu.
376 * @param idx Index to set.
377 */
378 void SetSelectedIndex(const u32 idx);
379
380 void OnRender(render::Renderer::Ref &drawer, const i32 x, const i32 y) override;
381 void OnInput(const u64 keys_down, const u64 keys_up, const u64 keys_held, const TouchPoint touch_pos) override;
382 };
383}
High-level handle wrapper to a texture in SDL2.
Definition sdl2_Types.hpp:47
Type used to vary a value, from an initial value to a final one, following the shape of a sigmoid fun...
Definition ui_Types.hpp:166
Base class for all UI elements in Plutonium, providing basic functionality for all of them.
Definition elm_Element.hpp:35
Represents a menu.
Definition elm_Menu.hpp:132
void SetY(const i32 y)
Sets the Y position of the Menu.
Definition elm_Menu.hpp:291
MenuItem::Ref & GetSelectedItem()
Gets the selected item of the Menu.
Definition elm_Menu.hpp:360
static constexpr u32 DefaultScrollbarWidth
Definition elm_Menu.hpp:147
void AddItem(MenuItem::Ref &item)
Adds a new item to the Menu.
Definition elm_Menu.hpp:338
static constexpr float DefaultIconItemSizesFactor
Definition elm_Menu.hpp:140
static constexpr u32 DefaultShadowHeight
Definition elm_Menu.hpp:149
static constexpr u32 DefaultIconMargin
Definition elm_Menu.hpp:142
static constexpr Color DefaultScrollbarColor
Definition elm_Menu.hpp:136
void SetX(const i32 x)
Sets the X position of the Menu.
Definition elm_Menu.hpp:279
i32 GetHeight() override
Gets the height of the Element.
Definition elm_Menu.hpp:307
i32 GetWidth() override
Gets the width of the Element.
Definition elm_Menu.hpp:295
void SetOnSelectionChanged(OnSelectionChangedCallback on_selection_changed_cb)
Sets the font name to use for the Menu.
Definition elm_Menu.hpp:330
static constexpr u8 DefaultShadowBaseAlpha
Definition elm_Menu.hpp:150
static constexpr u32 DefaultTextMargin
Definition elm_Menu.hpp:143
static constexpr s64 DefaultMoveWaitTimeMs
Definition elm_Menu.hpp:152
void OnInput(const u64 keys_down, const u64 keys_up, const u64 keys_held, const TouchPoint touch_pos) override
Called before rendering the Element in order to handle input.
std::vector< MenuItem::Ref > & GetItems()
Gets the items of the Menu.
Definition elm_Menu.hpp:368
void SetWidth(const i32 width)
Sets the width of the Menu.
Definition elm_Menu.hpp:303
void OnRender(render::Renderer::Ref &drawer, const i32 x, const i32 y) override
Renders the Element on the screen.
static constexpr u8 DefaultItemAlphaIncrementSteps
Definition elm_Menu.hpp:138
void ClearItems()
Clears all items from the Menu.
i32 GetY() override
Gets the Y position of the Element.
Definition elm_Menu.hpp:283
static constexpr u8 DefaultLightScrollbarColorFactor
Definition elm_Menu.hpp:145
void ForceReloadItems()
Forces the Menu to reload all items.
Definition elm_Menu.hpp:350
i32 GetX() override
Gets the X position of the Element.
Definition elm_Menu.hpp:271
void SetSelectedIndex(const u32 idx)
Sets the selected index of the Menu.
Menu(const i32 x, const i32 y, const i32 width, const Color items_clr, const Color items_focus_clr, const i32 items_height, const u32 items_to_show)
Creates a new instance of a Menu.
Represents a menu item.
Definition elm_Menu.hpp:19
OnKeyCallback GetOnKeyCallback(const u32 idx)
Gets the OnKeyCallback at the specified index.
Definition elm_Menu.hpp:83
void SetName(const std::string &name)
Sets the name of the MenuItem.
Definition elm_Menu.hpp:57
bool HasIcon()
Gets whether the MenuItem has an icon.
Definition elm_Menu.hpp:124
u32 GetOnKeyCallbackCount()
Gets the amount of OnKeyCallbacks in the MenuItem.
Definition elm_Menu.hpp:74
sdl2::TextureHandle::Ref GetIconTexture()
Gets the icon of the MenuItem.
Definition elm_Menu.hpp:110
void AddOnKey(OnKeyCallback on_key_cb, const u64 key=HidNpadButton_A)
Adds a new OnKeyCallback to the MenuItem.
static constexpr Color DefaultColor
Definition elm_Menu.hpp:28
u64 GetOnKeyCallbackKey(const u32 idx)
Gets the key of the OnKeyCallback at the specified index.
Definition elm_Menu.hpp:97
void SetIcon(sdl2::TextureHandle::Ref icon)
Sets the icon of the MenuItem.
MenuItem(const std::string &name)
Creates a new instance of a MenuItem.
Definition elm_Menu.hpp:42
std::string GetName()
Gets the name of the MenuItem.
Definition elm_Menu.hpp:49
The main class dealing with rendering.
Definition render_Renderer.hpp:198
Definition sdl2_Types.hpp:17
Definition render_Renderer.hpp:15
Definition elm_Button.hpp:13
#define PU_SMART_CTOR(type)
Defines a static function (::New(...)) as a constructor for smart ptrs, also defines a custom type (:...
Definition pu_Include.hpp:19
#define PU_CLASS_POD_GETSET(fn_name, var_name, type)
Automatically defines a getter and setter function for a POD variable.
Definition pu_Include.hpp:45
#define PU_CLASS_POD_SET(fn_name, var_name, type)
Automatically defines a setter function for a POD variable.
Definition pu_Include.hpp:37
#define PU_CLASS_POD_GET(fn_name, var_name, type)
Automatically defines a getter function for a POD variable.
Definition pu_Include.hpp:29
Type encoding a RGBA-8888 color.
Definition ui_Types.hpp:61
u8 g
Definition ui_Types.hpp:63
u8 a
Definition ui_Types.hpp:65
u8 r
Definition ui_Types.hpp:62
Color WithAlpha(const u8 a)
Creates a new Color with this Color's RGB values and the specified alpha value.
Definition ui_Types.hpp:93
u8 b
Definition ui_Types.hpp:64
constexpr Color(const u8 r, const u8 g, const u8 b, const u8 a)
Creates a new Color with the specified values.
Definition ui_Types.hpp:79
Type encoding a touch point.
Definition ui_Types.hpp:120