HikoGUI
A low latency retained GUI
Loading...
Searching...
No Matches
scroll_widget.hpp
1// Copyright Take Vos 2021.
2// Distributed under the Boost Software License, Version 1.0.
3// (See accompanying file LICENSE_1_0.txt or copy at https://www.boost.org/LICENSE_1_0.txt)
4
5#pragma once
6
7#include "widget.hpp"
8#include "scroll_bar_widget.hpp"
9#include "scroll_aperture_widget.hpp"
10#include "scroll_delegate.hpp"
11#include "../GUI/gui_window.hpp"
12#include "../geometry/axis.hpp"
13
14namespace hi::inline v1 {
15
43template<axis Axis = axis::both, bool ControlsWindow = false>
44class scroll_widget final : public widget {
45public:
46 using super = widget;
48
49 static constexpr hi::axis axis = Axis;
50 static constexpr bool controls_window = ControlsWindow;
51
53 {
54 if (auto delegate = _delegate.lock()) {
55 delegate->deinit(*this);
56 }
57 }
58
66 scroll_widget(gui_window &window, widget *parent, std::weak_ptr<delegate_type> delegate = {}) noexcept :
67 super(window, parent), _delegate(std::move(delegate))
68 {
69 hi_axiom(is_gui_thread());
70 hi_axiom(parent);
71
72 // The scroll-widget will not draw itself, only its selected content.
73 semantic_layer = parent->semantic_layer;
74
75 _aperture = std::make_unique<scroll_aperture_widget>(window, this);
76 _horizontal_scroll_bar = std::make_unique<horizontal_scroll_bar_widget>(
77 window, this, _aperture->content_width, _aperture->aperture_width, _aperture->offset_x);
78 _vertical_scroll_bar = std::make_unique<vertical_scroll_bar_widget>(
79 window, this, _aperture->content_height, _aperture->aperture_height, _aperture->offset_y);
80
81 if (auto d = _delegate.lock()) {
82 d->init(*this);
83 }
84 }
85
95 template<typename Widget, typename... Args>
96 Widget &make_widget(Args &&...args) noexcept
97 {
98 return _aperture->make_widget<Widget>(std::forward<Args>(args)...);
99 }
100
102 [[nodiscard]] generator<widget *> children() const noexcept override
103 {
104 co_yield _aperture.get();
105 co_yield _vertical_scroll_bar.get();
106 co_yield _horizontal_scroll_bar.get();
107 }
108
109 widget_constraints const &set_constraints() noexcept override
110 {
111 _layout = {};
112 hilet aperture_constraints = _aperture->set_constraints();
113 hilet horizontal_constraints = _horizontal_scroll_bar->set_constraints();
114 hilet vertical_constraints = _vertical_scroll_bar->set_constraints();
115
116 _constraints = aperture_constraints;
117
118 // When there are scrollbars the widget minimum size becomes basically zero.
119 // However we should at least have enough room to fit in the scroll-bars length-wise.
120 if constexpr (any(axis & axis::horizontal)) {
121 _constraints.minimum.width() = horizontal_constraints.minimum.width();
122 inplace_max(_constraints.preferred.width(), horizontal_constraints.minimum.width());
123 inplace_max(_constraints.maximum.width(), horizontal_constraints.minimum.width());
124 }
125 if constexpr (any(axis & axis::vertical)) {
126 _constraints.minimum.height() = vertical_constraints.minimum.height();
127 inplace_max(_constraints.preferred.height(), vertical_constraints.minimum.height());
128 inplace_max(_constraints.maximum.height(), vertical_constraints.minimum.height());
129 }
130
131 // Make room for the thickness of the scroll-bars.
132 if constexpr (any(axis & axis::horizontal)) {
133 _constraints.minimum.height() += horizontal_constraints.preferred.height();
134 _constraints.preferred.height() += horizontal_constraints.preferred.height();
135 _constraints.maximum.height() += horizontal_constraints.preferred.height();
136 }
137 if constexpr (any(axis & axis::vertical)) {
138 _constraints.minimum.width() += vertical_constraints.preferred.width();
139 _constraints.preferred.width() += vertical_constraints.preferred.width();
140 _constraints.maximum.width() += vertical_constraints.preferred.width();
141 }
142 return _constraints;
143 }
144
145 void set_layout(widget_layout const &layout) noexcept override
146 {
147 if (compare_store(_layout, layout)) {
148 _horizontal_scroll_bar->visible = _aperture->x_axis_scrolls() and any(axis & axis::horizontal);
149 _vertical_scroll_bar->visible = _aperture->y_axis_scrolls() and any(axis & axis::vertical);
150 hilet both_bars_visible = *_horizontal_scroll_bar->visible and *_vertical_scroll_bar->visible;
151
152 hilet vertical_scroll_bar_width = _vertical_scroll_bar->constraints().preferred.width();
153 hilet horizontal_scroll_bar_height = _horizontal_scroll_bar->constraints().preferred.height();
154
155 // The aperture size grows to fill the size of the layout.
156 hilet aperture_size = extent2{
157 *_vertical_scroll_bar->visible ? layout.width() - vertical_scroll_bar_width : layout.width(),
158 *_horizontal_scroll_bar->visible ? layout.height() - horizontal_scroll_bar_height : layout.height()};
159 hilet aperture_offset = point2{0.0f, *_horizontal_scroll_bar->visible ? horizontal_scroll_bar_height : 0.0f};
160 _aperture_rectangle = aarectangle{aperture_offset, aperture_size};
161
162 // The length of the scroll-bar is the full length of the widget, or just the length of the aperture depending
163 // if the counter-part scroll-bar is visible.
164 hilet horizontal_scroll_bar_size =
165 extent2{both_bars_visible ? aperture_size.width() : layout.width(), horizontal_scroll_bar_height};
166 hilet vertical_scroll_bar_size =
167 extent2{vertical_scroll_bar_width, both_bars_visible ? aperture_size.height() : layout.height()};
168
169 _vertical_scroll_bar_rectangle = aarectangle{
170 point2{layout.width() - vertical_scroll_bar_size.width(), layout.height() - vertical_scroll_bar_size.height()},
171 vertical_scroll_bar_size};
172
173 _horizontal_scroll_bar_rectangle = aarectangle{point2{0.0f, 0.0f}, horizontal_scroll_bar_size};
174
175 if constexpr (controls_window) {
176 window.set_resize_border_priority(
177 true, not *_vertical_scroll_bar->visible, not *_horizontal_scroll_bar->visible, true);
178 }
179 }
180
181 _aperture->set_layout(layout.transform(_aperture_rectangle));
182 if (*_vertical_scroll_bar->visible) {
183 _vertical_scroll_bar->set_layout(layout.transform(_vertical_scroll_bar_rectangle));
184 }
185 if (*_horizontal_scroll_bar->visible) {
186 _horizontal_scroll_bar->set_layout(layout.transform(_horizontal_scroll_bar_rectangle));
187 }
188 }
189
190 void draw(draw_context const &context) noexcept
191 {
192 if (*visible) {
193 _vertical_scroll_bar->draw(context);
194 _horizontal_scroll_bar->draw(context);
195 _aperture->draw(context);
196 }
197 }
198
199 [[nodiscard]] hitbox hitbox_test(point3 position) const noexcept override
200 {
201 hi_axiom(is_gui_thread());
202
203 if (*visible and *enabled) {
204 auto r = _aperture->hitbox_test_from_parent(position);
205 r = _horizontal_scroll_bar->hitbox_test_from_parent(position, r);
206 r = _vertical_scroll_bar->hitbox_test_from_parent(position, r);
207
208 if (layout().contains(position)) {
209 r = std::max(r, hitbox{this, position});
210 }
211 return r;
212
213 } else {
214 return {};
215 }
216 }
217 // @endprivatesection
218private:
220
221 aarectangle _aperture_rectangle;
223
224 aarectangle _horizontal_scroll_bar_rectangle;
226
227 aarectangle _vertical_scroll_bar_rectangle;
229};
230
231template<bool ControlsWindow = false>
232using vertical_scroll_widget = scroll_widget<axis::vertical, ControlsWindow>;
233
234template<bool ControlsWindow = false>
235using horizontal_scroll_widget = scroll_widget<axis::horizontal, ControlsWindow>;
236
237} // namespace hi::inline v1
#define hilet
Invariant should be the default for variables.
Definition required.hpp:23
STL namespace.
A return value for a generator-function.
Definition generator.hpp:27
constexpr float & height() noexcept
Access the y-as-height element from the extent.
Definition extent.hpp:151
constexpr float & width() noexcept
Access the x-as-width element from the extent.
Definition extent.hpp:140
Definition gui_window.hpp:40
The scroll widget allows a content widget to be shown in less space than is required.
Definition scroll_widget.hpp:44
scroll_widget(gui_window &window, widget *parent, std::weak_ptr< delegate_type > delegate={}) noexcept
Constructs an empty scroll widget.
Definition scroll_widget.hpp:66
Widget & make_widget(Args &&...args) noexcept
Add a content widget directly to this scroll widget.
Definition scroll_widget.hpp:96
Definition scroll_delegate.hpp:16
An interactive graphical object as part of the user-interface.
Definition widget.hpp:40
int semantic_layer
The draw layer of the widget.
Definition widget.hpp:87
T max(T... args)