1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/radiobox.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  15 #include "wx/radiobox.h" 
  18     #include "wx/tooltip.h" 
  22 #include "wx/gtk/private.h" 
  23 #include "wx/gtk/private/gtk2-compat.h" 
  25 #include <gdk/gdkkeysyms.h> 
  26 #if GTK_CHECK_VERSION(3,0,0) 
  27 #include <gdk/gdkkeysyms-compat.h> 
  30 //----------------------------------------------------------------------------- 
  31 // wxGTKRadioButtonInfo 
  32 //----------------------------------------------------------------------------- 
  33 // structure internally used by wxRadioBox to store its child buttons 
  35 class wxGTKRadioButtonInfo 
: public wxObject
 
  38     wxGTKRadioButtonInfo( GtkRadioButton 
* abutton
, const wxRect 
& arect 
) 
  39     : button( abutton 
), rect( arect 
) {} 
  41     GtkRadioButton 
* button
; 
  45 //----------------------------------------------------------------------------- 
  47 //----------------------------------------------------------------------------- 
  49 #include "wx/listimpl.cpp" 
  50 WX_DEFINE_LIST( wxRadioBoxButtonsInfoList 
) 
  52 extern bool          g_blockEventsOnDrag
; 
  54 //----------------------------------------------------------------------------- 
  56 //----------------------------------------------------------------------------- 
  59 static void gtk_radiobutton_clicked_callback( GtkToggleButton 
*button
, wxRadioBox 
*rb 
) 
  61     if (!rb
->m_hasVMT
) return; 
  62     if (g_blockEventsOnDrag
) return; 
  64     if (!gtk_toggle_button_get_active(button
)) return; 
  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
->HandleWindowEvent(event
); 
  74 //----------------------------------------------------------------------------- 
  76 //----------------------------------------------------------------------------- 
  79 static gint 
gtk_radiobox_keypress_callback( GtkWidget 
*widget
, GdkEventKey 
*gdk_event
, wxRadioBox 
*rb 
) 
  81     if (!rb
->m_hasVMT
) return FALSE
; 
  82     if (g_blockEventsOnDrag
) return FALSE
; 
  84     if ( ((gdk_event
->keyval 
== GDK_Tab
) || 
  85           (gdk_event
->keyval 
== GDK_ISO_Left_Tab
)) && 
  86          rb
->GetParent() && (rb
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) ) 
  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
) != 0 ); 
  94         new_event
.SetCurrentFocus( rb 
); 
  95         return rb
->GetParent()->HandleWindowEvent(new_event
); 
  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
)) 
 106     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= rb
->m_buttonsInfo
.GetFirst(); 
 107     while( node 
&& GTK_WIDGET( node
->GetData()->button 
) != widget 
) 
 109         node 
= node
->GetNext(); 
 116     if ((gdk_event
->keyval 
== GDK_Up
) || 
 117         (gdk_event
->keyval 
== GDK_Left
)) 
 119         if (node 
== rb
->m_buttonsInfo
.GetFirst()) 
 120             node 
= rb
->m_buttonsInfo
.GetLast(); 
 122             node 
= node
->GetPrevious(); 
 126         if (node 
== rb
->m_buttonsInfo
.GetLast()) 
 127             node 
= rb
->m_buttonsInfo
.GetFirst(); 
 129             node 
= node
->GetNext(); 
 132     GtkWidget 
*button 
= (GtkWidget
*) node
->GetData()->button
; 
 134     gtk_widget_grab_focus( button 
); 
 141 static gint 
