Fixed bug that caused wrong block of cells to be selected if the
[wxWidgets.git] / src / gtk1 / spinbutt.cpp
CommitLineData
8e189077
RR
1/////////////////////////////////////////////////////////////////////////////
2// Name: spinbutt.cpp
3// Purpose: wxSpinButton
4// Author: Robert
5// Modified by:
6// RCS-ID: $Id$
7// Copyright: (c) Robert Roebling
6380910c 8// Licence: wxWindows licence
8e189077
RR
9/////////////////////////////////////////////////////////////////////////////
10
11#ifdef __GNUG__
12#pragma implementation "spinbutt.h"
13#endif
14
15#include "wx/spinbutt.h"
dcf924a3
RR
16
17#ifdef wxUSE_SPINBTN
18
8e189077 19#include "wx/utils.h"
f5368809 20#include <math.h>
8e189077 21
83624f79
RR
22#include "gdk/gdk.h"
23#include "gtk/gtk.h"
24
acfd422a
RR
25//-----------------------------------------------------------------------------
26// idle system
27//-----------------------------------------------------------------------------
28
29extern void wxapp_install_idle_handler();
30extern bool g_isIdle;
31
8e189077
RR
32//-----------------------------------------------------------------------------
33// data
34//-----------------------------------------------------------------------------
35
36extern bool g_blockEventsOnDrag;
37
738f9e5a 38static const float sensitivity = 0.02;
6380910c 39
8e189077
RR
40//-----------------------------------------------------------------------------
41// "value_changed"
42//-----------------------------------------------------------------------------
43
44static void gtk_spinbutt_callback( GtkWidget *WXUNUSED(widget), wxSpinButton *win )
6380910c 45{
acfd422a
RR
46 if (g_isIdle) wxapp_install_idle_handler();
47
a2053b27 48 if (!win->m_hasVMT) return;
83624f79 49 if (g_blockEventsOnDrag) return;
828f655f 50
83624f79 51 float diff = win->m_adjust->value - win->m_oldPos;
6380910c 52 if (fabs(diff) < sensitivity) return;
828f655f 53 win->m_oldPos = win->m_adjust->value;
6380910c 54
83624f79 55 wxEventType command = wxEVT_NULL;
6380910c 56
83624f79 57 float line_step = win->m_adjust->step_increment;
6380910c
VZ
58
59 if (fabs(diff-line_step) < sensitivity) command = wxEVT_SCROLL_LINEDOWN;
60 else if (fabs(diff+line_step) < sensitivity) command = wxEVT_SCROLL_LINEUP;
83624f79 61 else command = wxEVT_SCROLL_THUMBTRACK;
8e189077 62
66d78146 63 int value = (int)ceil(win->m_adjust->value);
6380910c 64
83624f79
RR
65 wxSpinEvent event( command, win->GetId());
66 event.SetPosition( value );
83624f79 67 event.SetEventObject( win );
828f655f 68 win->GetEventHandler()->ProcessEvent( event );
e65cc56a
RR
69
70 /* always send a thumbtrack event */
71 if (command != wxEVT_SCROLL_THUMBTRACK)
72 {
73 command = wxEVT_SCROLL_THUMBTRACK;
74 wxSpinEvent event2( command, win->GetId());
75 event2.SetPosition( value );
76 event2.SetEventObject( win );
77 win->GetEventHandler()->ProcessEvent( event2 );
78 }
8e189077
RR
79}
80
81//-----------------------------------------------------------------------------
82// wxSpinButton
83//-----------------------------------------------------------------------------
84
85IMPLEMENT_DYNAMIC_CLASS(wxSpinButton,wxControl)
31528cd3 86IMPLEMENT_DYNAMIC_CLASS(wxSpinEvent, wxScrollEvent);
8e189077
RR
87
88BEGIN_EVENT_TABLE(wxSpinButton, wxControl)
89 EVT_SIZE(wxSpinButton::OnSize)
90END_EVENT_TABLE()
91
31528cd3
VZ
92bool wxSpinButton::Create(wxWindow *parent,
93 wxWindowID id,
94 const wxPoint& pos,
95 const wxSize& size,
96 long style,
97 const wxString& name)
8e189077 98{
83624f79 99 m_needParent = TRUE;
6380910c 100
83624f79 101 wxSize new_size = size;
19da4326 102 new_size.x = 15;
6380910c 103 if (new_size.y == -1)
738f9e5a 104 new_size.y = 26;
6380910c 105
2259e007 106 if (!PreCreation( parent, pos, new_size ) ||
4dcaf11a
RR
107 !CreateBase( parent, id, pos, new_size, style, wxDefaultValidator, name ))
108 {
223d09f6 109 wxFAIL_MSG( wxT("wxXX creation failed") );
4dcaf11a
RR
110 return FALSE;
111 }
8e189077 112
83624f79 113 m_oldPos = 0.0;
8e189077 114
83624f79 115 m_adjust = (GtkAdjustment*) gtk_adjustment_new( 0.0, 0.0, 100.0, 1.0, 5.0, 0.0);
6380910c 116
83624f79 117 m_widget = gtk_spin_button_new( m_adjust, 0, 0 );
6380910c 118
83624f79 119 gtk_spin_button_set_wrap( GTK_SPIN_BUTTON(m_widget), (m_windowStyle & wxSP_WRAP) );
6380910c
VZ
120
121 gtk_signal_connect( GTK_OBJECT (m_adjust),
83624f79 122 "value_changed",
6380910c
VZ
123 (GtkSignalFunc) gtk_spinbutt_callback,
124 (gpointer) this );
125
f03fc89f 126 m_parent->DoAddChild( this );
6380910c 127
83624f79 128 PostCreation();
6380910c 129
83624f79 130 SetBackgroundColour( parent->GetBackgroundColour() );
8e189077 131
83624f79 132 Show( TRUE );
6380910c 133
83624f79 134 return TRUE;
8e189077
RR
135}
136
8e189077
RR
137int wxSpinButton::GetMin() const
138{
223d09f6 139 wxCHECK_MSG( (m_widget != NULL), 0, wxT("invalid spin button") );
6380910c 140
66d78146 141 return (int)ceil(m_adjust->lower);
8e189077
RR
142}
143
144int wxSpinButton::GetMax() const
145{
223d09f6 146 wxCHECK_MSG( (m_widget != NULL), 0, wxT("invalid spin button") );
6380910c 147
66d78146 148 return (int)ceil(m_adjust->upper);
8e189077
RR
149}
150
151int wxSpinButton::GetValue() const
152{
223d09f6 153 wxCHECK_MSG( (m_widget != NULL), 0, wxT("invalid spin button") );
6380910c 154
66d78146 155 return (int)ceil(m_adjust->value);
8e189077
RR
156}
157
158void wxSpinButton::SetValue( int value )
159{
223d09f6 160 wxCHECK_RET( (m_widget != NULL), wxT("invalid spin button") );
6380910c 161
83624f79
RR
162 float fpos = (float)value;
163 m_oldPos = fpos;
6380910c
VZ
164 if (fabs(fpos-m_adjust->value) < sensitivity) return;
165
83624f79 166 m_adjust->value = fpos;
6380910c 167
83624f79 168 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "value_changed" );
8e189077
RR
169}
170
171void wxSpinButton::SetRange(int minVal, int maxVal)
172{
223d09f6 173 wxCHECK_RET( (m_widget != NULL), wxT("invalid spin button") );
6380910c 174
83624f79
RR
175 float fmin = (float)minVal;
176 float fmax = (float)maxVal;
6380910c
VZ
177
178 if ((fabs(fmin-m_adjust->lower) < sensitivity) &&
179 (fabs(fmax-m_adjust->upper) < sensitivity))
83624f79
RR
180 {
181 return;
182 }
6380910c 183
83624f79
RR
184 m_adjust->lower = fmin;
185 m_adjust->upper = fmax;
8e189077 186
83624f79 187 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "changed" );
65045edd
RR
188
189 // these two calls are required due to some bug in GTK
190 Refresh();
191 SetFocus();
8e189077
RR
192}
193
194void wxSpinButton::OnSize( wxSizeEvent &WXUNUSED(event) )
195{
223d09f6 196 wxCHECK_RET( (m_widget != NULL), wxT("invalid spin button") );
6380910c 197
19da4326 198 m_width = 15;
83624f79 199 gtk_widget_set_usize( m_widget, m_width, m_height );
8e189077
RR
200}
201
202bool wxSpinButton::IsOwnGtkWindow( GdkWindow *window )
203{
83624f79 204 return GTK_SPIN_BUTTON(m_widget)->panel == window;
8e189077
RR
205}
206
207void wxSpinButton::ApplyWidgetStyle()
208{
83624f79
RR
209 SetWidgetStyle();
210 gtk_widget_set_style( m_widget, m_widgetStyle );
8e189077
RR
211}
212
31528cd3 213#endif