1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/generic/spinctlg.cpp 
   3 // Purpose:     implements wxSpinCtrl as a composite control 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2001 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // License:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  28     #include "wx/textctrl.h" 
  31 #include "wx/spinctrl.h" 
  35 IMPLEMENT_DYNAMIC_CLASS(wxSpinDoubleEvent
, wxNotifyEvent
) 
  37 // There are port-specific versions for the wxSpinCtrl, so exclude the 
  38 // contents of this file in those cases 
  39 #if !defined(wxHAS_NATIVE_SPINCTRL) || !defined(wxHAS_NATIVE_SPINCTRLDOUBLE) 
  41 #include "wx/spinbutt.h" 
  45 // ---------------------------------------------------------------------------- 
  47 // ---------------------------------------------------------------------------- 
  49 // the margin between the text control and the spin 
  50 static const wxCoord MARGIN 
= 2; 
  52 #define SPINCTRLBUT_MAX 32000 // large to avoid wrap around trouble 
  54 // ---------------------------------------------------------------------------- 
  55 // wxSpinCtrlTextGeneric: text control used by spin control 
  56 // ---------------------------------------------------------------------------- 
  58 class wxSpinCtrlTextGeneric 
: public wxTextCtrl
 
  61     wxSpinCtrlTextGeneric(wxSpinCtrlGenericBase 
*spin
, const wxString
& value
) 
  62         : wxTextCtrl(spin
->GetParent(), wxID_ANY
, value
, wxDefaultPosition
, 
  63                      wxDefaultSize
, wxTE_NOHIDESEL
|wxTE_PROCESS_ENTER
) 
  67         // remove the default minsize, the spinctrl will have one instead 
  68         SetSizeHints(wxDefaultCoord
, wxDefaultCoord
); 
  71     virtual ~wxSpinCtrlTextGeneric() 
  73         // MSW sends extra kill focus event on destroy 
  75             m_spin
->m_textCtrl 
= NULL
; 
  80     void OnTextEnter(wxCommandEvent
& event
) 
  83             m_spin
->OnTextEnter(event
); 
  86     void OnChar( wxKeyEvent 
&event 
) 
  89             m_spin
->OnTextChar(event
); 
  92     void OnKillFocus(wxFocusEvent
& event
) 
  96             m_spin
->SyncSpinToText(); 
  97             m_spin
->DoSendEvent(); 
 103     wxSpinCtrlGenericBase 
*m_spin
; 
 106     DECLARE_EVENT_TABLE() 
 109 BEGIN_EVENT_TABLE(wxSpinCtrlTextGeneric
, wxTextCtrl
) 
 110     EVT_TEXT_ENTER(wxID_ANY
, wxSpinCtrlTextGeneric::OnTextEnter
) 
 112     EVT_CHAR(wxSpinCtrlTextGeneric::OnChar
) 
 114     EVT_KILL_FOCUS(wxSpinCtrlTextGeneric::OnKillFocus
) 
 117 // ---------------------------------------------------------------------------- 
 118 // wxSpinCtrlButtonGeneric: spin button used by spin control 
 119 // ---------------------------------------------------------------------------- 
 121 class wxSpinCtrlButtonGeneric 
: public wxSpinButton
 
 124     wxSpinCtrlButtonGeneric(wxSpinCtrlGenericBase 
*spin
, int style
) 
 125         : wxSpinButton(spin
->GetParent(), wxID_ANY
, wxDefaultPosition
, 
 126                        wxDefaultSize
, style 
| wxSP_VERTICAL
) 
 130         SetRange(-SPINCTRLBUT_MAX
, SPINCTRLBUT_MAX
); 
 132         // remove the default minsize, the spinctrl will have one instead 
 133         SetSizeHints(wxDefaultCoord
, wxDefaultCoord
); 
 136     void OnSpinButton(wxSpinEvent
& event
) 
 139             m_spin
->OnSpinButton(event
); 
 142     wxSpinCtrlGenericBase 
