feat: Support mouse scroll. (#201)

This commit is contained in:
Arthur Sonzogni 2021-09-08 09:36:37 +02:00 committed by GitHub
parent ed28bad02a
commit 4d50dadb41
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 237 additions and 109 deletions

View File

@ -9,6 +9,7 @@ example(gallery)
example(homescreen) example(homescreen)
example(input) example(input)
example(menu) example(menu)
example(menu_in_frame)
example(menu2) example(menu2)
example(menu_multiple) example(menu_multiple)
example(menu_entries) example(menu_entries)

View File

@ -15,22 +15,20 @@ struct CheckboxState {
}; };
int main(int argc, const char* argv[]) { int main(int argc, const char* argv[]) {
int size = 30; std::vector<CheckboxState> states(30);
std::vector<CheckboxState> states(size);
auto container = Container::Vertical({}); auto container = Container::Vertical({});
for (int i = 0; i < size; ++i) { for (int i = 0; i < 30; ++i) {
states[i].checked = false; states[i].checked = false;
container->Add( container->Add(
Checkbox("Checkbox" + std::to_string(i), &states[i].checked)); Checkbox("Checkbox" + std::to_string(i), &states[i].checked));
} }
auto component = Renderer(container, [&] { auto renderer = Renderer(container, [&] {
return container->Render() | frame | ftxui::size(HEIGHT, LESS_THAN, 10) | return container->Render() | frame | size(HEIGHT, LESS_THAN, 10) | border;
border;
}); });
auto screen = ScreenInteractive::FitComponent(); auto screen = ScreenInteractive::FitComponent();
screen.Loop(component); screen.Loop(renderer);
return 0; return 0;
} }

View File

