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