gtk_radiobutton_focus_out( GtkWidget 
* WXUNUSED(widget
), 
 142                                        GdkEventFocus 
*WXUNUSED(event
), 
 145     // NB: This control is composed of several GtkRadioButton widgets and 
 146     //     when focus changes from one of them to another in the same 
 147     //     wxRadioBox, we get a focus-out event followed by focus-in for 
 148     //     another GtkRadioButton owned by the same control. We don't want 
 149     //     to generate two spurious wxEVT_SET_FOCUS events in this case, 
 150     //     so we defer sending wx events until idle time. 
 151     win
->GTKHandleFocusOut(); 
 153     // never stop the signal emission, it seems to break the kbd handling 
 154     // inside the radiobox 
 160 static gint 
gtk_radiobutton_focus_in( GtkWidget 
* WXUNUSED(widget
), 
 161                                       GdkEventFocus 
*WXUNUSED(event
), 
 164     win
->GTKHandleFocusIn(); 
 166     // never stop the signal emission, it seems to break the kbd handling 
 167     // inside the radiobox 
 173 static void gtk_radiobutton_size_allocate( GtkWidget 
*widget
, 
 174                                            GtkAllocation 
* alloc
, 
 177     for ( wxRadioBoxButtonsInfoList::compatibility_iterator node 
= win
->m_buttonsInfo
.GetFirst(); 
 179           node 
= node
->GetNext()) 
 181         if (widget 
== GTK_WIDGET(node
->GetData()->button
)) 
 183             const wxPoint origin 
= win
->GetPosition(); 
 184             wxRect rect 
= wxRect( alloc
->x 
- origin
.x
, alloc
->y 
- origin
.y
, 
 185                                   alloc
->width
, alloc
->height 
); 
 186             node
->GetData()->rect 
= rect
; 
 194 //----------------------------------------------------------------------------- 
 196 //----------------------------------------------------------------------------- 
 198 IMPLEMENT_DYNAMIC_CLASS(wxRadioBox
,wxControl
) 
 200 bool wxRadioBox::Create( wxWindow 
*parent
, wxWindowID id
, 
 201                          const wxString
& title
, 
 202                          const wxPoint 
&pos
, const wxSize 
&size
, 
 203                          const wxArrayString
& choices
, int majorDim
, 
 204                          long style
, const wxValidator
& validator
, 
 205                          const wxString 
&name 
) 
 207     wxCArrayString 
chs(choices
); 
 209     return Create( parent
, id
, title
, pos
, size
, chs
.GetCount(), 
 210                    chs
.GetStrings(), majorDim
, style
, validator
, name 
); 
 213 bool wxRadioBox::Create( wxWindow 
*parent
, wxWindowID id
, const wxString
& title
, 
 214                          const wxPoint 
&pos
, const wxSize 
&size
, 
 215                          int n
, const wxString choices
[], int majorDim
, 
 216                          long style
, const wxValidator
& validator
, 
 217                          const wxString 
&name 
) 
 219     if (!PreCreation( parent
, pos
, size 
) || 
 220         !CreateBase( parent
, id
, pos
, size
, style
, validator
, name 
)) 
 222         wxFAIL_MSG( wxT("wxRadioBox creation failed") ); 
 226     m_widget 
= GTKCreateFrame(title
); 
 227     g_object_ref(m_widget
); 
 228     wxControl::SetLabel(title
); 
 229     if ( HasFlag(wxNO_BORDER
) ) 
 231         // If we don't do this here, the wxNO_BORDER style is ignored in Show() 
 232         gtk_frame_set_shadow_type(GTK_FRAME(m_widget
), GTK_SHADOW_NONE
); 
 236     // majorDim may be 0 if all trailing parameters were omitted, so don't 
 237     // assert here but just use the correct value for it 
 238     SetMajorDim(majorDim 
== 0 ? n 
: majorDim
, style
); 
 241     unsigned int num_of_cols 
= GetColumnCount(); 
 242     unsigned int num_of_rows 
= GetRowCount(); 
 244     GtkRadioButton 
*rbtn 
= NULL
; 
 246     GtkWidget 
*table 
= gtk_table_new( num_of_rows
, num_of_cols
, FALSE 
); 
 247     gtk_table_set_col_spacings( GTK_TABLE(table
), 1 ); 
 248     gtk_table_set_row_spacings( GTK_TABLE(table
), 1 ); 
 249     gtk_widget_show( table 
); 
 250     gtk_container_add( GTK_CONTAINER(m_widget
), table 
); 
 253     GSList 
*radio_button_group 
= NULL
; 
 254     for (unsigned int i 
= 0; i 
< (unsigned int)n
; i
++) 
 257             radio_button_group 
= gtk_radio_button_get_group( GTK_RADIO_BUTTON(rbtn
) ); 
 260         for ( wxString::const_iterator pc 
= choices
[i
].begin(); 
 261               pc 
!= choices
[i
].end(); ++pc 
) 
 263             if ( *pc 
!= wxT('&') ) 
 267         rbtn 
= GTK_RADIO_BUTTON( gtk_radio_button_new_with_label( radio_button_group
, wxGTK_CONV( label 
) ) ); 
 268         gtk_widget_show( GTK_WIDGET(rbtn
) ); 
 270         g_signal_connect (rbtn
, "key_press_event", 
 271                           G_CALLBACK (gtk_radiobox_keypress_callback
), this); 
 273         m_buttonsInfo
.Append( new wxGTKRadioButtonInfo( rbtn
, wxRect() ) ); 
 275         if (HasFlag(wxRA_SPECIFY_COLS
)) 
 277             int left 
= i%num_of_cols
; 
 278             int right 
= (i%num_of_cols
) + 1; 
 279             int top 
= i
/num_of_cols
; 
 280             int bottom 
= (i
/num_of_cols
)+1; 
 281             gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(rbtn
), left
, right
, top
, bottom
, 
 282                   GTK_FILL
, GTK_FILL
, 1, 1 ); 
 286             int left 
= i
/num_of_rows
; 
 287             int right 
= (i
/num_of_rows
) + 1; 
 288             int top 
= i%num_of_rows
; 
 289             int bottom 
= (i%num_of_rows
)+1; 
 290             gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(rbtn
), left
, right
, top
, bottom
, 
 291                   GTK_FILL
, GTK_FILL
, 1, 1 ); 
 294         ConnectWidget( GTK_WIDGET(rbtn
) ); 
 297             gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(rbtn
), TRUE 
); 
 299         g_signal_connect (rbtn
, "clicked", 
 300                           G_CALLBACK (gtk_radiobutton_clicked_callback
), this); 
 301         g_signal_connect (rbtn
, "focus_in_event", 
 302                           G_CALLBACK (gtk_radiobutton_focus_in
), this); 
 303         g_signal_connect (rbtn
, "focus_out_event", 
 304                           G_CALLBACK (gtk_radiobutton_focus_out
), this); 
 305         g_signal_connect (rbtn
, "size_allocate", 
 306                           G_CALLBACK (gtk_radiobutton_size_allocate
), this); 
 309     m_parent
