]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/scrolbar.cpp
mouse and cursor additions for cocoa, see #10361
[wxWidgets.git] / src / gtk / 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// For compilers that support precompilation, includes "wx.h".
11#include "wx/wxprec.h"
12
13#if wxUSE_SCROLLBAR
14
15#include "wx/scrolbar.h"
16
17#ifndef WX_PRECOMP
18 #include "wx/utils.h"
19#endif
20
21#include "wx/gtk/private.h"
22
23//-----------------------------------------------------------------------------
24// "value_changed" from scrollbar
25//-----------------------------------------------------------------------------
26
27extern "C" {
28static void
29gtk_value_changed(GtkRange* range, wxScrollBar* win)
30{
31 wxEventType eventType = win->GetScrollEventType(range);
32 if (eventType != wxEVT_NULL)
33 {
34 const int orient = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
35 const int value = win->GetThumbPosition();
36 const int id = win->GetId();
37
38 // first send the specific event for the user action
39 wxScrollEvent evtSpec(eventType, id, value, orient);
40 evtSpec.SetEventObject(win);
41 win->HandleWindowEvent(evtSpec);
42
43 if (!win->m_isScrolling)
44 {
45 // and if it's over also send a general "changed" event
46 wxScrollEvent evtChanged(wxEVT_SCROLL_CHANGED, id, value, orient);
47 evtChanged.SetEventObject(win);
48 win->HandleWindowEvent(evtChanged);
49 }
50 }
51}
52}
53
54//-----------------------------------------------------------------------------
55// "button_press_event" from scrollbar
56//-----------------------------------------------------------------------------
57
58extern "C" {
59static gboolean
60gtk_button_press_event(GtkRange*, GdkEventButton*, wxScrollBar* win)
61{
62 win->m_mouseButtonDown = true;
63 return false;
64}
65}
66
67//-----------------------------------------------------------------------------
68// "event_after" from scrollbar
69//-----------------------------------------------------------------------------
70
71extern "C" {
72static void
73gtk_event_after(GtkRange* range, GdkEvent* event, wxScrollBar* win)
74{
75 if (event->type == GDK_BUTTON_RELEASE)
76 {
77 g_signal_handlers_block_by_func(range, (void*)gtk_event_after, win);
78
79 const int value = win->GetThumbPosition();
80 const int orient = win->HasFlag(wxSB_VERTICAL) ? wxVERTICAL : wxHORIZONTAL;
81 const int id = win->GetId();
82
83 wxScrollEvent evtRel(wxEVT_SCROLL_THUMBRELEASE, id, value, orient);
84 evtRel.SetEventObject(win);
85 win->HandleWindowEvent(evtRel);
86
87 wxScrollEvent evtChanged(wxEVT_SCROLL_CHANGED, id, value, orient);
88 evtChanged.SetEventObject(win);
89 win->HandleWindowEvent(evtChanged);
90 }
91}
92}
93
94//-----------------------------------------------------------------------------
95// "button_release_event" from scrollbar
96//-----------------------------------------------------------------------------
97
98extern "C" {
99static gboolean
100gtk_button_release_event(GtkRange* range, GdkEventButton*, wxScrollBar* win)
101{
102 win->m_mouseButtonDown = false;
103 // If thumb tracking
104 if (win->m_isScrolling)
105 {
106 win->m_isScrolling = false;
107 // Hook up handler to send thumb release event after this emission is finished.
108 // To allow setting scroll position from event handler, sending event must
109 // be deferred until after the GtkRange handler for this signal has run
110 g_signal_handlers_unblock_by_func(range, (void*)gtk_event_after, win);
111 }
112
113 return false;
114}
115}
116
117//-----------------------------------------------------------------------------
118// wxScrollBar
119//-----------------------------------------------------------------------------
120
121IMPLEMENT_DYNAMIC_CLASS(wxScrollBar,wxControl)
122
123wxScrollBar::wxScrollBar()
124{
125}
126
127wxScrollBar::~wxScrollBar()
128{
129}
130
131bool wxScrollBar::Create(wxWindow *parent, wxWindowID id,
132 const wxPoint& pos, const wxSize& size,
133 long style, const wxValidator& validator, const wxString& name )
134{
135 if (!PreCreation( parent, pos, size ) ||
136 !CreateBase( parent, id, pos, size, style, validator, name ))
137 {
138 wxFAIL_MSG( wxT("wxScrollBar creation failed") );
139 return false;
140 }
141
142 const bool isVertical = (style & wxSB_VERTICAL) != 0;
143 if (isVertical)
144 m_widget = gtk_vscrollbar_new( (GtkAdjustment *) NULL );
145 else
146 m_widget = gtk_hscrollbar_new( (GtkAdjustment *) NULL );
147 g_object_ref(m_widget);
148
149 m_scrollBar[0] = (GtkRange*)m_widget;
150
151 g_signal_connect_after(m_widget, "value_changed",
152 G_CALLBACK(gtk_value_changed), this);
153 g_signal_connect(m_widget, "button_press_event",
154 G_CALLBACK(gtk_button_press_event), this);
155 g_signal_connect(m_widget, "button_release_event",
156 G_CALLBACK(gtk_button_release_event), this);
157
158 gulong handler_id;
159 handler_id = g_signal_connect(
160 m_widget, "event_after", G_CALLBACK(gtk_event_after), this);
161 g_signal_handler_block(m_widget, handler_id);
162
163 m_parent->DoAddChild( this );
164
165 PostCreation(size);
166
167 return true;
168}
169
170int wxScrollBar::GetThumbPosition() const
171{
172 GtkAdjustment* adj = ((GtkRange*)m_widget)->adjustment;
173 return int(adj->value + 0.5);
174}
175
176int wxScrollBar::GetThumbSize() const
177{
178 GtkAdjustment* adj = ((GtkRange*)m_widget)->adjustment;
179 return int(adj->page_size);
180}
181
182int wxScrollBar::GetPageSize() const
183{
184 GtkAdjustment* adj = ((GtkRange*)m_widget)->adjustment;
185 return int(adj->page_increment);
186}
187
188int wxScrollBar::GetRange() const
189{
190 GtkAdjustment* adj = ((GtkRange*)m_widget)->adjustment;
191 return int(adj->upper);
192}
193
194void wxScrollBar::SetThumbPosition( int viewStart )
195{
196 if (GetThumbPosition() != viewStart)
197 {
198 g_signal_handlers_block_by_func(m_widget,
199 (gpointer)gtk_value_changed, this);
200
201 gtk_range_set_value((GtkRange*)m_widget, viewStart);
202 m_scrollPos[0] = gtk_range_get_value((GtkRange*)m_widget);
203
204 g_signal_handlers_unblock_by_func(m_widget,
205 (gpointer)gtk_value_changed, this);
206 }
207}
208
209void wxScrollBar::SetScrollbar(int position, int thumbSize, int range, int pageSize, bool)
210{
211 if (range == 0)
212 {
213 // GtkRange requires upper > lower
214 range =
215 thumbSize = 1;
216 }
217 GtkAdjustment* adj = ((GtkRange*)m_widget)->adjustment;
218 adj->step_increment = 1;
219 adj->page_increment = pageSize;
220 adj->page_size = thumbSize;
221 adj->value = position;
222 g_signal_handlers_block_by_func(m_widget, (void*)gtk_value_changed, this);
223 gtk_range_set_range((GtkRange*)m_widget, 0, range);
224 m_scrollPos[0] = adj->value;
225 g_signal_handlers_unblock_by_func(m_widget, (void*)gtk_value_changed, this);
226}
227
228void wxScrollBar::SetPageSize( int pageLength )
229{
230 SetScrollbar(GetThumbPosition(), GetThumbSize(), GetRange(), pageLength);
231}
232
233void wxScrollBar::SetRange(int range)
234{
235 SetScrollbar(GetThumbPosition(), GetThumbSize(), range, GetPageSize());
236}
237
238GdkWindow *wxScrollBar::GTKGetWindow(wxArrayGdkWindows& WXUNUSED(windows)) const
239{
240 return m_widget->window;
241}
242
243// static
244wxVisualAttributes
245wxScrollBar::GetClassDefaultAttributes(wxWindowVariant WXUNUSED(variant))
246{
247 return GetDefaultAttributesFromGTKWidget(gtk_vscrollbar_new);
248}
249
250#endif // wxUSE_SCROLLBAR