1 #include <msp/core/algorithm.h>
2 #include <msp/core/maputils.h>
3 #include <msp/core/refptr.h>
6 #include "draghandle.h"
11 #include "indicator.h"
16 #include "progressbar.h"
27 TypeRegistry<Panel::Loader::AddChildType, Panel::Loader &> Panel::widget_registry;
28 bool Panel::widget_registry_init_done = false;
32 input_type = INPUT_NAVIGATION;
41 Layout &Panel::get_or_create_layout()
46 layout->set_container(*this);
52 void Panel::autosize_special(const Part &part, Geometry &ageom) const
54 if(part.get_name()=="children" && layout)
55 layout->autosize(ageom);
58 void Panel::render_special(const Part &part, GL::Renderer &renderer) const
60 if(part.get_name()=="children")
62 for(const Child *c: children)
63 if(c->widget->is_visible())
64 c->widget->render(renderer);
68 bool Panel::navigate(Navigation nav)
70 if(Container::navigate(nav))
73 if(nav==NAV_UP || nav==NAV_DOWN || nav==NAV_LEFT || nav==NAV_RIGHT)
75 int nav_x = (nav==NAV_RIGHT ? 1 : nav==NAV_LEFT ? -1 : 0);
76 int nav_y = (nav==NAV_UP ? 1 : nav==NAV_DOWN ? -1 : 0);
78 int origin_x, origin_y, origin_dim;
81 const Geometry &fgeom = input_focus->get_geometry();
82 origin_x = fgeom.x+(nav_x*0.5+0.5)*fgeom.w;
83 origin_y = fgeom.y+(nav_y*0.5+0.5)*fgeom.h;
84 origin_dim = abs(nav_x)*fgeom.h+abs(nav_y)*fgeom.w;
88 origin_x = geom.w*(0.5-nav_x*0.5);
89 origin_y = geom.h*(0.5-nav_y*0.5);
90 origin_dim = abs(nav_x)*geom.h+abs(nav_y)*geom.w;
93 if(pointer_grabbed && pointer_focus==input_focus)
96 Widget *sibling = find_next_child(origin_x, origin_y, origin_dim, nav_x, nav_y);
97 if(!sibling && input_focus)
99 const Geometry &fgeom = input_focus->get_geometry();
100 origin_x -= fgeom.w*(nav_x*0.5);
101 origin_y -= fgeom.h*(nav_y*0.5);
102 sibling = find_next_child(origin_x, origin_y, origin_dim, nav_x, nav_y);
107 set_input_focus(sibling);
108 if(Panel *panel = dynamic_cast<Panel *>(sibling))
109 panel->navigate(nav);
113 else if(nav==NAV_NEXT || nav==NAV_PREVIOUS)
115 auto i = find(nav_order, input_focus);
119 if(i!=nav_order.end())
121 if(i==nav_order.end())
122 i = nav_order.begin();
126 if(i==nav_order.begin())
139 Widget *Panel::find_next_child(int origin_x, int origin_y, int origin_dim, int nav_x, int nav_y) const
141 Widget *sibling = nullptr;
143 for(const Child *c: children)
145 if(c->widget==input_focus || !c->widget->is_focusable())
148 const Geometry &cgeom = c->widget->get_geometry();
149 int dx = compute_delta(cgeom.x, cgeom.w, origin_x, origin_dim, nav_x);
150 int dy = compute_delta(cgeom.y, cgeom.h, origin_y, origin_dim, nav_y);
153 if(nav_y && nav_y*dy>=0)
154 score = nav_y*dy+abs(dx)*4;
155 else if(nav_x && nav_x*dx>=0)
156 score = nav_x*dx+abs(dy)*4;
158 if(score>=0 && (!sibling || score<best_score))
168 int Panel::compute_delta(int pos, int dim, int origin_pos, int origin_dim, int nav)
171 return pos+dim-origin_pos;
173 return pos-origin_pos;
174 else if(pos+dim<origin_pos-origin_dim/2)
175 return pos+dim+origin_dim/2-origin_pos;
176 else if(pos>origin_pos+origin_dim/2)
177 return pos-origin_pos-origin_dim/2;
182 void Panel::on_size_change()
188 void Panel::on_child_added(Widget &wdg)
190 if(wdg.get_input_type()!=INPUT_NONE)
191 nav_order.push_back(&wdg);
195 layout->add_widget(wdg);
196 signal_autosize_changed.emit();
200 void Panel::on_child_removed(Widget &wdg)
202 auto i = std::remove(nav_order.begin(), nav_order.end(), &wdg);
203 if(i!=nav_order.end())
204 nav_order.erase(i, nav_order.end());
208 layout->remove_widget(wdg);
209 signal_autosize_changed.emit();
214 Panel::Loader::Loader(Panel &p, map<string, Widget *> &m):
215 DataFile::DerivedObjectLoader<Panel, Widget::Loader>(p),
219 if(!widget_registry_init_done)
221 widget_registry_init_done = true;
222 register_child_type<Button>("button");
223 register_child_type<DragHandle>("draghandle");
224 register_child_type<Dropdown>("dropdown");
225 register_child_type<Entry>("entry");
226 register_child_type<HSlider>("hslider");
227 register_child_type<Image>("image");
228 register_child_type<Indicator>("indicator");
229 register_child_type<Label>("label");
230 register_child_type<List>("list");
231 register_child_type<Panel>("panel");
232 register_child_type<ProgressBar>("progressbar");
233 register_child_type<Toggle>("toggle");
234 register_child_type<VSlider>("vslider");
237 add("column", &Loader::arrangement<Column>);
238 add("constraint",&Loader::constraint);
239 add("expand", &Loader::expand);
240 add("ghost", &Loader::ghost);
241 add("gravity", &Loader::gravity);
242 add("grid", &Loader::grid);
243 add("layout", &Loader::layout);
244 add("row", &Loader::arrangement<Row>);
245 add("stack", &Loader::arrangement<Stack>);
246 widget_registry.invoke_all(*this);
249 Widget &Panel::Loader::get_last_widget()
252 throw logic_error("no widget loaded");
258 void Panel::Loader::arrangement()
260 T arr(obj.get_or_create_layout());
261 ArrangedLoader<T> ldr(*this, arr);
265 void Panel::Loader::constraint(Layout::ConstraintType type, const string &n)
267 Widget &src = get_last_widget();
268 Widget &tgt = *get_item(wdg_map, n);
269 obj.get_or_create_layout().add_constraint(src, type, tgt);
272 void Panel::Loader::expand(bool h, bool v)
274 obj.get_or_create_layout().set_expand(get_last_widget(), h, v);
277 void Panel::Loader::ghost(bool g)
279 obj.get_or_create_layout().set_ghost(get_last_widget(), g);
282 void Panel::Loader::gravity(int h, int v)
284 obj.get_or_create_layout().set_gravity(get_last_widget(), h, v);
287 void Panel::Loader::grid(size_t cols)
289 Grid grd(obj.get_or_create_layout(), cols);
290 ArrangedLoader<Grid> ldr(*this, grd);
294 void Panel::Loader::layout()
296 Layout::Loader ldr(obj.get_or_create_layout(), wdg_map);
301 void Panel::Loader::unnamed_child<Panel>()
303 RefPtr<Panel> pnl = new Panel();
304 load_sub(*pnl, wdg_map);
306 last_widget = pnl.release();
311 Panel::ArrangedLoader<T>::ArrangedLoader(Loader &ldr, T &arr):
314 add_auxiliary_loader(ldr);
315 add_auxiliary_loader(arr_loader);