->DoAddChild( this ); 
 316 wxRadioBox::~wxRadioBox() 
 318     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 321         GtkWidget 
*button 
= GTK_WIDGET( node
->GetData()->button 
); 
 322         gtk_widget_destroy( button 
); 
 323         node 
= node
->GetNext(); 
 325     WX_CLEAR_LIST( wxRadioBoxButtonsInfoList
, m_buttonsInfo 
); 
 328 bool wxRadioBox::Show( bool show 
) 
 330     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid radiobox") ); 
 332     if (!wxControl::Show(show
)) 
 338     if ( HasFlag(wxNO_BORDER
) ) 
 339         gtk_widget_hide( m_widget 
); 
 341     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 344         GtkWidget 
*button 
= GTK_WIDGET( node
->GetData()->button 
); 
 347             gtk_widget_show( button 
); 
 349             gtk_widget_hide( button 
); 
 351         node 
= node
->GetNext(); 
 357 void wxRadioBox::SetSelection( int n 
) 
 359     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid radiobox") ); 
 361     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( n 
); 
 363     wxCHECK_RET( node
, wxT("radiobox wrong index") ); 
 365     GtkToggleButton 
*button 
= GTK_TOGGLE_BUTTON( node
->GetData()->button 
); 
 369     gtk_toggle_button_set_active( button
, 1 ); 
 374 int wxRadioBox::GetSelection(void) const 
 376     wxCHECK_MSG( m_widget 
!= NULL
, wxNOT_FOUND
, wxT("invalid radiobox") ); 
 380     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 383         GtkToggleButton 
