]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: src/gtk/radiobox.cpp | |
3 | // Purpose: | |
4 | // Author: Robert Roebling | |
5 | // Id: $Id$ | |
6 | // Copyright: (c) 1998 Robert Roebling | |
7 | // Licence: wxWindows licence | |
8 | ///////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | // For compilers that support precompilation, includes "wx.h". | |
11 | #include "wx/wxprec.h" | |
12 | ||
13 | #if wxUSE_RADIOBOX | |
14 | ||
15 | #include "wx/radiobox.h" | |
16 | ||
17 | #ifndef WX_PRECOMP | |
18 | #include "wx/log.h" | |
19 | #include "wx/frame.h" | |
20 | #include "wx/dialog.h" | |
21 | #endif | |
22 | ||
23 | #if wxUSE_TOOLTIPS | |
24 | #include "wx/tooltip.h" | |
25 | #endif | |
26 | ||
27 | #include "wx/gtk/private.h" | |
28 | #include <gdk/gdkkeysyms.h> | |
29 | ||
30 | //----------------------------------------------------------------------------- | |
31 | // wxGTKRadioButtonInfo | |
32 | //----------------------------------------------------------------------------- | |
33 | // structure internally used by wxRadioBox to store its child buttons | |
34 | ||
35 | class wxGTKRadioButtonInfo : public wxObject | |
36 | { | |
37 | public: | |
38 | wxGTKRadioButtonInfo( GtkRadioButton * abutton, const wxRect & arect ) | |
39 | : button( abutton ), rect( arect ) {} | |
40 | ||
41 | GtkRadioButton * button; | |
42 | wxRect rect; | |
43 | }; | |
44 | ||
45 | //----------------------------------------------------------------------------- | |
46 | // data | |
47 | //----------------------------------------------------------------------------- | |
48 | ||
49 | #include "wx/listimpl.cpp" | |
50 | WX_DEFINE_LIST( wxRadioBoxButtonsInfoList ) | |
51 | ||
52 | extern bool g_blockEventsOnDrag; | |
53 | ||
54 | //----------------------------------------------------------------------------- | |
55 | // "clicked" | |
56 | //----------------------------------------------------------------------------- | |
57 | ||
58 | extern "C" { | |
59 | static void gtk_radiobutton_clicked_callback( GtkToggleButton *button, wxRadioBox *rb ) | |
60 | { | |
61 | if (!rb->m_hasVMT) return; | |
62 | if (g_blockEventsOnDrag) return; | |
63 | ||
64 | if (!button->active) return; | |
65 | ||
66 | wxCommandEvent event( wxEVT_COMMAND_RADIOBOX_SELECTED, rb->GetId() ); | |
67 | event.SetInt( rb->GetSelection() ); | |
68 | event.SetString( rb->GetStringSelection() ); | |
69 | event.SetEventObject( rb ); | |
70 | rb->GetEventHandler()->ProcessEvent(event); | |
71 | } | |
72 | } | |
73 | ||
74 | //----------------------------------------------------------------------------- | |
75 | // "key_press_event" | |
76 | //----------------------------------------------------------------------------- | |
77 | ||
78 | extern "C" { | |
79 | static gint gtk_radiobox_keypress_callback( GtkWidget *widget, GdkEventKey *gdk_event, wxRadioBox *rb ) | |
80 | { | |
81 | if (!rb->m_hasVMT) return FALSE; | |
82 | if (g_blockEventsOnDrag) return FALSE; | |
83 | ||
84 | if ( ((gdk_event->keyval == GDK_Tab) || | |
85 | (gdk_event->keyval == GDK_ISO_Left_Tab)) && | |
86 | rb->GetParent() && (rb->GetParent()->HasFlag( wxTAB_TRAVERSAL)) ) | |
87 | { | |
88 | wxNavigationKeyEvent new_event; | |
89 | new_event.SetEventObject( rb->GetParent() ); | |
90 | // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB | |
91 | new_event.SetDirection( (gdk_event->keyval == GDK_Tab) ); | |
92 | // CTRL-TAB changes the (parent) window, i.e. switch notebook page | |
93 | new_event.SetWindowChange( (gdk_event->state & GDK_CONTROL_MASK) ); | |
94 | new_event.SetCurrentFocus( rb ); | |
95 | return rb->GetParent()->GetEventHandler()->ProcessEvent( new_event ); | |
96 | } | |
97 | ||
98 | if ((gdk_event->keyval != GDK_Up) && | |
99 | (gdk_event->keyval != GDK_Down) && | |
100 | (gdk_event->keyval != GDK_Left) && | |
101 | (gdk_event->keyval != GDK_Right)) | |
102 | { | |
103 | return FALSE; | |
104 | } | |
105 | ||
106 | wxRadioBoxButtonsInfoList::compatibility_iterator node = rb->m_buttonsInfo.GetFirst(); | |
107 | while( node && GTK_WIDGET( node->GetData()->button ) != widget ) | |
108 | { | |
109 | node = node->GetNext(); | |
110 | } | |
111 | if (!node) | |
112 | { | |
113 | return FALSE; | |
114 | } | |
115 | ||
116 | if ((gdk_event->keyval == GDK_Up) || | |
117 | (gdk_event->keyval == GDK_Left)) | |
118 | { | |
119 | if (node == rb->m_buttonsInfo.GetFirst()) | |
120 | node = rb->m_buttonsInfo.GetLast(); | |
121 | else | |
122 | node = node->GetPrevious(); | |
123 | } | |
124 | else | |
125 | { | |
126 | if (node == rb->m_buttonsInfo.GetLast()) | |
127 | node = rb->m_buttonsInfo.GetFirst(); | |
128 | else | |
129 | node = node->GetNext(); | |
130 | } | |
131 | ||
132 | GtkWidget *button = (GtkWidget*) node->GetData()->button; | |
133 | ||
134 | gtk_widget_grab_focus( button ); | |
135 | ||
136 | return TRUE; | |
137 | } | |
138 | } | |
139 | ||
140 | extern "C" { | |
141 | static gint gtk_radiobutton_focus_in( GtkWidget *widget, | |
142 | GdkEvent *WXUNUSED(event), | |
143 | wxRadioBox *win ) | |
144 | { | |
145 | if ( win->m_lostFocus ) | |
146 | { | |
147 | // no, we didn't really lose it | |
148 | win->m_lostFocus = FALSE; | |
149 | } | |
150 | else if ( !win->m_hasFocus ) | |
151 | { | |
152 | win->m_hasFocus = true; | |
153 | ||
154 | wxFocusEvent event( wxEVT_SET_FOCUS, win->GetId() ); | |
155 | event.SetEventObject( win ); | |
156 | ||
157 | // never stop the signal emission, it seems to break the kbd handling | |
158 | // inside the radiobox | |
159 | (void)win->GetEventHandler()->ProcessEvent( event ); | |
160 | } | |
161 | ||
162 | return FALSE; | |
163 | } | |
164 | } | |
165 | ||
166 | extern "C" { | |
167 | static gint gtk_radiobutton_focus_out( GtkWidget *widget, | |
168 | GdkEvent *WXUNUSED(event), | |
169 | wxRadioBox *win ) | |
170 | { | |
171 | // wxASSERT_MSG( win->m_hasFocus, _T("got focus out without any focus in?") ); | |
172 | // Replace with a warning, else we dump core a lot! | |
173 | // if (!win->m_hasFocus) | |
174 | // wxLogWarning(_T("Radiobox got focus out without any focus in.") ); | |
175 | ||
176 | // we might have lost the focus, but may be not - it may have just gone to | |
177 | // another button in the same radiobox, so we'll check for it in the next | |
178 | // idle iteration (leave m_hasFocus == true for now) | |
179 | win->m_lostFocus = true; | |
180 | ||
181 | return FALSE; | |
182 | } | |
183 | } | |
184 | ||
185 | extern "C" { | |
186 | static void gtk_radiobutton_size_allocate( GtkWidget *widget, | |
187 | GtkAllocation * alloc, | |
188 | wxRadioBox *win ) | |
189 | { | |
190 | for ( wxRadioBoxButtonsInfoList::compatibility_iterator node = win->m_buttonsInfo.GetFirst(); | |
191 | node; | |
192 | node = node->GetNext()) | |
193 | { | |
194 | if (widget == GTK_WIDGET(node->GetData()->button)) | |
195 | { | |
196 | const wxPoint origin = win->GetPosition(); | |
197 | wxRect rect = wxRect( alloc->x - origin.x, alloc->y - origin.y, | |
198 | alloc->width, alloc->height ); | |
199 | node->GetData()->rect = rect; | |
200 | break; | |
201 | } | |
202 | } | |
203 | } | |
204 | } | |
205 | ||
206 | ||
207 | //----------------------------------------------------------------------------- | |
208 | // wxRadioBox | |
209 | //----------------------------------------------------------------------------- | |
210 | ||
211 | IMPLEMENT_DYNAMIC_CLASS(wxRadioBox,wxControl) | |
212 | ||
213 | void wxRadioBox::Init() | |
214 | { | |
215 | m_hasFocus = | |
216 | m_lostFocus = false; | |
217 | } | |
218 | ||
219 | bool wxRadioBox::Create( wxWindow *parent, wxWindowID id, | |
220 | const wxString& title, | |
221 | const wxPoint &pos, const wxSize &size, | |
222 | const wxArrayString& choices, int majorDim, | |
223 | long style, const wxValidator& validator, | |
224 | const wxString &name ) | |
225 | { | |
226 | wxCArrayString chs(choices); | |
227 | ||
228 | return Create( parent, id, title, pos, size, chs.GetCount(), | |
229 | chs.GetStrings(), majorDim, style, validator, name ); | |
230 | } | |
231 | ||
232 | bool wxRadioBox::Create( wxWindow *parent, wxWindowID id, const wxString& title, | |
233 | const wxPoint &pos, const wxSize &size, | |
234 | int n, const wxString choices[], int majorDim, | |
235 | long style, const wxValidator& validator, | |
236 | const wxString &name ) | |
237 | { | |
238 | if (!PreCreation( parent, pos, size ) || | |
239 | !CreateBase( parent, id, pos, size, style, validator, name )) | |
240 | { | |
241 | wxFAIL_MSG( wxT("wxRadioBox creation failed") ); | |
242 | return false; | |
243 | } | |
244 | ||
245 | m_widget = GTKCreateFrame(title); | |
246 | wxControl::SetLabel(title); | |
247 | if ( HasFlag(wxNO_BORDER) ) | |
248 | { | |
249 | // If we don't do this here, the wxNO_BORDER style is ignored in Show() | |
250 | gtk_frame_set_shadow_type(GTK_FRAME(m_widget), GTK_SHADOW_NONE); | |
251 | } | |
252 | ||
253 | ||
254 | // majorDim may be 0 if all trailing parameters were omitted, so don't | |
255 | // assert here but just use the correct value for it | |
256 | SetMajorDim(majorDim == 0 ? n : majorDim, style); | |
257 | ||
258 | ||
259 | unsigned int num_of_cols = GetColumnCount(); | |
260 | unsigned int num_of_rows = GetRowCount(); | |
261 | ||
262 | GtkRadioButton *rbtn = (GtkRadioButton*) NULL; | |
263 | ||
264 | GtkWidget *table = gtk_table_new( num_of_rows, num_of_cols, FALSE ); | |
265 | gtk_table_set_col_spacings( GTK_TABLE(table), 1 ); | |
266 | gtk_table_set_row_spacings( GTK_TABLE(table), 1 ); | |
267 | gtk_widget_show( table ); | |
268 | gtk_container_add( GTK_CONTAINER(m_widget), table ); | |
269 | ||
270 | wxString label; | |
271 | GSList *radio_button_group = (GSList *) NULL; | |
272 | for (unsigned int i = 0; i < (unsigned int)n; i++) | |
273 | { | |
274 | if ( i != 0 ) | |
275 | radio_button_group = gtk_radio_button_get_group( GTK_RADIO_BUTTON(rbtn) ); | |
276 | ||
277 | label.Empty(); | |
278 | for ( const wxChar *pc = choices[i]; *pc; pc++ ) | |
279 | { | |
280 | if ( *pc != wxT('&') ) | |
281 | label += *pc; | |
282 | } | |
283 | ||
284 | rbtn = GTK_RADIO_BUTTON( gtk_radio_button_new_with_label( radio_button_group, wxGTK_CONV( label ) ) ); | |
285 | gtk_widget_show( GTK_WIDGET(rbtn) ); | |
286 | ||
287 | g_signal_connect (rbtn, "key_press_event", | |
288 | G_CALLBACK (gtk_radiobox_keypress_callback), this); | |
289 | ||
290 | m_buttonsInfo.Append( new wxGTKRadioButtonInfo( rbtn, wxRect() ) ); | |
291 | ||
292 | if (HasFlag(wxRA_SPECIFY_COLS)) | |
293 | { | |
294 | int left = i%num_of_cols; | |
295 | int right = (i%num_of_cols) + 1; | |
296 | int top = i/num_of_cols; | |
297 | int bottom = (i/num_of_cols)+1; | |
298 | gtk_table_attach( GTK_TABLE(table), GTK_WIDGET(rbtn), left, right, top, bottom, | |
299 | GTK_FILL, GTK_FILL, 1, 1 ); | |
300 | } | |
301 | else | |
302 | { | |
303 | int left = i/num_of_rows; | |
304 | int right = (i/num_of_rows) + 1; | |
305 | int top = i%num_of_rows; | |
306 | int bottom = (i%num_of_rows)+1; | |
307 | gtk_table_attach( GTK_TABLE(table), GTK_WIDGET(rbtn), left, right, top, bottom, | |
308 | GTK_FILL, GTK_FILL, 1, 1 ); | |
309 | } | |
310 | ||
311 | ConnectWidget( GTK_WIDGET(rbtn) ); | |
312 | ||
313 | if (!i) | |
314 | gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(rbtn), TRUE ); | |
315 | ||
316 | g_signal_connect (rbtn, "clicked", | |
317 | G_CALLBACK (gtk_radiobutton_clicked_callback), this); | |
318 | g_signal_connect (rbtn, "focus_in_event", | |
319 | G_CALLBACK (gtk_radiobutton_focus_in), this); | |
320 | g_signal_connect (rbtn, "focus_out_event", | |
321 | G_CALLBACK (gtk_radiobutton_focus_out), this); | |
322 | g_signal_connect (rbtn, "size_allocate", | |
323 | G_CALLBACK (gtk_radiobutton_size_allocate), this); | |
324 | } | |
325 | ||
326 | m_parent->DoAddChild( this ); | |
327 | ||
328 | PostCreation(size); | |
329 | ||
330 | return true; | |
331 | } | |
332 | ||
333 | wxRadioBox::~wxRadioBox() | |
334 | { | |
335 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
336 | while (node) | |
337 | { | |
338 | GtkWidget *button = GTK_WIDGET( node->GetData()->button ); | |
339 | gtk_widget_destroy( button ); | |
340 | node = node->GetNext(); | |
341 | } | |
342 | WX_CLEAR_LIST( wxRadioBoxButtonsInfoList, m_buttonsInfo ); | |
343 | } | |
344 | ||
345 | bool wxRadioBox::Show( bool show ) | |
346 | { | |
347 | wxCHECK_MSG( m_widget != NULL, false, wxT("invalid radiobox") ); | |
348 | ||
349 | if (!wxControl::Show(show)) | |
350 | { | |
351 | // nothing to do | |
352 | return false; | |
353 | } | |
354 | ||
355 | if ( HasFlag(wxNO_BORDER) ) | |
356 | gtk_widget_hide( m_widget ); | |
357 | ||
358 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
359 | while (node) | |
360 | { | |
361 | GtkWidget *button = GTK_WIDGET( node->GetData()->button ); | |
362 | ||
363 | if (show) | |
364 | gtk_widget_show( button ); | |
365 | else | |
366 | gtk_widget_hide( button ); | |
367 | ||
368 | node = node->GetNext(); | |
369 | } | |
370 | ||
371 | return true; | |
372 | } | |
373 | ||
374 | void wxRadioBox::SetFocus() | |
375 | { | |
376 | wxCHECK_RET( m_widget != NULL, wxT("invalid radiobox") ); | |
377 | ||
378 | if (m_buttonsInfo.GetCount() == 0) return; | |
379 | ||
380 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
381 | while (node) | |
382 | { | |
383 | GtkToggleButton *button = GTK_TOGGLE_BUTTON( node->GetData()->button ); | |
384 | if (button->active) | |
385 | { | |
386 | gtk_widget_grab_focus( GTK_WIDGET(button) ); | |
387 | return; | |
388 | } | |
389 | node = node->GetNext(); | |
390 | } | |
391 | } | |
392 | ||
393 | void wxRadioBox::SetSelection( int n ) | |
394 | { | |
395 | wxCHECK_RET( m_widget != NULL, wxT("invalid radiobox") ); | |
396 | ||
397 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( n ); | |
398 | ||
399 | wxCHECK_RET( node, wxT("radiobox wrong index") ); | |
400 | ||
401 | GtkToggleButton *button = GTK_TOGGLE_BUTTON( node->GetData()->button ); | |
402 | ||
403 | GtkDisableEvents(); | |
404 | ||
405 | gtk_toggle_button_set_active( button, 1 ); | |
406 | ||
407 | GtkEnableEvents(); | |
408 | } | |
409 | ||
410 | int wxRadioBox::GetSelection(void) const | |
411 | { | |
412 | wxCHECK_MSG( m_widget != NULL, wxNOT_FOUND, wxT("invalid radiobox") ); | |
413 | ||
414 | int count = 0; | |
415 | ||
416 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
417 | while (node) | |
418 | { | |
419 | GtkToggleButton *button = GTK_TOGGLE_BUTTON( node->GetData()->button ); | |
420 | if (button->active) return count; | |
421 | count++; | |
422 | node = node->GetNext(); | |
423 | } | |
424 | ||
425 | wxFAIL_MSG( wxT("wxRadioBox none selected") ); | |
426 | ||
427 | return wxNOT_FOUND; | |
428 | } | |
429 | ||
430 | wxString wxRadioBox::GetString(unsigned int n) const | |
431 | { | |
432 | wxCHECK_MSG( m_widget != NULL, wxEmptyString, wxT("invalid radiobox") ); | |
433 | ||
434 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( n ); | |
435 | ||
436 | wxCHECK_MSG( node, wxEmptyString, wxT("radiobox wrong index") ); | |
437 | ||
438 | GtkLabel *label = GTK_LABEL(GTK_BIN(node->GetData()->button)->child); | |
439 | ||
440 | wxString str( wxGTK_CONV_BACK( gtk_label_get_text(label) ) ); | |
441 | ||
442 | return str; | |
443 | } | |
444 | ||
445 | void wxRadioBox::SetLabel( const wxString& label ) | |
446 | { | |
447 | wxCHECK_RET( m_widget != NULL, wxT("invalid radiobox") ); | |
448 | ||
449 | GTKSetLabelForFrame(GTK_FRAME(m_widget), label); | |
450 | } | |
451 | ||
452 | void wxRadioBox::SetString(unsigned int item, const wxString& label) | |
453 | { | |
454 | wxCHECK_RET( m_widget != NULL, wxT("invalid radiobox") ); | |
455 | ||
456 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( item ); | |
457 | ||
458 | wxCHECK_RET( node, wxT("radiobox wrong index") ); | |
459 | ||
460 | GtkLabel *g_label = GTK_LABEL(GTK_BIN(node->GetData()->button)->child); | |
461 | ||
462 | gtk_label_set_text( g_label, wxGTK_CONV( label ) ); | |
463 | } | |
464 | ||
465 | bool wxRadioBox::Enable( bool enable ) | |
466 | { | |
467 | if ( !wxControl::Enable( enable ) ) | |
468 | return false; | |
469 | ||
470 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
471 | while (node) | |
472 | { | |
473 | GtkButton *button = GTK_BUTTON( node->GetData()->button ); | |
474 | GtkLabel *label = GTK_LABEL(GTK_BIN(button)->child); | |
475 | ||
476 | gtk_widget_set_sensitive( GTK_WIDGET(button), enable ); | |
477 | gtk_widget_set_sensitive( GTK_WIDGET(label), enable ); | |
478 | node = node->GetNext(); | |
479 | } | |
480 | ||
481 | return true; | |
482 | } | |
483 | ||
484 | bool wxRadioBox::Enable(unsigned int item, bool enable) | |
485 | { | |
486 | wxCHECK_MSG( m_widget != NULL, false, wxT("invalid radiobox") ); | |
487 | ||
488 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( item ); | |
489 | ||
490 | wxCHECK_MSG( node, false, wxT("radiobox wrong index") ); | |
491 | ||
492 | GtkButton *button = GTK_BUTTON( node->GetData()->button ); | |
493 | GtkLabel *label = GTK_LABEL(GTK_BIN(button)->child); | |
494 | ||
495 | gtk_widget_set_sensitive( GTK_WIDGET(button), enable ); | |
496 | gtk_widget_set_sensitive( GTK_WIDGET(label), enable ); | |
497 | ||
498 | return true; | |
499 | } | |
500 | ||
501 | bool wxRadioBox::IsItemEnabled(unsigned int item) const | |
502 | { | |
503 | wxCHECK_MSG( m_widget != NULL, false, wxT("invalid radiobox") ); | |
504 | ||
505 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( item ); | |
506 | ||
507 | wxCHECK_MSG( node, false, wxT("radiobox wrong index") ); | |
508 | ||
509 | GtkButton *button = GTK_BUTTON( node->GetData()->button ); | |
510 | ||
511 | // don't use GTK_WIDGET_IS_SENSITIVE() here, we want to return true even if | |
512 | // the parent radiobox is disabled | |
513 | return GTK_WIDGET_SENSITIVE(GTK_WIDGET(button)); | |
514 | } | |
515 | ||
516 | bool wxRadioBox::Show(unsigned int item, bool show) | |
517 | { | |
518 | wxCHECK_MSG( m_widget != NULL, false, wxT("invalid radiobox") ); | |
519 | ||
520 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( item ); | |
521 | ||
522 | wxCHECK_MSG( node, false, wxT("radiobox wrong index") ); | |
523 | ||
524 | GtkWidget *button = GTK_WIDGET( node->GetData()->button ); | |
525 | ||
526 | if (show) | |
527 | gtk_widget_show( button ); | |
528 | else | |
529 | gtk_widget_hide( button ); | |
530 | ||
531 | return true; | |
532 | } | |
533 | ||
534 | bool wxRadioBox::IsItemShown(unsigned int item) const | |
535 | { | |
536 | wxCHECK_MSG( m_widget != NULL, false, wxT("invalid radiobox") ); | |
537 | ||
538 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.Item( item ); | |
539 | ||
540 | wxCHECK_MSG( node, false, wxT("radiobox wrong index") ); | |
541 | ||
542 | GtkButton *button = GTK_BUTTON( node->GetData()->button ); | |
543 | ||
544 | return GTK_WIDGET_VISIBLE(GTK_WIDGET(button)); | |
545 | } | |
546 | ||
547 | unsigned int wxRadioBox::GetCount() const | |
548 | { | |
549 | return m_buttonsInfo.GetCount(); | |
550 | } | |
551 | ||
552 | void wxRadioBox::GtkDisableEvents() | |
553 | { | |
554 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
555 | while (node) | |
556 | { | |
557 | g_signal_handlers_disconnect_by_func (node->GetData()->button, | |
558 | (gpointer) gtk_radiobutton_clicked_callback, | |
559 | this); | |
560 | ||
561 | node = node->GetNext(); | |
562 | } | |
563 | } | |
564 | ||
565 | void wxRadioBox::GtkEnableEvents() | |
566 | { | |
567 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
568 | while (node) | |
569 | { | |
570 | g_signal_connect (node->GetData()->button, "clicked", | |
571 | G_CALLBACK (gtk_radiobutton_clicked_callback), this); | |
572 | ||
573 | node = node->GetNext(); | |
574 | } | |
575 | } | |
576 | ||
577 | void wxRadioBox::DoApplyWidgetStyle(GtkRcStyle *style) | |
578 | { | |
579 | GTKFrameApplyWidgetStyle(GTK_FRAME(m_widget), style); | |
580 | ||
581 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
582 | while (node) | |
583 | { | |
584 | GtkWidget *widget = GTK_WIDGET( node->GetData()->button ); | |
585 | ||
586 | gtk_widget_modify_style( widget, style ); | |
587 | gtk_widget_modify_style(GTK_BIN(widget)->child, style); | |
588 | ||
589 | node = node->GetNext(); | |
590 | } | |
591 | } | |
592 | ||
593 | bool wxRadioBox::GTKWidgetNeedsMnemonic() const | |
594 | { | |
595 | return true; | |
596 | } | |
597 | ||
598 | void wxRadioBox::GTKWidgetDoSetMnemonic(GtkWidget* w) | |
599 | { | |
600 | GTKFrameSetMnemonicWidget(GTK_FRAME(m_widget), w); | |
601 | } | |
602 | ||
603 | #if wxUSE_TOOLTIPS | |
604 | void wxRadioBox::ApplyToolTip(GtkTooltips * WXUNUSED(tips), const wxChar *tip) | |
605 | { | |
606 | // set this tooltip for all radiobuttons which don't have their own tips | |
607 | unsigned n = 0; | |
608 | for ( wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
609 | node; | |
610 | node = node->GetNext(), n++ ) | |
611 | { | |
612 | if ( !GetItemToolTip(n) ) | |
613 | { | |
614 | wxToolTip::Apply(GTK_WIDGET(node->GetData()->button), | |
615 | wxConvCurrent->cWX2MB(tip)); | |
616 | } | |
617 | } | |
618 | } | |
619 | ||
620 | void wxRadioBox::DoSetItemToolTip(unsigned int n, wxToolTip *tooltip) | |
621 | { | |
622 | wxCharBuffer buf; | |
623 | if ( !tooltip ) | |
624 | tooltip = GetToolTip(); | |
625 | if ( tooltip ) | |
626 | buf = wxGTK_CONV(tooltip->GetTip()); | |
627 | ||
628 | wxToolTip::Apply(GTK_WIDGET(m_buttonsInfo[n]->button), buf); | |
629 | } | |
630 | ||
631 | #endif // wxUSE_TOOLTIPS | |
632 | ||
633 | GdkWindow *wxRadioBox::GTKGetWindow(wxArrayGdkWindows& windows) const | |
634 | { | |
635 | windows.push_back(m_widget->window); | |
636 | ||
637 | wxRadioBoxButtonsInfoList::compatibility_iterator node = m_buttonsInfo.GetFirst(); | |
638 | while (node) | |
639 | { | |
640 | GtkWidget *button = GTK_WIDGET( node->GetData()->button ); | |
641 | ||
642 | windows.push_back(button->window); | |
643 | ||
644 | node = node->GetNext(); | |
645 | } | |
646 | ||
647 | return NULL; | |
648 | } | |
649 | ||
650 | void wxRadioBox::OnInternalIdle() | |
651 | { | |
652 | wxControl::OnInternalIdle(); | |
653 | ||
654 | if ( m_lostFocus ) | |
655 | { | |
656 | m_hasFocus = false; | |
657 | m_lostFocus = false; | |
658 | ||
659 | wxFocusEvent event( wxEVT_KILL_FOCUS, GetId() ); | |
660 | event.SetEventObject( this ); | |
661 | ||
662 | (void)GetEventHandler()->ProcessEvent( event ); | |
663 | } | |
664 | } | |
665 | ||
666 | // static | |
667 | wxVisualAttributes | |
668 | wxRadioBox::GetClassDefaultAttributes(wxWindowVariant WXUNUSED(variant)) | |
669 | { | |
670 | wxVisualAttributes attr; | |
671 | // NB: we need toplevel window so that GTK+ can find the right style | |
672 | GtkWidget *wnd = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
673 | GtkWidget* widget = gtk_radio_button_new_with_label(NULL, ""); | |
674 | gtk_container_add(GTK_CONTAINER(wnd), widget); | |
675 | attr = GetDefaultAttributesFromGTKWidget(widget); | |
676 | gtk_widget_destroy(wnd); | |
677 | return attr; | |
678 | } | |
679 | ||
680 | int wxRadioBox::GetItemFromPoint(const wxPoint& point) const | |
681 | { | |
682 | const wxPoint pt = ScreenToClient(point); | |
683 | unsigned n = 0; | |
684 | for ( wxRadioBoxButtonsInfoList::compatibility_iterator | |
685 | node = m_buttonsInfo.GetFirst(); node; node = node->GetNext(), n++ ) | |
686 | { | |
687 | if ( m_buttonsInfo[n]->rect.Contains(pt) ) | |
688 | return n; | |
689 | } | |
690 | ||
691 | return wxNOT_FOUND; | |
692 | } | |
693 | ||
694 | #endif // wxUSE_RADIOBOX |