1 /////////////////////////////////////////////////////////////////////////////
7 // Copyright: (c) Robert Roebling
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx.h".
12 #include "wx/wxprec.h"
14 #include "wx/spinctrl.h"
20 #include "wx/textctrl.h" // for wxEVT_COMMAND_TEXT_UPDATED
22 #include "wx/gtk/private.h"
24 //-----------------------------------------------------------------------------
26 //-----------------------------------------------------------------------------
28 extern void wxapp_install_idle_handler();
31 static const float sensitivity
= 0.02;
33 //-----------------------------------------------------------------------------
35 //-----------------------------------------------------------------------------
37 extern bool g_blockEventsOnDrag
;
39 //-----------------------------------------------------------------------------
41 //-----------------------------------------------------------------------------
44 static void gtk_spinctrl_callback( GtkWidget
*WXUNUSED(widget
), wxSpinCtrl
*win
)
46 if (g_isIdle
) wxapp_install_idle_handler();
48 if (!win
->m_hasVMT
) return;
49 if (g_blockEventsOnDrag
) return;
51 wxCommandEvent
event( wxEVT_COMMAND_SPINCTRL_UPDATED
, win
->GetId());
52 event
.SetEventObject( win
);
54 // note that we don't use wxSpinCtrl::GetValue() here because it would
55 // adjust the value to fit into the control range and this means that we
56 // would never be able to enter an "invalid" value in the control, even
57 // temporarily - and trying to enter 10 into the control which accepts the
58 // values in range 5..50 is then, ummm, quite challenging (hint: you can't
60 event
.SetInt( (int)ceil(win
->m_adjust
->value
) );
61 win
->GetEventHandler()->ProcessEvent( event
);
65 //-----------------------------------------------------------------------------
67 //-----------------------------------------------------------------------------
71 gtk_spinctrl_text_changed_callback( GtkWidget
*WXUNUSED(widget
), wxSpinCtrl
*win
)
73 if (!win
->m_hasVMT
) return;
76 wxapp_install_idle_handler();
78 wxCommandEvent
event( wxEVT_COMMAND_TEXT_UPDATED
, win
->GetId() );
79 event
.SetEventObject( win
);
82 event
.SetInt( (int)ceil(win
->m_adjust
->value
) );
83 win
->GetEventHandler()->ProcessEvent( event
);
87 //-----------------------------------------------------------------------------
89 //-----------------------------------------------------------------------------
91 IMPLEMENT_DYNAMIC_CLASS(wxSpinCtrl
,wxControl
)
93 BEGIN_EVENT_TABLE(wxSpinCtrl
, wxControl
)
94 EVT_CHAR(wxSpinCtrl::OnChar
)
97 bool wxSpinCtrl::Create(wxWindow
*parent
, wxWindowID id
,
98 const wxString
& value
,
99 const wxPoint
& pos
, const wxSize
& size
,
101 int min
, int max
, int initial
,
102 const wxString
& name
)
105 m_acceptsFocus
= TRUE
;
107 if (!PreCreation( parent
, pos
, size
) ||
108 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
110 wxFAIL_MSG( wxT("wxSpinCtrl creation failed") );
116 m_adjust
= (GtkAdjustment
*) gtk_adjustment_new( initial
, min
, max
, 1.0, 5.0, 0.0);
118 m_widget
= gtk_spin_button_new( m_adjust
, 1, 0 );
120 gtk_spin_button_set_wrap( GTK_SPIN_BUTTON(m_widget
),
121 (int)(m_windowStyle
& wxSP_WRAP
) );
125 m_parent
->DoAddChild( this );
134 void wxSpinCtrl::GtkDisableEvents()
136 g_signal_handlers_disconnect_by_func (m_adjust
,
137 (gpointer
) gtk_spinctrl_callback
,
139 g_signal_handlers_disconnect_by_func (m_widget
,
140 (gpointer
) gtk_spinctrl_text_changed_callback
,
144 void wxSpinCtrl::GtkEnableEvents()
146 g_signal_connect (m_adjust
, "value_changed",
147 G_CALLBACK (gtk_spinctrl_callback
),
149 g_signal_connect (m_widget
, "changed",
150 G_CALLBACK (gtk_spinctrl_text_changed_callback
),
154 int wxSpinCtrl::GetMin() const
156 wxCHECK_MSG( (m_widget
!= NULL
), 0, wxT("invalid spin button") );
158 return (int)ceil(m_adjust
->lower
);
161 int wxSpinCtrl::GetMax() const
163 wxCHECK_MSG( (m_widget
!= NULL
), 0, wxT("invalid spin button") );
165 return (int)ceil(m_adjust
->upper
);
168 int wxSpinCtrl::GetValue() const
170 wxCHECK_MSG( (m_widget
!= NULL
), 0, wxT("invalid spin button") );
172 gtk_spin_button_update( GTK_SPIN_BUTTON(m_widget
) );
174 return (int)ceil(m_adjust
->value
);
177 void wxSpinCtrl::SetValue( const wxString
& value
)
179 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid spin button") );
182 if ( (wxSscanf(value
, wxT("%d"), &n
) == 1) )
189 // invalid number - set text as is (wxMSW compatible)
191 gtk_entry_set_text( GTK_ENTRY(m_widget
), wxGTK_CONV( value
) );
196 void wxSpinCtrl::SetValue( int value
)
198 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid spin button") );
200 float fpos
= (float)value
;
202 if (fabs(fpos
-m_adjust
->value
) < sensitivity
) return;
204 m_adjust
->value
= fpos
;
207 g_signal_emit_by_name (m_adjust
, "value_changed");
211 void wxSpinCtrl::SetSelection(long from
, long to
)
213 // translate from wxWidgets conventions to GTK+ ones: (-1, -1) means the
215 if ( from
== -1 && to
== -1 )
221 gtk_editable_select_region( GTK_EDITABLE(m_widget
), (gint
)from
, (gint
)to
);
224 void wxSpinCtrl::SetRange(int minVal
, int maxVal
)
226 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid spin button") );
228 float fmin
= (float)minVal
;
229 float fmax
= (float)maxVal
;
231 if ((fabs(fmin
-m_adjust
->lower
) < sensitivity
) &&
232 (fabs(fmax
-m_adjust
->upper
) < sensitivity
))
237 m_adjust
->lower
= fmin
;
238 m_adjust
->upper
= fmax
;
240 g_signal_emit_by_name (m_adjust
, "changed");
242 // these two calls are required due to some bug in GTK
247 void wxSpinCtrl::OnChar( wxKeyEvent
&event
)
249 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid spin ctrl") );
251 if (event
.GetKeyCode() == WXK_RETURN
)
253 wxWindow
*top_frame
= m_parent
;
254 while (top_frame
->GetParent() && !(top_frame
->IsTopLevel()))
255 top_frame
= top_frame
->GetParent();
257 if ( GTK_IS_WINDOW(top_frame
->m_widget
) )
259 GtkWindow
*window
= GTK_WINDOW(top_frame
->m_widget
);
262 GtkWidget
*widgetDef
= window
->default_widget
;
266 gtk_widget_activate(widgetDef
);
273 if ((event
.GetKeyCode() == WXK_RETURN
) && (m_windowStyle
& wxPROCESS_ENTER
))
275 wxCommandEvent
evt( wxEVT_COMMAND_TEXT_ENTER
, m_windowId
);
276 evt
.SetEventObject(this);
277 GtkSpinButton
*gsb
= GTK_SPIN_BUTTON(m_widget
);
278 wxString val
= wxGTK_CONV_BACK( gtk_entry_get_text( &gsb
->entry
) );
279 evt
.SetString( val
);
280 if (GetEventHandler()->ProcessEvent(evt
)) return;
286 bool wxSpinCtrl::IsOwnGtkWindow( GdkWindow
*window
)
288 if (GTK_SPIN_BUTTON(m_widget
)->entry
.text_area
== window
) return TRUE
;
290 if (GTK_SPIN_BUTTON(m_widget
)->panel
== window
) return TRUE
;
295 wxSize
wxSpinCtrl::DoGetBestSize() const
297 wxSize
ret( wxControl::DoGetBestSize() );
298 wxSize
best(95, ret
.y
);
305 wxSpinCtrl::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
307 // TODO: overload to accept functions like gtk_spin_button_new?
308 // Until then use a similar type
309 return GetDefaultAttributesFromGTKWidget(gtk_entry_new
, true);