*button 
= GTK_TOGGLE_BUTTON( node
->GetData()->button 
); 
 384         if (gtk_toggle_button_get_active(button
)) return count
; 
 386         node 
= node
->GetNext(); 
 389     wxFAIL_MSG( wxT("wxRadioBox none selected") ); 
 394 wxString 
wxRadioBox::GetString(unsigned int n
) const 
 396     wxCHECK_MSG( m_widget 
!= NULL
, wxEmptyString
, wxT("invalid radiobox") ); 
 398     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( n 
); 
 400     wxCHECK_MSG( node
, wxEmptyString
, wxT("radiobox wrong index") ); 
 402     GtkLabel
* label 
= GTK_LABEL(gtk_bin_get_child(GTK_BIN(node
->GetData()->button
))); 
 404     wxString 
str( wxGTK_CONV_BACK( gtk_label_get_text(label
) ) ); 
 409 void wxRadioBox::SetLabel( const wxString
& label 
) 
 411     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid radiobox") ); 
 413     GTKSetLabelForFrame(GTK_FRAME(m_widget
), label
); 
 416 void wxRadioBox::SetString(unsigned int item
, const wxString
& label
) 
 418     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid radiobox") ); 
 420     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( item 
); 
 422     wxCHECK_RET( node
, wxT("radiobox wrong index") ); 
 424     GtkLabel
* g_label 
= GTK_LABEL(gtk_bin_get_child(GTK_BIN(node
->GetData()->button
))); 
 426     gtk_label_set_text( g_label
, wxGTK_CONV( label 
) ); 
 429 bool wxRadioBox::Enable( bool enable 
) 
 431     if ( !wxControl::Enable( enable 
) ) 
 434     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 437         GtkButton 
*button 
= GTK_BUTTON( node
->GetData()->button 
); 
 438         GtkLabel 
*label 
= GTK_LABEL(gtk_bin_get_child(GTK_BIN(button
))); 
 440         gtk_widget_set_sensitive( GTK_WIDGET(button
), enable 
); 
 441         gtk_widget_set_sensitive( GTK_WIDGET(label
), enable 
); 
 442         node 
= node
->GetNext(); 
 451 bool wxRadioBox::Enable(unsigned int item
, bool enable
) 
 453     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid radiobox") ); 
 455     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( item 
); 
 457     wxCHECK_MSG( node
, false, wxT("radiobox wrong index") ); 
 459     GtkButton 
*button 
= GTK_BUTTON( node
->GetData()->button 
); 
 460     GtkLabel 
*label 
= GTK_LABEL(gtk_bin_get_child(GTK_BIN(button
))); 
 462     gtk_widget_set_sensitive( GTK_WIDGET(button
), enable 
); 
 463     gtk_widget_set_sensitive( GTK_WIDGET(label
), enable 
); 
 468 bool wxRadioBox::IsItemEnabled(unsigned int item
) const 
 470     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid radiobox") ); 
 472     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( item 
); 
 474     wxCHECK_MSG( node
, false, wxT("radiobox wrong index") ); 
 476     GtkButton 
*button 
= GTK_BUTTON( node
->GetData()->button 
); 
 478     // don't use GTK_WIDGET_IS_SENSITIVE() here, we want to return true even if 
 479     // the parent radiobox is disabled 
 480     return gtk_widget_get_sensitive(GTK_WIDGET(button
)) != 0; 
 483 bool wxRadioBox::Show(unsigned int item
, bool show
) 
 485     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid radiobox") ); 
 487     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( item 
); 
 489     wxCHECK_MSG( node
, false, wxT("radiobox wrong index") ); 
 491     GtkWidget 
*button 
= GTK_WIDGET( node
->GetData()->button 
); 
 494         gtk_widget_show( button 
); 
 496         gtk_widget_hide( button 
); 
 501 bool wxRadioBox::IsItemShown(unsigned int item
) const 
 503     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid radiobox") ); 
 505     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.Item( item 
); 
 507     wxCHECK_MSG( node
, false, wxT("radiobox wrong index") ); 
 509     GtkButton 