@ -137,17 +137,18 @@ int main(int argc, const char* argv[]) {
int compiler_selected = 0; int compiler_selected = 0;
Component compiler = Radiobox(&compiler_entries, &compiler_selected); Component compiler = Radiobox(&compiler_entries, &compiler_selected);
std::array<std::string, 4> options_label = { std::array<std::string, 8> options_label = {
"-Wall", "-Wall",
"-Werror", "-Werror",
"-lpthread", "-lpthread",
"-O3", "-O3",
"-Wabi-tag",
"-Wno-class-conversion",
"-Wcomma-subscript",
"-Wno-conversion-null",
}; };
std::array<bool, 4> options_state = { std::array<bool, 8> options_state = {
false, false, false, false, false, false, false, false, false,
false,
false,
false,
}; };
std::vector<std::string> input_entries; std::vector<std::string> input_entries;
@ -170,6 +171,10 @@ int main(int argc, const char* argv[]) {
Checkbox(&options_label[1], &options_state[1]), Checkbox(&options_label[1], &options_state[1]),
Checkbox(&options_label[2], &options_state[2]), Checkbox(&options_label[2], &options_state[2]),
Checkbox(&options_label[3], &options_state[3]), Checkbox(&options_label[3], &options_state[3]),
Checkbox(&options_label[4], &options_state[4]),
Checkbox(&options_label[5], &options_state[5]),
Checkbox(&options_label[6], &options_state[6]),
Checkbox(&options_label[7], &options_state[7]),
}); });
auto compiler_component = Container::Horizontal({ auto compiler_component = Container::Horizontal({
@ -189,7 +194,7 @@ int main(int argc, const char* argv[]) {
// Compiler // Compiler
line.push_back(text(compiler_entries[compiler_selected]) | bold); line.push_back(text(compiler_entries[compiler_selected]) | bold);
// flags // flags
for (int i = 0; i < 4; ++i) { for (int i = 0; i < 8; ++i) {
if (options_state[i]) { if (options_state[i]) {
line.push_back(text(" ")); line.push_back(text(" "));
line.push_back(text(options_label[i]) | dim); line.push_back(text(options_label[i]) | dim);
@ -210,7 +215,7 @@ int main(int argc, const char* argv[]) {
auto compiler_renderer = Renderer(compiler_component, [&] { auto compiler_renderer = Renderer(compiler_component, [&] {
auto compiler_win = window(text("Compiler"), compiler->Render() | frame); auto compiler_win = window(text("Compiler"), compiler->Render() | frame);
auto flags_win = window(text("Flags"), flags->Render()); auto flags_win = window(text("Flags"), flags->Render() | frame);
auto executable_win = window(text("Executable:"), executable_->Render()); auto executable_win = window(text("Executable:"), executable_->Render());
auto input_win = auto input_win =
window(text("Input"), window(text("Input"),
@ -228,14 +233,14 @@ int main(int argc, const char* argv[]) {
})); }));
return vbox({ return vbox({
hbox({ hbox({
compiler_win | size(HEIGHT, LESS_THAN, 6), compiler_win,
flags_win, flags_win,
vbox({ vbox({
executable_win | size(WIDTH, EQUAL, 20), executable_win | size(WIDTH, EQUAL, 20),
input_win | size(WIDTH, EQUAL, 60), input_win | size(WIDTH, EQUAL, 60),
}), }),
filler(), filler(),
}), }) | size(HEIGHT, LESS_THAN, 6),
hflow(render_command()) | flex_grow, hflow(render_command()) | flex_grow,
}) | }) |
flex_grow | border; flex_grow | border;

View File

@ -0,0 +1,32 @@
#include <memory> // for shared_ptr, __shared_ptr_access
#include <string> // for string, basic_string, operator+, to_string
#include <vector> // for vector
#include "ftxui/component/captured_mouse.hpp" // for ftxui
#include "ftxui/component/component.hpp" // for Radiobox, Renderer
#include "ftxui/component/component_base.hpp" // for ComponentBase
#include "ftxui/component/screen_interactive.hpp" // for ScreenInteractive
#include "ftxui/dom/elements.hpp" // for operator|, Element, size, border, frame, HEIGHT, LESS_THAN
using namespace ftxui;
int main(int argc, const char* argv[]) {
std::vector<std::string> entries;
int selected = 0;
for (int i = 0; i < 30; ++i)
entries.push_back("Entry " + std::to_string(i));
auto radiobox = Menu(&entries, &selected);
auto renderer = Renderer(radiobox, [&] {
return radiobox->Render() | frame | size(HEIGHT, LESS_THAN, 10) | border;
});
auto screen = ScreenInteractive::FitComponent();
screen.Loop(renderer);
return 0;
}
// Copyright 2020 Arthur Sonzogni. All rights reserved.
// Use of this source code is governed by the MIT license that can be found in
// the LICENSE file.

View File

@ -45,8 +45,11 @@ struct ButtonOption {
struct CheckboxOption { struct CheckboxOption {
std::string style_checked = ""; ///< Prefix for a "checked" state. std::string style_checked = ""; ///< Prefix for a "checked" state.
std::string style_unchecked = ""; ///< Prefix for a "unchecked" state. std::string style_unchecked = ""; ///< Prefix for a "unchecked" state.
Decorator style_focused = inverted; ///< Decorator used when focused. Decorator style_normal = nothing; ///< style.
Decorator style_unfocused = nothing; ///< Decorator used when unfocused. Decorator style_focused = inverted; ///< Style when focused.
Decorator style_selected = bold; ///< Style when selected.
Decorator style_selected_focused =
Decorator(inverted) | bold; ///< Style when selected and focused.
/// Called when the user change the state. /// Called when the user change the state.
std::function<void()> on_change = []() {}; std::function<void()> on_change = []() {};
@ -73,8 +76,11 @@ struct InputOption {
struct RadioboxOption { struct RadioboxOption {
std::string style_checked = ""; ///< Prefix for a "checked" state. std::string style_checked = ""; ///< Prefix for a "checked" state.
std::string style_unchecked = ""; ///< Prefix for a "unchecked" state. std::string style_unchecked = ""; ///< Prefix for a "unchecked" state.
Decorator style_focused = inverted; ///< Decorator used when focused. Decorator style_normal = nothing; ///< style.
Decorator style_unfocused = nothing; ///< Decorator used when unfocused. Decorator style_focused = inverted; ///< Style when focused.
Decorator style_selected = bold; ///< Style when selected.
Decorator style_selected_focused =
Decorator(inverted) | bold; ///< Style when selected and focused.
/// Called when the selected entry changes. /// Called when the selected entry changes.
std::function<void()> on_change = []() {}; std::function<void()> on_change = []() {};

View File

@ -33,8 +33,12 @@ class CheckboxBase : public ComponentBase {
// Component implementation. // Component implementation.
Element Render() override { Element Render() override {
bool is_focused = Focused(); bool is_focused = Focused();
auto style = is_focused ? option_->style_focused : option_->style_unfocused; bool is_active = Active();
auto focus_management = is_focused ? focus : *state_ ? select : nothing; auto style = is_focused ? (hovered_ ? option_->style_selected_focused
: option_->style_selected)
: (hovered_ ? option_->style_focused
: option_->style_normal);
auto focus_management = is_focused ? focus : is_active ? select : nothing;
return hbox(text(*state_ ? option_->style_checked return hbox(text(*state_ ? option_->style_checked
: option_->style_unchecked), : option_->style_unchecked),
text(*label_) | style | focus_management) | text(*label_) | style | focus_management) |
@ -45,6 +49,7 @@ class CheckboxBase : public ComponentBase {
if (event.is_mouse()) if (event.is_mouse())
return OnMouseEvent(event); return OnMouseEvent(event);
hovered_ = false;
if (event == Event::Character(' ') || event == Event::Return) { if (event == Event::Character(' ') || event == Event::Return) {
*state_ = !*state_; *state_ = !*state_;
option_->on_change(); option_->on_change();
@ -54,12 +59,13 @@ class CheckboxBase : public ComponentBase {
} }
bool OnMouseEvent(Event event) { bool OnMouseEvent(Event event) {
hovered_ = box_.Contain(event.mouse().x, event.mouse().y);
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
if (!box_.Contain(event.mouse().x, event.mouse().y))
return false;
TakeFocus(); if (!hovered_)
return false;
if (event.mouse().button == Mouse::Left && if (event.mouse().button == Mouse::Left &&
event.mouse().motion == Mouse::Pressed) { event.mouse().motion == Mouse::Pressed) {
@ -75,8 +81,9 @@ class CheckboxBase : public ComponentBase {
ConstStringRef label_; ConstStringRef label_;
bool* const state_; bool* const state_;
Box box_; bool hovered_ = false;
Ref<CheckboxOption> option_; Ref<CheckboxOption> option_;
Box box_;
}; };
} // namespace } // namespace

View File

@ -54,11 +54,7 @@ class ContainerBase : public ComponentBase {
virtual bool EventHandler(Event) { return false; } virtual bool EventHandler(Event) { return false; }
virtual bool OnMouseEvent(Event event) { virtual bool OnMouseEvent(Event event) {
for (Component& child : children_) { return ComponentBase::OnEvent(event);
if (child->OnEvent(event))
return true;
}
return false;
} }
int selected_ = 0; int selected_ = 0;
@ -111,6 +107,27 @@ class VerticalContainer : public ContainerBase {
*selector_ = std::max(0, std::min(int(children_.size()) - 1, *selector_)); *selector_ = std::max(0, std::min(int(children_.size()) - 1, *selector_));
return old_selected != *selector_; return old_selected != *selector_;
} }
bool OnMouseEvent(Event event) override {
if (ContainerBase::OnMouseEvent(event))
return true;
if (event.mouse().button != Mouse::WheelUp &&
event.mouse().button != Mouse::WheelDown) {
return false;
}
if (!Focusable())
return false;
if (event.mouse().button == Mouse::WheelUp)
MoveSelector(-1);
if (event.mouse().button == Mouse::WheelDown)
MoveSelector(+1);
*selector_ = std::max(0, std::min(int(children_.size()) - 1, *selector_));
return true;
}
}; };
class HorizontalContainer : public ContainerBase { class HorizontalContainer : public ContainerBase {

View File

@ -52,14 +52,14 @@ class WideInputBase : public ComponentBase {
if (content.size() == 0) { if (content.size() == 0) {
if (is_focused) if (is_focused)
return text(*placeholder_) | focus | dim | inverted | main_decorator | return text(*placeholder_) | focus | dim | inverted | main_decorator |
reflect(input_box_); reflect(box_);
else else
return text(*placeholder_) | dim | main_decorator | reflect(input_box_); return text(*placeholder_) | dim | main_decorator | reflect(box_);
} }
// Not focused. // Not focused.
if (!is_focused) if (!is_focused)
return text(content) | main_decorator | reflect(input_box_); return text(content) | main_decorator | reflect(box_);
std::wstring part_before_cursor = content.substr(0, cursor_position()); std::wstring part_before_cursor = content.substr(0, cursor_position());
std::wstring part_at_cursor = cursor_position() < (int)content.size() std::wstring part_at_cursor = cursor_position() < (int)content.size()
@ -76,7 +76,7 @@ class WideInputBase : public ComponentBase {
text(part_before_cursor), text(part_before_cursor),
text(part_at_cursor) | underlined | focused | reflect(cursor_box_), text(part_at_cursor) | underlined | focused | reflect(cursor_box_),
text(part_after_cursor) text(part_after_cursor)
) | flex | inverted | frame | main_decorator | reflect(input_box_); ) | flex | inverted | frame | main_decorator | reflect(box_);
// clang-format on // clang-format on
} }
@ -153,7 +153,7 @@ class WideInputBase : public ComponentBase {
bool OnMouseEvent(Event event) { bool OnMouseEvent(Event event) {
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
if (!input_box_.Contain(event.mouse().x, event.mouse().y)) if (!box_.Contain(event.mouse().x, event.mouse().y))
return false; return false;
TakeFocus(); TakeFocus();
@ -177,7 +177,7 @@ class WideInputBase : public ComponentBase {
WideStringRef content_; WideStringRef content_;
ConstStringRef placeholder_; ConstStringRef placeholder_;
Box input_box_; Box box_;
Box cursor_box_; Box cursor_box_;
Ref<InputOption> option_; Ref<InputOption> option_;
}; };

View File

@ -26,7 +26,7 @@ class MenuBase : public ComponentBase {
MenuBase(ConstStringListRef entries, int* selected, Ref<MenuOption> option) MenuBase(ConstStringListRef entries, int* selected, Ref<MenuOption> option)
: entries_(entries), selected_(selected), option_(option) {} : entries_(entries), selected_(selected), option_(option) {}
Element Render() { Element Render() override {
Elements elements; Elements elements;
bool is_menu_focused = Focused(); bool is_menu_focused = Focused();
boxes_.resize(entries_.size()); boxes_.resize(entries_.size());
@ -45,18 +45,17 @@ class MenuBase : public ComponentBase {
elements.push_back(text(icon + entries_[i]) | style | focus_management | elements.push_back(text(icon + entries_[i]) | style | focus_management |
reflect(boxes_[i])); reflect(boxes_[i]));
} }
return vbox(std::move(elements)); return vbox(std::move(elements)) | reflect(box_);
} }
bool OnEvent(Event event) { bool OnEvent(Event event) override {
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
if (event.is_mouse()) if (event.is_mouse())
return OnMouseEvent(event); return OnMouseEvent(event);
if (!Focused()) if (Focused()) {
return false;
int old_selected = *selected_; int old_selected = *selected_;
if (event == Event::ArrowUp || event == Event::Character('k')) if (event == Event::ArrowUp || event == Event::Character('k'))
(*selected_)--; (*selected_)--;
@ -74,6 +73,7 @@ class MenuBase : public ComponentBase {
option_->on_change(); option_->on_change();
return true; return true;
} }
}
if (event == Event::Return) { if (event == Event::Return) {
option_->on_enter(); option_->on_enter();
@ -84,6 +84,15 @@ class MenuBase : public ComponentBase {
} }
bool OnMouseEvent(Event event) { bool OnMouseEvent(Event event) {
if (event.mouse().button == Mouse::WheelDown ||
event.mouse().button == Mouse::WheelUp) {
return OnMouseWheel(event);
}
if (event.mouse().button != Mouse::None &&
event.mouse().button != Mouse::Left) {
return false;
}
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
for (int i = 0; i < int(boxes_.size()); ++i) { for (int i = 0; i < int(boxes_.size()); ++i) {
@ -104,6 +113,23 @@ class MenuBase : public ComponentBase {
return false; return false;
} }
bool OnMouseWheel(Event event) {
if (!box_.Contain(event.mouse().x, event.mouse().y))
return false;
int old_selected = *selected_;
if (event.mouse().button == Mouse::WheelUp)
(*selected_)--;
if (event.mouse().button == Mouse::WheelDown)
(*selected_)++;
*selected_ = std::max(0, std::min(int(entries_.size()) - 1, *selected_));
if (*selected_ != old_selected)
option_->on_change();
return true;
}
bool Focusable() const final { return entries_.size(); } bool Focusable() const final { return entries_.size(); }
int& focused_entry() { return option_->focused_entry(); } int& focused_entry() { return option_->focused_entry(); }
@ -113,6 +139,7 @@ class MenuBase : public ComponentBase {
Ref<MenuOption> option_; Ref<MenuOption> option_;
std::vector<Box> boxes_; std::vector<Box> boxes_;
Box box_;
}; };
/// @brief A list of text. The focused element is selected. /// @brief A list of text. The focused element is selected.
@ -158,10 +185,10 @@ Component MenuEntry(ConstStringRef label, Ref<MenuEntryOption> option) {
private: private:
Element Render() override { Element Render() override {
bool focused = Focused(); bool focused = Focused();
auto style = auto style = hovered_ ? (focused ? option_->style_selected_focused
hovered_ ? (focused ? option_->style_selected_focused
: option_->style_selected) : option_->style_selected)
: (focused ? option_->style_focused : option_->style_normal); : (focused ? option_->style_focused
: option_->style_normal);
auto focus_management = focused ? select : nothing; auto focus_management = focused ? select : nothing;
auto label = focused ? "> " + (*label_) // auto label = focused ? "> " + (*label_) //
: " " + (*label_); : " " + (*label_);

View File

@ -37,19 +37,24 @@ class RadioboxBase : public ComponentBase {
if (option_->style_unchecked == "") if (option_->style_unchecked == "")
option_->style_unchecked = "( )"; option_->style_unchecked = "( )";
#endif #endif
hovered_ = *selected_;
} }
private: private:
Element Render() override { Element Render() override {
Elements elements; Elements elements;
bool is_focused = Focused(); bool is_menu_focused = Focused();
boxes_.resize(entries_.size()); boxes_.resize(entries_.size());
for (size_t i = 0; i < entries_.size(); ++i) { for (size_t i = 0; i < entries_.size(); ++i) {
auto style = (focused_entry() == int(i) && is_focused) bool is_focused = (focused_entry() == int(i)) && is_menu_focused;
? option_->style_focused bool is_selected = (hovered_ == int(i));
: option_->style_unfocused;
auto focus_management = (focused_entry() != int(i)) ? nothing auto style = is_selected ? (is_focused ? option_->style_selected_focused
: is_focused ? focus : option_->style_selected)
: (is_focused ? option_->style_focused
: option_->style_normal);
auto focus_management = !is_selected ? nothing
: is_menu_focused ? focus
: select; : select;
const std::string& symbol = *selected_ == int(i) const std::string& symbol = *selected_ == int(i)
@ -58,37 +63,39 @@ class RadioboxBase : public ComponentBase {
elements.push_back(hbox(text(symbol), text(entries_[i]) | style) | elements.push_back(hbox(text(symbol), text(entries_[i]) | style) |
focus_management | reflect(boxes_[i])); focus_management | reflect(boxes_[i]));
} }
return vbox(std::move(elements)); return vbox(std::move(elements)) | reflect(box_);
} }
bool OnEvent(Event event) override { bool OnEvent(Event event) override {
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
if (event.is_mouse()) if (event.is_mouse())
return OnMouseEvent(event); return OnMouseEvent(event);
if (!Focused()) if (Focused()) {
return false; int old_hovered = hovered_;
int new_focused = focused_entry();
if (event == Event::ArrowUp || event == Event::Character('k')) if (event == Event::ArrowUp || event == Event::Character('k'))
new_focused--; (hovered_)--;
if (event == Event::ArrowDown || event == Event::Character('j')) if (event == Event::ArrowDown || event == Event::Character('j'))
new_focused++; (hovered_)++;
if (event == Event::Tab && entries_.size()) if (event == Event::Tab && entries_.size())
new_focused = (new_focused + 1) % entries_.size(); hovered_ = (hovered_ + 1) % entries_.size();
if (event == Event::TabReverse && entries_.size()) if (event == Event::TabReverse && entries_.size())
new_focused = (new_focused + entries_.size() - 1) % entries_.size(); hovered_ = (hovered_ + entries_.size() - 1) % entries_.size();
new_focused = std::max(0, std::min(int(entries_.size()) - 1, new_focused)); hovered_ = std::max(0, std::min(int(entries_.size()) - 1, hovered_));
if (focused_entry() != new_focused) { if (hovered_ != old_hovered) {
focused_entry() = new_focused; focused_entry() = hovered_;
option_->on_change();
return true; return true;
} }
}
if (event == Event::Character(' ') || event == Event::Return) { if (event == Event::Character(' ') || event == Event::Return) {
*selected_ = focused_entry(); *selected_ = hovered_;
//*selected_ = focused_entry();
option_->on_change(); option_->on_change();
} }
@ -98,35 +105,58 @@ class RadioboxBase : public ComponentBase {
bool OnMouseEvent(Event event) { bool OnMouseEvent(Event event) {
if (!CaptureMouse(event)) if (!CaptureMouse(event))
return false; return false;
if (event.mouse().button == Mouse::WheelDown ||
event.mouse().button == Mouse::WheelUp) {
return OnMouseWheel(event);
}
for (int i = 0; i < int(boxes_.size()); ++i) { for (int i = 0; i < int(boxes_.size()); ++i) {
if (!boxes_[i].Contain(event.mouse().x, event.mouse().y)) if (!boxes_[i].Contain(event.mouse().x, event.mouse().y))
continue; continue;
TakeFocus();
focused_entry() = i; focused_entry() = i;
TakeFocus();
if (event.mouse().button == Mouse::Left && if (event.mouse().button == Mouse::Left &&
event.mouse().motion == Mouse::Pressed) { event.mouse().motion == Mouse::Released) {
cursor_position = i;
TakeFocus();
if (*selected_ != i) { if (*selected_ != i) {
*selected_ = i; *selected_ = i;
option_->on_change(); option_->on_change();
} }
return true; return true;
} }
} }
return false; return false;
} }
bool OnMouseWheel(Event event) {
if (!box_.Contain(event.mouse().x, event.mouse().y))
return false;
int old_hovered = hovered_;
if (event.mouse().button == Mouse::WheelUp)
(hovered_)--;
if (event.mouse().button == Mouse::WheelDown)
(hovered_)++;
hovered_ = std::max(0, std::min(int(entries_.size()) - 1, hovered_));
if (hovered_ != old_hovered)
option_->on_change();
return true;
}
bool Focusable() const final { return entries_.size(); } bool Focusable() const final { return entries_.size(); }
int& focused_entry() { return option_->focused_entry(); } int& focused_entry() { return option_->focused_entry(); }
ConstStringListRef entries_; ConstStringListRef entries_;
int* const selected_; int* selected_;
int hovered_;
int cursor_position = 0;
std::vector<Box> boxes_; std::vector<Box> boxes_;
Box box_;
Ref<RadioboxOption> option_; Ref<RadioboxOption> option_;
}; };

View File

@ -43,7 +43,7 @@ class ResizableSplitLeftBase : public ComponentBase {
} }
if (captured_mouse_) { if (captured_mouse_) {
*main_size_ = event.mouse().x - global_box_.x_min; *main_size_ = event.mouse().x - box_.x_min;
return true; return true;
} }
@ -56,7 +56,7 @@ class ResizableSplitLeftBase : public ComponentBase {
separator() | reflect(separator_box_), separator() | reflect(separator_box_),
child_->Render() | xflex, child_->Render() | xflex,
}) | }) |
reflect(global_box_); reflect(box_);
}; };
private: private:
@ -65,7 +65,7 @@ class ResizableSplitLeftBase : public ComponentBase {
int* const main_size_; int* const main_size_;
CapturedMouse captured_mouse_; CapturedMouse captured_mouse_;
Box separator_box_; Box separator_box_;
Box global_box_; Box box_;
}; };
class ResizableSplitRightBase : public ComponentBase { class ResizableSplitRightBase : public ComponentBase {
@ -99,7 +99,7 @@ class ResizableSplitRightBase : public ComponentBase {
} }
if (captured_mouse_) { if (captured_mouse_) {
*main_size_ = global_box_.x_max - event.mouse().x; *main_size_ = box_.x_max - event.mouse().x;
return true; return true;
} }
@ -112,7 +112,7 @@ class ResizableSplitRightBase : public ComponentBase {
separator() | reflect(separator_box_), separator() | reflect(separator_box_),
main_->Render() | size(WIDTH, EQUAL, *main_size_), main_->Render() | size(WIDTH, EQUAL, *main_size_),
}) | }) |
reflect(global_box_); reflect(box_);
}; };
private: private:
@ -121,7 +121,7 @@ class ResizableSplitRightBase : public ComponentBase {
int* const main_size_; int* const main_size_;
CapturedMouse captured_mouse_; CapturedMouse captured_mouse_;
Box separator_box_; Box separator_box_;
Box global_box_; Box box_;
}; };
class ResizableSplitTopBase : public ComponentBase { class ResizableSplitTopBase : public ComponentBase {
@ -155,7 +155,7 @@ class ResizableSplitTopBase : public ComponentBase {
} }
if (captured_mouse_) { if (captured_mouse_) {
*main_size_ = event.mouse().y - global_box_.y_min; *main_size_ = event.mouse().y - box_.y_min;
return true; return true;
} }
@ -168,7 +168,7 @@ class ResizableSplitTopBase : public ComponentBase {
separator() | reflect(separator_box_), separator() | reflect(separator_box_),
child_->Render() | yflex, child_->Render() | yflex,
}) | }) |
reflect(global_box_); reflect(box_);
}; };
private: private:
@ -177,7 +177,7 @@ class ResizableSplitTopBase : public ComponentBase {
int* const main_size_; int* const main_size_;
CapturedMouse captured_mouse_; CapturedMouse captured_mouse_;
Box separator_box_; Box separator_box_;
Box global_box_; Box box_;
}; };
class ResizableSplitBottomBase : public ComponentBase { class ResizableSplitBottomBase : public ComponentBase {
@ -211,7 +211,7 @@ class ResizableSplitBottomBase : public ComponentBase {
} }
if (captured_mouse_) { if (captured_mouse_) {
*main_size_ = global_box_.y_max - event.mouse().y; *main_size_ = box_.y_max - event.mouse().y;
return true; return true;
} }
@ -224,7 +224,7 @@ class ResizableSplitBottomBase : public ComponentBase {
separator() | reflect(separator_box_), separator() | reflect(separator_box_),
main_->Render() | size(HEIGHT, EQUAL, *main_size_), main_->Render() | size(HEIGHT, EQUAL, *main_size_),
}) | }) |
reflect(global_box_); reflect(box_);
}; };
private: private:
@ -233,7 +233,7 @@ class ResizableSplitBottomBase : public ComponentBase {
int* const main_size_; int* const main_size_;
CapturedMouse captured_mouse_; CapturedMouse captured_mouse_;
Box separator_box_; Box separator_box_;
Box global_box_; Box box_;
}; };
} // namespace } // namespace

View File

@ -22,8 +22,13 @@ class Reflect : public Node {
void SetBox(Box box) final { void SetBox(Box box) final {
reflected_box_ = box; reflected_box_ = box;
Node::SetBox(reflected_box_); Node::SetBox(box);
children_[0]->SetBox(reflected_box_); children_[0]->SetBox(box);
}
void Render(Screen& screen) final {
reflected_box_ = Box::Intersection(screen.stencil, reflected_box_);
return Node::Render(screen);
} }
private: private: