]> git.tdb.fi Git - libs/gltk.git/blob - source/panel.cpp
Add Widget::set_focus
[libs/gltk.git] / source / panel.cpp
1 /* $Id$
2
3 This file is part of libmspgltk
4 Copyright © 2007  Mikko Rasa, Mikkosoft Productions
5 Distributed under the LGPL
6 */
7
8 #include <algorithm>
9 #include <msp/core/refptr.h>
10 #include "button.h"
11 #include "dropdown.h"
12 #include "entry.h"
13 #include "hslider.h"
14 #include "indicator.h"
15 #include "label.h"
16 #include "list.h"
17 #include "panel.h"
18 #include "part.h"
19 #include "table.h"
20 #include "toggle.h"
21 #include "vslider.h"
22
23 using namespace std;
24
25 namespace Msp {
26 namespace GLtk {
27
28 Panel::Panel(const Resources &r):
29         Widget(r),
30         pointer_focus(0),
31         pointer_grab(0),
32         input_focus(0)
33 {
34         update_style();
35 }
36
37 Panel::~Panel()
38 {
39         while(!children.empty())
40                 delete children.front();
41 }
42
43 void Panel::add(Widget &wdg)
44 {
45         set_parent(wdg, this);
46         children.push_back(&wdg);
47 }
48
49 void Panel::remove(Widget &wdg)
50 {
51         list<Widget *>::iterator i=find(children.begin(), children.end(), &wdg);
52         if(i==children.end())
53                 throw InvalidState("That Widget is not in this Panel");
54
55         if(&wdg==pointer_focus)
56                 set_pointer_focus(0, 0);
57         if(&wdg==input_focus)
58                 set_input_focus(0);
59
60         set_parent(wdg, 0);
61         children.erase(i);
62 }
63
64 void Panel::raise(Widget &wdg)
65 {
66         list<Widget *>::iterator i=find(children.begin(), children.end(), &wdg);
67         if(i==children.end())
68                 throw InvalidState("That Widget is not in this Panel");
69
70         children.erase(i);
71         children.push_back(&wdg);
72 }
73
74 void Panel::button_press(int x, int y, unsigned btn)
75 {
76         if(pointer_grab>0)
77         {
78                 const Geometry &cgeom=pointer_focus->get_geometry();
79                 pointer_focus->button_press(x-cgeom.x, y-cgeom.y, btn);
80         }
81         else if(geom.is_inside_relative(x, y))
82         {
83                 if(Widget *wdg=get_child_at(x, y))
84                 {
85                         set_pointer_focus(wdg, btn);
86                         set_input_focus(wdg);
87
88                         const Geometry &cgeom=wdg->get_geometry();
89                         wdg->button_press(x-cgeom.x, y-cgeom.y, btn);
90                 }
91         }
92 }
93
94 void Panel::button_release(int x, int y, unsigned btn)
95 {
96         if(pointer_grab>0)
97         {
98                 const Geometry &cgeom=pointer_focus->get_geometry();
99                 pointer_focus->button_release(x-cgeom.x, y-cgeom.y, btn);
100
101                 if(btn==pointer_grab)
102                         set_pointer_focus(get_child_at(x, y), 0);
103         }
104         else if(geom.is_inside_relative(x, y))
105         {
106                 if(Widget *wdg=get_child_at(x, y))
107                 {
108                         const Geometry &cgeom=wdg->get_geometry();
109                         wdg->button_release(x-cgeom.x, y-cgeom.y, btn);
110                 }
111         }
112 }
113
114 void Panel::pointer_motion(int x, int y)
115 {
116         if(pointer_grab>0)
117         {
118                 const Geometry &cgeom=pointer_focus->get_geometry();
119                 pointer_focus->pointer_motion(x-cgeom.x, y-cgeom.y);
120         }
121         else if(geom.is_inside_relative(x, y))
122         {
123                 Widget *wdg=get_child_at(x, y);
124                 set_pointer_focus(wdg, 0);
125                 if(wdg)
126                 {
127                         const Geometry &cgeom=wdg->get_geometry();
128                         wdg->pointer_motion(x-cgeom.x, y-cgeom.y);
129                 }
130         }
131 }
132
133 void Panel::pointer_leave()
134 {
135         set_pointer_focus(0, 0);
136 }
137
138 void Panel::key_press(unsigned key, unsigned mod, wchar_t ch)
139 {
140         if(input_focus)
141                 input_focus->key_press(key, mod, ch);
142 }
143
144 void Panel::key_release(unsigned key, unsigned mod)
145 {
146         if(input_focus)
147                 input_focus->key_release(key, mod);
148 }
149
150 void Panel::focus_out()
151 {
152         set_input_focus(0);
153 }
154
155 void Panel::child_hidden(Widget &wdg)
156 {
157         if(&wdg==pointer_focus)
158                 set_pointer_focus(0, 0);
159 }
160
161 void Panel::grab_pointer(Widget &wdg)
162 {
163         if(pointer_grab==0 || pointer_focus==&wdg)
164                 set_pointer_focus(&wdg, 255);
165         else
166                 throw InvalidState("Pointer is already grabbed");
167 }
168
169 void Panel::ungrab_pointer(Widget &wdg)
170 {
171         if(pointer_focus==&wdg)
172                 set_pointer_focus(0, 0);
173         else if(pointer_grab>0)
174                 throw Exception("Someone is trying to steal the pointer!");
175 }
176
177 void Panel::grab_focus(Widget &wdg)
178 {
179         list<Widget *>::iterator i=find(children.begin(), children.end(), &wdg);
180         if(i==children.end())
181                 throw InvalidState("That Widget is not in this Panel");
182         
183         set_input_focus(&wdg);
184         if(parent)
185                 parent->grab_focus(*this);
186 }
187
188 void Panel::render_special(const Part &part) const
189 {
190         if(part.get_name()=="children")
191         {
192                 for(list<Widget *>::const_iterator i=children.begin(); i!=children.end(); ++i)
193                         if((*i)->is_visible())
194                                 (*i)->render();
195         }
196 }
197
198 void Panel::set_pointer_focus(Widget *wdg, int grab)
199 {
200         if(grab>0 && !wdg)
201                 throw InvalidParameterValue("Can't grab on null widget");
202
203         if(wdg!=pointer_focus)
204         {
205                 if(pointer_focus)
206                         pointer_focus->pointer_leave();
207
208                 pointer_focus=wdg;
209
210                 if(pointer_focus)
211                         pointer_focus->pointer_enter();
212         }
213
214         pointer_grab=grab;
215 }
216
217 void Panel::set_input_focus(Widget *wdg)
218 {
219         if(wdg!=input_focus)
220         {
221                 if(input_focus)
222                         input_focus->focus_out();
223
224                 input_focus=wdg;
225
226                 if(input_focus)
227                         input_focus->focus_in();
228         }
229 }
230
231 Widget *Panel::get_child_at(int x, int y)
232 {
233         for(list<Widget *>::reverse_iterator i=children.rbegin(); i!=children.rend(); ++i)
234                 if((*i)->is_visible() && (*i)->get_geometry().is_inside(x, y))
235                         return *i;
236
237         return 0;
238 }
239
240
241 Panel::Loader::Loader(Panel &p, map<string, Widget *> &m):
242         Widget::Loader(p),
243         pnl(p),
244         wdg_map(m)
245 {
246         add("button",    &Loader::child<Button>);
247         add("dropdown",  &Loader::child<Dropdown>);
248         add("entry",     &Loader::child<Entry>);
249         add("hslider",   &Loader::child<HSlider>);
250         add("indicator", &Loader::child<Indicator>);
251         add("label",     &Loader::child<Label>);
252         add("list",      &Loader::child<List>);
253         add("panel",     &Loader::panel);
254         add("table",     &Loader::child<Table>);
255         add("toggle",    &Loader::child<Toggle>);
256         add("vslider",   &Loader::child<VSlider>);
257 }
258
259 template<typename T>
260 void Panel::Loader::child(const string &n)
261 {
262         RefPtr<T> chl=new T(pnl.res);
263         load_sub(*chl);
264         pnl.add(*chl.get());
265         wdg_map[n]=chl.release();
266 }
267
268 void Panel::Loader::panel(const string &n)
269 {
270         RefPtr<Panel> p=new Panel(pnl.res);
271         load_sub(*p, wdg_map);
272         pnl.add(*p.get());
273         wdg_map[n]=p.release();
274 }
275
276 } // namespace GLtk
277 } // namespace Msp