diff --git a/src/ui/elements/ui_button.cpp b/src/ui/elements/ui_button.cpp index eae1174..8158f45 100644 --- a/src/ui/elements/ui_button.cpp +++ b/src/ui/elements/ui_button.cpp @@ -4,9 +4,6 @@ namespace recompui { - static const std::string_view hover_state = "hover"; - static const std::string_view disabled_state = "disabled"; - Button::Button(const std::string &text, ButtonStyle style, Element *parent) : Element(parent, Events(EventType::Click, EventType::Hover, EventType::Enable), "button") { this->style = style; diff --git a/src/ui/elements/ui_clickable.cpp b/src/ui/elements/ui_clickable.cpp index 9c800b2..d917820 100644 --- a/src/ui/elements/ui_clickable.cpp +++ b/src/ui/elements/ui_clickable.cpp @@ -2,9 +2,6 @@ namespace recompui { - static const std::string_view hover_state = "hover"; - static const std::string_view disabled_state = "disabled"; - Clickable::Clickable(Element *parent, bool draggable) : Element(parent, Events(EventType::Click, EventType::Hover, EventType::Enable, draggable ? EventType::Drag : EventType::None)) { if (draggable) { set_drag(Drag::Drag); diff --git a/src/ui/elements/ui_radio.cpp b/src/ui/elements/ui_radio.cpp index dae8f37..080d90a 100644 --- a/src/ui/elements/ui_radio.cpp +++ b/src/ui/elements/ui_radio.cpp @@ -2,4 +2,101 @@ namespace recompui { + // RadioOption + + RadioOption::RadioOption(std::string_view name, uint32_t index, Element *parent) : Element(parent, Events(EventType::Click, EventType::Focus, EventType::Hover, EventType::Enable), "label") { + this->index = index; + + set_text(name); + set_cursor(Cursor::Pointer); + set_font_size(28.0f); + set_letter_spacing(3.08f); + set_line_height(28.0f); + set_font_weight(700); + set_border_color(Color{ 242, 242, 242, 255 }); + set_border_bottom_width(0.0f); + set_color(Color{ 255, 255, 255, 153 }); + hover_style.set_color(Color{ 255, 255, 255, 204 }); + checked_style.set_color(Color{ 255, 255, 255, 255 }); + checked_style.set_border_bottom_width(1.0f); + + add_style(&hover_style, { hover_state }); + add_style(&checked_style, { checked_state }); + } + + void RadioOption::set_pressed_callback(std::function callback) { + pressed_callback = callback; + } + + void RadioOption::set_selected_state(bool enable) { + set_style_enabled(checked_state, enable); + } + + void RadioOption::process_event(const Event &e) { + switch (e.type) { + case EventType::Click: + pressed_callback(index); + break; + case EventType::Hover: + set_style_enabled(hover_state, std::get(e.variant).active); + break; + case EventType::Enable: + set_style_enabled(disabled_state, !std::get(e.variant).active); + break; + default: + break; + } + } + + // Radio + + void Radio::set_index_internal(uint32_t index, bool setup, bool trigger_callbacks) { + if (this->index != index || setup) { + options[this->index]->set_selected_state(false); + this->index = index; + options[index]->set_selected_state(true); + + if (trigger_callbacks) { + for (const auto &function : index_changed_callbacks) { + function(index); + } + } + } + } + + void Radio::option_selected(uint32_t index) { + set_index_internal(index, false, true); + } + + Radio::Radio(Element *parent) : Container(FlexDirection::Row, JustifyContent::FlexStart, parent) { + set_gap(12.0f); + } + + Radio::~Radio() { + + } + + void Radio::add_option(std::string_view name) { + RadioOption *option = get_current_context().create_element(name, uint32_t(options.size()), this); + option->set_pressed_callback(std::bind(&Radio::option_selected, this, std::placeholders::_1)); + options.emplace_back(option); + + // The first option was added, select it. + if (options.size() == 1) { + set_index_internal(0, true, false); + } + } + + void Radio::set_index(uint32_t index) { + set_index_internal(index, false, false); + } + + uint32_t Radio::get_index() const { + return index; + } + + void Radio::add_index_changed_callback(std::function callback) { + index_changed_callbacks.emplace_back(callback); + } + }; \ No newline at end of file diff --git a/src/ui/elements/ui_radio.h b/src/ui/elements/ui_radio.h index f8a50cb..ba3a704 100644 --- a/src/ui/elements/ui_radio.h +++ b/src/ui/elements/ui_radio.h @@ -1,7 +1,38 @@ #pragma once -#include "ui_element.h" +#include "ui_container.h" +#include "ui_label.h" namespace recompui { + class RadioOption : public Element { + private: + Style hover_style; + Style checked_style; + std::function pressed_callback = nullptr; + uint32_t index = 0; + protected: + virtual void process_event(const Event &e) override; + public: + RadioOption(std::string_view name, uint32_t index, Element *parent); + void set_pressed_callback(std::function callback); + void set_selected_state(bool enable); + }; + + class Radio : public Container { + private: + std::vector options; + uint32_t index = 0; + std::vector> index_changed_callbacks; + + void set_index_internal(uint32_t index, bool setup, bool trigger_callbacks); + void option_selected(uint32_t index); + public: + Radio(Element *parent); + virtual ~Radio(); + void add_option(std::string_view name); + void set_index(uint32_t index); + uint32_t get_index() const; + void add_index_changed_callback(std::function callback); + }; } // namespace recompui \ No newline at end of file diff --git a/src/ui/elements/ui_toggle.cpp b/src/ui/elements/ui_toggle.cpp index 17a6ac6..62c378c 100644 --- a/src/ui/elements/ui_toggle.cpp +++ b/src/ui/elements/ui_toggle.cpp @@ -4,10 +4,6 @@ namespace recompui { - static const std::string_view checked_state = "checked"; - static const std::string_view hover_state = "hover"; - static const std::string_view disabled_state = "disabled"; - Toggle::Toggle(Element *parent) : Element(parent, Events(EventType::Click, EventType::Hover, EventType::Enable), "button") { set_width(162.0f); set_height(72.0f); diff --git a/src/ui/elements/ui_types.h b/src/ui/elements/ui_types.h index c296d9b..3ad2f54 100644 --- a/src/ui/elements/ui_types.h +++ b/src/ui/elements/ui_types.h @@ -5,6 +5,10 @@ namespace recompui { + constexpr std::string_view checked_state = "checked"; + constexpr std::string_view hover_state = "hover"; + constexpr std::string_view disabled_state = "disabled"; + struct Color { uint8_t r = 255; uint8_t g = 255; diff --git a/src/ui/ui_config_sub_menu.cpp b/src/ui/ui_config_sub_menu.cpp index ab52e84..ff0d12c 100644 --- a/src/ui/ui_config_sub_menu.cpp +++ b/src/ui/ui_config_sub_menu.cpp @@ -19,6 +19,7 @@ void ConfigOptionElement::process_event(const Event &e) { } ConfigOptionElement::ConfigOptionElement(Element *parent) : Element(parent, Events(EventType::Hover)) { + set_gap(8.0f); set_min_height(100.0f); name_label = get_current_context().create_element