]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk1/scrolbar.cpp
Fix memory leak when a spacer is added, and crash when a window is added before wxSiz...
[wxWidgets.git] / src / gtk1 / scrolbar.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/gtk/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#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
12#pragma implementation "scrolbar.h"
13#endif
14
15// For compilers that support precompilation, includes "wx.h".
16#include "wx/wxprec.h"
17
18#if wxUSE_SCROLLBAR
19
20#include "wx/scrolbar.h"
21
22#include "wx/utils.h"
23#include "wx/math.h"
24#include "wx/gtk/private.h"
25
26//-----------------------------------------------------------------------------
27// idle system
28//-----------------------------------------------------------------------------
29
30extern void wxapp_install_idle_handler();
31extern bool g_isIdle;
32
33//-----------------------------------------------------------------------------
34// data
35//-----------------------------------------------------------------------------
36
37extern bool g_blockEventsOnDrag;
38extern bool g_blockEventsOnScroll;
39static wxEventType g_currentUpDownEvent = wxEVT_NULL;
40
41static const float sensitivity = 0.02;
42
43//-----------------------------------------------------------------------------
44// "value_changed"
45//-----------------------------------------------------------------------------
46
47// FIXME: is GtkScrollType really passed to us as 2nd argument?
48
49extern "C" {
50static void gtk_scrollbar_callback( GtkAdjustment *adjust,
51 SCROLLBAR_CBACK_ARG
52 wxScrollBar *win )
53{
54 if (g_isIdle) wxapp_install_idle_handler();
55
56 if (!win->m_hasVMT) return;
57 if (g_blockEventsOnDrag) return;
58
59 float diff = adjust->value - win->m_oldPos;
60 if (fabs(diff) < sensitivity) return;
61
62 win->m_oldPos = adjust->value;
63
64 wxEventType command = GtkScrollTypeToWx(GET_SCROLL_TYPE(win->m_widget));
65
66 double dvalue = adjust->value;
67 int value = (int)(dvalue < 0 ? dvalue - 0.5 : dvalue + 0.5);
68
69 int orient = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
70
71 // throw a LINEUP / LINEDOWN event if necessary
72 if (g_currentUpDownEvent != wxEVT_NULL)
73 {
74 wxScrollEvent event( g_currentUpDownEvent, win->GetId(), value, orient );
75 event.SetEventObject( win );
76 win->GetEventHandler()->ProcessEvent( event );
77 }
78
79 // throw other event (wxEVT_SCROLL_THUMBTRACK)
80 wxScrollEvent event( command, win->GetId(), value, orient );
81 event.SetEventObject( win );
82 win->GetEventHandler()->ProcessEvent( event );
83
84/*
85 wxCommandEvent cevent( wxEVT_COMMAND_SCROLLBAR_UPDATED, win->GetId() );
86 cevent.SetEventObject( win );
87 win->ProcessEvent( cevent );
88*/
89}
90}
91
92//-----------------------------------------------------------------------------
93// "button_press_event" from slider
94//-----------------------------------------------------------------------------
95extern "C" {
96static gint gtk_scrollbar_button_press_callback( GtkRange *widget,
97 GdkEventButton *gdk_event,
98 wxScrollBar *win )
99{
100 if (g_isIdle) wxapp_install_idle_handler();
101
102 // check if a LINEUP/LINEDOWN event must be thrown
103 // I suppose here the size of scrollbar top/bottom buttons is 16px height
104 if (gdk_event->type == GDK_BUTTON_PRESS && gdk_event->button == 1)
105 {
106 int scroll_height, mouse_pos;
107
108 // get the mouse position when the click is done
109 if (win->HasFlag(wxSB_VERTICAL))
110 {
111 scroll_height = GTK_WIDGET(widget)->allocation.height - 16;
112 mouse_pos = (int)gdk_event->y;
113 }
114 else
115 {
116 scroll_height = GTK_WIDGET(widget)->allocation.width - 16;
117 mouse_pos = (int)gdk_event->x;
118 }
119
120 // compare mouse position to scrollbar height
121 if (mouse_pos > scroll_height)
122 g_currentUpDownEvent = wxEVT_SCROLL_LINEDOWN;
123 else if (mouse_pos < 16)
124 g_currentUpDownEvent = wxEVT_SCROLL_LINEUP;
125 }
126
127#ifndef __WXGTK20__
128 // There is no slider field any more
129 win->m_isScrolling = (gdk_event->window == widget->slider);
130#endif
131
132 return FALSE;
133}
134}
135
136//-----------------------------------------------------------------------------
137// "button_release_event" from slider
138//-----------------------------------------------------------------------------
139
140extern "C" {
141static gint
142gtk_scrollbar_button_release_callback( GtkRange *WXUNUSED(widget),
143 GdkEventButton *WXUNUSED(gdk_event),
144 wxScrollBar *win )
145{
146 if (g_isIdle)
147 wxapp_install_idle_handler();
148
149 if (win->m_isScrolling)
150 {
151 wxEventType command = wxEVT_SCROLL_THUMBRELEASE;
152 int value = (int)ceil(win->m_adjust->value);
153 int orient = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
154
155 wxScrollEvent event( command, win->GetId(), value, orient );
156 event.SetEventObject( win );
157 win->GetEventHandler()->ProcessEvent( event );
158 }
159
160 win->m_isScrolling = FALSE;
161
162 // reset the LINEUP/LINEDOWN flag when the mouse button is released
163 g_currentUpDownEvent = wxEVT_NULL;
164
165 return FALSE;
166}
167}
168
169//-----------------------------------------------------------------------------
170// wxScrollBar
171//-----------------------------------------------------------------------------
172
173IMPLEMENT_DYNAMIC_CLASS(wxScrollBar,wxControl)
174
175wxScrollBar::~wxScrollBar()
176{
177}
178
179bool wxScrollBar::Create(wxWindow *parent, wxWindowID id,
180 const wxPoint& pos, const wxSize& size,
181 long style, const wxValidator& validator, const wxString& name )
182{
183 m_needParent = TRUE;
184 m_acceptsFocus = TRUE;
185
186 if (!PreCreation( parent, pos, size ) ||
187 !CreateBase( parent, id, pos, size, style, validator, name ))
188 {
189 wxFAIL_MSG( wxT("wxScrollBar creation failed") );
190 return FALSE;
191 }
192
193 m_oldPos = 0.0;
194
195 if ((style & wxSB_VERTICAL) == wxSB_VERTICAL)
196 m_widget = gtk_vscrollbar_new( (GtkAdjustment *) NULL );
197 else
198 m_widget = gtk_hscrollbar_new( (GtkAdjustment *) NULL );
199
200 m_adjust = gtk_range_get_adjustment( GTK_RANGE(m_widget) );
201
202 gtk_signal_connect( GTK_OBJECT(m_adjust),
203 "value_changed",
204 (GtkSignalFunc) gtk_scrollbar_callback,
205 (gpointer) this );
206 gtk_signal_connect( GTK_OBJECT(m_widget),
207 "button_press_event",
208 (GtkSignalFunc)gtk_scrollbar_button_press_callback,
209 (gpointer) this );
210 gtk_signal_connect( GTK_OBJECT(m_widget),
211 "button_release_event",
212 (GtkSignalFunc)gtk_scrollbar_button_release_callback,
213 (gpointer) this );
214
215 m_parent->DoAddChild( this );
216
217 PostCreation(size);
218
219 return TRUE;
220}
221
222int wxScrollBar::GetThumbPosition() const
223{
224 double val = m_adjust->value;
225 return (int)(val < 0 ? val - 0.5 : val + 0.5);
226}
227
228int wxScrollBar::GetThumbSize() const
229{
230 return (int)(m_adjust->page_size+0.5);
231}
232
233int wxScrollBar::GetPageSize() const
234{
235 return (int)(m_adjust->page_increment+0.5);
236}
237
238int wxScrollBar::GetRange() const
239{
240 return (int)(m_adjust->upper+0.5);
241}
242
243void wxScrollBar::SetThumbPosition( int viewStart )
244{
245 if (m_isScrolling) return;
246
247 float fpos = (float)viewStart;
248 m_oldPos = fpos;
249 if (fabs(fpos-m_adjust->value) < 0.2) return;
250 m_adjust->value = fpos;
251
252 gtk_signal_disconnect_by_func( GTK_OBJECT(m_adjust),
253 (GtkSignalFunc) gtk_scrollbar_callback,
254 (gpointer) this );
255
256 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "value_changed" );
257
258 gtk_signal_connect( GTK_OBJECT(m_adjust),
259 "value_changed",
260 (GtkSignalFunc) gtk_scrollbar_callback,
261 (gpointer) this );
262}
263
264void wxScrollBar::SetScrollbar( int position, int thumbSize, int range, int pageSize,
265 bool WXUNUSED(refresh) )
266{
267 float fpos = (float)position;
268 float frange = (float)range;
269 float fthumb = (float)thumbSize;
270 float fpage = (float)pageSize;
271
272 if ((fabs(frange-m_adjust->upper) < 0.2) &&
273 (fabs(fthumb-m_adjust->page_size) < 0.2) &&
274 (fabs(fpage-m_adjust->page_increment) < 0.2))
275 {
276 SetThumbPosition( position );
277 return;
278 }
279
280 m_oldPos = fpos;
281
282 m_adjust->lower = 0.0;
283 m_adjust->upper = frange;
284 m_adjust->value = fpos;
285 m_adjust->step_increment = 1.0;
286 m_adjust->page_increment = (float)(wxMax(fpage,0));
287 m_adjust->page_size = fthumb;
288
289 gtk_signal_emit_by_name( GTK_OBJECT(m_adjust), "changed" );
290}
291
292/* Backward compatibility */
293int wxScrollBar::GetValue() const
294{
295 return GetThumbPosition();
296}
297
298void wxScrollBar::SetValue( int viewStart )
299{
300 SetThumbPosition( viewStart );
301}
302
303void wxScrollBar::GetValues( int *viewStart, int *viewLength, int *objectLength, int *pageLength ) const
304{
305 int pos = (int)(m_adjust->value+0.5);
306 int thumb = (int)(m_adjust->page_size+0.5);
307 int page = (int)(m_adjust->page_increment+0.5);
308 int range = (int)(m_adjust->upper+0.5);
309
310 *viewStart = pos;
311 *viewLength = range;
312 *objectLength = thumb;
313 *pageLength = page;
314}
315
316int wxScrollBar::GetViewLength() const
317{
318 return (int)(m_adjust->upper+0.5);
319}
320
321int wxScrollBar::GetObjectLength() const
322{
323 return (int)(m_adjust->page_size+0.5);
324}
325
326void wxScrollBar::SetPageSize( int pageLength )
327{
328 int pos = (int)(m_adjust->value+0.5);
329 int thumb = (int)(m_adjust->page_size+0.5);
330 int range = (int)(m_adjust->upper+0.5);
331 SetScrollbar( pos, thumb, range, pageLength );
332}
333
334void wxScrollBar::SetObjectLength( int objectLength )
335{
336 int pos = (int)(m_adjust->value+0.5);
337 int page = (int)(m_adjust->page_increment+0.5);
338 int range = (int)(m_adjust->upper+0.5);
339 SetScrollbar( pos, objectLength, range, page );
340}
341
342void wxScrollBar::SetViewLength( int viewLength )
343{
344 int pos = (int)(m_adjust->value+0.5);
345 int thumb = (int)(m_adjust->page_size+0.5);
346 int page = (int)(m_adjust->page_increment+0.5);
347 SetScrollbar( pos, thumb, viewLength, page );
348}
349
350bool wxScrollBar::IsOwnGtkWindow( GdkWindow *window )
351{
352 GtkRange *range = GTK_RANGE(m_widget);
353 return ( (window == GTK_WIDGET(range)->window)
354#ifndef __WXGTK20__
355 || (window == range->trough)
356 || (window == range->slider)
357 || (window == range->step_forw)
358 || (window == range->step_back)
359#endif // GTK+ 1.x
360 );
361}
362
363wxSize wxScrollBar::DoGetBestSize() const
364{
365 return wxControl::DoGetBestSize();
366}
367
368// static
369wxVisualAttributes
370wxScrollBar::GetClassDefaultAttributes(wxWindowVariant WXUNUSED(variant))
371{
372 return GetDefaultAttributesFromGTKWidget(gtk_vscrollbar_new);
373}
374
375#endif