*m_spin
; 
 145     DECLARE_EVENT_TABLE() 
 148 BEGIN_EVENT_TABLE(wxSpinCtrlButtonGeneric
, wxSpinButton
) 
 149     EVT_SPIN_UP(  wxID_ANY
, wxSpinCtrlButtonGeneric::OnSpinButton
) 
 150     EVT_SPIN_DOWN(wxID_ANY
, wxSpinCtrlButtonGeneric::OnSpinButton
) 
 153 // ============================================================================ 
 154 // wxSpinCtrlGenericBase 
 155 // ============================================================================ 
 157 // ---------------------------------------------------------------------------- 
 158 // wxSpinCtrlGenericBase creation 
 159 // ---------------------------------------------------------------------------- 
 161 void wxSpinCtrlGenericBase::Init() 
 167     m_snap_to_ticks 
= false; 
 168     m_format        
= wxS("%g"); 
 176 bool wxSpinCtrlGenericBase::Create(wxWindow 
*parent
, 
 178                                    const wxString
& value
, 
 179                                    const wxPoint
& pos
, const wxSize
& size
, 
 181                                    double min
, double max
, double initial
, 
 183                                    const wxString
& name
) 
 185     // don't use borders for this control itself, it wouldn't look good with 
 186     // the text control borders (but we might want to use style border bits to 
 187     // select the text control style) 
 188     if ( !wxControl::Create(parent
, id
, wxDefaultPosition
, wxDefaultSize
, 
 189                             wxBORDER_NONE
, wxDefaultValidator
, name
) ) 
 197     m_increment 
= increment
; 
 199     m_textCtrl   
= new wxSpinCtrlTextGeneric(this, value
); 
 200     m_spinButton 
= new wxSpinCtrlButtonGeneric(this, style
); 
 202     m_spin_value 
= m_spinButton
->GetValue(); 
 204     // the string value overrides the numeric one (for backwards compatibility 
 205     // reasons and also because it is simpler to satisfy the string value which 
 206     // comes much sooner in the list of arguments and leave the initial 
 207     // parameter unspecified) 
 208     if ( !value
.empty() ) 
 211         if ( value
.ToDouble(&d
) ) 
 214             m_textCtrl
->SetValue(wxString::Format(m_format
, m_value
)); 
 218     SetInitialSize(size
); 
 221     // have to disable this window to avoid interfering it with message 
 222     // processing to the text and the button... but pretend it is enabled to 
 223     // make IsEnabled() return true 
 224     wxControl::Enable(false); // don't use non virtual Disable() here! 
 227     // we don't even need to show this window itself - and not doing it avoids 
 228     // that it overwrites the text control 
 229     wxControl::Show(false); 
 234 wxSpinCtrlGenericBase::~wxSpinCtrlGenericBase() 
 236     // delete the controls now, don't leave them alive even though they would 
 237     // still be eventually deleted by our parent - but it will be too late, the 
 238     // user code expects them to be gone now 
 242         // null this since MSW sends KILL_FOCUS on deletion, see ~wxSpinCtrlTextGeneric 
 243         wxDynamicCast(m_textCtrl
, wxSpinCtrlTextGeneric
)->m_spin 
= NULL
; 
 245         wxSpinCtrlTextGeneric 
*text 
= (wxSpinCtrlTextGeneric
*)m_textCtrl
; 
 254 // ---------------------------------------------------------------------------- 
 256 // ---------------------------------------------------------------------------- 
 258 wxSize 
wxSpinCtrlGenericBase::DoGetBestSize() const 
 260     wxSize sizeBtn  
= m_spinButton
->GetBestSize(), 
 261            sizeText 
= m_textCtrl
->GetBestSize(); 
 263     return wxSize(sizeBtn
.x 
+ sizeText
.x 
+ MARGIN
, sizeText
.y
); 
 266 void wxSpinCtrlGenericBase::DoMoveWindow(int x
, int y
, int width
, int height
) 
 268     wxControl::DoMoveWindow(x
, y
, width
, height
); 
 270     // position the subcontrols inside the client area 
 271     wxSize sizeBtn 
= m_spinButton
->GetSize(); 
 273     wxCoord wText 
