Implemented wxSpinButton for wxMotif.
[wxWidgets.git] / src / motif / spinbutt.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: spinbutt.cpp
3 // Purpose: wxSpinButton
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 17/09/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 #ifdef __GNUG__
13 #pragma implementation "spinbutt.h"
14 #endif
15
16 #include "wx/spinbutt.h"
17 #include "wx/spinctrl.h"
18 #include "wx/timer.h"
19
20 #ifdef __VMS__
21 #pragma message disable nosimpint
22 #endif
23 #include <Xm/ArrowBG.h>
24 #include <Xm/ArrowB.h>
25 #ifdef __VMS__
26 #pragma message enable nosimpint
27 #endif
28
29 #include "wx/motif/private.h"
30
31 // helper class
32 enum ArrowDirection
33 {
34 wxARROW_UP,
35 wxARROW_DOWN,
36 wxARROW_LEFT,
37 wxARROW_RIGHT
38 };
39
40 class wxArrowButtonTimer;
41 class wxArrowButton;
42
43 // ----------------------------------------------------------------------------
44 // wxArrowButtonTimer
45 // ----------------------------------------------------------------------------
46
47 static const unsigned int TICK_BEFORE_START = 10;
48 static const unsigned int TICK_BEFORE_EXPONENTIAL = 40;
49 static const unsigned int MAX_INCREMENT = 150;
50 static const unsigned int TICK_INTERVAL = 113;
51
52 class wxArrowButtonTimer : public wxTimer
53 {
54 public:
55 wxArrowButtonTimer( wxArrowButton* btn, int sign )
56 : m_sign( sign ),
57 m_button( btn )
58 { Reset(); };
59
60 void Notify();
61 void Reset() { m_ticks = 0; m_increment = 1; }
62 private:
63 unsigned int m_ticks;
64 unsigned int m_increment;
65 int m_sign;
66 wxArrowButton* m_button;
67 };
68
69 // ----------------------------------------------------------------------------
70 // wxArrowButton
71 // ----------------------------------------------------------------------------
72
73 class wxArrowButton : public wxControl
74 {
75 friend class wxArrowButtonTimer;
76 public:
77 wxArrowButton( int increment )
78 : m_increment( increment ),
79 m_timer( 0 ) {}
80
81 wxArrowButton( wxSpinButton* parent, wxWindowID id, ArrowDirection d,
82 const wxPoint& pos = wxDefaultPosition,
83 const wxSize& size = wxDefaultSize, int increment = 1 )
84 : wxControl(),
85 m_increment( increment ),
86 m_timer( 0 )
87 {
88 Create( parent, id, d, pos, size );
89 }
90
91 ~wxArrowButton()
92 { delete m_timer; }
93
94 bool Create( wxSpinButton* parent, wxWindowID id, ArrowDirection d,
95 const wxPoint& pos = wxDefaultPosition,
96 const wxSize& size = wxDefaultSize );
97 private:
98 // creates a new timer object, or stops the currently running one
99 wxTimer* GetFreshTimer();
100 wxSpinButton* GetSpinButton() { return (wxSpinButton*)GetParent(); }
101 static void SpinButtonCallback( Widget w, XtPointer clientData,
102 XtPointer WXUNUSED(ptr) );
103 static void StartTimerCallback( Widget w, XtPointer clientData,
104 XtPointer WXUNUSED(ptr) );
105
106 static void StopTimerCallback( Widget w, XtPointer clientData,
107 XtPointer WXUNUSED(ptr) );
108
109 int m_increment;
110 wxArrowButtonTimer* m_timer;
111 };
112
113 // ----------------------------------------------------------------------------
114 // wxArrowButtonTimer implementation
115 // ----------------------------------------------------------------------------
116
117 void wxArrowButtonTimer::Notify()
118 {
119 ++m_ticks;
120 if( m_ticks < TICK_BEFORE_START ) return;
121 // increment every other tick
122 if( m_ticks <= TICK_BEFORE_EXPONENTIAL && m_ticks & 1 )
123 return;
124 if( m_ticks > TICK_BEFORE_EXPONENTIAL )
125 m_increment = 2 * m_increment;
126 if( m_increment >= MAX_INCREMENT ) m_increment = MAX_INCREMENT;
127 m_button->GetSpinButton()->Increment( m_sign * m_increment );
128 }
129
130 // ----------------------------------------------------------------------------
131 // wxArrowButton implementation
132 // ----------------------------------------------------------------------------
133
134 wxTimer* wxArrowButton::GetFreshTimer()
135 {
136 if( m_timer )
137 {
138 m_timer->Stop();
139 m_timer->Reset();
140 }
141 else
142 m_timer = new wxArrowButtonTimer( this, m_increment );
143
144 return m_timer;
145 }
146
147 void wxArrowButton::SpinButtonCallback( Widget w, XtPointer clientData,
148 XtPointer WXUNUSED(ptr) )
149 {
150 if( !wxGetWindowFromTable( w ) )
151 // Widget has been deleted!
152 return;
153
154 wxArrowButton* btn = (wxArrowButton*)clientData;
155
156 btn->GetSpinButton()->Increment( btn->m_increment );
157 }
158
159 void wxArrowButton::StartTimerCallback( Widget w, XtPointer clientData,
160 XtPointer WXUNUSED(ptr) )
161 {
162 if( !wxGetWindowFromTable( w ) )
163 // Widget has been deleted!
164 return;
165
166 wxArrowButton* btn = (wxArrowButton*)clientData;
167 btn->GetFreshTimer()->Start( TICK_INTERVAL );
168 }
169
170 void wxArrowButton::StopTimerCallback( Widget w, XtPointer clientData,
171 XtPointer WXUNUSED(ptr) )
172 {
173 if( !wxGetWindowFromTable( w ) )
174 // Widget has been deleted!
175 return;
176
177 wxArrowButton* btn = (wxArrowButton*)clientData;
178 delete btn->m_timer;
179 btn->m_timer = 0;
180 }
181
182 bool wxArrowButton::Create( wxSpinButton* parent, wxWindowID id,
183 ArrowDirection d,
184 const wxPoint& pos, const wxSize& size )
185 {
186 int arrow_dir;
187
188 switch( d )
189 {
190 case wxARROW_UP:
191 arrow_dir = XmARROW_UP;
192 break;
193 case wxARROW_DOWN:
194 arrow_dir = XmARROW_DOWN;
195 break;
196 case wxARROW_LEFT:
197 arrow_dir = XmARROW_LEFT;
198 break;
199 case wxARROW_RIGHT:
200 arrow_dir = XmARROW_RIGHT;
201 break;
202 }
203
204 if( parent ) parent->AddChild( this );
205
206 Widget parentWidget = (Widget) parent->GetClientWidget();
207 m_mainWidget = (WXWidget) XtVaCreateManagedWidget( "XmArrowButton",
208 xmArrowButtonWidgetClass,
209 parentWidget,
210 XmNarrowDirection, arrow_dir,
211 NULL );
212
213 XtAddCallback( (Widget) m_mainWidget,
214 XmNactivateCallback, (XtCallbackProc) SpinButtonCallback,
215 (XtPointer) this );
216 XtAddCallback( (Widget) m_mainWidget,
217 XmNarmCallback, (XtCallbackProc) StartTimerCallback,
218 (XtPointer) this );
219 XtAddCallback( (Widget) m_mainWidget,
220 XmNactivateCallback, (XtCallbackProc) StopTimerCallback,
221 (XtPointer) this );
222
223 AttachWidget( parent, m_mainWidget, (WXWidget) NULL,
224 pos.x, pos.y, size.x, size.y );
225
226 return TRUE;
227 }
228
229 // ----------------------------------------------------------------------------
230 // wxSpinButton
231 // ----------------------------------------------------------------------------
232
233 IMPLEMENT_DYNAMIC_CLASS(wxSpinButton, wxControl);
234 IMPLEMENT_DYNAMIC_CLASS(wxSpinEvent, wxNotifyEvent);
235
236 static void CalcSizes( wxPoint pt, wxSize sz,
237 wxPoint& pt1, wxSize& sz1,
238 wxPoint& pt2, wxSize& sz2,
239 bool isVertical )
240 {
241 typedef int wxSize::* CDPTR1;
242 typedef int wxPoint::* CDPTR2;
243
244 sz1 = sz2 = sz;
245 pt2 = pt1 = pt;
246
247 CDPTR1 szm = isVertical ? &wxSize::y : &wxSize::x;
248 CDPTR2 ptm = isVertical ? &wxPoint::y : &wxPoint::x;
249 int dim = sz.*szm, half = dim/2;
250
251 sz1.*szm = half;
252 sz2.*szm = dim - half;
253 pt2.*ptm += half + 1;
254 }
255
256 bool wxSpinButton::Create( wxWindow *parent, wxWindowID id,
257 const wxPoint& pos, const wxSize& size,
258 long style, const wxString& name )
259 {
260 m_windowStyle = style;
261
262 wxSize newSize = GetBestSize();
263 if( size.x != -1 ) newSize.x = size.x;
264 if( size.y != -1 ) newSize.y = size.y;
265
266 if( !wxControl::Create( parent, id, pos, newSize, style ) )
267 {
268 return FALSE;
269 }
270
271 SetName(name);
272
273 InitBase();
274
275 m_windowId = ( id == -1 ) ? NewControlId() : id;
276
277 bool isVert = IsVertical();
278 wxPoint pt1, pt2;
279 wxSize sz1, sz2;
280 CalcSizes( wxPoint(0,0), newSize, pt1, sz1, pt2, sz2, isVert );
281 m_up = new wxArrowButton( this, -1, isVert ? wxARROW_UP : wxARROW_LEFT,
282 pt1, sz1, 1 );
283 m_down = new wxArrowButton( this, -1,
284 isVert ? wxARROW_DOWN : wxARROW_RIGHT,
285 pt2, sz2, -1 );
286
287 return TRUE;
288 }
289
290 wxSpinButton::~wxSpinButton()
291 {
292 }
293
294 void wxSpinButton::DoMoveWindow(int x, int y, int width, int height)
295 {
296 wxControl::DoMoveWindow( x, y, width, height );
297
298 wxPoint pt1, pt2;
299 wxSize sz1, sz2;
300
301 CalcSizes( wxPoint(0,0), wxSize(width,height), pt1,
302 sz1, pt2, sz2, IsVertical() );
303 m_up->SetSize( pt1.x, pt1.y, sz1.x, sz1.y );
304 m_down->SetSize( pt2.x, pt2.y, sz2.x, sz2.y );
305 }
306
307 void wxSpinButton::DoSetSize(int x, int y, int width, int height,
308 int sizeFlags)
309 {
310 if( sizeFlags & wxSIZE_AUTO_WIDTH && width == -1 )
311 width = 30;
312 if( sizeFlags & wxSIZE_AUTO_HEIGHT && height == -1 )
313 height = 30;
314 if( sizeFlags & wxSIZE_USE_EXISTING && width == -1 )
315 width = GetSize().x;
316 if( sizeFlags & wxSIZE_USE_EXISTING && height == -1 )
317 height = GetSize().y;
318
319 wxControl::DoSetSize(x, y, width, height, 0);
320 }
321
322 void wxSpinButton::Increment( int delta )
323 {
324 if( m_pos < m_min ) m_pos = m_min;
325 if( m_pos > m_max ) m_pos = m_max;
326
327 int npos = m_pos + delta;
328
329 if( npos < m_min ) npos = m_min;
330 if( npos > m_max ) npos = m_max;
331 if( npos == m_pos ) return;
332
333 wxSpinEvent event( delta > 0 ? wxEVT_SCROLL_LINEUP : wxEVT_SCROLL_LINEDOWN,
334 m_windowId );
335 event.SetPosition( npos );
336 event.SetEventObject( this );
337
338 GetEventHandler()->ProcessEvent( event );
339
340 if( event.IsAllowed() )
341 {
342 m_pos = npos;
343 event.SetEventType( wxEVT_SCROLL_THUMBTRACK );
344 event.SetPosition( m_pos );
345
346 GetEventHandler()->ProcessEvent( event );
347 }
348 }
349
350 wxSize wxSpinButton::DoGetBestSize() const
351 {
352 return IsVertical() ? wxSize( 24, 34 ) : wxSize( 34, 24 );
353 }
354
355 // Attributes
356 ////////////////////////////////////////////////////////////////////////////
357
358 int wxSpinButton::GetValue() const
359 {
360 return m_pos;
361 }
362
363 void wxSpinButton::SetValue(int val)
364 {
365 m_pos = val;
366 }
367
368 void wxSpinButton::SetRange(int minVal, int maxVal)
369 {
370 wxSpinButtonBase::SetRange(minVal, maxVal);
371 }
372
373 void wxSpinButton::ChangeFont(bool WXUNUSED(keepOriginalSize))
374 {
375 // TODO
376 }
377
378 void wxSpinButton::ChangeBackgroundColour()
379 {
380 wxControl::ChangeBackgroundColour();
381 }
382
383 void wxSpinButton::ChangeForegroundColour()
384 {
385 // TODO
386 }