]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/univ/radiobox.cpp
Only send keypress events to focus widget, not its parents
[wxWidgets.git] / src / univ / radiobox.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/univ/radiobox.cpp
3// Purpose: wxRadioBox implementation
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 11.09.00
7// RCS-ID: $Id$
8// Copyright: (c) 2000 SciTech Software, Inc. (www.scitechsoft.com)
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20#include "wx/wxprec.h"
21
22#ifdef __BORLANDC__
23 #pragma hdrstop
24#endif
25
26#if wxUSE_RADIOBOX
27
28#include "wx/radiobox.h"
29
30#ifndef WX_PRECOMP
31 #include "wx/dcclient.h"
32 #include "wx/radiobut.h"
33 #include "wx/validate.h"
34 #include "wx/arrstr.h"
35#endif
36
37#include "wx/tooltip.h"
38
39#include "wx/univ/theme.h"
40#include "wx/univ/renderer.h"
41#include "wx/univ/inphand.h"
42#include "wx/univ/colschem.h"
43
44// ----------------------------------------------------------------------------
45// constants
46// ----------------------------------------------------------------------------
47
48static const int BUTTON_BORDER_X = 2;
49static const int BUTTON_BORDER_Y = 4;
50
51static const int BOX_BORDER_X = 2;
52static const int BOX_BORDER_Y = 2;
53
54// ----------------------------------------------------------------------------
55// wxRadioBox event handler
56// ----------------------------------------------------------------------------
57
58class wxRadioHookHandler : public wxEvtHandler
59{
60public:
61 wxRadioHookHandler(wxRadioBox *radio) { m_radio = radio; }
62
63 virtual bool ProcessEvent(wxEvent& event)
64 {
65 // we intercept the command events from radio buttons
66 if ( event.GetEventType() == wxEVT_COMMAND_RADIOBUTTON_SELECTED )
67 {
68 m_radio->OnRadioButton(event);
69 }
70 else if ( event.GetEventType() == wxEVT_KEY_DOWN )
71 {
72 if ( m_radio->OnKeyDown((wxKeyEvent &)event) )
73 {
74 return true;
75 }
76 }
77
78 // just pass it on
79 return GetNextHandler()->ProcessEvent(event);
80 }
81
82private:
83 wxRadioBox *m_radio;
84};
85
86// ============================================================================
87// implementation
88// ============================================================================
89
90IMPLEMENT_DYNAMIC_CLASS(wxRadioBox, wxControl)
91
92// ----------------------------------------------------------------------------
93// wxRadioBox creation
94// ----------------------------------------------------------------------------
95
96void wxRadioBox::Init()
97{
98 m_selection = -1;
99}
100
101wxRadioBox::wxRadioBox(wxWindow *parent, wxWindowID id, const wxString& title,
102 const wxPoint& pos, const wxSize& size,
103 const wxArrayString& choices,
104 int majorDim, long style,
105 const wxValidator& val, const wxString& name)
106{
107 wxCArrayString chs(choices);
108
109 Init();
110
111 (void)Create(parent, id, title, pos, size, chs.GetCount(),
112 chs.GetStrings(), majorDim, style, val, name);
113}
114
115bool wxRadioBox::Create(wxWindow *parent,
116 wxWindowID id,
117 const wxString& title,
118 const wxPoint& pos,
119 const wxSize& size,
120 const wxArrayString& choices,
121 int majorDim,
122 long style,
123 const wxValidator& val,
124 const wxString& name)
125{
126 wxCArrayString chs(choices);
127
128 return Create(parent, id, title, pos, size, chs.GetCount(),
129 chs.GetStrings(), majorDim, style, val, name);
130}
131
132bool wxRadioBox::Create(wxWindow *parent,
133 wxWindowID id,
134 const wxString& title,
135 const wxPoint& pos,
136 const wxSize& size,
137 int n,
138 const wxString *choices,
139 int majorDim,
140 long style,
141 const wxValidator& wxVALIDATOR_PARAM(val),
142 const wxString& name)
143{
144 // for compatibility with the other ports which don't handle (yet?)
145 // wxRA_LEFTTORIGHT and wxRA_TOPTOBOTTOM flags, we add them ourselves if
146 // not specified
147 if ( !(style & (wxRA_LEFTTORIGHT | wxRA_TOPTOBOTTOM)) )
148 {
149 // horizontal radiobox use left to right layout
150 if ( style & wxRA_HORIZONTAL )
151 {
152 style |= wxRA_LEFTTORIGHT;
153 }
154 else if ( style & wxRA_VERTICAL )
155 {
156 style |= wxRA_TOPTOBOTTOM;
157 }
158 else
159 {
160 wxFAIL_MSG( _T("you must specify wxRA_XXX style!") );
161
162 // use default
163 style = wxRA_HORIZONTAL | wxRA_LEFTTORIGHT;
164 }
165 }
166
167 if ( !wxStaticBox::Create(parent, id, title, pos, size, style, name) )
168 return false;
169
170#if wxUSE_VALIDATORS
171 SetValidator(val);
172#endif // wxUSE_VALIDATORS
173
174 Append(n, choices);
175
176 // majorDim default value is 0 which means make one row/column
177 SetMajorDim(majorDim == 0 ? n : majorDim, style);
178
179 if ( size == wxDefaultSize )
180 {
181 SetClientSize(DoGetBestClientSize());
182 }
183
184 // Need to move the radiobox in order to move the radio buttons
185 wxPoint actualPos = GetPosition();
186 wxSize actualSize = GetSize();
187 DoMoveWindow(actualPos.x, actualPos.y, actualSize.x, actualSize.y);
188
189 // radiobox should already have selection so select at least one item
190 SetSelection(0);
191
192 return true;
193}
194
195wxRadioBox::~wxRadioBox()
196{
197 // remove the event handlers we pushed on them from all buttons and delete
198 // the buttons themselves: this must be done as the user code expects them
199 // to disappear now and not some time later when they will be deleted by
200 // our (common) parent
201 unsigned int count = m_buttons.GetCount();
202 for ( unsigned int n = 0; n < count; n++ )
203 {
204 m_buttons[n]->PopEventHandler(true /* delete it */);
205
206 delete m_buttons[n];
207 }
208}
209
210// ----------------------------------------------------------------------------
211// wxRadioBox init
212// ----------------------------------------------------------------------------
213
214void wxRadioBox::Append(int count, const wxString *choices)
215{
216 if ( !count )
217 return;
218
219 wxWindow *parent = GetParent();
220 m_buttons.Alloc(count);
221 for ( int n = 0; n < count; n++ )
222 {
223 // make the first button in the box the start of new group by giving it
224 // wxRB_GROUP style
225 wxRadioButton *btn = new wxRadioButton(parent, wxID_ANY, choices[n],
226 wxDefaultPosition,
227 wxDefaultSize,
228 n == 0 ? wxRB_GROUP : 0);
229
230 // we want to get the events from the buttons to translate it into
231 btn->PushEventHandler(new wxRadioHookHandler(this));
232 m_buttons.Add(btn);
233 }
234}
235
236// ----------------------------------------------------------------------------
237// selection
238// ----------------------------------------------------------------------------
239
240void wxRadioBox::SetSelection(int n)
241{
242 wxCHECK_RET( IsValid(n), _T("invalid index in wxRadioBox::SetSelection") );
243
244 m_selection = n;
245
246 wxRadioButton *btn = m_buttons[n];
247
248 // the selected button is always focused in the radiobox
249 btn->SetFocus();
250
251 // this will also unselect the previously selected button in our group
252 btn->SetValue(true);
253}
254
255int wxRadioBox::GetSelection() const
256{
257 return m_selection;
258}
259
260void wxRadioBox::SendRadioEvent()
261{
262 wxCHECK_RET( m_selection != -1, _T("no active radio button") );
263
264 wxCommandEvent event(wxEVT_COMMAND_RADIOBOX_SELECTED, GetId());
265 InitCommandEvent(event);
266 event.SetInt(m_selection);
267 event.SetString(GetString(m_selection));
268
269 Command(event);
270}
271
272void wxRadioBox::OnRadioButton(wxEvent& event)
273{
274 int n = m_buttons.Index((wxRadioButton *)event.GetEventObject());
275 wxCHECK_RET( n != wxNOT_FOUND, _T("click from alien radio button") );
276
277 m_selection = n;
278
279 SendRadioEvent();
280}
281
282// ----------------------------------------------------------------------------
283// methods forwarded to the buttons
284// ----------------------------------------------------------------------------
285
286wxString wxRadioBox::GetString(unsigned int n) const
287{
288 wxCHECK_MSG( IsValid(n), wxEmptyString,
289 _T("invalid index in wxRadioBox::GetString") );
290
291 return m_buttons[n]->GetLabel();
292}
293
294void wxRadioBox::SetString(unsigned int n, const wxString& label)
295{
296 wxCHECK_RET( IsValid(n), _T("invalid index in wxRadioBox::SetString") );
297
298 m_buttons[n]->SetLabel(label);
299}
300
301bool wxRadioBox::Enable(unsigned int n, bool enable)
302{
303 wxCHECK_MSG( IsValid(n), false, _T("invalid index in wxRadioBox::Enable") );
304
305 return m_buttons[n]->Enable(enable);
306}
307
308bool wxRadioBox::IsItemEnabled(unsigned int n) const
309{
310 wxCHECK_MSG( IsValid(n), false, _T("invalid index in wxRadioBox::IsItemEnabled") );
311
312 return m_buttons[n]->IsEnabled();
313}
314
315bool wxRadioBox::Show(unsigned int n, bool show)
316{
317 wxCHECK_MSG( IsValid(n), false, _T("invalid index in wxRadioBox::Show") );
318
319 return m_buttons[n]->Show(show);
320}
321
322bool wxRadioBox::IsItemShown(unsigned int n) const
323{
324 wxCHECK_MSG( IsValid(n), false, _T("invalid index in wxRadioBox::IsItemShown") );
325
326 return m_buttons[n]->IsShown();
327}
328
329// ----------------------------------------------------------------------------
330// methods forwarded to the static box
331// ----------------------------------------------------------------------------
332
333bool wxRadioBox::Enable(bool enable)
334{
335 if ( !wxStaticBox::Enable(enable) )
336 return false;
337
338 // also enable/disable the buttons
339 const unsigned int count = m_buttons.GetCount();
340 for ( unsigned int n = 0; n < count; n++ )
341 {
342 Enable(n, enable);
343 }
344
345 return true;
346}
347
348bool wxRadioBox::Show(bool show)
349{
350 if ( !wxStaticBox::Show(show) )
351 return false;
352
353 // also show/hide the buttons
354 const unsigned int count = m_buttons.GetCount();
355 for ( unsigned int n = 0; n < count; n++ )
356 {
357 Show(n, show);
358 }
359
360 return true;
361}
362
363wxString wxRadioBox::GetLabel() const
364{
365 return wxStaticBox::GetLabel();
366}
367
368void wxRadioBox::SetLabel(const wxString& label)
369{
370 wxStaticBox::SetLabel(label);
371}
372
373#if wxUSE_TOOLTIPS
374void wxRadioBox::DoSetToolTip(wxToolTip *tooltip)
375{
376 wxControl::DoSetToolTip(tooltip);
377
378 // Also set them for all Radio Buttons
379 const unsigned int count = m_buttons.GetCount();
380 for ( unsigned int n = 0; n < count; n++ )
381 {
382 if (tooltip)
383 m_buttons[n]->SetToolTip(tooltip->GetTip());
384 else
385 m_buttons[n]->SetToolTip(NULL);
386 }
387}
388#endif // wxUSE_TOOLTIPS
389
390// ----------------------------------------------------------------------------
391// buttons positioning
392// ----------------------------------------------------------------------------
393
394wxSize wxRadioBox::GetMaxButtonSize() const
395{
396 int widthMax, heightMax, width = 0, height = 0;
397 widthMax = heightMax = 0;
398
399 const unsigned int count = GetCount();
400 for ( unsigned int n = 0; n < count; n++ )
401 {
402 m_buttons[n]->GetBestSize(&width, &height);
403
404 if ( width > widthMax )
405 widthMax = width;
406 if ( height > heightMax )
407 heightMax = height;
408 }
409
410 return wxSize(widthMax + BUTTON_BORDER_X, heightMax + BUTTON_BORDER_Y);
411}
412
413wxSize wxRadioBox::DoGetBestClientSize() const
414{
415 wxSize sizeBtn = GetMaxButtonSize();
416
417 sizeBtn.x *= GetColumnCount();
418 sizeBtn.y *= GetRowCount();
419
420 // add a border around all buttons
421 sizeBtn.x += 2*BOX_BORDER_X;
422 sizeBtn.y += 2*BOX_BORDER_Y;
423
424 // account for the area taken by static box
425 wxRect rect = GetBorderGeometry();
426 sizeBtn.x += rect.x + rect.width;
427 sizeBtn.y += rect.y + rect.height;
428
429 return sizeBtn;
430}
431
432void wxRadioBox::DoMoveWindow(int x0, int y0, int width, int height)
433{
434 wxStaticBox::DoMoveWindow(x0, y0, width, height);
435
436 wxSize sizeBtn = GetMaxButtonSize();
437 wxPoint ptOrigin = GetBoxAreaOrigin();
438 wxPoint clientOrigin = GetParent() ? GetParent()->GetClientAreaOrigin() : wxPoint(0,0);
439
440 x0 += ptOrigin.x + BOX_BORDER_X - clientOrigin.x;
441 y0 += ptOrigin.y + BOX_BORDER_Y - clientOrigin.y;
442
443 int x = x0,
444 y = y0;
445
446 const unsigned int count = GetCount();
447 for ( unsigned int n = 0; n < count; n++ )
448 {
449 m_buttons[n]->SetSize(x, y, sizeBtn.x, sizeBtn.y);
450
451 if ( GetWindowStyle() & wxRA_TOPTOBOTTOM )
452 {
453 // from top to bottom
454 if ( (n + 1) % GetRowCount() )
455 {
456 // continue in this column
457 y += sizeBtn.y;
458 }
459 else
460 {
461 // start a new column
462 x += sizeBtn.x;
463 y = y0;
464 }
465 }
466 else // wxRA_LEFTTORIGHT: mirror the code above
467 {
468 // from left to right
469 if ( (n + 1) % GetColumnCount() )
470 {
471 // continue in this row
472 x += sizeBtn.x;
473 }
474 else
475 {
476 // start a new row
477 y += sizeBtn.y;
478 x = x0;
479 }
480 }
481 }
482}
483
484// ----------------------------------------------------------------------------
485// keyboard navigation
486// ----------------------------------------------------------------------------
487
488bool wxRadioBox::OnKeyDown(wxKeyEvent& event)
489{
490 wxDirection dir;
491 switch ( event.GetKeyCode() )
492 {
493 case WXK_UP:
494 dir = wxUP;
495 break;
496
497 case WXK_LEFT:
498 dir = wxLEFT;
499 break;
500
501 case WXK_DOWN:
502 dir = wxDOWN;
503 break;
504
505 case WXK_RIGHT:
506 dir = wxRIGHT;
507 break;
508
509 default:
510 return false;
511 }
512
513 int selOld = GetSelection();
514 int selNew = GetNextItem(selOld, dir, GetWindowStyle());
515 if ( selNew != selOld )
516 {
517 SetSelection(selNew);
518
519 // emulate the button click
520 SendRadioEvent();
521 }
522
523 return true;
524}
525
526#endif // wxUSE_RADIOBOX