FTXUI  5.0.0
C++ functional terminal UI.
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
component_fuzzer.cpp
Go to the documentation of this file.
1// Copyright 2021 Arthur Sonzogni. All rights reserved.
2// Use of this source code is governed by the MIT license that can be found in
3// the LICENSE file.
4#include <cassert>
5#include <iostream>
6#include <vector>
9
10using namespace ftxui;
11namespace {
12
13bool GeneratorBool(const char*& data, size_t& size) {
14 if (size == 0)
15 return false;
16
17 auto out = bool(data[0] % 2);
18 data++;
19 size--;
20 return out;
21}
22
23std::string GeneratorString(const char*& data, size_t& size) {
24 int index = 0;
25 while (index < size && data[index])
26 ++index;
27
28 auto out = std::string(data, data + index);
29 data += index;
30 size -= index;
31
32 // The input component do not support invalid UTF8 yet.
33 try {
34 to_wstring(out);
35 } catch (...) {
36 return "0";
37 }
38 return std::move(out);
39}
40
41int GeneratorInt(const char* data, size_t size) {
42 if (size == 0)
43 return 0;
44 auto out = int(data[0]);
45 data++;
46 size--;
47 return out;
48}
49
50Color GeneratorColor(const char* data, size_t size) {
51 return Color::RGB(GeneratorInt(data, size), GeneratorInt(data, size),
52 GeneratorInt(data, size));
53}
54
55AnimatedColorOption GeneratorAnimatedColorOption(const char* data,
56 size_t size) {
58 option.enabled = GeneratorBool(data, size);
59 option.inactive = GeneratorColor(data, size);
60 option.active = GeneratorColor(data, size);
61 option.duration = std::chrono::milliseconds(GeneratorInt(data, size));
62 return option;
63}
64
65AnimatedColorsOption GeneratorAnimatedColorsOptions(const char* data,
66 size_t size) {
68 option.background = GeneratorAnimatedColorOption(data, size);
69 option.foreground = GeneratorAnimatedColorOption(data, size);
70 return option;
71}
72
73ButtonOption GeneratorButtonOption(const char* data, size_t size) {
74 ButtonOption option;
75 option.animated_colors = GeneratorAnimatedColorsOptions(data, size);
76 return option;
77}
78
79UnderlineOption GeneratorUnderlineOption(const char* data, size_t size) {
80 UnderlineOption option;
81 option.enabled = GeneratorBool(data, size);
82 option.color_active = GeneratorColor(data, size);
83 option.color_inactive = GeneratorColor(data, size);
84 option.leader_duration = std::chrono::milliseconds(GeneratorInt(data, size));
85 option.follower_duration =
86 std::chrono::milliseconds(GeneratorInt(data, size));
87 option.leader_delay = std::chrono::milliseconds(GeneratorInt(data, size));
88 option.follower_delay = std::chrono::milliseconds(GeneratorInt(data, size));
89 return option;
90}
91
92MenuEntryOption GeneratorMenuEntryOption(const char* data, size_t size) {
93 MenuEntryOption option;
94 option.animated_colors = GeneratorAnimatedColorsOptions(data, size);
95 return option;
96}
97
98MenuOption GeneratorMenuOption(const char* data, size_t size) {
99 MenuOption option;
100 option.underline = GeneratorUnderlineOption(data, size);
101 option.entries_option = GeneratorMenuEntryOption(data, size);
102 option.direction = static_cast<Direction>(GeneratorInt(data, size) % 4);
103 return option;
104}
105
106bool g_bool;
107int g_int;
108std::vector<std::string> g_list;
109
110Components GeneratorComponents(const char*& data, size_t& size, int depth);
111
112Component GeneratorComponent(const char*& data, size_t& size, int depth) {
113 depth--;
114 int value = GeneratorInt(data, size);
115 if (depth <= 0)
116 return Button(GeneratorString(data, size), [] {});
117
118 constexpr int value_max = 19;
119 value = (value % value_max + value_max) % value_max;
120 switch (value) {
121 case 0:
122 return Button(
123 GeneratorString(data, size), [] {},
124 GeneratorButtonOption(data, size));
125 case 1:
126 return Checkbox(GeneratorString(data, size), &g_bool);
127 case 2:
128 return Input(GeneratorString(data, size), GeneratorString(data, size));
129 case 3:
130 return Menu(&g_list, &g_int, GeneratorMenuOption(data, size));
131 case 4:
132 return Radiobox(&g_list, &g_int);
133 case 5:
134 return Toggle(&g_list, &g_int);
135 case 6:
136 return Slider(GeneratorString(data, size), &g_int,
137 GeneratorInt(data, size), GeneratorInt(data, size),
138 GeneratorInt(data, size));
139 case 7:
140 return ResizableSplitLeft(GeneratorComponent(data, size, depth - 1),
141 GeneratorComponent(data, size, depth - 1),
142 &g_int);
143 case 8:
144 return ResizableSplitRight(GeneratorComponent(data, size, depth - 1),
145 GeneratorComponent(data, size, depth - 1),
146 &g_int);
147 case 9:
148 return ResizableSplitTop(GeneratorComponent(data, size, depth - 1),
149 GeneratorComponent(data, size, depth - 1),
150 &g_int);
151 case 10:
152 return ResizableSplitBottom(GeneratorComponent(data, size, depth - 1),
153 GeneratorComponent(data, size, depth - 1),
154 &g_int);
155 case 11:
156 return Container::Vertical(GeneratorComponents(data, size, depth - 1));
157
158 case 12:
159 return Container::Vertical(GeneratorComponents(data, size, depth - 1),
160 &g_int);
161
162 case 13:
163 return Container::Horizontal(GeneratorComponents(data, size, depth - 1));
164 case 14:
165 return Container::Horizontal(GeneratorComponents(data, size, depth - 1),
166 &g_int);
167 case 15:
168 return Container::Tab(GeneratorComponents(data, size, depth - 1), &g_int);
169 case 16:
170 return Maybe(GeneratorComponent(data, size, depth - 1), &g_bool);
171 case 17:
172 return Dropdown(&g_list, &g_int);
173 case 18:
174 return Collapsible(GeneratorString(data, size),
175 GeneratorComponent(data, size, depth - 1),
176 GeneratorBool(data, size));
177 default:
178 assert(false);
179 }
180}
181
182Components GeneratorComponents(const char*& data, size_t& size, int depth) {
183 Components out;
184 if (depth > 0) {
185 while (size && GeneratorInt(data, size) % 2) {
186 out.push_back(GeneratorComponent(data, size, depth - 1));
187 }
188 }
189 return std::move(out);
190}
191
192} // namespace
193extern "C" int LLVMFuzzerTestOneInput(const char* data, size_t size) {
194 g_bool = GeneratorBool(data, size);
195 g_int = GeneratorInt(data, size);
196 g_list = {
197 "test_1", "test_2", "test_3", "test_4", "test_5",
198 };
199
200 int depth = 10;
201 auto component = GeneratorComponent(data, size, depth);
202
203 int width = GeneratorInt(data, size);
204 int height = GeneratorInt(data, size);
205
206 width %= 500;
207 width += 500;
208
209 height %= 500;
210 height += 500;
211
212 auto screen =
213 Screen::Create(Dimension::Fixed(width), Dimension::Fixed(height));
214
215 auto event_receiver = MakeReceiver<Task>();
216 {
217 auto parser = TerminalInputParser(event_receiver->MakeSender());
218 for (size_t i = 0; i < size; ++i)
219 parser.Add(data[i]);
220 }
221
222 Task event;
223 while (event_receiver->Receive(&event)) {
224 component->OnEvent(std::get<Event>(event));
225 auto document = component->Render();
226 Render(screen, document);
227 }
228 return 0; // Non-zero return values are reserved for future use.
229}
A class representing terminal colors.
Definition: color.hpp:21
int LLVMFuzzerTestOneInput(const char *data, size_t size)
Component Horizontal(Components children)
A list of components, drawn one by one horizontally and navigated horizontally using left/right arrow...
Definition: container.cpp:360
Component Vertical(Components children)
A list of components, drawn one by one vertically and navigated vertically using up/down arrow key or...
Definition: container.cpp:317
Component Tab(Components children, int *selector)
A list of components, where only one is drawn and interacted with at a time. The |selector| gives the...
Definition: container.cpp:405
Dimensions Fixed(int)
Definition: screen.cpp:372
AnimatedColorOption foreground
Decorator size(WidthOrHeight, Constraint, int value)
Apply a constraint on the size of an element.
Definition: size.cpp:90
Component Maybe(Component, const bool *show)
Decorate a component |child|. It is shown only when |show| is true.
Definition: maybe.cpp:75
Direction
Definition: direction.hpp:8
Component ResizableSplitTop(Component main, Component back, int *main_size)
An vertical split in between two components, configurable using the mouse.
Component Checkbox(CheckboxOption options)
Component Menu(MenuOption options)
A list of text. The focused element is selected.
Definition: menu.cpp:513
std::shared_ptr< ComponentBase > Component
Component Toggle(ConstStringListRef entries, int *selected)
An horizontal list of elements. The user can navigate through them.
Definition: menu.cpp:555
std::vector< Component > Components
Component Radiobox(RadioboxOption options)
A list of element, where only one can be selected.
Definition: radiobox.cpp:207
Component Button(ButtonOption options)
Draw a button. Execute a function when clicked.
Definition: button.cpp:175
AnimatedColorOption background
std::wstring to_wstring(const std::string &s)
Convert a std::wstring into a UTF8 std::string.
Definition: string.cpp:1637
Component Input(InputOption options={})
An input box for editing text.
Definition: input.cpp:564
Component ResizableSplitRight(Component main, Component back, int *main_size)
An horizontal split in between two components, configurable using the mouse.
Component Dropdown(ConstStringListRef entries, int *selected)
A dropdown menu.
Definition: dropdown.cpp:22
Component Slider(SliderOption< T > options)
A slider in any direction.
Definition: slider.cpp:338
std::variant< Event, Closure, AnimationTask > Task
Definition: task.hpp:14
AnimatedColorsOption animated_colors
Component ResizableSplitBottom(Component main, Component back, int *main_size)
An vertical split in between two components, configurable using the mouse.
void Render(Screen &screen, const Element &element)
Display an element on a ftxui::Screen.
Definition: node.cpp:47
Component ResizableSplitLeft(Component main, Component back, int *main_size)
An horizontal split in between two components, configurable using the mouse.
Component Collapsible(ConstStringRef label, Component child, Ref< bool > show=false)
Option for the MenuEntry component.
Option about a potentially animated color.
animation::Duration duration
Option for the AnimatedButton component.
AnimatedColorsOption animated_colors
Option for the Menu component.
MenuEntryOption entries_option
UnderlineOption underline
‍The index of the selected entry.
animation::Duration follower_duration
animation::Duration follower_delay
animation::Duration leader_duration
animation::Duration leader_delay