1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/spinctrl_osx.cpp
5 // Modified by: Mark Newsam (Based on GTK file)
7 // Copyright: (c) Robert Roebling
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 #include "wx/wxprec.h"
15 #include "wx/spinctrl.h"
18 #include "wx/textctrl.h"
19 #include "wx/containr.h"
22 #include "wx/spinbutt.h"
24 // ----------------------------------------------------------------------------
26 // ----------------------------------------------------------------------------
28 // the focus rect around a text may have 4 pixels in each direction
29 // we handle these problems right now in an extended vis region of a window
30 static const wxCoord TEXTBORDER
= 4 ;
31 // the margin between the text control and the spin
32 // HIG says 2px between text and stepper control,
33 // but a value of 3 leads to the same look as the
34 // spin controls in Apple's apps
35 static const wxCoord MARGIN
= 3;
37 // ----------------------------------------------------------------------------
38 // wxSpinCtrlText: text control used by spin control
39 // ----------------------------------------------------------------------------
41 class wxSpinCtrlText
: public wxTextCtrl
44 wxSpinCtrlText(wxSpinCtrl
*spin
, const wxString
& value
, int style
)
45 : wxTextCtrl(spin
, wxID_ANY
, value
, wxDefaultPosition
, wxSize(40, wxDefaultCoord
), style
)
49 // remove the default minsize, the spinctrl will have one instead
50 SetMinSize(wxDefaultSize
);
53 bool ProcessEvent(wxEvent
&event
)
55 // Hand button down events to wxSpinCtrl. Doesn't work.
56 if (event
.GetEventType() == wxEVT_LEFT_DOWN
&& m_spin
->ProcessEvent( event
))
59 return wxTextCtrl::ProcessEvent( event
);
63 void OnSetFocus(wxFocusEvent
& event
)
65 // delegate to parent control
66 event
.SetEventObject( GetParent() );
67 GetParent()->HandleWindowEvent(event
);
70 void OnKillFocus(wxFocusEvent
& event
)
73 if ( !GetValue().ToLong(&l
) )
75 // not a number at all
80 if (l
< m_spin
->GetMin())
82 if (l
> m_spin
->GetMax())
85 // Update text control
87 str
.Printf( wxT("%d"), (int)l
);
88 if (str
!= GetValue())
91 if (l
!= m_spin
->m_oldValue
)
93 // set value in spin button
94 // does that trigger an event?
95 m_spin
->m_btn
->SetValue( l
);
98 wxCommandEvent
cevent(wxEVT_COMMAND_SPINCTRL_UPDATED
, m_spin
->GetId());
99 cevent
.SetEventObject(m_spin
);
101 m_spin
->HandleWindowEvent(cevent
);
103 m_spin
->m_oldValue
= l
;
106 // delegate to parent control
107 event
.SetEventObject( GetParent() );
108 GetParent()->HandleWindowEvent(event
);
111 void OnTextChange(wxCommandEvent
& event
)
114 if ( m_spin
->GetTextValue(&val
) )
116 m_spin
->GetSpinButton()->SetValue(val
);
118 // If we're already processing a text update from m_spin,
119 // don't send it again, since we could end up recursing
121 if (event
.GetId() == m_spin
->GetId())
127 // Send event that the text was manually changed
128 wxCommandEvent
event(wxEVT_COMMAND_TEXT_UPDATED
, m_spin
->GetId());
129 event
.SetEventObject(m_spin
);
130 event
.SetString(m_spin
->GetText()->GetValue());
133 m_spin
->HandleWindowEvent(event
);
142 DECLARE_EVENT_TABLE()
145 BEGIN_EVENT_TABLE(wxSpinCtrlText
, wxTextCtrl
)
146 EVT_TEXT(wxID_ANY
, wxSpinCtrlText::OnTextChange
)
147 EVT_SET_FOCUS(wxSpinCtrlText::OnSetFocus
)
148 EVT_KILL_FOCUS(wxSpinCtrlText::OnKillFocus
)
151 // ----------------------------------------------------------------------------
152 // wxSpinCtrlButton: spin button used by spin control
153 // ----------------------------------------------------------------------------
155 class wxSpinCtrlButton
: public wxSpinButton
158 wxSpinCtrlButton(wxSpinCtrl
*spin
, int style
)
159 : wxSpinButton(spin
)
162 SetWindowStyle(style
| wxSP_VERTICAL
);
164 // TODO: The spin button gets truncated a little bit due to size
165 // differences so change it's default size a bit. SMALL still gets a
166 // bit truncated, but MINI seems to be too small... Readdress this
167 // when the textctrl issues are all sorted out.
168 //SetWindowVariant(wxWINDOW_VARIANT_SMALL);
170 // remove the default minsize, the spinctrl will have one instead
171 SetMinSize(wxDefaultSize
);
175 void OnSpinButton(wxSpinEvent
& eventSpin
)
177 int pos
= eventSpin
.GetPosition();
178 m_spin
->SetTextValue(pos
);
180 wxCommandEvent
event(wxEVT_COMMAND_SPINCTRL_UPDATED
, m_spin
->GetId());
181 event
.SetEventObject(m_spin
);
184 m_spin
->HandleWindowEvent(event
);
186 m_spin
->m_oldValue
= pos
;
192 DECLARE_EVENT_TABLE()
195 BEGIN_EVENT_TABLE(wxSpinCtrlButton
, wxSpinButton
)
196 EVT_SPIN(wxID_ANY
, wxSpinCtrlButton::OnSpinButton
)
199 BEGIN_EVENT_TABLE(wxSpinCtrl
, wxControl
)
200 WX_EVENT_TABLE_CONTROL_CONTAINER(wxSpinCtrl
)
203 WX_DELEGATE_TO_CONTROL_CONTAINER(wxSpinCtrl
, wxControl
)
206 // ============================================================================
208 // ============================================================================
210 // ----------------------------------------------------------------------------
211 // wxSpinCtrl creation
212 // ----------------------------------------------------------------------------
214 void wxSpinCtrl::Init()
218 WX_INIT_CONTROL_CONTAINER();
221 bool wxSpinCtrl::Create(wxWindow
*parent
,
223 const wxString
& value
,
230 const wxString
& name
)
232 if ( !wxControl::Create(parent
, id
, pos
, size
, style
,
233 wxDefaultValidator
, name
) )
238 // the string value overrides the numeric one (for backwards compatibility
239 // reasons and also because it is simpler to satisfy the string value which
240 // comes much sooner in the list of arguments and leave the initial
241 // parameter unspecified)
242 if ( !value
.empty() )
245 if ( value
.ToLong(&l
) )
249 wxSize csize
= size
;
250 m_text
= new wxSpinCtrlText(this, value
, style
& ( wxTE_PROCESS_ENTER
| wxALIGN_MASK
) );
251 m_btn
= new wxSpinCtrlButton(this, style
);
253 m_btn
->SetRange(min
, max
);
254 m_btn
->SetValue(initial
);
256 m_oldValue
= GetMin()-1;
258 if ( size
.x
== wxDefaultCoord
){
259 csize
.x
= m_text
->GetSize().x
+ MARGIN
+ m_btn
->GetSize().x
;
262 if ( size
.y
== wxDefaultCoord
) {
263 csize
.y
= m_text
->GetSize().y
+ 2 * TEXTBORDER
; //allow for text border highlights
264 if ( m_btn
->GetSize().y
> csize
.y
)
265 csize
.y
= m_btn
->GetSize().y
;
270 //MacPostControlCreate(pos, csize);
271 SetInitialSize(csize
);
276 wxSpinCtrl::~wxSpinCtrl()
278 // delete the controls now, don't leave them alive even though they would
279 // still be eventually deleted by our parent - but it will be too late, the
280 // user code expects them to be gone now
285 // ----------------------------------------------------------------------------
287 // ----------------------------------------------------------------------------
289 wxSize
wxSpinCtrl::DoGetBestSize() const
291 if (!m_btn
|| !m_text
)
294 wxSize sizeBtn
= m_btn
->GetBestSize(),
295 sizeText
= m_text
->GetBestSize();
297 sizeText
.y
+= 2 * TEXTBORDER
;
298 sizeText
.x
+= 2 * TEXTBORDER
;
301 if (sizeText
.y
> sizeBtn
.y
)
306 return wxSize(sizeBtn
.x
+ sizeText
.x
+ MARGIN
, height
);
309 void wxSpinCtrl::DoMoveWindow(int x
, int y
, int width
, int height
)
311 // position the subcontrols inside the client area
312 wxSize sizeBtn
= m_btn
->GetSize();
313 wxSize sizeText
= m_text
->GetSize();
315 wxControl::DoMoveWindow(x
, y
, width
, height
);
317 wxCoord wText
= width
- sizeBtn
.x
- MARGIN
- 2 * TEXTBORDER
;
319 m_text
->SetSize(TEXTBORDER
, (height
- sizeText
.y
) / 2, wText
, -1);
320 m_btn
->SetSize(0 + wText
+ MARGIN
+ TEXTBORDER
, (height
- sizeBtn
.y
) / 2 , -1, -1 );
323 // ----------------------------------------------------------------------------
324 // operations forwarded to the subcontrols
325 // ----------------------------------------------------------------------------
327 bool wxSpinCtrl::Enable(bool enable
)
329 if ( !wxControl::Enable(enable
) )
334 bool wxSpinCtrl::Show(bool show
)
336 if ( !wxControl::Show(show
) )
341 // ----------------------------------------------------------------------------
342 // value and range access
343 // ----------------------------------------------------------------------------
345 bool wxSpinCtrl::GetTextValue(int *val
) const
348 if ( !m_text
->GetValue().ToLong(&l
) )
350 // not a number at all
354 if ( l
< GetMin() || l
> GetMax() )
365 int wxSpinCtrl::GetValue() const
367 return m_btn
? m_btn
->GetValue() : 0;
370 int wxSpinCtrl::GetMin() const
372 return m_btn
? m_btn
->GetMin() : 0;
375 int wxSpinCtrl::GetMax() const
377 return m_btn
? m_btn
->GetMax() : 0;
380 // ----------------------------------------------------------------------------
381 // changing value and range
382 // ----------------------------------------------------------------------------
384 void wxSpinCtrl::SetTextValue(int val
)
386 wxCHECK_RET( m_text
, wxT("invalid call to wxSpinCtrl::SetTextValue") );
388 m_text
->SetValue(wxString::Format(wxT("%d"), val
));
391 m_text
->SetSelection(0, -1);
393 m_text
->SetInsertionPointEnd();
395 // and give focus to the control!
396 // m_text->SetFocus(); Why???? TODO.
399 void wxSpinCtrl::SetValue(int val
)
401 wxCHECK_RET( m_btn
, wxT("invalid call to wxSpinCtrl::SetValue") );
405 m_btn
->SetValue(val
);
409 void wxSpinCtrl::SetValue(const wxString
& text
)
411 wxCHECK_RET( m_text
, wxT("invalid call to wxSpinCtrl::SetValue") );
414 if ( text
.ToLong(&val
) && ((val
> INT_MIN
) && (val
< INT_MAX
)) )
418 else // not a number at all or out of range
420 m_text
->SetValue(text
);
421 m_text
->SetSelection(0, -1);
425 void wxSpinCtrl::SetRange(int min
, int max
)
427 wxCHECK_RET( m_btn
, wxT("invalid call to wxSpinCtrl::SetRange") );
429 m_btn
->SetRange(min
, max
);
432 void wxSpinCtrl::SetSelection(long from
, long to
)
434 // if from and to are both -1, it means (in wxWidgets) that all text should
436 if ( (from
== -1) && (to
== -1) )
440 m_text
->SetSelection(from
, to
);
443 #endif // wxUSE_SPINCTRL