]> git.saurik.com Git - wxWidgets.git/blame - src/generic/scrlwing.cpp
Applied #9011: Native wxListCtrl::HitTest on OS X
[wxWidgets.git] / src / generic / scrlwing.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
ed4b0fdc 2// Name: src/generic/scrlwing.cpp
d32e78bd 3// Purpose: wxScrolledWindow implementation
c801d85f 4// Author: Julian Smart
566d84a7
RL
5// Modified by: Vadim Zeitlin on 31.08.00: wxScrollHelper allows to implement.
6// Ron Lee on 10.4.02: virtual size / auto scrollbars et al.
c801d85f
KB
7// Created: 01/02/97
8// RCS-ID: $Id$
77ffb593 9// Copyright: (c) wxWidgets team
65571936 10// Licence: wxWindows licence
c801d85f
KB
11/////////////////////////////////////////////////////////////////////////////
12
d80cd92a
VZ
13// ============================================================================
14// declarations
15// ============================================================================
16
17// ----------------------------------------------------------------------------
18// headers
19// ----------------------------------------------------------------------------
20
c801d85f
KB
21// For compilers that support precompilation, includes "wx.h".
22#include "wx/wxprec.h"
23
c801d85f 24#ifdef __BORLANDC__
d80cd92a 25 #pragma hdrstop
c801d85f
KB
26#endif
27
de6185e2
WS
28#include "wx/scrolwin.h"
29
30#ifndef WX_PRECOMP
31 #include "wx/utils.h"
8e609c82 32 #include "wx/panel.h"
ed4b0fdc 33 #include "wx/dcclient.h"
7d616e99 34 #include "wx/timer.h"
ed2fbeb8 35 #include "wx/sizer.h"
7d616e99 36 #include "wx/settings.h"
4b316329 37#endif
ed4b0fdc 38
b9dac1ab
JS
39#ifdef __WXMAC__
40#include "wx/scrolbar.h"
41#endif
42
2a201ef8 43#include "wx/recguard.h"
c801d85f 44
48d1144b 45#ifdef __WXMSW__
1e6feb95 46 #include <windows.h> // for DLGC_WANTARROWS
7acf6a92 47 #include "wx/msw/winundef.h"
48d1144b
JS
48#endif
49
a91b47e8
JS
50#ifdef __WXMOTIF__
51// For wxRETAINED implementation
338dd992
JJ
52#ifdef __VMS__ //VMS's Xm.h is not (yet) compatible with C++
53 //This code switches off the compiler warnings
54# pragma message disable nosimpint
55#endif
a91b47e8 56#include <Xm/Xm.h>
338dd992
JJ
57#ifdef __VMS__
58# pragma message enable nosimpint
59#endif
a91b47e8
JS
60#endif
61
066f1b7a 62/*
ca65c044
WS
63 TODO PROPERTIES
64 style wxHSCROLL | wxVSCROLL
066f1b7a
SC
65*/
66
d80cd92a 67// ----------------------------------------------------------------------------
1e6feb95
VZ
68// wxScrollHelperEvtHandler: intercept the events from the window and forward
69// them to wxScrollHelper
d80cd92a
VZ
70// ----------------------------------------------------------------------------
71
349efbaa 72class WXDLLEXPORT wxScrollHelperEvtHandler : public wxEvtHandler
1e6feb95
VZ
73{
74public:
29e1398f 75 wxScrollHelperEvtHandler(wxScrollHelperBase *scrollHelper)
1e6feb95
VZ
76 {
77 m_scrollHelper = scrollHelper;
78 }
d80cd92a 79
1e6feb95
VZ
80 virtual bool ProcessEvent(wxEvent& event);
81
ca65c044 82 void ResetDrawnFlag() { m_hasDrawnWindow = false; }
349efbaa 83
1e6feb95 84private:
29e1398f 85 wxScrollHelperBase *m_scrollHelper;
349efbaa
VZ
86
87 bool m_hasDrawnWindow;
22f3361e 88
c0c133e1 89 wxDECLARE_NO_COPY_CLASS(wxScrollHelperEvtHandler);
1e6feb95
VZ
90};
91
4b316329 92#if wxUSE_TIMER
1e6feb95
VZ
93// ----------------------------------------------------------------------------
94// wxAutoScrollTimer: the timer used to generate a stream of scroll events when
95// a captured mouse is held outside the window
96// ----------------------------------------------------------------------------
97
98class wxAutoScrollTimer : public wxTimer
99{
100public:
29e1398f
VZ
101 wxAutoScrollTimer(wxWindow *winToScroll,
102 wxScrollHelperBase *scroll,
1e6feb95
VZ
103 wxEventType eventTypeToSend,
104 int pos, int orient);
105
106 virtual void Notify();
107
108private:
109 wxWindow *m_win;
29e1398f 110 wxScrollHelperBase *m_scrollHelper;
1e6feb95
VZ
111 wxEventType m_eventType;
112 int m_pos,
113 m_orient;
22f3361e 114
c0c133e1 115 wxDECLARE_NO_COPY_CLASS(wxAutoScrollTimer);
1e6feb95 116};
d80cd92a
VZ
117
118// ============================================================================
119// implementation
120// ============================================================================
121
122// ----------------------------------------------------------------------------
1e6feb95 123// wxAutoScrollTimer
d80cd92a
VZ
124// ----------------------------------------------------------------------------
125
1e6feb95 126wxAutoScrollTimer::wxAutoScrollTimer(wxWindow *winToScroll,
29e1398f 127 wxScrollHelperBase *scroll,
1e6feb95
VZ
128 wxEventType eventTypeToSend,
129 int pos, int orient)
c801d85f 130{
1e6feb95
VZ
131 m_win = winToScroll;
132 m_scrollHelper = scroll;
133 m_eventType = eventTypeToSend;
134 m_pos = pos;
135 m_orient = orient;
c801d85f
KB
136}
137
1e6feb95 138void wxAutoScrollTimer::Notify()
c801d85f 139{
1e6feb95
VZ
140 // only do all this as long as the window is capturing the mouse
141 if ( wxWindow::GetCapture() != m_win )
142 {
143 Stop();
144 }
145 else // we still capture the mouse, continue generating events
146 {
147 // first scroll the window if we are allowed to do it
148 wxScrollWinEvent event1(m_eventType, m_pos, m_orient);
149 event1.SetEventObject(m_win);
150 if ( m_scrollHelper->SendAutoScrollEvents(event1) &&
151 m_win->GetEventHandler()->ProcessEvent(event1) )
152 {
153 // and then send a pseudo mouse-move event to refresh the selection
154 wxMouseEvent event2(wxEVT_MOTION);
b19bcdeb 155 event2.SetPosition(wxGetMousePosition());
1e6feb95
VZ
156
157 // the mouse event coordinates should be client, not screen as
158 // returned by wxGetMousePosition
159 wxWindow *parentTop = m_win;
160 while ( parentTop->GetParent() )
161 parentTop = parentTop->GetParent();
162 wxPoint ptOrig = parentTop->GetPosition();
163 event2.m_x -= ptOrig.x;
164 event2.m_y -= ptOrig.y;
165
166 event2.SetEventObject(m_win);
167
168 // FIXME: we don't fill in the other members - ok?
169
170 m_win->GetEventHandler()->ProcessEvent(event2);
171 }
172 else // can't scroll further, stop
173 {
174 Stop();
175 }
176 }
177}
4b316329 178#endif
1e6feb95
VZ
179
180// ----------------------------------------------------------------------------
181// wxScrollHelperEvtHandler
182// ----------------------------------------------------------------------------
183
184bool wxScrollHelperEvtHandler::ProcessEvent(wxEvent& event)
185{
9a268018 186 wxEventType evType = event.GetEventType();
2feed004 187
349efbaa
VZ
188 // the explanation of wxEVT_PAINT processing hack: for historic reasons
189 // there are 2 ways to process this event in classes deriving from
190 // wxScrolledWindow. The user code may
191 //
192 // 1. override wxScrolledWindow::OnDraw(dc)
193 // 2. define its own OnPaint() handler
194 //
195 // In addition, in wxUniversal wxWindow defines OnPaint() itself and
196 // always processes the draw event, so we can't just try the window
197 // OnPaint() first and call our HandleOnPaint() if it doesn't process it
198 // (the latter would never be called in wxUniversal).
199 //
200 // So the solution is to have a flag telling us whether the user code drew
201 // anything in the window. We set it to true here but reset it to false in
202 // wxScrolledWindow::OnPaint() handler (which wouldn't be called if the
203 // user code defined OnPaint() in the derived class)
ca65c044 204 m_hasDrawnWindow = true;
349efbaa 205
ff186591
VZ
206 // pass it on to the real handler
207 bool processed = wxEvtHandler::ProcessEvent(event);
208
209 // always process the size events ourselves, even if the user code handles
210 // them as well, as we need to AdjustScrollbars()
211 //
212 // NB: it is important to do it after processing the event in the normal
213 // way as HandleOnSize() may generate a wxEVT_SIZE itself if the
214 // scrollbar[s] (dis)appear and it should be seen by the user code
215 // after this one
216 if ( evType == wxEVT_SIZE )
217 {
218 m_scrollHelper->HandleOnSize((wxSizeEvent &)event);
219
ca65c044 220 return true;
ff186591
VZ
221 }
222
223 if ( processed )
349efbaa
VZ
224 {
225 // normally, nothing more to do here - except if it was a paint event
226 // which wasn't really processed, then we'll try to call our
227 // OnDraw() below (from HandleOnPaint)
c6b81ceb 228 if ( m_hasDrawnWindow || event.IsCommandEvent() )
349efbaa 229 {
ca65c044 230 return true;
349efbaa
VZ
231 }
232 }
2feed004 233
e421922f
VZ
234 if ( evType == wxEVT_PAINT )
235 {
236 m_scrollHelper->HandleOnPaint((wxPaintEvent &)event);
ca65c044 237 return true;
e421922f 238 }
1e6feb95 239
06b9c2a2
VS
240 if ( evType == wxEVT_CHILD_FOCUS )
241 {
242 m_scrollHelper->HandleOnChildFocus((wxChildFocusEvent &)event);
243 return true;
244 }
245
35c71386
VZ
246 // reset the skipped flag (which might have been set to true in
247 // ProcessEvent() above) to be able to test it below
248 bool wasSkipped = event.GetSkipped();
249 if ( wasSkipped )
250 event.Skip(false);
251
e421922f
VZ
252 if ( evType == wxEVT_SCROLLWIN_TOP ||
253 evType == wxEVT_SCROLLWIN_BOTTOM ||
254 evType == wxEVT_SCROLLWIN_LINEUP ||
255 evType == wxEVT_SCROLLWIN_LINEDOWN ||
256 evType == wxEVT_SCROLLWIN_PAGEUP ||
257 evType == wxEVT_SCROLLWIN_PAGEDOWN ||
258 evType == wxEVT_SCROLLWIN_THUMBTRACK ||
259 evType == wxEVT_SCROLLWIN_THUMBRELEASE )
260 {
35c71386
VZ
261 m_scrollHelper->HandleOnScroll((wxScrollWinEvent &)event);
262 if ( !event.GetSkipped() )
263 {
264 // it makes sense to indicate that we processed the message as we
265 // did scroll the window (and also notice that wxAutoScrollTimer
266 // relies on our return value to stop scrolling when we are at top
267 // or bottom already)
268 processed = true;
269 wasSkipped = false;
270 }
e421922f 271 }
1e6feb95 272
e421922f
VZ
273 if ( evType == wxEVT_ENTER_WINDOW )
274 {
275 m_scrollHelper->HandleOnMouseEnter((wxMouseEvent &)event);
276 }
277 else if ( evType == wxEVT_LEAVE_WINDOW )
278 {
279 m_scrollHelper->HandleOnMouseLeave((wxMouseEvent &)event);
280 }
281#if wxUSE_MOUSEWHEEL
602e3365
RR
282 // Use GTK's own scroll wheel handling in GtkScrolledWindow
283#ifndef __WXGTK20__
e421922f
VZ
284 else if ( evType == wxEVT_MOUSEWHEEL )
285 {
286 m_scrollHelper->HandleOnMouseWheel((wxMouseEvent &)event);
602e3365 287 return true;
e421922f 288 }
602e3365 289#endif
e421922f 290#endif // wxUSE_MOUSEWHEEL
e421922f
VZ
291 else if ( evType == wxEVT_CHAR )
292 {
293 m_scrollHelper->HandleOnChar((wxKeyEvent &)event);
35c71386
VZ
294 if ( !event.GetSkipped() )
295 {
296 processed = true;
297 wasSkipped = false;
298 }
1e6feb95
VZ
299 }
300
35c71386
VZ
301 if ( processed )
302 event.Skip(wasSkipped);
303
304 return processed;
1e6feb95
VZ
305}
306
29e1398f
VZ
307// ============================================================================
308// wxScrollHelperBase implementation
309// ============================================================================
310
1e6feb95 311// ----------------------------------------------------------------------------
29e1398f 312// wxScrollHelperBase construction
1e6feb95
VZ
313// ----------------------------------------------------------------------------
314
29e1398f 315wxScrollHelperBase::wxScrollHelperBase(wxWindow *win)
1e6feb95 316{
9a83f860 317 wxASSERT_MSG( win, wxT("associated window can't be NULL in wxScrollHelper") );
d32e78bd 318
1e6feb95
VZ
319 m_xScrollPixelsPerLine =
320 m_yScrollPixelsPerLine =
321 m_xScrollPosition =
322 m_yScrollPosition =
323 m_xScrollLines =
324 m_yScrollLines =
325 m_xScrollLinesPerPage =
139adb6a 326 m_yScrollLinesPerPage = 0;
1e6feb95
VZ
327
328 m_xScrollingEnabled =
ca65c044 329 m_yScrollingEnabled = true;
1e6feb95
VZ
330
331 m_scaleX =
139adb6a 332 m_scaleY = 1.0;
24ce4c18 333#if wxUSE_MOUSEWHEEL
d2c52078 334 m_wheelRotation = 0;
24ce4c18 335#endif
a58a12e9 336
1e6feb95 337 m_win =
29e1398f 338 m_targetWindow = NULL;
139adb6a 339
29e1398f 340 m_timerAutoScroll = NULL;
d7da9756 341
349efbaa
VZ
342 m_handler = NULL;
343
d32e78bd 344 m_win = win;
29e1398f
VZ
345
346 m_win->SetScrollHelper(static_cast<wxScrollHelper *>(this));
d32e78bd
VZ
347
348 // by default, the associated window is also the target window
349 DoSetTargetWindow(win);
1e6feb95 350}
d7da9756 351
29e1398f 352wxScrollHelperBase::~wxScrollHelperBase()
d80cd92a 353{
1e6feb95
VZ
354 StopAutoScrolling();
355
349efbaa 356 DeleteEvtHandler();
d80cd92a
VZ
357}
358
359// ----------------------------------------------------------------------------
360// setting scrolling parameters
361// ----------------------------------------------------------------------------
362
29e1398f
VZ
363void wxScrollHelperBase::SetScrollbars(int pixelsPerUnitX,
364 int pixelsPerUnitY,
365 int noUnitsX,
366 int noUnitsY,
367 int xPos,
368 int yPos,
369 bool noRefresh)
c801d85f 370{
b0486e0d
SN
371 int xpos, ypos;
372
373 CalcUnscrolledPosition(xPos, yPos, &xpos, &ypos);
139adb6a
RR
374 bool do_refresh =
375 (
c801d85f 376 (noUnitsX != 0 && m_xScrollLines == 0) ||
566d84a7 377 (noUnitsX < m_xScrollLines && xpos > pixelsPerUnitX * noUnitsX) ||
b0486e0d 378
c801d85f 379 (noUnitsY != 0 && m_yScrollLines == 0) ||
566d84a7 380 (noUnitsY < m_yScrollLines && ypos > pixelsPerUnitY * noUnitsY) ||
c801d85f 381 (xPos != m_xScrollPosition) ||
b0486e0d 382 (yPos != m_yScrollPosition)
139adb6a 383 );
a58a12e9 384
139adb6a
RR
385 m_xScrollPixelsPerLine = pixelsPerUnitX;
386 m_yScrollPixelsPerLine = pixelsPerUnitY;
387 m_xScrollPosition = xPos;
388 m_yScrollPosition = yPos;
a91b47e8 389
150c8d89
RL
390 int w = noUnitsX * pixelsPerUnitX;
391 int h = noUnitsY * pixelsPerUnitY;
392
2b5f62a0
VZ
393 // For better backward compatibility we set persisting limits
394 // here not just the size. It makes SetScrollbars 'sticky'
395 // emulating the old non-autoscroll behaviour.
2b43d588 396 // m_targetWindow->SetVirtualSizeHints( w, h );
a58a12e9 397
2b5f62a0
VZ
398 // The above should arguably be deprecated, this however we still need.
399
f18f464c
VZ
400 // take care not to set 0 virtual size, 0 means that we don't have any
401 // scrollbars and hence we should use the real size instead of the virtual
402 // one which is indicated by using wxDefaultCoord
403 m_targetWindow->SetVirtualSize( w ? w : wxDefaultCoord,
404 h ? h : wxDefaultCoord);
dc429f89 405
a58a12e9 406 if (do_refresh && !noRefresh)
ca65c044 407 m_targetWindow->Refresh(true, GetScrollRect());
a58a12e9 408
1f066698
VZ
409#ifndef __WXUNIVERSAL__
410 // If the target is not the same as the window with the scrollbars,
411 // then we need to update the scrollbars here, since they won't have
412 // been updated by SetVirtualSize().
413 if ( m_targetWindow != m_win )
414#endif // !__WXUNIVERSAL__
415 {
416 AdjustScrollbars();
417 }
418#ifndef __WXUNIVERSAL__
419 else
420 {
421 // otherwise this has been done by AdjustScrollbars, above
1f066698
VZ
422 }
423#endif // !__WXUNIVERSAL__
c801d85f
KB
424}
425
d80cd92a 426// ----------------------------------------------------------------------------
af4088f1 427// [target] window handling
d80cd92a 428// ----------------------------------------------------------------------------
ecab4dba 429
29e1398f 430void wxScrollHelperBase::DeleteEvtHandler()
ecab4dba 431{
248d771c
VZ
432 // search for m_handler in the handler list
433 if ( m_win && m_handler )
349efbaa 434 {
74f6bbf9 435 if ( m_win->RemoveEventHandler(m_handler) )
248d771c 436 {
74f6bbf9 437 delete m_handler;
248d771c 438 }
74f6bbf9
VZ
439 //else: something is very wrong, so better [maybe] leak memory than
440 // risk a crash because of double deletion
248d771c 441
74f6bbf9 442 m_handler = NULL;
349efbaa
VZ
443 }
444}
445
29e1398f 446void wxScrollHelperBase::ResetDrawnFlag()
16361ec9
VS
447{
448 wxCHECK_RET( m_handler, "invalid use of ResetDrawnFlag - no handler?" );
449 m_handler->ResetDrawnFlag();
450}
451
29e1398f 452void wxScrollHelperBase::DoSetTargetWindow(wxWindow *target)
349efbaa 453{
ecab4dba 454 m_targetWindow = target;
8adc196b
SC
455#ifdef __WXMAC__
456 target->MacSetClipChildren( true ) ;
457#endif
349efbaa
VZ
458
459 // install the event handler which will intercept the events we're
af4088f1
VZ
460 // interested in (but only do it for our real window, not the target window
461 // which we scroll - we don't need to hijack its events)
462 if ( m_targetWindow == m_win )
13ff9344 463 {
248d771c
VZ
464 // if we already have a handler, delete it first
465 DeleteEvtHandler();
466
13ff9344
JS
467 m_handler = new wxScrollHelperEvtHandler(this);
468 m_targetWindow->PushEventHandler(m_handler);
469 }
349efbaa
VZ
470}
471
29e1398f 472void wxScrollHelperBase::SetTargetWindow(wxWindow *target)
349efbaa
VZ
473{
474 wxCHECK_RET( target, wxT("target window must not be NULL") );
475
476 if ( target == m_targetWindow )
477 return;
478
af4088f1 479 DoSetTargetWindow(target);
ecab4dba
RR
480}
481
29e1398f 482wxWindow *wxScrollHelperBase::GetTargetWindow() const
ecab4dba
RR
483{
484 return m_targetWindow;
485}
486
d80cd92a
VZ
487// ----------------------------------------------------------------------------
488// scrolling implementation itself
489// ----------------------------------------------------------------------------
490
29e1398f 491void wxScrollHelperBase::HandleOnScroll(wxScrollWinEvent& event)
c801d85f 492{
139adb6a 493 int nScrollInc = CalcScrollInc(event);
1e6feb95 494 if ( nScrollInc == 0 )
139adb6a 495 {
1e6feb95
VZ
496 // can't scroll further
497 event.Skip();
498
499 return;
139adb6a 500 }
c801d85f 501
ca65c044 502 bool needsRefresh = false;
1e6feb95
VZ
503 int dx = 0,
504 dy = 0;
b97dc9c8 505 int orient = event.GetOrientation();
139adb6a
RR
506 if (orient == wxHORIZONTAL)
507 {
1e6feb95
VZ
508 if ( m_xScrollingEnabled )
509 {
510 dx = -m_xScrollPixelsPerLine * nScrollInc;
511 }
139adb6a 512 else
1e6feb95 513 {
ca65c044 514 needsRefresh = true;
1e6feb95 515 }
139adb6a 516 }
c801d85f 517 else
139adb6a 518 {
1e6feb95
VZ
519 if ( m_yScrollingEnabled )
520 {
521 dy = -m_yScrollPixelsPerLine * nScrollInc;
522 }
139adb6a 523 else
1e6feb95 524 {
ca65c044 525 needsRefresh = true;
1e6feb95
VZ
526 }
527 }
528
b97dc9c8
VS
529 if ( !needsRefresh )
530 {
531 // flush all pending repaints before we change m_{x,y}ScrollPosition, as
532 // otherwise invalidated area could be updated incorrectly later when
533 // ScrollWindow() makes sure they're repainted before scrolling them
93727870
SC
534#ifdef __WXMAC__
535 // wxWindowMac is taking care of making sure the update area is correctly
536 // set up, while not forcing an immediate redraw
537#else
b97dc9c8 538 m_targetWindow->Update();
93727870 539#endif
b97dc9c8
VS
540 }
541
542 if (orient == wxHORIZONTAL)
543 {
544 m_xScrollPosition += nScrollInc;
545 m_win->SetScrollPos(wxHORIZONTAL, m_xScrollPosition);
546 }
547 else
548 {
549 m_yScrollPosition += nScrollInc;
550 m_win->SetScrollPos(wxVERTICAL, m_yScrollPosition);
551 }
552
1e6feb95
VZ
553 if ( needsRefresh )
554 {
ca65c044 555 m_targetWindow->Refresh(true, GetScrollRect());
1e6feb95
VZ
556 }
557 else
558 {
d5d58a93 559 m_targetWindow->ScrollWindow(dx, dy, GetScrollRect());
3d2b9c20 560 }
c801d85f
KB
561}
562
29e1398f 563int wxScrollHelperBase::CalcScrollInc(wxScrollWinEvent& event)
c801d85f 564{
ecab4dba
RR
565 int pos = event.GetPosition();
566 int orient = event.GetOrientation();
c801d85f 567
ecab4dba 568 int nScrollInc = 0;
1a8caf94 569 if (event.GetEventType() == wxEVT_SCROLLWIN_TOP)
c801d85f 570 {
ecab4dba
RR
571 if (orient == wxHORIZONTAL)
572 nScrollInc = - m_xScrollPosition;
573 else
574 nScrollInc = - m_yScrollPosition;
1a8caf94
RR
575 } else
576 if (event.GetEventType() == wxEVT_SCROLLWIN_BOTTOM)
577 {
ecab4dba
RR
578 if (orient == wxHORIZONTAL)
579 nScrollInc = m_xScrollLines - m_xScrollPosition;
580 else
581 nScrollInc = m_yScrollLines - m_yScrollPosition;
1a8caf94
RR
582 } else
583 if (event.GetEventType() == wxEVT_SCROLLWIN_LINEUP)
584 {
ecab4dba 585 nScrollInc = -1;
1a8caf94
RR
586 } else
587 if (event.GetEventType() == wxEVT_SCROLLWIN_LINEDOWN)
588 {
ecab4dba 589 nScrollInc = 1;
1a8caf94
RR
590 } else
591 if (event.GetEventType() == wxEVT_SCROLLWIN_PAGEUP)
592 {
ecab4dba
RR
593 if (orient == wxHORIZONTAL)
594 nScrollInc = -GetScrollPageSize(wxHORIZONTAL);
595 else
596 nScrollInc = -GetScrollPageSize(wxVERTICAL);
1a8caf94
RR
597 } else
598 if (event.GetEventType() == wxEVT_SCROLLWIN_PAGEDOWN)
599 {
ecab4dba
RR
600 if (orient == wxHORIZONTAL)
601 nScrollInc = GetScrollPageSize(wxHORIZONTAL);
602 else
603 nScrollInc = GetScrollPageSize(wxVERTICAL);
1a8caf94
RR
604 } else
605 if ((event.GetEventType() == wxEVT_SCROLLWIN_THUMBTRACK) ||
606 (event.GetEventType() == wxEVT_SCROLLWIN_THUMBRELEASE))
607 {
ecab4dba
RR
608 if (orient == wxHORIZONTAL)
609 nScrollInc = pos - m_xScrollPosition;
610 else
611 nScrollInc = pos - m_yScrollPosition;
c801d85f 612 }
88150e60 613
ecab4dba
RR
614 if (orient == wxHORIZONTAL)
615 {
2e9c0c01 616 if ( m_xScrollPosition + nScrollInc < 0 )
ecab4dba 617 {
2e9c0c01
VZ
618 // As -ve as we can go
619 nScrollInc = -m_xScrollPosition;
620 }
621 else // check for the other bound
622 {
623 const int posMax = m_xScrollLines - m_xScrollLinesPerPage;
624 if ( m_xScrollPosition + nScrollInc > posMax )
878ddad5 625 {
2e9c0c01
VZ
626 // As +ve as we can go
627 nScrollInc = posMax - m_xScrollPosition;
878ddad5 628 }
ecab4dba 629 }
9d9355c6 630 }
2e9c0c01 631 else // wxVERTICAL
ecab4dba 632 {
2e9c0c01 633 if ( m_yScrollPosition + nScrollInc < 0 )
d80cd92a 634 {
2e9c0c01
VZ
635 // As -ve as we can go
636 nScrollInc = -m_yScrollPosition;
ecab4dba 637 }
2e9c0c01 638 else // check for the other bound
be9b0663 639 {
2e9c0c01
VZ
640 const int posMax = m_yScrollLines - m_yScrollLinesPerPage;
641 if ( m_yScrollPosition + nScrollInc > posMax )
642 {
643 // As +ve as we can go
644 nScrollInc = posMax - m_yScrollPosition;
645 }
be9b0663 646 }
9d9355c6 647 }
9d9355c6 648
ecab4dba 649 return nScrollInc;
c801d85f
KB
650}
651
29e1398f 652void wxScrollHelperBase::DoPrepareDC(wxDC& dc)
c801d85f 653{
1e6feb95 654 wxPoint pt = dc.GetDeviceOrigin();
807a572e
RR
655#ifdef __WXGTK__
656 // It may actually be correct to always query
69367c56 657 // the m_sign from the DC here, but I leave the
807a572e
RR
658 // #ifdef GTK for now.
659 if (m_win->GetLayoutDirection() == wxLayout_RightToLeft)
660 dc.SetDeviceOrigin( pt.x + m_xScrollPosition * m_xScrollPixelsPerLine,
661 pt.y - m_yScrollPosition * m_yScrollPixelsPerLine );
662 else
663#endif
664 dc.SetDeviceOrigin( pt.x - m_xScrollPosition * m_xScrollPixelsPerLine,
665 pt.y - m_yScrollPosition * m_yScrollPixelsPerLine );
139adb6a 666 dc.SetUserScale( m_scaleX, m_scaleY );
c801d85f
KB
667}
668
29e1398f 669void wxScrollHelperBase::SetScrollRate( int xstep, int ystep )
566d84a7
RL
670{
671 int old_x = m_xScrollPixelsPerLine * m_xScrollPosition;
672 int old_y = m_yScrollPixelsPerLine * m_yScrollPosition;
673
674 m_xScrollPixelsPerLine = xstep;
675 m_yScrollPixelsPerLine = ystep;
676
677 int new_x = m_xScrollPixelsPerLine * m_xScrollPosition;
678 int new_y = m_yScrollPixelsPerLine * m_yScrollPosition;
679
680 m_win->SetScrollPos( wxHORIZONTAL, m_xScrollPosition );
681 m_win->SetScrollPos( wxVERTICAL, m_yScrollPosition );
682 m_targetWindow->ScrollWindow( old_x - new_x, old_y - new_y );
683
684 AdjustScrollbars();
685}
686
29e1398f 687void wxScrollHelperBase::GetScrollPixelsPerUnit (int *x_unit, int *y_unit) const
c801d85f 688{
a0bc2c1d
VZ
689 if ( x_unit )
690 *x_unit = m_xScrollPixelsPerLine;
691 if ( y_unit )
692 *y_unit = m_yScrollPixelsPerLine;
c801d85f
KB
693}
694
5713b349 695
29e1398f 696int wxScrollHelperBase::GetScrollLines( int orient ) const
5713b349
RR
697{
698 if ( orient == wxHORIZONTAL )
699 return m_xScrollLines;
700 else
701 return m_yScrollLines;
702}
703
29e1398f 704int wxScrollHelperBase::GetScrollPageSize(int orient) const
c801d85f
KB
705{
706 if ( orient == wxHORIZONTAL )
707 return m_xScrollLinesPerPage;
708 else
709 return m_yScrollLinesPerPage;
710}
711
29e1398f 712void wxScrollHelperBase::SetScrollPageSize(int orient, int pageSize)
c801d85f
KB
713{
714 if ( orient == wxHORIZONTAL )
715 m_xScrollLinesPerPage = pageSize;
716 else
717 m_yScrollLinesPerPage = pageSize;
718}
719
29e1398f 720void wxScrollHelperBase::EnableScrolling (bool x_scroll, bool y_scroll)
c801d85f 721{
29e1398f
VZ
722 m_xScrollingEnabled = x_scroll;
723 m_yScrollingEnabled = y_scroll;
724}
a58a12e9 725
29e1398f
VZ
726// Where the current view starts from
727void wxScrollHelperBase::DoGetViewStart (int *x, int *y) const
728{
729 if ( x )
730 *x = m_xScrollPosition;
731 if ( y )
732 *y = m_yScrollPosition;
733}
c801d85f 734
29e1398f
VZ
735void wxScrollHelperBase::DoCalcScrolledPosition(int x, int y,
736 int *xx, int *yy) const
737{
738 if ( xx )
739 *xx = x - m_xScrollPosition * m_xScrollPixelsPerLine;
740 if ( yy )
741 *yy = y - m_yScrollPosition * m_yScrollPixelsPerLine;
742}
8ea45699 743
29e1398f
VZ
744void wxScrollHelperBase::DoCalcUnscrolledPosition(int x, int y,
745 int *xx, int *yy) const
746{
747 if ( xx )
748 *xx = x + m_xScrollPosition * m_xScrollPixelsPerLine;
749 if ( yy )
750 *yy = y + m_yScrollPosition * m_yScrollPixelsPerLine;
751}
a58a12e9 752
29e1398f
VZ
753// ----------------------------------------------------------------------------
754// geometry
755// ----------------------------------------------------------------------------
139adb6a 756
29e1398f
VZ
757bool wxScrollHelperBase::ScrollLayout()
758{
759 if ( m_win->GetSizer() && m_targetWindow == m_win )
c801d85f 760 {
29e1398f
VZ
761 // If we're the scroll target, take into account the
762 // virtual size and scrolled position of the window.
139adb6a 763
29e1398f
VZ
764 int x = 0, y = 0, w = 0, h = 0;
765 CalcScrolledPosition(0,0, &x,&y);
766 m_win->GetVirtualSize(&w, &h);
767 m_win->GetSizer()->SetDimension(x, y, w, h);
768 return true;
b97dc9c8 769 }
d2c52078 770
29e1398f
VZ
771 // fall back to default for LayoutConstraints
772 return m_win->wxWindow::Layout();
773}
b97dc9c8 774
29e1398f
VZ
775void wxScrollHelperBase::ScrollDoSetVirtualSize(int x, int y)
776{
777 m_win->wxWindow::DoSetVirtualSize( x, y );
778 AdjustScrollbars();
d32e78bd
VZ
779
780 if (m_win->GetAutoLayout())
781 m_win->Layout();
782}
783
784// wxWindow's GetBestVirtualSize returns the actual window size,
785// whereas we want to return the virtual size
29e1398f 786wxSize wxScrollHelperBase::ScrollGetBestVirtualSize() const
d32e78bd
VZ
787{
788 wxSize clientSize(m_win->GetClientSize());
789 if ( m_win->GetSizer() )
790 clientSize.IncTo(m_win->GetSizer()->CalcMin());
791
792 return clientSize;
793}
794
d80cd92a
VZ
795// ----------------------------------------------------------------------------
796// event handlers
797// ----------------------------------------------------------------------------
798
799// Default OnSize resets scrollbars, if any
29e1398f 800void wxScrollHelperBase::HandleOnSize(wxSizeEvent& WXUNUSED(event))
d80cd92a 801{
c376d80f 802 if ( m_targetWindow->GetAutoLayout() )
2b5f62a0 803 {
c376d80f 804 wxSize size = m_targetWindow->GetBestVirtualSize();
d32e78bd 805
c376d80f 806 // This will call ::Layout() and ::AdjustScrollbars()
168f82ce 807 m_win->SetVirtualSize( size );
2b5f62a0
VZ
808 }
809 else
c376d80f 810 {
2b5f62a0 811 AdjustScrollbars();
c376d80f 812 }
d80cd92a
VZ
813}
814
815// This calls OnDraw, having adjusted the origin according to the current
816// scroll position
29e1398f 817void wxScrollHelperBase::HandleOnPaint(wxPaintEvent& WXUNUSED(event))
d80cd92a 818{
af4088f1
VZ
819 // don't use m_targetWindow here, this is always called for ourselves
820 wxPaintDC dc(m_win);
1e6feb95 821 DoPrepareDC(dc);
d80cd92a
VZ
822
823 OnDraw(dc);
824}
825
438e3558
VZ
826// kbd handling: notice that we use OnChar() and not OnKeyDown() for
827// compatibility here - if we used OnKeyDown(), the programs which process
828// arrows themselves in their OnChar() would never get the message and like
829// this they always have the priority
29e1398f 830void wxScrollHelperBase::HandleOnChar(wxKeyEvent& event)
d80cd92a 831{
e2495f72
VZ
832 // prepare the event this key press maps to
833 wxScrollWinEvent newEvent;
d80cd92a 834
e2495f72
VZ
835 newEvent.SetPosition(0);
836 newEvent.SetEventObject(m_win);
56dade3c 837
e2495f72
VZ
838 // this is the default, it's changed to wxHORIZONTAL below if needed
839 newEvent.SetOrientation(wxVERTICAL);
d80cd92a 840
e2495f72
VZ
841 // some key events result in scrolling in both horizontal and vertical
842 // direction, e.g. Ctrl-{Home,End}, if this flag is true we should generate
843 // a second event in horizontal direction in addition to the primary one
844 bool sendHorizontalToo = false;
39c869a6 845
4995ca80 846 switch ( event.GetKeyCode() )
d80cd92a
VZ
847 {
848 case WXK_PAGEUP:
e2495f72 849 newEvent.SetEventType(wxEVT_SCROLLWIN_PAGEUP);
d80cd92a
VZ
850 break;
851
852 case WXK_PAGEDOWN:
e2495f72 853 newEvent.SetEventType(wxEVT_SCROLLWIN_PAGEDOWN);
d80cd92a
VZ
854 break;
855
d80cd92a 856 case WXK_HOME:
e2495f72 857 newEvent.SetEventType(wxEVT_SCROLLWIN_TOP);
d80cd92a 858
e2495f72 859 sendHorizontalToo = event.ControlDown();
d80cd92a
VZ
860 break;
861
e2495f72
VZ
862 case WXK_END:
863 newEvent.SetEventType(wxEVT_SCROLLWIN_BOTTOM);
d80cd92a 864
e2495f72 865 sendHorizontalToo = event.ControlDown();
d80cd92a
VZ
866 break;
867
868 case WXK_LEFT:
e2495f72
VZ
869 newEvent.SetOrientation(wxHORIZONTAL);
870 // fall through
871
872 case WXK_UP:
873 newEvent.SetEventType(wxEVT_SCROLLWIN_LINEUP);
d80cd92a
VZ
874 break;
875
876 case WXK_RIGHT:
e2495f72
VZ
877 newEvent.SetOrientation(wxHORIZONTAL);
878 // fall through
879
880 case WXK_DOWN:
881 newEvent.SetEventType(wxEVT_SCROLLWIN_LINEDOWN);
d80cd92a
VZ
882 break;
883
884 default:
e2495f72 885 // not a scrolling key
d80cd92a 886 event.Skip();
e2495f72 887 return;
d80cd92a 888 }
39c869a6 889
e2495f72 890 m_win->ProcessWindowEvent(newEvent);
39c869a6 891
e2495f72 892 if ( sendHorizontalToo )
39c869a6 893 {
e2495f72
VZ
894 newEvent.SetOrientation(wxHORIZONTAL);
895 m_win->ProcessWindowEvent(newEvent);
39c869a6 896 }
d80cd92a 897}
d2c52078 898
1e6feb95
VZ
899// ----------------------------------------------------------------------------
900// autoscroll stuff: these functions deal with sending fake scroll events when
901// a captured mouse is being held outside the window
902// ----------------------------------------------------------------------------
903
29e1398f 904bool wxScrollHelperBase::SendAutoScrollEvents(wxScrollWinEvent& event) const
1e6feb95
VZ
905{
906 // only send the event if the window is scrollable in this direction
907 wxWindow *win = (wxWindow *)event.GetEventObject();
908 return win->HasScrollbar(event.GetOrientation());
909}
910
29e1398f 911void wxScrollHelperBase::StopAutoScrolling()
1e6feb95 912{
4b316329 913#if wxUSE_TIMER
1e6feb95
VZ
914 if ( m_timerAutoScroll )
915 {
916 delete m_timerAutoScroll;
d3b9f782 917 m_timerAutoScroll = NULL;
1e6feb95 918 }
4b316329 919#endif
1e6feb95 920}
d2c52078 921
29e1398f 922void wxScrollHelperBase::HandleOnMouseEnter(wxMouseEvent& event)
d2c52078 923{
1e6feb95
VZ
924 StopAutoScrolling();
925
926 event.Skip();
927}
d2c52078 928
29e1398f 929void wxScrollHelperBase::HandleOnMouseLeave(wxMouseEvent& event)
1e6feb95
VZ
930{
931 // don't prevent the usual processing of the event from taking place
932 event.Skip();
933
934 // when a captured mouse leave a scrolled window we start generate
935 // scrolling events to allow, for example, extending selection beyond the
936 // visible area in some controls
937 if ( wxWindow::GetCapture() == m_targetWindow )
938 {
939 // where is the mouse leaving?
940 int pos, orient;
941 wxPoint pt = event.GetPosition();
942 if ( pt.x < 0 )
943 {
944 orient = wxHORIZONTAL;
945 pos = 0;
946 }
947 else if ( pt.y < 0 )
948 {
949 orient = wxVERTICAL;
950 pos = 0;
951 }
952 else // we're lower or to the right of the window
953 {
954 wxSize size = m_targetWindow->GetClientSize();
955 if ( pt.x > size.x )
956 {
957 orient = wxHORIZONTAL;
958 pos = m_xScrollLines;
959 }
960 else if ( pt.y > size.y )
961 {
962 orient = wxVERTICAL;
963 pos = m_yScrollLines;
964 }
965 else // this should be impossible
966 {
967 // but seems to happen sometimes under wxMSW - maybe it's a bug
968 // there but for now just ignore it
969
9a83f860 970 //wxFAIL_MSG( wxT("can't understand where has mouse gone") );
1e6feb95
VZ
971
972 return;
973 }
974 }
975
976 // only start the auto scroll timer if the window can be scrolled in
977 // this direction
978 if ( !m_targetWindow->HasScrollbar(orient) )
979 return;
980
4b316329 981#if wxUSE_TIMER
1e6feb95
VZ
982 delete m_timerAutoScroll;
983 m_timerAutoScroll = new wxAutoScrollTimer
984 (
985 m_targetWindow, this,
986 pos == 0 ? wxEVT_SCROLLWIN_LINEUP
987 : wxEVT_SCROLLWIN_LINEDOWN,
988 pos,
989 orient
990 );
991 m_timerAutoScroll->Start(50); // FIXME: make configurable
534b127c
WS
992#else
993 wxUnusedVar(pos);
4b316329 994#endif
1e6feb95
VZ
995 }
996}
997
e421922f
VZ
998#if wxUSE_MOUSEWHEEL
999
29e1398f 1000void wxScrollHelperBase::HandleOnMouseWheel(wxMouseEvent& event)
1e6feb95 1001{
d2c52078 1002 m_wheelRotation += event.GetWheelRotation();
1e6feb95 1003 int lines = m_wheelRotation / event.GetWheelDelta();
d2c52078
RD
1004 m_wheelRotation -= lines * event.GetWheelDelta();
1005
1e6feb95
VZ
1006 if (lines != 0)
1007 {
1e6feb95 1008
b6b85bdc
JS
1009 wxScrollWinEvent newEvent;
1010
ac6f6ffc 1011 newEvent.SetPosition(0);
6682e732 1012 newEvent.SetOrientation( event.GetWheelAxis() == 0 ? wxVERTICAL : wxHORIZONTAL);
687706f5 1013 newEvent.SetEventObject(m_win);
b6b85bdc 1014
9b9337da 1015 if (event.IsPageScroll())
4b056ef5
RD
1016 {
1017 if (lines > 0)
687706f5 1018 newEvent.SetEventType(wxEVT_SCROLLWIN_PAGEUP);
4b056ef5 1019 else
687706f5 1020 newEvent.SetEventType(wxEVT_SCROLLWIN_PAGEDOWN);
4b056ef5 1021
b6b85bdc 1022 m_win->GetEventHandler()->ProcessEvent(newEvent);
4b056ef5
RD
1023 }
1024 else
1025 {
1026 lines *= event.GetLinesPerAction();
1027 if (lines > 0)
687706f5 1028 newEvent.SetEventType(wxEVT_SCROLLWIN_LINEUP);
4b056ef5 1029 else
687706f5 1030 newEvent.SetEventType(wxEVT_SCROLLWIN_LINEDOWN);
b6b85bdc 1031
4b056ef5
RD
1032 int times = abs(lines);
1033 for (; times > 0; times--)
1034 m_win->GetEventHandler()->ProcessEvent(newEvent);
1035 }
d2c52078
RD
1036 }
1037}
1038
e421922f
VZ
1039#endif // wxUSE_MOUSEWHEEL
1040
29e1398f 1041void wxScrollHelperBase::HandleOnChildFocus(wxChildFocusEvent& event)
06b9c2a2
VS
1042{
1043 // this event should be processed by all windows in parenthood chain,
1044 // e.g. so that nested wxScrolledWindows work correctly
1045 event.Skip();
1046
1047 // find the immediate child under which the window receiving focus is:
1048 wxWindow *win = event.GetWindow();
02a33891
VS
1049
1050 if ( win == m_targetWindow )
1051 return; // nothing to do
1052
2e1517a2 1053#if defined( __WXOSX__ ) && wxUSE_SCROLLBAR
380e1c0c
JS
1054 if (wxDynamicCast(win, wxScrollBar))
1055 return;
1056#endif
1057
3ec704b4
RD
1058 // Fixing ticket: http://trac.wxwidgets.org/ticket/9563
1059 // When a child inside a wxControlContainer receives a focus, the
1060 // wxControlContainer generates an artificial wxChildFocusEvent for
1061 // itself, telling its parent that 'it' received the focus. The effect is
1062 // that this->HandleOnChildFocus is called twice, first with the
1063 // artificial wxChildFocusEvent and then with the original event. We need
1064 // to ignore the artificial event here or otherwise HandleOnChildFocus
1065 // would first scroll the target window to make the entire
1066 // wxControlContainer visible and immediately afterwards scroll the target
1067 // window again to make the child widget visible. This leads to ugly
1068 // flickering when using nested wxPanels/wxScrolledWindows.
1069 //
1070 // Ignore this event if 'win' is derived from wxControlContainer AND its
1071 // parent is the m_targetWindow AND 'win' is not actually reciving the
1072 // focus (win != FindFocus). TODO: This affects all wxControlContainer
1073 // objects, but wxControlContainer is not part of the wxWidgets RTTI and
1074 // so wxDynamicCast(win, wxControlContainer) does not compile. Find a way
1075 // to determine if 'win' derives from wxControlContainer. Until then,
1076 // testing if 'win' derives from wxPanel will probably get >90% of all
1077 // cases.
1078
1079 wxWindow *actual_focus=wxWindow::FindFocus();
1080 if (win != actual_focus &&
1081 wxDynamicCast(win, wxPanel) != 0 &&
1082 win->GetParent() == m_targetWindow)
1083 // if win is a wxPanel and receives the focus, it should not be
1084 // scrolled into view
29e1398f 1085 return;
3ec704b4 1086
888e9638 1087 const wxRect viewRect(m_targetWindow->GetClientRect());
3ec704b4
RD
1088
1089 // For composite controls such as wxComboCtrl we should try to fit the
1090 // entire control inside the visible area of the target window, not just
1091 // the focused child of the control. Otherwise we'd make only the textctrl
1092 // part of a wxComboCtrl visible and the button would still be outside the
1093 // scrolled area. But do so only if the parent fits *entirely* inside the
1094 // scrolled window. In other situations, such as nested wxPanel or
1095 // wxScrolledWindows, the parent might be way to big to fit inside the
1096 // scrolled window. If that is the case, then make only the focused window
1097 // visible
1098 if ( win->GetParent() != m_targetWindow)
06b9c2a2 1099 {
3ec704b4
RD
1100 wxWindow *parent=win->GetParent();
1101 wxSize parent_size=parent->GetSize();
888e9638
VZ
1102 if (parent_size.GetWidth() <= viewRect.GetWidth() &&
1103 parent_size.GetHeight() <= viewRect.GetHeight())
3ec704b4 1104 // make the immediate parent visible instead of the focused control
29e1398f 1105 win=parent;
06b9c2a2
VS
1106 }
1107
888e9638
VZ
1108 // make win position relative to the m_targetWindow viewing area instead of
1109 // its parent
1110 const wxRect
1111 winRect(m_targetWindow->ScreenToClient(win->GetScreenPosition()),
1112 win->GetSize());
1113
1114 // check if it's fully visible
1115 if ( viewRect.Contains(winRect) )
1116 {
1117 // it is, nothing to do
1118 return;
1119 }
1120
1121 // check if we can make it fully visible: this is only possible if it's not
1122 // larger than our view area
1123 if ( winRect.GetWidth() > viewRect.GetWidth() ||
1124 winRect.GetHeight() > viewRect.GetHeight() )
1125 {
1126 // we can't make it fit so avoid scrolling it at all, this is only
1127 // going to be confusing and not helpful
1128 return;
1129 }
1130
1131
1132 // do make the window fit inside the view area by scrolling to it
06b9c2a2
VS
1133 int stepx, stepy;
1134 GetScrollPixelsPerUnit(&stepx, &stepy);
1135
06b9c2a2
VS
1136 int startx, starty;
1137 GetViewStart(&startx, &starty);
1138
1139 // first in vertical direction:
1140 if ( stepy > 0 )
1141 {
1142 int diff = 0;
1143
888e9638 1144 if ( winRect.GetTop() < 0 )
06b9c2a2 1145 {
888e9638 1146 diff = winRect.GetTop();
06b9c2a2 1147 }
888e9638 1148 else if ( winRect.GetBottom() > viewRect.GetHeight() )
06b9c2a2 1149 {
888e9638 1150 diff = winRect.GetBottom() - viewRect.GetHeight() + 1;
06b9c2a2
VS
1151 // round up to next scroll step if we can't get exact position,
1152 // so that the window is fully visible:
1153 diff += stepy - 1;
1154 }
1155
1156 starty = (starty * stepy + diff) / stepy;
1157 }
1158
1159 // then horizontal:
1160 if ( stepx > 0 )
1161 {
1162 int diff = 0;
1163
888e9638 1164 if ( winRect.GetLeft() < 0 )
06b9c2a2 1165 {
888e9638 1166 diff = winRect.GetLeft();
06b9c2a2 1167 }
888e9638 1168 else if ( winRect.GetRight() > viewRect.GetWidth() )
06b9c2a2 1169 {
888e9638 1170 diff = winRect.GetRight() - viewRect.GetWidth() + 1;
06b9c2a2
VS
1171 // round up to next scroll step if we can't get exact position,
1172 // so that the window is fully visible:
1173 diff += stepx - 1;
1174 }
1175
1176 startx = (startx * stepx + diff) / stepx;
1177 }
1178
1179 Scroll(startx, starty);
1180}
1181
29e1398f
VZ
1182
1183#ifdef wxHAS_GENERIC_SCROLLWIN
1184
1185// ----------------------------------------------------------------------------
1186// wxScrollHelper implementation
1187// ----------------------------------------------------------------------------
1188
1189wxScrollHelper::wxScrollHelper(wxWindow *winToScroll)
1190 : wxScrollHelperBase(winToScroll)
1191{
1192 m_xVisibility =
1193 m_yVisibility = wxSHOW_SB_DEFAULT;
1194}
1195
1196void wxScrollHelper::DoShowScrollbars(wxScrollbarVisibility horz,
1197 wxScrollbarVisibility vert)
1198{
1199 if ( horz != m_xVisibility || vert != m_yVisibility )
1200 {
1201 m_xVisibility = horz;
1202 m_yVisibility = vert;
1203
1204 AdjustScrollbars();
1205 }
1206}
1207
1208void
1209wxScrollHelper::DoAdjustScrollbar(int orient,
1210 int clientSize,
1211 int virtSize,
1ddb6d28 1212 int pixelsPerUnit,
29e1398f
VZ
1213 int& scrollUnits,
1214 int& scrollPosition,
1ddb6d28 1215 int& scrollLinesPerPage,
29e1398f
VZ
1216 wxScrollbarVisibility visibility)
1217{
29e1398f 1218 // scroll lines per page: if 0, no scrolling is needed
29e1398f 1219 // check if we need scrollbar in this direction at all
1ddb6d28 1220 if ( pixelsPerUnit == 0 || clientSize >= virtSize )
29e1398f
VZ
1221 {
1222 // scrolling is disabled or unnecessary
1223 scrollUnits =
1224 scrollPosition = 0;
1ddb6d28 1225 scrollLinesPerPage = 0;
29e1398f
VZ
1226 }
1227 else // might need scrolling
1228 {
1229 // Round up integer division to catch any "leftover" client space.
1230 scrollUnits = (virtSize + pixelsPerUnit - 1) / pixelsPerUnit;
1231
1232 // Calculate the number of fully scroll units
1ddb6d28 1233 scrollLinesPerPage = clientSize / pixelsPerUnit;
29e1398f 1234
1ddb6d28 1235 if ( scrollLinesPerPage >= scrollUnits )
29e1398f
VZ
1236 {
1237 // we're big enough to not need scrolling
1238 scrollUnits =
1239 scrollPosition = 0;
1ddb6d28 1240 scrollLinesPerPage = 0;
29e1398f
VZ
1241 }
1242 else // we do need a scrollbar
1243 {
1ddb6d28
VZ
1244 if ( scrollLinesPerPage < 1 )
1245 scrollLinesPerPage = 1;
29e1398f
VZ
1246
1247 // Correct position if greater than extent of canvas minus
1248 // the visible portion of it or if below zero
1ddb6d28 1249 const int posMax = scrollUnits - scrollLinesPerPage;
29e1398f
VZ
1250 if ( scrollPosition > posMax )
1251 scrollPosition = posMax;
1252 else if ( scrollPosition < 0 )
1253 scrollPosition = 0;
1254 }
1255 }
1256
1ddb6d28
VZ
1257 // in wxSHOW_SB_NEVER case don't show the scrollbar even if it's needed, in
1258 // wxSHOW_SB_ALWAYS case show the scrollbar even if it's not needed by
1259 // passing a special range value to SetScrollbar()
36e5a9a7 1260 int range;
1ddb6d28
VZ
1261 switch ( visibility )
1262 {
1263 case wxSHOW_SB_NEVER:
1264 range = 0;
1265 break;
1266
36e5a9a7
VZ
1267 case wxSHOW_SB_ALWAYS:
1268 range = scrollUnits ? scrollUnits : -1;
1269 break;
1270
1271 default:
1272 wxFAIL_MSG( wxS("unknown scrollbar visibility") );
1273 // fall through
1274
1ddb6d28
VZ
1275 case wxSHOW_SB_DEFAULT:
1276 range = scrollUnits;
1277 break;
1278
1ddb6d28 1279 }
29e1398f 1280
1ddb6d28 1281 m_win->SetScrollbar(orient, scrollPosition, scrollLinesPerPage, range);
29e1398f
VZ
1282}
1283
1284void wxScrollHelper::AdjustScrollbars()
1285{
1286 static wxRecursionGuardFlag s_flagReentrancy;
1287 wxRecursionGuard guard(s_flagReentrancy);
1288 if ( guard.IsInside() )
1289 {
1290 // don't reenter AdjustScrollbars() while another call to
1291 // AdjustScrollbars() is in progress because this may lead to calling
1292 // ScrollWindow() twice and this can really happen under MSW if
1293 // SetScrollbar() call below adds or removes the scrollbar which
1294 // changes the window size and hence results in another
1295 // AdjustScrollbars() call
1296 return;
1297 }
1298
1299 int oldXScroll = m_xScrollPosition;
1300 int oldYScroll = m_yScrollPosition;
1301
1302 // we may need to readjust the scrollbars several times as enabling one of
1303 // them reduces the area available for the window contents and so can make
1304 // the other scrollbar necessary now although it wasn't necessary before
1305 //
1306 // VZ: normally this loop should be over in at most 2 iterations, I don't
1307 // know why do we need 5 of them
1308 for ( int iterationCount = 0; iterationCount < 5; iterationCount++ )
1309 {
1310 wxSize clientSize = GetTargetSize();
1311 const wxSize virtSize = m_targetWindow->GetVirtualSize();
1312
1313 // this block of code tries to work around the following problem: the
1314 // window could have been just resized to have enough space to show its
1315 // full contents without the scrollbars, but its client size could be
1316 // not big enough because it does have the scrollbars right now and so
1317 // the scrollbars would remain even though we don't need them any more
1318 //
1319 // to prevent this from happening, check if we have enough space for
1320 // everything without the scrollbars and explicitly disable them then
1321 const wxSize availSize = GetSizeAvailableForScrollTarget(
1322 m_win->GetSize() - m_win->GetWindowBorderSize());
1323 if ( availSize != clientSize )
1324 {
1325 if ( availSize.x >= virtSize.x && availSize.y >= virtSize.y )
1326 {
1327 // this will be enough to make the scrollbars disappear below
1328 // and then the client size will indeed become equal to the
1329 // full available size
1330 clientSize = availSize;
1331 }
1332 }
1333
1334
1335 DoAdjustScrollbar(wxHORIZONTAL,
1336 clientSize.x,
1337 virtSize.x,
1338 m_xScrollPixelsPerLine,
1339 m_xScrollLines,
1340 m_xScrollPosition,
1ddb6d28 1341 m_xScrollLinesPerPage,
29e1398f
VZ
1342 m_xVisibility);
1343
1344 DoAdjustScrollbar(wxVERTICAL,
1345 clientSize.y,
1346 virtSize.y,
1347 m_yScrollPixelsPerLine,
1348 m_yScrollLines,
1349 m_yScrollPosition,
1ddb6d28 1350 m_yScrollLinesPerPage,
29e1398f
VZ
1351 m_yVisibility);
1352
1353
1354 // If a scrollbar (dis)appeared as a result of this, we need to adjust
1355 // them again but if the client size didn't change, then we're done
1356 if ( GetTargetSize() == clientSize )
1357 break;
1358 }
1359
1360#ifdef __WXMOTIF__
1361 // Sorry, some Motif-specific code to implement a backing pixmap
1362 // for the wxRETAINED style. Implementing a backing store can't
1363 // be entirely generic because it relies on the wxWindowDC implementation
1364 // to duplicate X drawing calls for the backing pixmap.
1365
1366 if ( m_targetWindow->GetWindowStyle() & wxRETAINED )
1367 {
1368 Display* dpy = XtDisplay((Widget)m_targetWindow->GetMainWidget());
1369
1370 int totalPixelWidth = m_xScrollLines * m_xScrollPixelsPerLine;
1371 int totalPixelHeight = m_yScrollLines * m_yScrollPixelsPerLine;
1372 if (m_targetWindow->GetBackingPixmap() &&
1373 !((m_targetWindow->GetPixmapWidth() == totalPixelWidth) &&
1374 (m_targetWindow->GetPixmapHeight() == totalPixelHeight)))
1375 {
1376 XFreePixmap (dpy, (Pixmap) m_targetWindow->GetBackingPixmap());
1377 m_targetWindow->SetBackingPixmap((WXPixmap) 0);
1378 }
1379
1380 if (!m_targetWindow->GetBackingPixmap() &&
1381 (m_xScrollLines != 0) && (m_yScrollLines != 0))
1382 {
1383 int depth = wxDisplayDepth();
1384 m_targetWindow->SetPixmapWidth(totalPixelWidth);
1385 m_targetWindow->SetPixmapHeight(totalPixelHeight);
1386 m_targetWindow->SetBackingPixmap((WXPixmap) XCreatePixmap (dpy, RootWindow (dpy, DefaultScreen (dpy)),
1387 m_targetWindow->GetPixmapWidth(), m_targetWindow->GetPixmapHeight(), depth));
1388 }
1389
1390 }
1391#endif // Motif
1392
1393 if (oldXScroll != m_xScrollPosition)
1394 {
1395 if (m_xScrollingEnabled)
1396 m_targetWindow->ScrollWindow( m_xScrollPixelsPerLine * (oldXScroll - m_xScrollPosition), 0,
1397 GetScrollRect() );
1398 else
1399 m_targetWindow->Refresh(true, GetScrollRect());
1400 }
1401
1402 if (oldYScroll != m_yScrollPosition)
1403 {
1404 if (m_yScrollingEnabled)
1405 m_targetWindow->ScrollWindow( 0, m_yScrollPixelsPerLine * (oldYScroll-m_yScrollPosition),
1406 GetScrollRect() );
1407 else
1408 m_targetWindow->Refresh(true, GetScrollRect());
1409 }
1410}
1411
1412void wxScrollHelper::DoScroll( int x_pos, int y_pos )
1413{
1414 if (!m_targetWindow)
1415 return;
1416
1417 if (((x_pos == -1) || (x_pos == m_xScrollPosition)) &&
1418 ((y_pos == -1) || (y_pos == m_yScrollPosition))) return;
1419
1420 int w = 0, h = 0;
1421 GetTargetSize(&w, &h);
1422
1423 // compute new position:
1424 int new_x = m_xScrollPosition;
1425 int new_y = m_yScrollPosition;
1426
1427 if ((x_pos != -1) && (m_xScrollPixelsPerLine))
1428 {
1429 new_x = x_pos;
1430
1431 // Calculate page size i.e. number of scroll units you get on the
1432 // current client window
1433 int noPagePositions = w/m_xScrollPixelsPerLine;
1434 if (noPagePositions < 1) noPagePositions = 1;
1435
1436 // Correct position if greater than extent of canvas minus
1437 // the visible portion of it or if below zero
1438 new_x = wxMin( m_xScrollLines-noPagePositions, new_x );
1439 new_x = wxMax( 0, new_x );
1440 }
1441 if ((y_pos != -1) && (m_yScrollPixelsPerLine))
1442 {
1443 new_y = y_pos;
1444
1445 // Calculate page size i.e. number of scroll units you get on the
1446 // current client window
1447 int noPagePositions = h/m_yScrollPixelsPerLine;
1448 if (noPagePositions < 1) noPagePositions = 1;
1449
1450 // Correct position if greater than extent of canvas minus
1451 // the visible portion of it or if below zero
1452 new_y = wxMin( m_yScrollLines-noPagePositions, new_y );
1453 new_y = wxMax( 0, new_y );
1454 }
1455
1456 if ( new_x == m_xScrollPosition && new_y == m_yScrollPosition )
1457 return; // nothing to do, the position didn't change
1458
1459 // flush all pending repaints before we change m_{x,y}ScrollPosition, as
1460 // otherwise invalidated area could be updated incorrectly later when
1461 // ScrollWindow() makes sure they're repainted before scrolling them
1462 m_targetWindow->Update();
1463
1464 // update the position and scroll the window now:
1465 if (m_xScrollPosition != new_x)
1466 {
1467 int old_x = m_xScrollPosition;
1468 m_xScrollPosition = new_x;
1469 m_win->SetScrollPos( wxHORIZONTAL, new_x );
1470 m_targetWindow->ScrollWindow( (old_x-new_x)*m_xScrollPixelsPerLine, 0,
1471 GetScrollRect() );
1472 }
1473
1474 if (m_yScrollPosition != new_y)
1475 {
1476 int old_y = m_yScrollPosition;
1477 m_yScrollPosition = new_y;
1478 m_win->SetScrollPos( wxVERTICAL, new_y );
1479 m_targetWindow->ScrollWindow( 0, (old_y-new_y)*m_yScrollPixelsPerLine,
1480 GetScrollRect() );
1481 }
1482}
1483
1484#endif // wxHAS_GENERIC_SCROLLWIN
1485
1e6feb95 1486// ----------------------------------------------------------------------------
16361ec9 1487// wxScrolled<T> and wxScrolledWindow implementation
1e6feb95
VZ
1488// ----------------------------------------------------------------------------
1489
16361ec9 1490wxSize wxScrolledT_Helper::FilterBestSize(const wxWindow *win,
29e1398f 1491 const wxScrollHelper *helper,
16361ec9 1492 const wxSize& origBest)
7d616e99
VS
1493{
1494 // NB: We don't do this in WX_FORWARD_TO_SCROLL_HELPER, because not
1495 // all scrollable windows should behave like this, only those that
1496 // contain children controls within scrollable area
1497 // (i.e., wxScrolledWindow) and other some scrollable windows may
1498 // have different DoGetBestSize() implementation (e.g. wxTreeCtrl).
1499
16361ec9 1500 wxSize best = origBest;
7d616e99 1501
16361ec9 1502 if ( win->GetAutoLayout() )
7d616e99
VS
1503 {
1504 // Only use the content to set the window size in the direction
1505 // where there's no scrolling; otherwise we're going to get a huge
1506 // window in the direction in which scrolling is enabled
1507 int ppuX, ppuY;
16361ec9 1508 helper->GetScrollPixelsPerUnit(&ppuX, &ppuY);
7d616e99
VS
1509
1510 // NB: This code used to use *current* size if min size wasn't
1511 // specified, presumably to get some reasonable (i.e., larger than
1512 // minimal) size. But that's a wrong thing to do in GetBestSize(),
1513 // so we use minimal size as specified. If the app needs some
1514 // minimal size for its scrolled window, it should set it and put
1515 // the window into sizer as expandable so that it can use all space
1516 // available to it.
1517 //
1518 // See also http://svn.wxwidgets.org/viewvc/wx?view=rev&revision=45864
1519
16361ec9 1520 wxSize minSize = win->GetMinSize();
7d616e99
VS
1521
1522 if ( ppuX > 0 )
1523 best.x = minSize.x + wxSystemSettings::GetMetric(wxSYS_VSCROLL_X);
1524
1525 if ( ppuY > 0 )
1526 best.y = minSize.y + wxSystemSettings::GetMetric(wxSYS_HSCROLL_Y);
1527 }
1528
1529 return best;
1530}
1531
0cf5b099 1532#ifdef __WXMSW__
f2a6c918 1533WXLRESULT wxScrolledT_Helper::FilterMSWWindowProc(WXUINT nMsg, WXLRESULT rc)
0cf5b099 1534{
4676948b 1535#ifndef __WXWINCE__
0cf5b099
VZ
1536 // we need to process arrows ourselves for scrolling
1537 if ( nMsg == WM_GETDLGCODE )
1538 {
1539 rc |= DLGC_WANTARROWS;
1540 }
4676948b 1541#endif
0cf5b099
VZ
1542 return rc;
1543}
0cf5b099 1544#endif // __WXMSW__
16361ec9
VS
1545
1546// NB: skipping wxScrolled<T> in wxRTTI information because being a templte,
1547// it doesn't and can't implement wxRTTI support
1548IMPLEMENT_DYNAMIC_CLASS(wxScrolledWindow, wxPanel)