Scrolling fixes.
[wxWidgets.git] / src / gtk1 / scrolbar.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: scrolbar.cpp
3 // Purpose:
4 // Author: Robert Roebling
5 // Id: $Id$
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
9
10
11 #ifdef __GNUG__
12 #pragma implementation "scrolbar.h"
13 #endif
14
15 #include "wx/scrolbar.h"
16
17 #if wxUSE_SCROLLBAR
18
19 #include "wx/utils.h"
20
21 #include <math.h>
22
23 #include <gdk/gdk.h>
24 #include <gtk/gtk.h>
25
26 //-----------------------------------------------------------------------------
27 // idle system
28 //-----------------------------------------------------------------------------
29
30 extern void wxapp_install_idle_handler();
31 extern bool g_isIdle;
32
33 //-----------------------------------------------------------------------------
34 // data
35 //-----------------------------------------------------------------------------
36
37 extern bool g_blockEventsOnDrag;
38 extern bool g_blockEventsOnScroll;
39
40 static const float sensitivity = 0.02;
41
42 //-----------------------------------------------------------------------------
43 // "value_changed"
44 //-----------------------------------------------------------------------------
45
46 static void gtk_scrollbar_callback( GtkAdjustment *adjust, wxScrollBar *win )
47 {
48 if (g_isIdle) wxapp_install_idle_handler();
49
50 if (!win->m_hasVMT) return;
51 if (g_blockEventsOnDrag) return;
52
53 float diff = adjust->value - win->m_oldPos;
54 if (fabs(diff) < sensitivity) return;
55
56 win->m_oldPos = adjust->value;
57
58 GtkRange *range = GTK_RANGE( win->m_widget );
59
60 wxEventType command = wxEVT_SCROLL_THUMBTRACK;
61 if (range->scroll_type == GTK_SCROLL_STEP_BACKWARD) command = wxEVT_SCROLL_LINEUP;
62 else if (range->scroll_type == GTK_SCROLL_STEP_FORWARD) command = wxEVT_SCROLL_LINEDOWN;
63 else if (range->scroll_type == GTK_SCROLL_PAGE_BACKWARD) command = wxEVT_SCROLL_PAGEUP;
64 else if (range->scroll_type == GTK_SCROLL_PAGE_FORWARD) command = wxEVT_SCROLL_PAGEDOWN;
65
66 int value = (int)ceil(adjust->value);
67
68 int orient = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
69
70 wxScrollEvent event( command, win->GetId(), value, orient );
71 event.SetEventObject( win );
72 win->GetEventHandler()->ProcessEvent( event );
73
74 /*
75 wxCommandEvent cevent( wxEVT_COMMAND_SCROLLBAR_UPDATED, win->GetId() );
76 cevent.SetEventObject( win );
77 win->ProcessEvent( cevent );
78 */
79 }
80
81 //-----------------------------------------------------------------------------
82 // "button_press_event" from slider
83 //-----------------------------------------------------------------------------
84
85 static gint gtk_scrollbar_button_press_callback( GtkRange *widget,
86 GdkEventButton *gdk_event,
87 wxScrollBar *win )
88 {
89 if (g_isIdle) wxapp_install_idle_handler();
90
91 // g_blockEventsOnScroll = TRUE; doesn't work in DialogEd
92
93 win->m_isScrolling = (gdk_event->window == widget->slider);
94
95 return FALSE;
96 }
97
98 //-----------------------------------------------------------------------------
99 // "button_release_event" from slider
100 //-----------------------------------------------------------------------------
101
102 static gint gtk_scrollbar_button_release_callback( GtkRange *WXUNUSED(widget),
103 GdkEventButton *WXUNUSED(gdk_event),
104 wxScrollBar *win )
105 {
106 if (g_isIdle) wxapp_install_idle_handler();
107
108 // g_blockEventsOnScroll = FALSE;
109
110 if (win->m_isScrolling)
111 {
112 wxEventType command = wxEVT_SCROLL_THUMBTRACK;
113 int value = (int)ceil(win->m_adjust->value);
114 int dir = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
115
116 wxScrollEvent event( command, value, dir );
117 event.SetScrolling( FALSE );
118 event.SetEventObject( win );
119 win->GetEventHandler()->ProcessEvent( event );
120 }
121
122 win->m_isScrolling = FALSE;
123
124 return FALSE;
125 }
126
127 //-----------------------------------------------------------------------------
128 // wxScrollBar
129 //-----------------------------------------------------------------------------
130
131 IMPLEMENT_DYNAMIC_CLASS(wxScrollBar,wxControl)
132
133 wxScrollBar::~wxScrollBar(void)
134 {
135 }
136
137 bool wxScrollBar::Create(wxWindow *parent, wxWindowID id,
138 const wxPoint& pos, const wxSize& size,
139 long style, const wxValidator& validator, const wxString& name )
140 {
141 m_needParent = TRUE;
142 m_acceptsFocus = TRUE;
143
144 if (!PreCreation( parent, pos, size ) ||
145 !CreateBase( parent, id, pos, size, style, validator, name ))
146 {
147 wxFAIL_MSG( wxT("wxScrollBar creation failed") );
148 return FALSE;
149 }
150
151 m_oldPos = 0.0;
152
153 if ((style & wxSB_VERTICAL) == wxSB_VERTICAL)
154 m_widget = gtk_vscrollbar_new( (GtkAdjustment *) NULL );
155 else
156 m_widget = gtk_hscrollbar_new( (GtkAdjustment *) NULL );
157
158 m_adjust = gtk_range_get_adjustment( GTK_RANGE(m_widget) );
159
160 gtk_signal_connect( GTK_OBJECT(m_adjust),
161 "value_changed",
162 (GtkSignalFunc) gtk_scrollbar_callback,
163 (gpointer) this );
164
165 gtk_signal_connect( GTK_OBJECT(m_widget),
166 "button_press_event",
167 (GtkSignalFunc)gtk_scrollbar_button_press_callback,
168 (gpointer) this );
169
170 gtk_signal_connect( GTK_OBJECT(m_widget),
171 "button_release_event",
172 (GtkSignalFunc)gtk_scrollbar_button_release_callback,
173 (gpointer) this );
174
175 m_parent->DoAddChild( this );
176
177 PostCreation();
178
179 SetBackgroundColour( parent->GetBackgroundColour() );
180
181 Show( TRUE );
182
183 return TRUE;
184 }
185
186 int wxScrollBar::GetThumbPosition(void) const
187 {
188 return (int)(m_adjust->value+0.5);
189 }
190
191 int wxScrollBar::GetThumbSize() const
192 {
193 return (int)(m_adjust->page_size+0.5);
194 }
195
196 int wxScrollBar::GetPageSize() const
197 {
198 return (int)(m_adjust->page_increment+0.5);
199 }
200
201 int wxScrollBar::GetRange() const
202 {
203 return (int)(m_adjust->upper+0.5);
204 }
205
206 void wxScrollBar::SetThumbPosition( int viewStart )
207 {
208 if (m_isScrolling) return;
209
210 float fpos = (float)viewStart;
211 m_oldPos = fpos;
212 if (fabs(fpos-m_adjust->value) < 0.2) return;
213 m_adjust->value = fpos;
214
215 gtk_signal_disconnect_by_func( GTK_OBJECT(m_adjust),
216 (GtkSignalFunc) gtk_scrollbar_callback,
217 (gpointer) this );
218
219 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "value_changed" );
220
221 gtk_signal_connect( GTK_OBJECT(m_adjust),
222 "value_changed",
223 (GtkSignalFunc) gtk_scrollbar_callback,
224 (gpointer) this );
225 }
226
227 void wxScrollBar::SetScrollbar( int position, int thumbSize, int range, int pageSize,
228 bool WXUNUSED(refresh) )
229 {
230 float fpos = (float)position;
231 float frange = (float)range;
232 float fthumb = (float)thumbSize;
233 float fpage = (float)pageSize;
234
235 if ((fabs(frange-m_adjust->upper) < 0.2) &&
236 (fabs(fthumb-m_adjust->page_size) < 0.2) &&
237 (fabs(fpage-m_adjust->page_increment) < 0.2))
238 {
239 SetThumbPosition( position );
240 return;
241 }
242
243 m_oldPos = fpos;
244
245 m_adjust->lower = 0.0;
246 m_adjust->upper = frange;
247 m_adjust->value = fpos;
248 m_adjust->step_increment = 1.0;
249 m_adjust->page_increment = (float)(wxMax(fpage,0));
250 m_adjust->page_size = fthumb;
251
252 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "changed" );
253 }
254
255 /* Backward compatibility */
256 int wxScrollBar::GetValue(void) const
257 {
258 return GetThumbPosition();
259 }
260
261 void wxScrollBar::SetValue( int viewStart )
262 {
263 SetThumbPosition( viewStart );
264 }
265
266 void wxScrollBar::GetValues( int *viewStart, int *viewLength, int *objectLength, int *pageLength ) const
267 {
268 int pos = (int)(m_adjust->value+0.5);
269 int thumb = (int)(m_adjust->page_size+0.5);
270 int page = (int)(m_adjust->page_increment+0.5);
271 int range = (int)(m_adjust->upper+0.5);
272
273 *viewStart = pos;
274 *viewLength = range;
275 *objectLength = thumb;
276 *pageLength = page;
277 }
278
279 int wxScrollBar::GetViewLength() const
280 {
281 return (int)(m_adjust->upper+0.5);
282 }
283
284 int wxScrollBar::GetObjectLength() const
285 {
286 return (int)(m_adjust->page_size+0.5);
287 }
288
289 void wxScrollBar::SetPageSize( int pageLength )
290 {
291 int pos = (int)(m_adjust->value+0.5);
292 int thumb = (int)(m_adjust->page_size+0.5);
293 int range = (int)(m_adjust->upper+0.5);
294 SetScrollbar( pos, thumb, range, pageLength );
295 }
296
297 void wxScrollBar::SetObjectLength( int objectLength )
298 {
299 int pos = (int)(m_adjust->value+0.5);
300 int page = (int)(m_adjust->page_increment+0.5);
301 int range = (int)(m_adjust->upper+0.5);
302 SetScrollbar( pos, objectLength, range, page );
303 }
304
305 void wxScrollBar::SetViewLength( int viewLength )
306 {
307 int pos = (int)(m_adjust->value+0.5);
308 int thumb = (int)(m_adjust->page_size+0.5);
309 int page = (int)(m_adjust->page_increment+0.5);
310 SetScrollbar( pos, thumb, viewLength, page );
311 }
312
313 bool wxScrollBar::IsOwnGtkWindow( GdkWindow *window )
314 {
315 GtkRange *range = GTK_RANGE(m_widget);
316 return ( (window == GTK_WIDGET(range)->window) ||
317 (window == range->trough) ||
318 (window == range->slider) ||
319 (window == range->step_forw) ||
320 (window == range->step_back) );
321 }
322
323 void wxScrollBar::ApplyWidgetStyle()
324 {
325 SetWidgetStyle();
326 gtk_widget_set_style( m_widget, m_widgetStyle );
327 }
328
329 #endif