= width 
- sizeBtn
.x
; 
 274     m_textCtrl
->SetSize(x
, y
, wText
, height
); 
 275     m_spinButton
->SetSize(x 
+ wText 
+ MARGIN
, y
, wxDefaultCoord
, height
); 
 278 // ---------------------------------------------------------------------------- 
 279 // operations forwarded to the subcontrols 
 280 // ---------------------------------------------------------------------------- 
 282 bool wxSpinCtrlGenericBase::Enable(bool enable
) 
 284     if ( !wxControl::Enable(enable
) ) 
 287     m_spinButton
->Enable(enable
); 
 288     m_textCtrl
->Enable(enable
); 
 293 bool wxSpinCtrlGenericBase::Show(bool show
) 
 295     if ( !wxControl::Show(show
) ) 
 298     // under GTK Show() is called the first time before we are fully 
 302         m_spinButton
->Show(show
); 
 303         m_textCtrl
->Show(show
); 
 309 bool wxSpinCtrlGenericBase::Reparent(wxWindowBase 
*newParent
) 
 313         m_spinButton
->Reparent(newParent
); 
 314         m_textCtrl
->Reparent(newParent
); 
 320 // ---------------------------------------------------------------------------- 
 321 // Handle sub controls events 
 322 // ---------------------------------------------------------------------------- 
 324 void wxSpinCtrlGenericBase::OnSpinButton(wxSpinEvent
& event
) 
 328     // Sync the textctrl since the user expects that the button will modify 
 329     // what they see in the textctrl. 
 330     if ( m_textCtrl 
&& m_textCtrl
->IsModified() ) 
 333     int spin_value 
= event
.GetPosition(); 
 334     double step 
= (event
.GetEventType() == wxEVT_SCROLL_LINEUP
) ? 1 : -1; 
 336     // Use the spinbutton's acceleration, if any, but not if wrapping around 
 337     if (((spin_value 
>= 0) && (m_spin_value 
>= 0)) || ((spin_value 
<= 0) && (m_spin_value 
<= 0))) 
 338         step 
*= abs(spin_value 
- m_spin_value
); 
 340     double value 
= m_value 
+ step
*m_increment
; 
 342     // we can always reach the ends using the spinbutton 
 343     if (value 
< m_min
) value 
= m_min
; 
 344     if (value 
> m_max
) value 
= m_max
; 
 346     // Ignore the edges when it wraps since the up/down event may be opposite 
 347     // They are in GTK and Mac 
 348     if (abs(spin_value 
- m_spin_value
) > SPINCTRLBUT_MAX
) 
 350         m_spin_value 
= spin_value
; 
 354     m_spin_value 
= spin_value
; 
 356     if (InRange(value
) && DoSetValue(value
)) 
 360 void wxSpinCtrlGenericBase::OnTextEnter(wxCommandEvent
& event
) 
 367 void wxSpinCtrlGenericBase::OnTextChar(wxKeyEvent
& event
) 
 369     double value 
= m_value
; 
 370     switch ( event
.GetKeyCode() ) 
 373             value 
+= m_increment
; 
 377             value 
-= m_increment
; 
 381             value 
+= m_increment 
* 10.0; 
 385             value 
-= m_increment 
* 10.0; 
 393     if ( m_textCtrl 
&& m_textCtrl
->IsModified() ) 
 396     if ( DoSetValue(value
) ) 
 400 // ---------------------------------------------------------------------------- 
 401 // Textctrl functions 
 402 // ---------------------------------------------------------------------------- 
 404 void wxSpinCtrlGenericBase::SyncSpinToText() 
 410     if ( m_textCtrl
->GetValue().ToDouble(&textValue
) ) 
 412         if (textValue 
> m_max
) 
 414         else if (textValue 
< m_min
) 
 417         if (m_value 
!= textValue
) 
 419             DoSetValue(textValue
); 
 424         // textctrl is out of sync, discard and reset 
 429 // ---------------------------------------------------------------------------- 
 430 // changing value and range 
 431 // ---------------------------------------------------------------------------- 
 433 void wxSpinCtrlGenericBase::SetValue(const wxString
& text
) 
 435     wxCHECK_RET( m_textCtrl
, _T("invalid call to wxSpinCtrl::SetValue") ); 
 438     if ( text
.ToDouble(&val
) && InRange(val
) ) 
 442     else // not a number at all or out of range 
 444         m_textCtrl
