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);