FTXUI  4.1.1
C++ functional terminal UI.
Loading...
Searching...
No Matches
radiobox.cpp
Go to the documentation of this file.
1#include <functional> // for function
2#include <memory> // for allocator_traits<>::value_type
3#include <utility> // for move
4#include <vector> // for vector
5
6#include "ftxui/component/captured_mouse.hpp" // for CapturedMouse
7#include "ftxui/component/component.hpp" // for Make, Radiobox
8#include "ftxui/component/component_base.hpp" // for ComponentBase
9#include "ftxui/component/component_options.hpp" // for RadioboxOption, EntryState
10#include "ftxui/component/event.hpp" // for Event, Event::ArrowDown, Event::ArrowUp, Event::End, Event::Home, Event::PageDown, Event::PageUp, Event::Return, Event::Tab, Event::TabReverse
11#include "ftxui/component/mouse.hpp" // for Mouse, Mouse::WheelDown, Mouse::WheelUp, Mouse::Left, Mouse::Released
12#include "ftxui/component/screen_interactive.hpp" // for Component
13#include "ftxui/dom/elements.hpp" // for operator|, reflect, Element, vbox, Elements, focus, nothing, select
14#include "ftxui/screen/box.hpp" // for Box
15#include "ftxui/screen/util.hpp" // for clamp
16#include "ftxui/util/ref.hpp" // for Ref, ConstStringListRef
17
18namespace ftxui {
19
20namespace {
21/// @brief A list of selectable element. One and only one can be selected at
22/// the same time.
23/// @ingroup component
24class RadioboxBase : public ComponentBase {
25 public:
26 RadioboxBase(ConstStringListRef entries,
27 int* selected,
28 Ref<RadioboxOption> option)
29 : entries_(entries), selected_(selected), option_(std::move(option)) {}
30
31 private:
32 Element Render() override {
33 Clamp();
34 Elements elements;
35 const bool is_menu_focused = Focused();
36 for (int i = 0; i < size(); ++i) {
37 const bool is_focused = (focused_entry() == i) && is_menu_focused;
38 const bool is_selected = (hovered_ == i);
39 auto focus_management = !is_selected ? nothing
40 : is_menu_focused ? focus
41 : select;
42 auto state = EntryState{
43 entries_[i],
44 *selected_ == i,
45 is_selected,
46 is_focused,
47 };
48 auto element =
49 (option_->transform ? option_->transform
51
52 elements.push_back(element | focus_management | reflect(boxes_[i]));
53 }
54 return vbox(std::move(elements)) | reflect(box_);
55 }
56
57 // NOLINTNEXTLINE(readability-function-cognitive-complexity)
58 bool OnEvent(Event event) override {
59 Clamp();
60 if (!CaptureMouse(event)) {
61 return false;
62 }
63
64 if (event.is_mouse()) {
65 return OnMouseEvent(event);
66 }
67
68 if (Focused()) {
69 const int old_hovered = hovered_;
70 if (event == Event::ArrowUp || event == Event::Character('k')) {
71 (hovered_)--;
72 }
73 if (event == Event::ArrowDown || event == Event::Character('j')) {
74 (hovered_)++;
75 }
76 if (event == Event::PageUp) {
77 (hovered_) -= box_.y_max - box_.y_min;
78 }
79 if (event == Event::PageDown) {
80 (hovered_) += box_.y_max - box_.y_min;
81 }
82 if (event == Event::Home) {
83 (hovered_) = 0;
84 }
85 if (event == Event::End) {
86 (hovered_) = size() - 1;
87 }
88 if (event == Event::Tab && size()) {
89 hovered_ = (hovered_ + 1) % size();
90 }
91 if (event == Event::TabReverse && size()) {
92 hovered_ = (hovered_ + size() - 1) % size();
93 }
94
95 hovered_ = util::clamp(hovered_, 0, size() - 1);
96
97 if (hovered_ != old_hovered) {
98 focused_entry() = hovered_;
99 option_->on_change();
100 return true;
101 }
102 }
103
104 if (event == Event::Character(' ') || event == Event::Return) {
105 *selected_ = hovered_;
106 option_->on_change();
107 return true;
108 }
109
110 return false;
111 }
112
113 bool OnMouseEvent(Event event) {
114 if (event.mouse().button == Mouse::WheelDown ||
115 event.mouse().button == Mouse::WheelUp) {
116 return OnMouseWheel(event);
117 }
118
119 for (int i = 0; i < size(); ++i) {
120 if (!boxes_[i].Contain(event.mouse().x, event.mouse().y)) {
121 continue;
122 }
123
124 TakeFocus();
125 focused_entry() = i;
126 if (event.mouse().button == Mouse::Left &&
127 event.mouse().motion == Mouse::Released) {
128 if (*selected_ != i) {
129 *selected_ = i;
130 option_->on_change();
131 }
132
133 return true;
134 }
135 }
136 return false;
137 }
138
139 bool OnMouseWheel(Event event) {
140 if (!box_.Contain(event.mouse().x, event.mouse().y)) {
141 return false;
142 }
143
144 const int old_hovered = hovered_;
145
146 if (event.mouse().button == Mouse::WheelUp) {
147 (hovered_)--;
148 }
149 if (event.mouse().button == Mouse::WheelDown) {
150 (hovered_)++;
151 }
152
153 hovered_ = util::clamp(hovered_, 0, size() - 1);
154
155 if (hovered_ != old_hovered) {
156 option_->on_change();
157 }
158
159 return true;
160 }
161
162 void Clamp() {
163 boxes_.resize(size());
164 *selected_ = util::clamp(*selected_, 0, size() - 1);
165 focused_entry() = util::clamp(focused_entry(), 0, size() - 1);
166 hovered_ = util::clamp(hovered_, 0, size() - 1);
167 }
168
169 bool Focusable() const final { return entries_.size(); }
170 int& focused_entry() { return option_->focused_entry(); }
171 int size() const { return int(entries_.size()); }
172
173 ConstStringListRef entries_;
174 int* selected_;
175 int hovered_ = *selected_;
176 std::vector<Box> boxes_;
177 Box box_;
178 Ref<RadioboxOption> option_;
179};
180
181} // namespace
182
183/// @brief A list of element, where only one can be selected.
184/// @param entries The list of entries in the list.
185/// @param selected The index of the currently selected element.
186/// @param option Additional optional parameters.
187/// @ingroup component
188/// @see RadioboxBase
189///
190/// ### Example
191///
192/// ```cpp
193/// auto screen = ScreenInteractive::TerminalOutput();
194/// std::vector<std::string> entries = {
195/// "entry 1",
196/// "entry 2",
197/// "entry 3",
198/// };
199/// int selected = 0;
200/// auto menu = Radiobox(&entries, &selected);
201/// screen.Loop(menu);
202/// ```
203///
204/// ### Output
205///
206/// ```bash
207/// ◉ entry 1
208/// ○ entry 2
209/// ○ entry 3
210/// ```
212 int* selected,
213 Ref<RadioboxOption> option) {
214 return Make<RadioboxBase>(entries, selected, std::move(option));
215}
216
217} // namespace ftxui
218
219// Copyright 2020 Arthur Sonzogni. All rights reserved.
220// Use of this source code is governed by the MIT license that can be found in
221// the LICENSE file.
An adapter. Reference a list of strings.
Definition ref.hpp:83
An adapter. Own or reference an mutable object.
Definition ref.hpp:27
constexpr const T & clamp(const T &v, const T &lo, const T &hi)
Definition util.hpp:6
Component Radiobox(ConstStringListRef entries, int *selected_, Ref< RadioboxOption > option={})
A list of element, where only one can be selected.
Definition radiobox.cpp:211
Element nothing(Element element)
A decoration doing absolutely nothing.
Definition util.cpp:27
std::shared_ptr< T > Make(Args &&... args)
Definition component.hpp:25
std::shared_ptr< Node > Element
Definition elements.hpp:19
Element focus(Element)
Definition frame.cpp:83
std::vector< Element > Elements
Definition elements.hpp:20
Decorator reflect(Box &box)
Definition reflect.cpp:39
void Render(Screen &screen, const Element &element)
Display an element on a ftxui::Screen.
Definition node.cpp:44
Decorator size(Direction, Constraint, int value)
Apply a constraint on the size of an element.
Definition size.cpp:85
Element select(Element)
Definition frame.cpp:38
std::shared_ptr< ComponentBase > Component
Element vbox(Elements)
A container displaying elements vertically one by one.
Definition vbox.cpp:78
static const Event TabReverse
Definition event.hpp:52
static const Event PageUp
Definition event.hpp:58
static const Event ArrowUp
Definition event.hpp:38
static const Event Tab
Definition event.hpp:51
static const Event ArrowDown
Definition event.hpp:39
static const Event End
Definition event.hpp:56
static const Event Home
Definition event.hpp:55
static const Event PageDown
Definition event.hpp:59
static const Event Return
Definition event.hpp:49
static RadioboxOption Simple()
Option for standard Radiobox.
std::function< Element(const EntryState &)> transform