1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxScrolledWindow implementation
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation
14 #pragma implementation "scrolwin.h"
17 // For compilers that support precompilation, includes "wx.h".
18 #include "wx/wxprec.h"
21 #include "wx/dcclient.h"
27 #include "wx/generic/scrolwin.h"
30 #if !USE_SHARED_LIBRARY
31 BEGIN_EVENT_TABLE(wxScrolledWindow
, wxPanel
)
32 EVT_SCROLLWIN(wxScrolledWindow::OnScroll
)
33 EVT_SIZE(wxScrolledWindow::OnSize
)
34 EVT_PAINT(wxScrolledWindow::OnPaint
)
37 IMPLEMENT_DYNAMIC_CLASS(wxScrolledWindow
, wxPanel
)
45 // For wxRETAINED implementation
49 wxScrolledWindow::wxScrolledWindow()
51 m_xScrollPixelsPerLine
= 0;
52 m_yScrollPixelsPerLine
= 0;
53 m_xScrollingEnabled
= TRUE
;
54 m_yScrollingEnabled
= TRUE
;
55 m_xScrollPosition
= 0;
56 m_yScrollPosition
= 0;
59 m_xScrollLinesPerPage
= 0;
60 m_yScrollLinesPerPage
= 0;
65 bool wxScrolledWindow::Create(wxWindow
*parent
, wxWindowID id
,
71 m_xScrollPixelsPerLine
= 0;
72 m_yScrollPixelsPerLine
= 0;
73 m_xScrollingEnabled
= TRUE
;
74 m_yScrollingEnabled
= TRUE
;
75 m_xScrollPosition
= 0;
76 m_yScrollPosition
= 0;
79 m_xScrollLinesPerPage
= 0;
80 m_yScrollLinesPerPage
= 0;
84 return wxPanel::Create(parent
, id
, pos
, size
, style
, name
);
88 * pixelsPerUnitX/pixelsPerUnitY: number of pixels per unit (e.g. pixels per text line)
89 * noUnitsX/noUnitsY: : no. units per scrollbar
91 void wxScrolledWindow::SetScrollbars (int pixelsPerUnitX
, int pixelsPerUnitY
,
92 int noUnitsX
, int noUnitsY
,
93 int xPos
, int yPos
, bool noRefresh
)
97 (noUnitsX
!= 0 && m_xScrollLines
== 0) ||
98 (noUnitsX
< m_xScrollLines
) ||
99 (noUnitsY
!= 0 && m_yScrollLines
== 0) ||
100 (noUnitsY
< m_yScrollLines
) ||
101 (xPos
!= m_xScrollPosition
) ||
102 (yPos
!= m_yScrollPosition
) ||
103 (pixelsPerUnitX
!= m_xScrollPixelsPerLine
) ||
104 (pixelsPerUnitY
!= m_yScrollPixelsPerLine
)
107 m_xScrollPixelsPerLine
= pixelsPerUnitX
;
108 m_yScrollPixelsPerLine
= pixelsPerUnitY
;
109 m_xScrollPosition
= xPos
;
110 m_yScrollPosition
= yPos
;
111 m_xScrollLines
= noUnitsX
;
112 m_yScrollLines
= noUnitsY
;
115 // Sorry, some Motif-specific code to implement a backing pixmap
116 // for the wxRETAINED style. Implementing a backing store can't
117 // be entirely generic because it relies on the wxWindowDC implementation
118 // to duplicate X drawing calls for the backing pixmap.
120 if ((m_windowStyle
& wxRETAINED
) == wxRETAINED
)
122 Display
* dpy
= XtDisplay((Widget
) GetMainWidget());
124 int totalPixelWidth
= m_xScrollLines
* m_xScrollPixelsPerLine
;
125 int totalPixelHeight
= m_yScrollLines
* m_yScrollPixelsPerLine
;
126 if (m_backingPixmap
&&
127 !((m_pixmapWidth
== totalPixelWidth
) &&
128 (m_pixmapHeight
== totalPixelHeight
)))
130 XFreePixmap (dpy
, (Pixmap
) m_backingPixmap
);
131 m_backingPixmap
= (WXPixmap
) 0;
134 if (!m_backingPixmap
&&
135 (noUnitsX
!= 0) && (noUnitsY
!= 0))
137 int depth
= wxDisplayDepth();
138 m_pixmapWidth
= totalPixelWidth
;
139 m_pixmapHeight
= totalPixelHeight
;
140 m_backingPixmap
= (WXPixmap
) XCreatePixmap (dpy
, RootWindow (dpy
, DefaultScreen (dpy
)),
141 m_pixmapWidth
, m_pixmapHeight
, depth
);
149 if (do_refresh
&& !noRefresh
)
154 UpdateWindow ((HWND
) GetHWND());
158 void wxScrolledWindow::OnScroll(wxScrollWinEvent
& event
)
160 int orient
= event
.GetOrientation();
162 int nScrollInc
= CalcScrollInc(event
);
163 if (nScrollInc
== 0) return;
165 if (orient
== wxHORIZONTAL
)
167 int newPos
= m_xScrollPosition
+ nScrollInc
;
168 SetScrollPos(wxHORIZONTAL
, newPos
, TRUE
);
172 int newPos
= m_yScrollPosition
+ nScrollInc
;
173 SetScrollPos(wxVERTICAL
, newPos
, TRUE
);
176 if (orient
== wxHORIZONTAL
)
178 m_xScrollPosition
+= nScrollInc
;
182 m_yScrollPosition
+= nScrollInc
;
185 if (orient
== wxHORIZONTAL
)
187 if (m_xScrollingEnabled
)
188 ScrollWindow(-m_xScrollPixelsPerLine
* nScrollInc
, 0, (const wxRect
*) NULL
);
194 if (m_yScrollingEnabled
)
195 ScrollWindow(0, -m_yScrollPixelsPerLine
* nScrollInc
, (const wxRect
*) NULL
);
201 int wxScrolledWindow::CalcScrollInc(wxScrollWinEvent
& event
)
203 int pos
= event
.GetPosition();
204 int orient
= event
.GetOrientation();
207 switch (event
.GetEventType())
209 case wxEVT_SCROLLWIN_TOP
:
211 if (orient
== wxHORIZONTAL
)
212 nScrollInc
= - m_xScrollPosition
;
214 nScrollInc
= - m_yScrollPosition
;
217 case wxEVT_SCROLLWIN_BOTTOM
:
219 if (orient
== wxHORIZONTAL
)
220 nScrollInc
= m_xScrollLines
- m_xScrollPosition
;
222 nScrollInc
= m_yScrollLines
- m_yScrollPosition
;
225 case wxEVT_SCROLLWIN_LINEUP
:
230 case wxEVT_SCROLLWIN_LINEDOWN
:
235 case wxEVT_SCROLLWIN_PAGEUP
:
237 if (orient
== wxHORIZONTAL
)
238 nScrollInc
= -GetScrollPageSize(wxHORIZONTAL
);
240 nScrollInc
= -GetScrollPageSize(wxVERTICAL
);
243 case wxEVT_SCROLLWIN_PAGEDOWN
:
245 if (orient
== wxHORIZONTAL
)
246 nScrollInc
= GetScrollPageSize(wxHORIZONTAL
);
248 nScrollInc
= GetScrollPageSize(wxVERTICAL
);
251 case wxEVT_SCROLLWIN_THUMBTRACK
:
253 if (orient
== wxHORIZONTAL
)
254 nScrollInc
= pos
- m_xScrollPosition
;
256 nScrollInc
= pos
- m_yScrollPosition
;
265 if (orient
== wxHORIZONTAL
)
267 if (m_xScrollPixelsPerLine
> 0) {
269 GetClientSize(&w
, &h
);
271 int nMaxWidth
= m_xScrollLines
*m_xScrollPixelsPerLine
;
272 int noPositions
= (int) ( ((nMaxWidth
- w
)/(float)m_xScrollPixelsPerLine
) + 0.5 );
276 if ( (m_xScrollPosition
+ nScrollInc
) < 0 )
277 nScrollInc
= -m_xScrollPosition
; // As -ve as we can go
278 else if ( (m_xScrollPosition
+ nScrollInc
) > noPositions
)
279 nScrollInc
= noPositions
- m_xScrollPosition
; // As +ve as we can go
286 if (m_yScrollPixelsPerLine
> 0) {
288 GetClientSize(&w
, &h
);
290 int nMaxHeight
= m_yScrollLines
*m_yScrollPixelsPerLine
;
291 int noPositions
= (int) ( ((nMaxHeight
- h
)/(float)m_yScrollPixelsPerLine
) + 0.5 );
295 if ( (m_yScrollPosition
+ nScrollInc
) < 0 )
296 nScrollInc
= -m_yScrollPosition
; // As -ve as we can go
297 else if ( (m_yScrollPosition
+ nScrollInc
) > noPositions
)
298 nScrollInc
= noPositions
- m_yScrollPosition
; // As +ve as we can go
307 // Adjust the scrollbars - new version.
308 void wxScrolledWindow::AdjustScrollbars()
311 GetClientSize(&w
, &h
);
313 int oldXScroll
= m_xScrollPosition
;
314 int oldYScroll
= m_yScrollPosition
;
316 if (m_xScrollLines
> 0)
318 // Calculate page size i.e. number of scroll units you get on the
319 // current client window
320 int noPagePositions
= (int) ( (w
/(float)m_xScrollPixelsPerLine
) + 0.5 );
321 if (noPagePositions
< 1) noPagePositions
= 1;
323 // Correct position if greater than extent of canvas minus
324 // the visible portion of it or if below zero
325 m_xScrollPosition
= wxMin( m_xScrollLines
-noPagePositions
, m_xScrollPosition
);
326 m_xScrollPosition
= wxMax( 0, m_xScrollPosition
);
328 SetScrollbar(wxHORIZONTAL
, m_xScrollPosition
, noPagePositions
, m_xScrollLines
);
329 // The amount by which we scroll when paging
330 SetScrollPageSize(wxHORIZONTAL
, noPagePositions
);
334 m_xScrollPosition
= 0;
335 SetScrollbar (wxHORIZONTAL
, 0, 0, 0, FALSE
);
338 if (m_yScrollLines
> 0)
340 // Calculate page size i.e. number of scroll units you get on the
341 // current client window
342 int noPagePositions
= (int) ( (h
/(float)m_yScrollPixelsPerLine
) + 0.5 );
343 if (noPagePositions
< 1) noPagePositions
= 1;
345 // Correct position if greater than extent of canvas minus
346 // the visible portion of it or if below zero
347 m_yScrollPosition
= wxMin( m_yScrollLines
-noPagePositions
, m_yScrollPosition
);
348 m_yScrollPosition
= wxMax( 0, m_yScrollPosition
);
350 SetScrollbar(wxVERTICAL
, m_yScrollPosition
, noPagePositions
, m_yScrollLines
);
351 // The amount by which we scroll when paging
352 SetScrollPageSize(wxVERTICAL
, noPagePositions
);
356 m_yScrollPosition
= 0;
357 SetScrollbar (wxVERTICAL
, 0, 0, 0, FALSE
);
360 if (oldXScroll
!= m_xScrollPosition
)
362 if (m_xScrollingEnabled
)
363 ScrollWindow( m_xScrollPixelsPerLine
* (oldXScroll
-m_xScrollPosition
), 0, (const wxRect
*) NULL
);
368 if (oldYScroll
!= m_yScrollPosition
)
370 if (m_yScrollingEnabled
)
371 ScrollWindow( 0, m_yScrollPixelsPerLine
* (oldYScroll
-m_yScrollPosition
), (const wxRect
*) NULL
);
377 // Default OnSize resets scrollbars, if any
378 void wxScrolledWindow::OnSize(wxSizeEvent
& WXUNUSED(event
))
380 #if wxUSE_CONSTRAINTS
381 if (GetAutoLayout()) Layout();
387 // This calls OnDraw, having adjusted the origin according to the current
389 void wxScrolledWindow::OnPaint(wxPaintEvent
& WXUNUSED(event
))
397 // Override this function if you don't want to have wxScrolledWindow
398 // automatically change the origin according to the scroll position.
399 void wxScrolledWindow::PrepareDC(wxDC
& dc
)
401 dc
.SetDeviceOrigin( -m_xScrollPosition
* m_xScrollPixelsPerLine
,
402 -m_yScrollPosition
* m_yScrollPixelsPerLine
);
403 dc
.SetUserScale( m_scaleX
, m_scaleY
);
406 #if WXWIN_COMPATIBILITY
407 void wxScrolledWindow::GetScrollUnitsPerPage (int *x_page
, int *y_page
) const
409 *x_page
= GetScrollPageSize(wxHORIZONTAL
);
410 *y_page
= GetScrollPageSize(wxVERTICAL
);
413 void wxScrolledWindow::CalcUnscrolledPosition(int x
, int y
, float *xx
, float *yy
) const
416 *xx
= (float)(x
+ m_xScrollPosition
* m_xScrollPixelsPerLine
);
418 *yy
= (float)(y
+ m_yScrollPosition
* m_yScrollPixelsPerLine
);
420 #endif // WXWIN_COMPATIBILITY
422 void wxScrolledWindow::GetScrollPixelsPerUnit (int *x_unit
, int *y_unit
) const
425 *x_unit
= m_xScrollPixelsPerLine
;
427 *y_unit
= m_yScrollPixelsPerLine
;
430 int wxScrolledWindow::GetScrollPageSize(int orient
) const
432 if ( orient
== wxHORIZONTAL
)
433 return m_xScrollLinesPerPage
;
435 return m_yScrollLinesPerPage
;
438 void wxScrolledWindow::SetScrollPageSize(int orient
, int pageSize
)
440 if ( orient
== wxHORIZONTAL
)
441 m_xScrollLinesPerPage
= pageSize
;
443 m_yScrollLinesPerPage
= pageSize
;
447 * Scroll to given position (scroll position, not pixel position)
449 void wxScrolledWindow::Scroll( int x_pos
, int y_pos
)
451 if (((x_pos
== -1) || (x_pos
== m_xScrollPosition
)) &&
452 ((y_pos
== -1) || (y_pos
== m_yScrollPosition
))) return;
455 GetClientSize(&w
, &h
);
459 int old_x
= m_xScrollPosition
;
460 m_xScrollPosition
= x_pos
;
462 // Calculate page size i.e. number of scroll units you get on the
463 // current client window
464 int noPagePositions
= (int) ( (w
/(float)m_xScrollPixelsPerLine
) + 0.5 );
465 if (noPagePositions
< 1) noPagePositions
= 1;
467 // Correct position if greater than extent of canvas minus
468 // the visible portion of it or if below zero
469 m_xScrollPosition
= wxMin( m_xScrollLines
-noPagePositions
, m_xScrollPosition
);
470 m_xScrollPosition
= wxMax( 0, m_xScrollPosition
);
472 SetScrollPos( wxHORIZONTAL
, m_xScrollPosition
, TRUE
);
474 ScrollWindow( (old_x
-m_xScrollPosition
)*m_xScrollPixelsPerLine
, 0 );
478 int old_y
= m_yScrollPosition
;
479 m_yScrollPosition
= y_pos
;
481 // Calculate page size i.e. number of scroll units you get on the
482 // current client window
483 int noPagePositions
= (int) ( (h
/(float)m_yScrollPixelsPerLine
) + 0.5 );
484 if (noPagePositions
< 1) noPagePositions
= 1;
486 // Correct position if greater than extent of canvas minus
487 // the visible portion of it or if below zero
488 m_yScrollPosition
= wxMin( m_yScrollLines
-noPagePositions
, m_yScrollPosition
);
489 m_yScrollPosition
= wxMax( 0, m_yScrollPosition
);
491 SetScrollPos( wxVERTICAL
, m_yScrollPosition
, TRUE
);
493 ScrollWindow( 0, (old_y
-m_yScrollPosition
)*m_yScrollPixelsPerLine
);
498 // ::UpdateWindow ((HWND) GetHWND());
504 void wxScrolledWindow::EnableScrolling (bool x_scroll
, bool y_scroll
)
506 m_xScrollingEnabled
= x_scroll
;
507 m_yScrollingEnabled
= y_scroll
;
510 void wxScrolledWindow::GetVirtualSize (int *x
, int *y
) const
513 *x
= m_xScrollPixelsPerLine
* m_xScrollLines
;
515 *y
= m_yScrollPixelsPerLine
* m_yScrollLines
;
518 // Where the current view starts from
519 void wxScrolledWindow::ViewStart (int *x
, int *y
) const
522 *x
= m_xScrollPosition
;
524 *y
= m_yScrollPosition
;
527 void wxScrolledWindow::CalcScrolledPosition(int x
, int y
, int *xx
, int *yy
) const
530 *xx
= x
- m_xScrollPosition
* m_xScrollPixelsPerLine
;
532 *yy
= y
- m_yScrollPosition
* m_yScrollPixelsPerLine
;
535 void wxScrolledWindow::CalcUnscrolledPosition(int x
, int y
, int *xx
, int *yy
) const
538 *xx
= x
+ m_xScrollPosition
* m_xScrollPixelsPerLine
;
540 *yy
= y
+ m_yScrollPosition
* m_yScrollPixelsPerLine
;