->SetValue(text
); 
 445         m_textCtrl
->SetSelection(0, -1); 
 449 bool wxSpinCtrlGenericBase::DoSetValue(double val
) 
 451     wxCHECK_MSG( m_textCtrl
, false, _T("invalid call to wxSpinCtrl::SetValue") ); 
 456     if ( m_snap_to_ticks 
&& (m_increment 
!= 0) ) 
 458         double snap_value 
= val 
/ m_increment
; 
 460         if (wxFinite(snap_value
)) // FIXME what to do about a failure? 
 462             if ((snap_value 
- floor(snap_value
)) < (ceil(snap_value
) - snap_value
)) 
 463                 val 
= floor(snap_value
) * m_increment
; 
 465                 val 
= ceil(snap_value
) * m_increment
; 
 469     wxString 
str(wxString::Format(m_format
.c_str(), val
)); 
 471     if ((val 
!= m_value
) || (str 
!= m_textCtrl
->GetValue())) 
 474         str
.ToDouble( &m_value 
);    // wysiwyg for textctrl 
 475         m_textCtrl
->SetValue( str 
); 
 476         m_textCtrl
->DiscardEdits(); 
 483 void wxSpinCtrlGenericBase::DoSetRange(double min
, double max
) 
 489 void wxSpinCtrlGenericBase::DoSetIncrement(double inc
) 
 494 void wxSpinCtrlGenericBase::SetSnapToTicks(bool snap_to_ticks
) 
 496     m_snap_to_ticks 
= snap_to_ticks
; 
 500 void wxSpinCtrlGenericBase::SetSelection(long from
, long to
) 
 502     wxCHECK_RET( m_textCtrl
, _T("invalid call to wxSpinCtrl::SetSelection") ); 
 504     m_textCtrl
->SetSelection(from
, to
); 
 507 #ifndef wxHAS_NATIVE_SPINCTRL 
 509 //----------------------------------------------------------------------------- 
 511 //----------------------------------------------------------------------------- 
 513 IMPLEMENT_DYNAMIC_CLASS(wxSpinCtrl
, wxSpinCtrlGenericBase
) 
 515 void wxSpinCtrl::DoSendEvent() 
 517     wxSpinEvent 
event( wxEVT_COMMAND_SPINCTRL_UPDATED
, GetId()); 
 518     event
.SetEventObject( this ); 
 519     event
.SetPosition((int)(m_value 
+ 0.5)); // FIXME should be SetValue 
 520     event
.SetString(m_textCtrl
->GetValue()); 
 521     GetEventHandler()->ProcessEvent( event 
); 
 524 #endif // !wxHAS_NATIVE_SPINCTRL 
 526 //----------------------------------------------------------------------------- 
 528 //----------------------------------------------------------------------------- 
 530 IMPLEMENT_DYNAMIC_CLASS(wxSpinCtrlDouble
, wxSpinCtrlGenericBase
) 
 532 void wxSpinCtrlDouble::DoSendEvent() 
 534     wxSpinDoubleEvent 
event( wxEVT_COMMAND_SPINCTRLDOUBLE_UPDATED
, GetId()); 
 535     event
.SetEventObject( this ); 
 536     event
.SetValue(m_value
); 
 537     event
.SetString(m_textCtrl
->GetValue()); 
 538     GetEventHandler()->ProcessEvent( event 
); 
 541 void wxSpinCtrlDouble::SetDigits(unsigned digits
) 
 543     wxCHECK_RET( digits 
<= 20, "too many digits for wxSpinCtrlDouble" ); 
 545     m_format
.Printf(wxT("%%0.%ulf"), digits
); 
 550 #endif // wxUSE_SPINBTN 
 552 #endif // !wxPort-with-native-spinctrl 
 554 #endif // wxUSE_SPINCTRL