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"
20 #include "wx/dialog.h"
24 #include "wx/tooltip.h"
27 #include "wx/gtk/private.h"
28 #include <gdk/gdkkeysyms.h>
30 #include "wx/gtk/win_gtk.h"
32 //-----------------------------------------------------------------------------
33 // wxGTKRadioButtonInfo
34 //-----------------------------------------------------------------------------
35 // structure internally used by wxRadioBox to store its child buttons
37 class wxGTKRadioButtonInfo
: public wxObject
40 wxGTKRadioButtonInfo( GtkRadioButton
* abutton
, const wxRect
& arect
)
41 : button( abutton
), rect( arect
) {}
43 GtkRadioButton
* button
;
47 //-----------------------------------------------------------------------------
49 //-----------------------------------------------------------------------------
51 #include "wx/listimpl.cpp"
52 WX_DEFINE_LIST( wxRadioBoxButtonsInfoList
);
54 extern bool g_blockEventsOnDrag
;
56 //-----------------------------------------------------------------------------
58 //-----------------------------------------------------------------------------
61 static void gtk_radiobutton_clicked_callback( GtkToggleButton
*button
, wxRadioBox
*rb
)
63 if (g_isIdle
) wxapp_install_idle_handler();
65 if (!rb
->m_hasVMT
) return;
66 if (g_blockEventsOnDrag
) return;
68 if (!button
->active
) return;
70 wxCommandEvent
event( wxEVT_COMMAND_RADIOBOX_SELECTED
, rb
->GetId() );
71 event
.SetInt( rb
->GetSelection() );
72 event
.SetString( rb
->GetStringSelection() );
73 event
.SetEventObject( rb
);
74 rb
->GetEventHandler()->ProcessEvent(event
);
78 //-----------------------------------------------------------------------------
80 //-----------------------------------------------------------------------------
83 static gint
gtk_radiobox_keypress_callback( GtkWidget
*widget
, GdkEventKey
*gdk_event
, wxRadioBox
*rb
)
85 // don't need to install idle handler, its done from "event" signal
87 if (!rb
->m_hasVMT
) return FALSE
;
88 if (g_blockEventsOnDrag
) return FALSE
;
90 if ( ((gdk_event
->keyval
== GDK_Tab
) ||
91 (gdk_event
->keyval
== GDK_ISO_Left_Tab
)) &&
92 rb
->GetParent() && (rb
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) )
94 wxNavigationKeyEvent new_event
;
95 new_event
.SetEventObject( rb
->GetParent() );
96 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
97 new_event
.SetDirection( (gdk_event
->keyval
== GDK_Tab
) );
98 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
99 new_event
.SetWindowChange( (gdk_event
->state
& GDK_CONTROL_MASK
) );
100 new_event
.SetCurrentFocus( rb
);
101 return rb
->GetParent()->GetEventHandler()->ProcessEvent( new_event
);
104 if ((gdk_event
->keyval
!= GDK_Up
) &&
105 (gdk_event
->keyval
!= GDK_Down
) &&
106 (gdk_event
->keyval
!= GDK_Left
) &&
107 (gdk_event
->keyval
!= GDK_Right
))
112 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= rb
->m_buttonsInfo
.GetFirst();
113 while( node
&& GTK_WIDGET( node
->GetData()->button
) != widget
)
115 node
= node
->GetNext();
122 g_signal_stop_emission_by_name (widget
, "key_press_event");
124 if ((gdk_event
->keyval
== GDK_Up
) ||
125 (gdk_event
->keyval
== GDK_Left
))
127 if (node
== rb
->m_buttonsInfo
.GetFirst())
128 node
= rb
->m_buttonsInfo
.GetLast();
130 node
= node
->GetPrevious();
134 if (node
== rb
->m_buttonsInfo
.GetLast())
135 node
= rb
->m_buttonsInfo
.GetFirst();
137 node
= node
->GetNext();
140 GtkWidget
*button
= (GtkWidget
*) node
->GetData()->button
;
142 gtk_widget_grab_focus( button
);
149 static gint
gtk_radiobutton_focus_in( GtkWidget
*widget
,
150 GdkEvent
*WXUNUSED(event
),
153 if ( win
->m_lostFocus
)
155 // no, we didn't really lose it
156 win
->m_lostFocus
= FALSE
;
158 else if ( !win
->m_hasFocus
)
160 win
->m_hasFocus
= true;
162 wxFocusEvent
event( wxEVT_SET_FOCUS
, win
->GetId() );
163 event
.SetEventObject( win
);
165 // never stop the signal emission, it seems to break the kbd handling
166 // inside the radiobox
167 (void)win
->GetEventHandler()->ProcessEvent( event
);
175 static gint
gtk_radiobutton_focus_out( GtkWidget
*widget
,
176 GdkEvent
*WXUNUSED(event
),
179 // wxASSERT_MSG( win->m_hasFocus, _T("got focus out without any focus in?") );
180 // Replace with a warning, else we dump core a lot!
181 // if (!win->m_hasFocus)
182 // wxLogWarning(_T("Radiobox got focus out without any focus in.") );
184 // we might have lost the focus, but may be not - it may have just gone to
185 // another button in the same radiobox, so we'll check for it in the next
186 // idle iteration (leave m_hasFocus == true for now)
187 win
->m_lostFocus
= true;
194 static void gtk_radiobutton_size_allocate( GtkWidget
*widget
,
195 GtkAllocation
* alloc
,
199 for ( wxRadioBoxButtonsInfoList
::compatibility_iterator node
= win
->m_buttonsInfo
.GetFirst();
201 node
= node
->GetNext(), n
++ )
203 if( widget
== GTK_WIDGET(node
->GetData()->button
) )
205 const wxPoint origin
= win
->GetPosition();
206 wxRect rect
= wxRect( alloc
->x
- origin
.x
, alloc
->y
- origin
.y
,
207 alloc
->width
, alloc
->height
);
208 node
->GetData()->rect
= rect
;
216 //-----------------------------------------------------------------------------
218 //-----------------------------------------------------------------------------
220 IMPLEMENT_DYNAMIC_CLASS(wxRadioBox
,wxControl
)
222 void wxRadioBox
::Init()
225 m_acceptsFocus
= true;
231 bool wxRadioBox
::Create( wxWindow
*parent
, wxWindowID id
,
232 const wxString
& title
,
233 const wxPoint
&pos
, const wxSize
&size
,
234 const wxArrayString
& choices
, int majorDim
,
235 long style
, const wxValidator
& validator
,
236 const wxString
&name
)
238 wxCArrayString
chs(choices
);
240 return Create( parent
, id
, title
, pos
, size
, chs
.GetCount(),
241 chs
.GetStrings(), majorDim
, style
, validator
, name
);
244 bool wxRadioBox
::Create( wxWindow
*parent
, wxWindowID id
, const wxString
& title
,
245 const wxPoint
&pos
, const wxSize
&size
,
246 int n
, const wxString choices
[], int majorDim
,
247 long style
, const wxValidator
& validator
,
248 const wxString
&name
)
250 if (!PreCreation( parent
, pos
, size
) ||
251 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
253 wxFAIL_MSG( wxT("wxRadioBox creation failed") );
257 m_widget
= GTKCreateFrame(title
);
258 wxControl
::SetLabel(title
);
260 // majorDim may be 0 if all trailing parameters were omitted, so don't
261 // assert here but just use the correct value for it
262 SetMajorDim(majorDim
== 0 ? n
: majorDim
, style
);
265 unsigned int num_of_cols
= GetColumnCount();
266 unsigned int num_of_rows
= GetRowCount();
268 GtkRadioButton
*rbtn
= (GtkRadioButton
*) NULL
;
270 GtkWidget
*table
= gtk_table_new( num_of_rows
, num_of_cols
, FALSE
);
271 gtk_table_set_col_spacings( GTK_TABLE(table
), 1 );
272 gtk_table_set_row_spacings( GTK_TABLE(table
), 1 );
273 gtk_widget_show( table
);
274 gtk_container_add( GTK_CONTAINER(m_widget
), table
);
277 GSList
*radio_button_group
= (GSList
*) NULL
;
278 for (unsigned int i
= 0; i
< (unsigned int)n
; i
++)
281 radio_button_group
= gtk_radio_button_get_group( GTK_RADIO_BUTTON(rbtn
) );
284 for ( const wxChar
*pc
= choices
[i
]; *pc
; pc
++ )
286 if ( *pc
!= wxT('&') )
290 rbtn
= GTK_RADIO_BUTTON( gtk_radio_button_new_with_label( radio_button_group
, wxGTK_CONV( label
) ) );
291 gtk_widget_show( GTK_WIDGET(rbtn
) );
293 g_signal_connect (rbtn
, "key_press_event",
294 G_CALLBACK (gtk_radiobox_keypress_callback
), this);
296 m_buttonsInfo
.Append( new wxGTKRadioButtonInfo( rbtn
, wxRect() ) );
298 if (HasFlag(wxRA_SPECIFY_COLS
))
300 int left
= i
%num_of_cols
;
301 int right
= (i
%num_of_cols
) + 1;
302 int top
= i
/num_of_cols
;
303 int bottom
= (i
/num_of_cols
)+1;
304 gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(rbtn
), left
, right
, top
, bottom
,
305 GTK_FILL
, GTK_FILL
, 1, 1 );
309 int left
= i
/num_of_rows
;
310 int right
= (i
/num_of_rows
) + 1;
311 int top
= i
%num_of_rows
;
312 int bottom
= (i
%num_of_rows
)+1;
313 gtk_table_attach( GTK_TABLE(table
), GTK_WIDGET(rbtn
), left
, right
, top
, bottom
,
314 GTK_FILL
, GTK_FILL
, 1, 1 );
317 ConnectWidget( GTK_WIDGET(rbtn
) );
320 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(rbtn
), TRUE
);
322 g_signal_connect (rbtn
, "clicked",
323 G_CALLBACK (gtk_radiobutton_clicked_callback
), this);
324 g_signal_connect (rbtn
, "focus_in_event",
325 G_CALLBACK (gtk_radiobutton_focus_in
), this);
326 g_signal_connect (rbtn
, "focus_out_event",
327 G_CALLBACK (gtk_radiobutton_focus_out
), this);
328 g_signal_connect (rbtn
, "size_allocate",
329 G_CALLBACK (gtk_radiobutton_size_allocate
), this);
332 m_parent
->DoAddChild( this );
339 wxRadioBox
::~wxRadioBox()
341 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
344 GtkWidget
*button
= GTK_WIDGET( node
->GetData()->button
);
345 gtk_widget_destroy( button
);
346 node
= node
->GetNext();
348 WX_CLEAR_LIST( wxRadioBoxButtonsInfoList
, m_buttonsInfo
);
351 bool wxRadioBox
::Show( bool show
)
353 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
355 if (!wxControl
::Show(show
))
361 if ( HasFlag(wxNO_BORDER
) )
362 gtk_widget_hide( m_widget
);
364 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
367 GtkWidget
*button
= GTK_WIDGET( node
->GetData()->button
);
370 gtk_widget_show( button
);
372 gtk_widget_hide( button
);
374 node
= node
->GetNext();
380 void wxRadioBox
::SetFocus()
382 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
384 if (m_buttonsInfo
.GetCount() == 0) return;
386 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
389 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData()->button
);
392 gtk_widget_grab_focus( GTK_WIDGET(button
) );
395 node
= node
->GetNext();
399 void wxRadioBox
::SetSelection( int n
)
401 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
403 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( n
);
405 wxCHECK_RET( node
, wxT("radiobox wrong index") );
407 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData()->button
);
411 gtk_toggle_button_set_active( button
, 1 );
416 int wxRadioBox
::GetSelection(void) const
418 wxCHECK_MSG( m_widget
!= NULL
, wxNOT_FOUND
, wxT("invalid radiobox") );
422 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
425 GtkToggleButton
*button
= GTK_TOGGLE_BUTTON( node
->GetData()->button
);
426 if (button
->active
) return count
;
428 node
= node
->GetNext();
431 wxFAIL_MSG( wxT("wxRadioBox none selected") );
436 wxString wxRadioBox
::GetString(unsigned int n
) const
438 wxCHECK_MSG( m_widget
!= NULL
, wxEmptyString
, wxT("invalid radiobox") );
440 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( n
);
442 wxCHECK_MSG( node
, wxEmptyString
, wxT("radiobox wrong index") );
444 GtkLabel
*label
= GTK_LABEL(GTK_BIN(node
->GetData()->button
)->child
);
446 wxString
str( wxGTK_CONV_BACK( gtk_label_get_text(label
) ) );
451 void wxRadioBox
::SetLabel( const wxString
& label
)
453 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
455 GTKSetLabelForFrame(GTK_FRAME(m_widget
), label
);
458 void wxRadioBox
::SetString(unsigned int item
, const wxString
& label
)
460 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid radiobox") );
462 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( item
);
464 wxCHECK_RET( node
, wxT("radiobox wrong index") );
466 GtkLabel
*g_label
= GTK_LABEL(GTK_BIN(node
->GetData()->button
)->child
);
468 gtk_label_set_text( g_label
, wxGTK_CONV( label
) );
471 bool wxRadioBox
::Enable( bool enable
)
473 if ( !wxControl
::Enable( enable
) )
476 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
479 GtkButton
*button
= GTK_BUTTON( node
->GetData()->button
);
480 GtkLabel
*label
= GTK_LABEL(GTK_BIN(button
)->child
);
482 gtk_widget_set_sensitive( GTK_WIDGET(button
), enable
);
483 gtk_widget_set_sensitive( GTK_WIDGET(label
), enable
);
484 node
= node
->GetNext();
490 bool wxRadioBox
::Enable(unsigned int item
, bool enable
)
492 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
494 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( item
);
496 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
498 GtkButton
*button
= GTK_BUTTON( node
->GetData()->button
);
499 GtkLabel
*label
= GTK_LABEL(GTK_BIN(button
)->child
);
501 gtk_widget_set_sensitive( GTK_WIDGET(button
), enable
);
502 gtk_widget_set_sensitive( GTK_WIDGET(label
), enable
);
507 bool wxRadioBox
::IsItemEnabled(unsigned int item
) const
509 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
511 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( item
);
513 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
515 GtkButton
*button
= GTK_BUTTON( node
->GetData()->button
);
517 // don't use GTK_WIDGET_IS_SENSITIVE() here, we want to return true even if
518 // the parent radiobox is disabled
519 return GTK_WIDGET_SENSITIVE(GTK_WIDGET(button
));
522 bool wxRadioBox
::Show(unsigned int item
, bool show
)
524 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
526 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( item
);
528 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
530 GtkWidget
*button
= GTK_WIDGET( node
->GetData()->button
);
533 gtk_widget_show( button
);
535 gtk_widget_hide( button
);
540 bool wxRadioBox
::IsItemShown(unsigned int item
) const
542 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid radiobox") );
544 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.Item( item
);
546 wxCHECK_MSG( node
, false, wxT("radiobox wrong index") );
548 GtkButton
*button
= GTK_BUTTON( node
->GetData()->button
);
550 return GTK_WIDGET_VISIBLE(GTK_WIDGET(button
));
553 unsigned int wxRadioBox
::GetCount() const
555 return m_buttonsInfo
.GetCount();
558 void wxRadioBox
::GtkDisableEvents()
560 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
563 g_signal_handlers_disconnect_by_func (node
->GetData()->button
,
564 (gpointer
) gtk_radiobutton_clicked_callback
,
567 node
= node
->GetNext();
571 void wxRadioBox
::GtkEnableEvents()
573 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
576 g_signal_connect (node
->GetData()->button
, "clicked",
577 G_CALLBACK (gtk_radiobutton_clicked_callback
), this);
579 node
= node
->GetNext();
583 void wxRadioBox
::DoApplyWidgetStyle(GtkRcStyle
*style
)
585 GTKFrameApplyWidgetStyle(GTK_FRAME(m_widget
), style
);
587 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
590 GtkWidget
*widget
= GTK_WIDGET( node
->GetData()->button
);
592 gtk_widget_modify_style( widget
, style
);
593 gtk_widget_modify_style(GTK_BIN(widget
)->child
, style
);
595 node
= node
->GetNext();
599 bool wxRadioBox
::GTKWidgetNeedsMnemonic() const
604 void wxRadioBox
::GTKWidgetDoSetMnemonic(GtkWidget
* w
)
606 GTKFrameSetMnemonicWidget(GTK_FRAME(m_widget
), w
);
610 void wxRadioBox
::ApplyToolTip(GtkTooltips
* WXUNUSED(tips
), const wxChar
*tip
)
612 // set this tooltip for all radiobuttons which don't have their own tips
614 for ( wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
616 node
= node
->GetNext(), n
++ )
618 if ( !GetItemToolTip(n
) )
620 wxToolTip
::Apply(GTK_WIDGET(node
->GetData()->button
),
621 wxConvCurrent
->cWX2MB(tip
));
626 void wxRadioBox
::DoSetItemToolTip(unsigned int n
, wxToolTip
*tooltip
)
630 tooltip
= GetToolTip();
632 buf
= wxGTK_CONV(tooltip
->GetTip());
634 wxToolTip
::Apply(GTK_WIDGET(m_buttonsInfo
[n
]->button
), buf
);
637 #endif // wxUSE_TOOLTIPS
639 GdkWindow
*wxRadioBox
::GTKGetWindow(wxArrayGdkWindows
& windows
) const
641 windows
.push_back(m_widget
->window
);
643 wxRadioBoxButtonsInfoList
::compatibility_iterator node
= m_buttonsInfo
.GetFirst();
646 GtkWidget
*button
= GTK_WIDGET( node
->GetData()->button
);
648 windows
.push_back(button
->window
);
650 node
= node
->GetNext();
656 void wxRadioBox
::OnInternalIdle()
658 wxControl
::OnInternalIdle();
665 wxFocusEvent
event( wxEVT_KILL_FOCUS
, GetId() );
666 event
.SetEventObject( this );
668 (void)GetEventHandler()->ProcessEvent( event
);
674 wxRadioBox
::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
676 wxVisualAttributes attr
;
677 // NB: we need toplevel window so that GTK+ can find the right style
678 GtkWidget
*wnd
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
679 GtkWidget
* widget
= gtk_radio_button_new_with_label(NULL
, "");
680 gtk_container_add(GTK_CONTAINER(wnd
), widget
);
681 attr
= GetDefaultAttributesFromGTKWidget(widget
);
682 gtk_widget_destroy(wnd
);
686 int wxRadioBox
::GetItemFromPoint(const wxPoint
& point
) const
688 const wxPoint pt
= ScreenToClient(point
);
690 for ( wxRadioBoxButtonsInfoList
::compatibility_iterator
691 node
= m_buttonsInfo
.GetFirst(); node
; node
= node
->GetNext(), n
++ )
693 if ( m_buttonsInfo
[n
]->rect
.Inside(pt
) )
700 #endif // wxUSE_RADIOBOX