]> git.saurik.com Git - wxWidgets.git/blame - src/gtk/checkbox.cpp
added a check which should prevent the crash of bug 555111
[wxWidgets.git] / src / gtk / checkbox.cpp
CommitLineData
c801d85f
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: checkbox.cpp
3// Purpose:
4// Author: Robert Roebling
dbf858b5 5// Id: $Id$
01111366 6// Copyright: (c) 1998 Robert Roebling
ff8bfdbb 7// Licence: wxWindows licence
c801d85f
KB
8/////////////////////////////////////////////////////////////////////////////
9
10
11#ifdef __GNUG__
12#pragma implementation "checkbox.h"
13#endif
14
1e6feb95 15#include "wx/defs.h"
c801d85f 16
dcf924a3
RR
17#if wxUSE_CHECKBOX
18
1e6feb95
VZ
19#include "wx/checkbox.h"
20
9e691f46 21#include "wx/gtk/private.h"
83624f79 22
acfd422a
RR
23//-----------------------------------------------------------------------------
24// idle system
25//-----------------------------------------------------------------------------
26
27extern void wxapp_install_idle_handler();
28extern bool g_isIdle;
29
66bd6b93
RR
30//-----------------------------------------------------------------------------
31// data
32//-----------------------------------------------------------------------------
33
d7fa7eaa
RR
34extern bool g_blockEventsOnDrag;
35extern wxCursor g_globalCursor;
36extern wxWindowGTK *g_delayedFocus;
66bd6b93 37
c801d85f 38//-----------------------------------------------------------------------------
e1e955e1 39// "clicked"
c801d85f
KB
40//-----------------------------------------------------------------------------
41
66bd6b93 42static void gtk_checkbox_clicked_callback( GtkWidget *WXUNUSED(widget), wxCheckBox *cb )
c801d85f 43{
acfd422a
RR
44 if (g_isIdle) wxapp_install_idle_handler();
45
a2053b27 46 if (!cb->m_hasVMT) return;
ff8bfdbb 47
b292e2f5 48 if (g_blockEventsOnDrag) return;
9864c56d
RR
49
50 if (cb->m_blockEvent) return;
ff8bfdbb 51
b292e2f5
RR
52 wxCommandEvent event(wxEVT_COMMAND_CHECKBOX_CLICKED, cb->GetId());
53 event.SetInt( cb->GetValue() );
54 event.SetEventObject(cb);
55 cb->GetEventHandler()->ProcessEvent(event);
6de97a3b 56}
c801d85f 57
e1e955e1
RR
58//-----------------------------------------------------------------------------
59// wxCheckBox
c801d85f
KB
60//-----------------------------------------------------------------------------
61
62IMPLEMENT_DYNAMIC_CLASS(wxCheckBox,wxControl)
63
2cc0e28f 64wxCheckBox::wxCheckBox()
c801d85f 65{
6de97a3b 66}
c801d85f 67
2cc0e28f
VZ
68bool wxCheckBox::Create(wxWindow *parent,
69 wxWindowID id,
70 const wxString &label,
71 const wxPoint &pos,
72 const wxSize &size,
73 long style,
74 const wxValidator& validator,
75 const wxString &name )
c801d85f 76{
b292e2f5
RR
77 m_needParent = TRUE;
78 m_acceptsFocus = TRUE;
9864c56d 79 m_blockEvent = FALSE;
ff8bfdbb 80
4dcaf11a
RR
81 if (!PreCreation( parent, pos, size ) ||
82 !CreateBase( parent, id, pos, size, style, validator, name ))
83 {
223d09f6 84 wxFAIL_MSG( wxT("wxCheckBox creation failed") );
1db8dc4a 85 return FALSE;
4dcaf11a 86 }
ce4169a4 87
2cc0e28f
VZ
88 wxControl::SetLabel( label );
89
90 if ( style & wxALIGN_RIGHT )
91 {
92 // VZ: as I don't know a way to create a right aligned checkbox with
93 // GTK we will create a checkbox without label and a label at the
94 // left of it
95 m_widgetCheckbox = gtk_check_button_new();
6de97a3b 96
3bce7509 97 m_widgetLabel = gtk_label_new(m_label.mbc_str());
2cc0e28f 98 gtk_misc_set_alignment(GTK_MISC(m_widgetLabel), 0.0, 0.5);
ff8bfdbb 99
2cc0e28f
VZ
100 m_widget = gtk_hbox_new(FALSE, 0);
101 gtk_box_pack_start(GTK_BOX(m_widget), m_widgetLabel, FALSE, FALSE, 3);
102 gtk_box_pack_start(GTK_BOX(m_widget), m_widgetCheckbox, FALSE, FALSE, 3);
103
2cc0e28f
VZ
104 gtk_widget_show( m_widgetLabel );
105 gtk_widget_show( m_widgetCheckbox );
106 }
107 else
108 {
3bce7509 109 m_widgetCheckbox = gtk_check_button_new_with_label( m_label.mbc_str() );
9e691f46 110 m_widgetLabel = BUTTON_CHILD( m_widgetCheckbox );
2cc0e28f
VZ
111 m_widget = m_widgetCheckbox;
112 }
113
2cc0e28f
VZ
114 gtk_signal_connect( GTK_OBJECT(m_widgetCheckbox),
115 "clicked",
116 GTK_SIGNAL_FUNC(gtk_checkbox_clicked_callback),
117 (gpointer *)this );
ff8bfdbb 118
f03fc89f 119 m_parent->DoAddChild( this );
ff8bfdbb 120
b292e2f5 121 PostCreation();
ff8bfdbb 122
db434467
RR
123 SetFont( parent->GetFont() );
124
125 wxSize size_best( DoGetBestSize() );
126 wxSize new_size( size );
127 if (new_size.x == -1)
128 new_size.x = size_best.x;
129 if (new_size.y == -1)
130 new_size.y = size_best.y;
131 if ((new_size.x != size.x) || (new_size.y != size.y))
132 SetSize( new_size.x, new_size.y );
133
b292e2f5
RR
134 SetBackgroundColour( parent->GetBackgroundColour() );
135 SetForegroundColour( parent->GetForegroundColour() );
f96aa4d9 136
b292e2f5 137 Show( TRUE );
ff8bfdbb 138
b292e2f5 139 return TRUE;
6de97a3b 140}
c801d85f 141
debe6624 142void wxCheckBox::SetValue( bool state )
c801d85f 143{
223d09f6 144 wxCHECK_RET( m_widgetCheckbox != NULL, wxT("invalid checkbox") );
ff8bfdbb 145
953704c1 146 if (state == GetValue())
8bf45ed1
VZ
147 return;
148
9864c56d 149 m_blockEvent = TRUE;
ae0bdb01 150
2cc0e28f 151 gtk_toggle_button_set_state( GTK_TOGGLE_BUTTON(m_widgetCheckbox), state );
1db8dc4a 152
9864c56d 153 m_blockEvent = FALSE;
6de97a3b 154}
c801d85f 155
83058c58 156bool wxCheckBox::GetValue() const
c801d85f 157{
223d09f6 158 wxCHECK_MSG( m_widgetCheckbox != NULL, FALSE, wxT("invalid checkbox") );
f96aa4d9 159
2cc0e28f 160 return GTK_TOGGLE_BUTTON(m_widgetCheckbox)->active;
6de97a3b 161}
c801d85f 162
83058c58
RR
163void wxCheckBox::SetLabel( const wxString& label )
164{
223d09f6 165 wxCHECK_RET( m_widgetLabel != NULL, wxT("invalid checkbox") );
f96aa4d9 166
b292e2f5 167 wxControl::SetLabel( label );
ff8bfdbb 168
3bce7509 169 gtk_label_set( GTK_LABEL(m_widgetLabel), GetLabel().mbc_str() );
83058c58
RR
170}
171
f03fc89f 172bool wxCheckBox::Enable( bool enable )
d3904ceb 173{
f03fc89f
VZ
174 if ( !wxControl::Enable( enable ) )
175 return FALSE;
ff8bfdbb 176
2cc0e28f 177 gtk_widget_set_sensitive( m_widgetLabel, enable );
f03fc89f
VZ
178
179 return TRUE;
d3904ceb
RR
180}
181
58614078 182void wxCheckBox::ApplyWidgetStyle()
868a2826 183{
b292e2f5 184 SetWidgetStyle();
2cc0e28f
VZ
185 gtk_widget_set_style( m_widgetCheckbox, m_widgetStyle );
186 gtk_widget_set_style( m_widgetLabel, m_widgetStyle );
f96aa4d9
RR
187}
188
2f073eb2
RR
189bool wxCheckBox::IsOwnGtkWindow( GdkWindow *window )
190{
9e691f46 191 return window == TOGGLE_BUTTON_EVENT_WIN(m_widget);
2f073eb2
RR
192}
193
194void wxCheckBox::OnInternalIdle()
195{
196 wxCursor cursor = m_cursor;
197 if (g_globalCursor.Ok()) cursor = g_globalCursor;
198
9e691f46
VZ
199 GdkWindow *event_window = TOGGLE_BUTTON_EVENT_WIN(m_widgetCheckbox);
200 if ( event_window && cursor.Ok() )
2f073eb2
RR
201 {
202 /* I now set the cursor the anew in every OnInternalIdle call
1db8dc4a
VZ
203 as setting the cursor in a parent window also effects the
204 windows above so that checking for the current cursor is
205 not possible. */
206
9e691f46 207 gdk_window_set_cursor( event_window, cursor.GetCursor() );
2f073eb2
RR
208 }
209
d7fa7eaa
RR
210 if (g_delayedFocus == this)
211 {
212 if (GTK_WIDGET_REALIZED(m_widget))
213 {
214 gtk_widget_grab_focus( m_widget );
215 g_delayedFocus = NULL;
216 }
217 }
218
2f073eb2
RR
219 UpdateWindowUI();
220}
221
f68586e5
VZ
222wxSize wxCheckBox::DoGetBestSize() const
223{
db434467 224 return wxControl::DoGetBestSize();
f68586e5
VZ
225}
226
dcf924a3 227#endif