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"
17 #include "wx/dialog.h"
21 #include "wx/gtk/private.h"
22 #include <gdk/gdkkeysyms.h>
24 #include "wx/gtk/win_gtk.h"
26 //-----------------------------------------------------------------------------
28 //-----------------------------------------------------------------------------
30 extern void wxapp_install_idle_handler();
33 //-----------------------------------------------------------------------------
35 //-----------------------------------------------------------------------------
37 extern bool g_blockEventsOnDrag
;
38 extern wxWindowGTK
*g_delayedFocus
;
40 //-----------------------------------------------------------------------------
42 //-----------------------------------------------------------------------------
45 static void gtk_radiobutton_clicked_callback( GtkToggleButton
*button
, wxRadioBox
*rb
)
47 if (g_isIdle
) wxapp_install_idle_handler();
49 if (!rb
->m_hasVMT
) return;
50 if (g_blockEventsOnDrag
) return;
52 if (!button
->active
) return;
54 wxCommandEvent
event( wxEVT_COMMAND_RADIOBOX_SELECTED
, rb
->GetId() );
55 event
.SetInt( rb
->GetSelection() );
56 event
.SetString( rb
->GetStringSelection() );
57 event
.SetEventObject( rb
);
58 rb
->GetEventHandler()->ProcessEvent(event
);
62 //-----------------------------------------------------------------------------
64 //-----------------------------------------------------------------------------
67 static gint
gtk_radiobox_keypress_callback( GtkWidget
*widget
, GdkEventKey
*gdk_event
, wxRadioBox
*rb
)
70 wxapp_install_idle_handler();
72 if (!rb
->m_hasVMT
) return FALSE
;
73 if (g_blockEventsOnDrag
) return FALSE
;
75 if ((gdk_event
->keyval
!= GDK_Up
) &&
76 (gdk_event
->keyval
!= GDK_Down
) &&
77 (gdk_event
->keyval
!= GDK_Left
) &&
78 (gdk_event
->keyval
!= GDK_Right
))
83 wxList::compatibility_iterator node
= rb
->m_boxes
.Find( (wxObject
*) widget
);
89 g_signal_stop_emission_by_name (widget
, "key_press_event");
91 if ((gdk_event
->keyval
== GDK_Up
) ||
92 (gdk_event
->keyval
== GDK_Left
))
94 if (node
== rb
->m_boxes
.GetFirst())
95 node
= rb
->m_boxes
.GetLast();
97 node
= node
->GetPrevious();
101 if (node
== rb
->m_boxes
.GetLast())
102 node
= rb
->m_boxes
.GetFirst();
104 node
= node
->GetNext();
107 GtkWidget
*button
= (GtkWidget
*) node
->GetData();
109 gtk_widget_grab_focus( button
);
116 static gint
gtk_radiobutton_focus_in( GtkWidget
*widget
,
117 GdkEvent
*WXUNUSED(event
),
120 if ( win
->m_lostFocus
)
122 // no, we didn't really lose it
123 win
->m_lostFocus
= FALSE
;
125 else if ( !win
->m_hasFocus
)
127 win
->m_hasFocus
= true;
129 wxFocusEvent
event( wxEVT_SET_FOCUS
, win
->GetId() );
130 event
.SetEventObject( win
);
132 // never stop the signal emission, it seems to break the kbd handling
133 // inside the radiobox
134 (void)win
->GetEventHandler()->ProcessEvent( event
);
142 static gint
gtk_radiobutton_focus_out( GtkWidget
*widget
,
143 GdkEvent
*WXUNUSED(event
),
146 // wxASSERT_MSG( win->m_hasFocus, _T("got focus out without any focus in?") );
147 // Replace with a warning, else we dump core a lot!
148 // if (!win->m_hasFocus)
149 // wxLogWarning(_T("Radiobox got focus out without any focus in.") );
151 // we might have lost the focus, but may be not - it may have just gone to
152 // another button in the same radiobox, so we'll check for it in the next
153 // idle iteration (leave m_hasFocus == true for now)
154 win
->m_lostFocus
= true;
160 //-----------------------------------------------------------------------------
162 //-----------------------------------------------------------------------------
164 IMPLEMENT_DYNAMIC_CLASS(wxRadioBox
,wxControl
)
166 void wxRadioBox::Init()
169 m_acceptsFocus
= true;
175 bool wxRadioBox::Create( wxWindow
*parent
, wxWindowID id
,
176 const wxString
& title
,
177 const wxPoint
&pos
, const wxSize
&size
,
178 const wxArrayString
& choices
, int majorDim
,
179 long style
, const wxValidator
& validator
,
180 const wxString
&name
)
182 wxCArrayString
chs(choices
);
184 return Create( parent
, id
, title
, pos
, size
, chs
.GetCount(),
185 chs
.GetStrings(), majorDim
, style
, validator
, name
);
188 bool wxRadioBox::Create( wxWindow
*parent
, wxWindowID id
, const wxString
& title
,
189 const wxPoint
&pos
, const wxSize
&size
,
190 int n
, const wxString choices
[], int majorDim
,
191 long style
, const wxValidator
& validator
,
192 const wxString
&name
)
194 if (!PreCreation( parent
, pos
, size
) ||
195 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
197 wxFAIL_MSG( wxT("wxRadioBox creation failed") );
201 m_widget
= gtk_frame_new(NULL
);
204 // majorDim may be 0 if all trailing parameters were omitted, so don't
205 // assert here but just use the correct value for it
206 SetMajorDim(majorDim
== 0 ? n
: majorDim
, style
);
209 int num_of_cols
= GetColumnCount();
210 int num_of_rows
= GetRowCount();
212 GtkRadioButton
*m_radio
= (GtkRadioButton
*) NULL
;
214 GtkWidget
*table
= gtk_table_new( num_of_rows
, num_of_cols
, FALSE
);
215 gtk_table_set_col_spacings( GTK_TABLE(table
), 1 );
216 gtk_table_set_row_spacings( GTK_TABLE(table
), 1 );
217 gtk_widget_show( table
);
218 gtk_container_add( GTK_CONTAINER(m_widget
), table
);
221 GSList
*radio_button_group
= (GSList
*) NULL
;
222 for (int i
= 0; i
< n
; i
++)
225 radio_button_group
= gtk_radio_button_group( GTK_RADIO_BUTTON(m_radio
) );
228 for ( const wxChar
*pc
= choices
[i
]; *pc
; pc
++ )
230 if ( *pc
!= wxT('&') )
234 m_radio
= GTK_RADIO_BUTTON( gtk_radio_button_new_with_label( radio_button_group
, wxGTK_CONV( label
) ) );
235 gtk_widget_show( GTK_WIDGET(m_radio
) );
237 g_signal_connect (m_radio
, "key_press_event",
238 G_CALLBACK (gtk_radiobox_keypress_callback
), this);
240 m_boxes
.Append( (wxObject
*) m_radio
);
242 if (HasFlag(wxRA_SPECIFY_COLS
))
244 int left
= i%num_of_cols
;
245 int right
= (i%num_of_cols
) + 1;
246 int top
= i
/num_of_cols
;
247 int bottom
= (i
/num_of_cols
)+1;
248 gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(m_radio
), left
, right
, top
, bottom
,
249 GTK_FILL
, GTK_FILL
, 1, 1 );
253 int left
= i
/num_of_rows
;
254 int right
= (i
/num_of_rows
) + 1;
255 int top
= i%num_of_rows
;
256 int bottom
= (i%num_of_rows
)+1;
257 gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(m_radio
), left
, right
, top
, bottom
,
258 GTK_FILL
, GTK_FILL
, 1, 1 );
261 ConnectWidget( GTK_WIDGET(m_radio
) );
264 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(m_radio
), TRUE
);
266 g_signal_connect (m_radio
, "clicked",
267 G_CALLBACK (gtk_radiobutton_clicked_callback
), this);
268 g_signal_connect (m_radio
, "focus_in_event",
269 G_CALLBACK (gtk_radiobutton_focus_in
), this);
270 g_signal_connect (m_radio
, "focus_out_event",
271 G_CALLBACK (gtk_radiobutton_focus_out
), this);
274 m_parent
->DoAddChild( this );
281 wxRadioBox::~wxRadioBox()
283 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
286 GtkWidget
*button
= GTK_WIDGET( node
->GetData() );
287 gtk_widget_destroy( button
);
288 node
= node
->GetNext();
292 bool wxRadioBox::Show( bool show
)
294 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
296 if (!wxControl::Show(show
))
302 if ( HasFlag(wxNO_BORDER
) )
303 gtk_widget_hide( m_widget
);
305 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
308 GtkWidget
*button
= GTK_WIDGET( node
->GetData() );
311 gtk_widget_show( button
);
313 gtk_widget_hide( button
);
315 node
= node
->GetNext();
321 void wxRadioBox::SetFocus()
323 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
325 if (m_boxes
.GetCount() == 0) return;
327 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
330 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData() );
333 gtk_widget_grab_focus( GTK_WIDGET(button
) );
336 node
= node
->GetNext();
340 void wxRadioBox::SetSelection( int n
)
342 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
344 wxList::compatibility_iterator node
= m_boxes
.Item( n
);
346 wxCHECK_RET( node
, wxT("radiobox wrong index") );
348 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData() );
352 gtk_toggle_button_set_active( button
, 1 );
357 int wxRadioBox::GetSelection(void) const
359 wxCHECK_MSG( m_widget
!= NULL
, wxNOT_FOUND
, wxT("invalid radiobox") );
363 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
366 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData() );
367 if (button
->active
) return count
;
369 node
= node
->GetNext();
372 wxFAIL_MSG( wxT("wxRadioBox none selected") );
377 wxString
wxRadioBox::GetString( int n
) const
379 wxCHECK_MSG( m_widget
!= NULL
, wxEmptyString
, wxT("invalid radiobox") );
381 wxList::compatibility_iterator node
= m_boxes
.Item( n
);
383 wxCHECK_MSG( node
, wxEmptyString
, wxT("radiobox wrong index") );
385 GtkLabel
*label
= GTK_LABEL(GTK_BIN(node
->GetData())->child
);
387 wxString
str( wxGTK_CONV_BACK( gtk_label_get_text(label
) ) );
392 void wxRadioBox::SetLabel( const wxString
& label
)
394 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
396 GTKSetLabelForFrame(GTK_FRAME(m_widget
), label
);
399 void wxRadioBox::SetString( int item
, const wxString
& label
)
401 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
403 wxList::compatibility_iterator node
= m_boxes
.Item( item
);
405 wxCHECK_RET( node
, wxT("radiobox wrong index") );
407 GtkLabel
*g_label
= GTK_LABEL(GTK_BIN(node
->GetData())->child
);
409 gtk_label_set( g_label
, wxGTK_CONV( label
) );
412 bool wxRadioBox::Enable( bool enable
)
414 if ( !wxControl::Enable( enable
) )
417 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
420 GtkButton
*button
= GTK_BUTTON( node
->GetData() );
421 GtkLabel
*label
= GTK_LABEL(GTK_BIN(button
)->child
);
423 gtk_widget_set_sensitive( GTK_WIDGET(button
), enable
);
424 gtk_widget_set_sensitive( GTK_WIDGET(label
), enable
);
425 node
= node
->GetNext();
431 bool wxRadioBox::Enable( int item
, bool enable
)
433 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
435 wxList::compatibility_iterator node
= m_boxes
.Item( item
);
437 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
439 GtkButton
*button
= GTK_BUTTON( node
->GetData() );
440 GtkLabel
*label
= GTK_LABEL(GTK_BIN(button
)->child
);
442 gtk_widget_set_sensitive( GTK_WIDGET(button
), enable
);
443 gtk_widget_set_sensitive( GTK_WIDGET(label
), enable
);
448 bool wxRadioBox::IsItemEnabled(int item
) const
450 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
452 wxList::compatibility_iterator node
= m_boxes
.Item( item
);
454 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
456 GtkButton
*button
= GTK_BUTTON( node
->GetData() );
458 // don't use GTK_WIDGET_IS_SENSITIVE() here, we want to return true even if
459 // the parent radiobox is disabled
460 return GTK_WIDGET_SENSITIVE(GTK_WIDGET(button
));
463 bool wxRadioBox::Show( int item
, bool show
)
465 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
467 wxList::compatibility_iterator node
= m_boxes
.Item( item
);
469 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
471 GtkWidget
*button
= GTK_WIDGET( node
->GetData() );
474 gtk_widget_show( button
);
476 gtk_widget_hide( button
);
481 bool wxRadioBox::IsItemShown(int item
) const
483 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
485 wxList::compatibility_iterator node
= m_boxes
.Item( item
);
487 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
489 GtkButton
*button
= GTK_BUTTON( node
->GetData() );
491 return GTK_WIDGET_VISIBLE(GTK_WIDGET(button
));
494 int wxRadioBox::GetCount() const
496 return m_boxes
.GetCount();
499 void wxRadioBox::GtkDisableEvents()
501 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
504 g_signal_handlers_disconnect_by_func (node
->GetData(),
505 (gpointer
) gtk_radiobutton_clicked_callback
,
508 node
= node
->GetNext();
512 void wxRadioBox::GtkEnableEvents()
514 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
517 g_signal_connect (node
->GetData(), "clicked",
518 G_CALLBACK (gtk_radiobutton_clicked_callback
), this);
520 node
= node
->GetNext();
524 void wxRadioBox::DoApplyWidgetStyle(GtkRcStyle
*style
)
526 gtk_widget_modify_style( m_widget
, style
);
527 gtk_widget_modify_style(GTK_FRAME(m_widget
)->label_widget
, style
);
529 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
532 GtkWidget
*widget
= GTK_WIDGET( node
->GetData() );
534 gtk_widget_modify_style( widget
, style
);
535 gtk_widget_modify_style(GTK_BIN(widget
)->child
, style
);
537 node
= node
->GetNext();
542 void wxRadioBox::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
544 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
547 GtkWidget
*widget
= GTK_WIDGET( node
->GetData() );
548 gtk_tooltips_set_tip( tips
, widget
, wxConvCurrent
->cWX2MB(tip
), (gchar
*) NULL
);
549 node
= node
->GetNext();
552 #endif // wxUSE_TOOLTIPS
554 bool wxRadioBox::IsOwnGtkWindow( GdkWindow
*window
)
556 if (window
== m_widget
->window
)
559 wxList::compatibility_iterator node
= m_boxes
.GetFirst();
562 GtkWidget
*button
= GTK_WIDGET( node
->GetData() );
564 if (window
== button
->window
)
567 node
= node
->GetNext();
573 void wxRadioBox::OnInternalIdle()
580 wxFocusEvent
event( wxEVT_KILL_FOCUS
, GetId() );
581 event
.SetEventObject( this );
583 (void)GetEventHandler()->ProcessEvent( event
);
586 if (g_delayedFocus
== this)
588 if (GTK_WIDGET_REALIZED(m_widget
))
590 g_delayedFocus
= NULL
;
598 wxRadioBox::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
600 wxVisualAttributes attr
;
601 // NB: we need toplevel window so that GTK+ can find the right style
602 GtkWidget
*wnd
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
603 GtkWidget
* widget
= gtk_radio_button_new_with_label(NULL
, "");
604 gtk_container_add(GTK_CONTAINER(wnd
), widget
);
605 attr
= GetDefaultAttributesFromGTKWidget(widget
);
606 gtk_widget_destroy(wnd
);
610 #endif // wxUSE_RADIOBOX