*button 
= GTK_BUTTON( node
->GetData()->button 
); 
 511     return gtk_widget_get_visible(GTK_WIDGET(button
)) != 0; 
 514 unsigned int wxRadioBox::GetCount() const 
 516     return m_buttonsInfo
.GetCount(); 
 519 void wxRadioBox::GtkDisableEvents() 
 521     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 524         g_signal_handlers_block_by_func(node
->GetData()->button
, 
 525             (gpointer
)gtk_radiobutton_clicked_callback
, this); 
 527         node 
= node
->GetNext(); 
 531 void wxRadioBox::GtkEnableEvents() 
 533     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 536         g_signal_handlers_unblock_by_func(node
->GetData()->button
, 
 537             (gpointer
)gtk_radiobutton_clicked_callback
, this); 
 539         node 
= node
->GetNext(); 
 543 void wxRadioBox::DoApplyWidgetStyle(GtkRcStyle 
*style
) 
 545     GTKFrameApplyWidgetStyle(GTK_FRAME(m_widget
), style
); 
 547     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 550         GtkWidget 
*widget 
= GTK_WIDGET( node
->GetData()->button 
); 
 552         GTKApplyStyle(widget
, style
); 
 553         GTKApplyStyle(gtk_bin_get_child(GTK_BIN(widget
)), style
); 
 555         node 
= node
->GetNext(); 
 559 bool wxRadioBox::GTKWidgetNeedsMnemonic() const 
 564 void wxRadioBox::GTKWidgetDoSetMnemonic(GtkWidget
* w
) 
 566     GTKFrameSetMnemonicWidget(GTK_FRAME(m_widget
), w
); 
 570 void wxRadioBox::GTKApplyToolTip(const char* tip
) 
 572     // set this tooltip for all radiobuttons which don't have their own tips 
 574     for ( wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 576           node 
= node
->GetNext(), n
++ ) 
 578         if ( !GetItemToolTip(n
) ) 
 580             wxToolTip::GTKApply(GTK_WIDGET(node
->GetData()->button
), tip
); 
 585 void wxRadioBox::DoSetItemToolTip(unsigned int n
, wxToolTip 
*tooltip
) 
 589         tooltip 
= GetToolTip(); 
 591         buf 
= wxGTK_CONV(tooltip
->GetTip()); 
 593     wxToolTip::GTKApply(GTK_WIDGET(m_buttonsInfo
[n
]->button
), buf
); 
 596 #endif // wxUSE_TOOLTIPS 
 598 GdkWindow 
*wxRadioBox::GTKGetWindow(wxArrayGdkWindows
& windows
) const 
 600     windows
.push_back(gtk_widget_get_window(m_widget
)); 
 602     wxRadioBoxButtonsInfoList::compatibility_iterator node 
= m_buttonsInfo
.GetFirst(); 
 605         GtkWidget 
*button 
= GTK_WIDGET( node
->GetData()->button 
); 
 607         // don't put NULL pointers in the 'windows' array! 
 608         if (gtk_widget_get_window(button
)) 
 609             windows
.push_back(gtk_widget_get_window(button
)); 
 611         node 
= node
->GetNext(); 
 619 wxRadioBox::GetClassDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
 621     wxVisualAttributes attr
; 
 622     // NB: we need toplevel window so that GTK+ can find the right style 
 623     GtkWidget 
*wnd 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 624     GtkWidget
* widget 
= gtk_radio_button_new_with_label(NULL
, ""); 
 625     gtk_container_add(GTK_CONTAINER(wnd
), widget
); 
 626     attr 
= GetDefaultAttributesFromGTKWidget(widget
); 
 627     gtk_widget_destroy(wnd
); 
 631 int wxRadioBox::GetItemFromPoint(const wxPoint
& point
) const 
 633     const wxPoint pt 
= ScreenToClient(point
); 
 635     for ( wxRadioBoxButtonsInfoList::compatibility_iterator
 
 636             node 
= m_buttonsInfo
.GetFirst(); node
; node 
= node
->GetNext(), n
++ ) 
 638         if ( m_buttonsInfo
[n
]->rect
.Contains(pt
) ) 
 645 #endif // wxUSE_RADIOBOX