1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        gtk/scrolwin.cpp 
   3 // Purpose:     wxScrolledWindow implementation 
   4 // Author:      Julian Smart 
   5 // Modified by: Ron Lee 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "scrolwin.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  32 #include "wx/dcclient.h" 
  34 #include "wx/scrolwin.h" 
  38 #include "wx/gtk/private.h" 
  39 #include "wx/gtk/win_gtk.h" 
  41 // ---------------------------------------------------------------------------- 
  43 // ---------------------------------------------------------------------------- 
  45 BEGIN_EVENT_TABLE(wxScrolledWindow
, wxPanel
) 
  46     EVT_SCROLLWIN(wxScrolledWindow::OnScroll
) 
  47     EVT_SIZE(wxScrolledWindow::OnSize
) 
  48     EVT_PAINT(wxScrolledWindow::OnPaint
) 
  49     EVT_CHAR(wxScrolledWindow::OnChar
) 
  52 IMPLEMENT_DYNAMIC_CLASS(wxScrolledWindow
, wxPanel
) 
  54 // ============================================================================ 
  56 // ============================================================================ 
  58 //----------------------------------------------------------------------------- 
  60 //----------------------------------------------------------------------------- 
  62 extern bool       g_blockEventsOnDrag
; 
  63 extern bool       g_blockEventsOnScroll
; 
  65 //----------------------------------------------------------------------------- 
  67 //----------------------------------------------------------------------------- 
  69 extern void wxapp_install_idle_handler(); 
  72 //----------------------------------------------------------------------------- 
  73 // "value_changed" from m_vAdjust 
  74 //----------------------------------------------------------------------------- 
  76 static void gtk_scrolled_window_vscroll_callback( GtkAdjustment 
*adjust
, 
  78                                                   wxScrolledWindow 
*win 
) 
  81         wxapp_install_idle_handler(); 
  83     if (g_blockEventsOnDrag
) return; 
  85     if (!win
->m_hasVMT
) return; 
  87     win
->GtkVScroll( adjust
->value
, 
  88             GET_SCROLL_TYPE(GTK_SCROLLED_WINDOW(win
->m_widget
)->vscrollbar
) ); 
  91 //----------------------------------------------------------------------------- 
  92 // "value_changed" from m_hAdjust 
  93 //----------------------------------------------------------------------------- 
  95 static void gtk_scrolled_window_hscroll_callback( GtkAdjustment 
*adjust
, 
  97                                                   wxScrolledWindow 
*win 
) 
 100         wxapp_install_idle_handler(); 
 102     if (g_blockEventsOnDrag
) return; 
 103     if (!win
->m_hasVMT
) return; 
 105     win
->GtkHScroll( adjust
->value
, 
 106             GET_SCROLL_TYPE(GTK_SCROLLED_WINDOW(win
->m_widget
)->hscrollbar
) ); 
 109 //----------------------------------------------------------------------------- 
 110 // "button_press_event" from scrollbar 
 111 //----------------------------------------------------------------------------- 
 113 static gint 
gtk_scrollbar_button_press_callback( GtkRange 
*widget
, 
 114                                                  GdkEventButton 
*gdk_event
, 
 118         wxapp_install_idle_handler(); 
 120     g_blockEventsOnScroll 
= TRUE
; 
 122     // FIXME: there is no slider field any more, what was meant here? 
 124     win
->m_isScrolling 
= (gdk_event
->window 
== widget
->slider
); 
 130 //----------------------------------------------------------------------------- 
 131 // "button_release_event" from scrollbar 
 132 //----------------------------------------------------------------------------- 
 134 static gint 
gtk_scrollbar_button_release_callback( GtkRange 
*widget
, 
 135                                                    GdkEventButton 
*WXUNUSED(gdk_event
), 
 138 //  don't test here as we can release the mouse while being over 
 139 //  a different window than the slider 
 141 //    if (gdk_event->window != widget->slider) return FALSE; 
 143     g_blockEventsOnScroll 
= FALSE
; 
 145     if (win
->m_isScrolling
) 
 147         wxEventType command 
= wxEVT_SCROLLWIN_THUMBRELEASE
; 
 151         GtkScrolledWindow 
*scrolledWindow 
= GTK_SCROLLED_WINDOW(win
->m_widget
); 
 152         if (widget 
== GTK_RANGE(scrolledWindow
->hscrollbar
)) 
 154             value 
= (int)(win
->m_hAdjust
->value
+0.5); 
 157         if (widget 
== GTK_RANGE(scrolledWindow
->vscrollbar
)) 
 159             value 
= (int)(win
->m_vAdjust
->value
+0.5); 
 163         wxScrollWinEvent 
event( command
, value
, dir 
); 
 164         event
.SetEventObject( win 
); 
 165         win
->GetEventHandler()->ProcessEvent( event 
); 
 168     win
->m_isScrolling 
= FALSE
; 
 173 //----------------------------------------------------------------------------- 
 174 // InsertChild for wxScrolledWindow 
 175 //----------------------------------------------------------------------------- 
 177 static void wxInsertChildInScrolledWindow( wxWindow
* parent
, wxWindow
* child 
) 
 179     // The window might have been scrolled already, do we 
 180     // have to adapt the position. 
 181     GtkPizza 
*pizza 
= GTK_PIZZA(parent
->m_wxwindow
); 
 182     child
->m_x 
+= pizza
->xoffset
; 
 183     child
->m_y 
+= pizza
->yoffset
; 
 185     gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
), 
 186                      GTK_WIDGET(child
->m_widget
), 
 193 // ---------------------------------------------------------------------------- 
 194 // wxScrolledWindow creation 
 195 // ---------------------------------------------------------------------------- 
 197 void wxScrolledWindow::Init() 
 199     m_xScrollPixelsPerLine 
= 0; 
 200     m_yScrollPixelsPerLine 
= 0; 
 201     m_xScrollingEnabled 
= TRUE
; 
 202     m_yScrollingEnabled 
= TRUE
; 
 203     m_xScrollPosition 
= 0; 
 204     m_yScrollPosition 
= 0; 
 205     m_xScrollLinesPerPage 
= 0; 
 206     m_yScrollLinesPerPage 
= 0; 
 207     m_targetWindow 
= (wxWindow
*) NULL
; 
 210     m_hasScrolling 
= TRUE
; 
 213 bool wxScrolledWindow::Create(wxWindow 
*parent
, 
 218                               const wxString
& name
) 
 222     if (!PreCreation( parent
, pos
, size 
) || 
 223         !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
 225         wxFAIL_MSG( wxT("wxWindow creation failed") ); 
 229     m_insertCallback 
= wxInsertChildInScrolledWindow
; 
 231     m_targetWindow 
= this; 
 233     m_widget 
= gtk_scrolled_window_new( (GtkAdjustment 
*) NULL
, (GtkAdjustment 
*) NULL 
); 
 234     GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS 
); 
 236     GtkScrolledWindow 
*scrolledWindow 
= GTK_SCROLLED_WINDOW(m_widget
); 
 238     GtkScrolledWindowClass 
*scroll_class 
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) ); 
 239     scroll_class
->scrollbar_spacing 
= 0; 
 241     gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC 
); 
 243     m_hAdjust 
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->hscrollbar
) ); 
 244     m_vAdjust 
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->vscrollbar
) ); 
 246     m_wxwindow 
= gtk_pizza_new(); 
 248     gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow 
); 
 250     GtkPizza 
*pizza 
= GTK_PIZZA(m_wxwindow
); 
 252     if (HasFlag(wxRAISED_BORDER
)) 
 254         gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_OUT 
); 
 256     else if (HasFlag(wxSUNKEN_BORDER
)) 
 258         gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_IN 
); 
 260     else if (HasFlag(wxSIMPLE_BORDER
)) 
 262         gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_THIN 
); 
 266         gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_NONE 
); 
 269     GTK_WIDGET_SET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS 
); 
 270     m_acceptsFocus 
= TRUE
; 
 272     // I _really_ don't want scrollbars in the beginning 
 273     m_vAdjust
->lower 
= 0.0; 
 274     m_vAdjust
->upper 
= 1.0; 
 275     m_vAdjust
->value 
= 0.0; 
 276     m_vAdjust
->step_increment 
= 1.0; 
 277     m_vAdjust
->page_increment 
= 2.0; 
 278     gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" ); 
 279     m_hAdjust
->lower 
= 0.0; 
 280     m_hAdjust
->upper 
= 1.0; 
 281     m_hAdjust
->value 
= 0.0; 
 282     m_hAdjust
->step_increment 
= 1.0; 
 283     m_hAdjust
->page_increment 
= 2.0; 
 284     gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" ); 
 286     // Handlers for new scrollbar values 
 290     // these handlers block mouse events to any window during scrolling such as 
 291     // motion events and prevent GTK and wxWindows from fighting over where the 
 294     gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_press_event", 
 295           (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this ); 
 297     gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_press_event", 
 298           (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this ); 
 300     gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_release_event", 
 301           (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this ); 
 303     gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_release_event", 
 304           (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this ); 
 306     gtk_widget_show( m_wxwindow 
); 
 309         m_parent
->DoAddChild( this ); 
 311     m_focusWidget 
= m_wxwindow
; 
 320 // ---------------------------------------------------------------------------- 
 321 // setting scrolling parameters 
 322 // ---------------------------------------------------------------------------- 
 324 void wxScrolledWindow::DoSetVirtualSize( int x
, int y 
) 
 326     wxPanel::DoSetVirtualSize( x
, y 
); 
 329 #if wxUSE_CONSTRAINTS 
 336  * pixelsPerUnitX/pixelsPerUnitY: number of pixels per unit (e.g. pixels per text line) 
 337  * noUnitsX/noUnitsY:        : no. units per scrollbar 
 339 void wxScrolledWindow::SetScrollbars( int pixelsPerUnitX
, int pixelsPerUnitY
, 
 340                                       int noUnitsX
, int noUnitsY
, 
 341                                       int xPos
, int yPos
, bool noRefresh 
) 
 343     int old_x 
= m_xScrollPixelsPerLine 
* m_xScrollPosition
; 
 344     int old_y 
= m_yScrollPixelsPerLine 
* m_yScrollPosition
; 
 346     m_xScrollPixelsPerLine 
= pixelsPerUnitX
; 
 347     m_yScrollPixelsPerLine 
= pixelsPerUnitY
; 
 349     m_hAdjust
->value 
= m_xScrollPosition 
= xPos
; 
 350     m_vAdjust
->value 
= m_yScrollPosition 
= yPos
; 
 352     m_targetWindow
->SetVirtualSizeHints( noUnitsX 
* pixelsPerUnitX
, noUnitsY 
* pixelsPerUnitY 
); 
 356         int new_x 
= m_xScrollPixelsPerLine 
* m_xScrollPosition
; 
 357         int new_y 
= m_yScrollPixelsPerLine 
* m_yScrollPosition
; 
 359         m_targetWindow
->ScrollWindow( old_x 
- new_x
, old_y 
- new_y 
); 
 363 void wxScrolledWindow::AdjustScrollbars() 
 368     m_targetWindow
->GetClientSize( &w
, &h 
); 
 369     m_targetWindow
->GetVirtualSize( &vw
, &vh 
); 
 371     if (m_xScrollPixelsPerLine 
== 0) 
 373         m_hAdjust
->upper 
= 1.0; 
 374         m_hAdjust
->page_size 
= 1.0; 
 378         m_hAdjust
->upper 
= vw 
/ m_xScrollPixelsPerLine
; 
 379         m_hAdjust
->page_size 
= (w 
/ m_xScrollPixelsPerLine
); 
 381         // If the scrollbar hits the right side, move the window 
 382         // right to keep it from over extending. 
 384         if( m_hAdjust
->value 
+ m_hAdjust
->page_size 
> m_hAdjust
->upper 
) 
 386             if (GetChildren().GetCount() == 0) 
 388                 m_hAdjust
->value 
= m_hAdjust
->upper 
- m_hAdjust
->page_size
; 
 389                 m_xScrollPosition 
= (int)m_hAdjust
->value
; 
 394     if (m_yScrollPixelsPerLine 
== 0) 
 396         m_vAdjust
->upper 
= 1.0; 
 397         m_vAdjust
->page_size 
= 1.0; 
 401         m_vAdjust
->upper 
= vh 
/ m_yScrollPixelsPerLine
; 
 402         m_vAdjust
->page_size 
= (h 
/ m_yScrollPixelsPerLine
); 
 404         if( m_vAdjust
->value 
+ m_vAdjust
->page_size 
> m_vAdjust
->upper 
) 
 406             if (GetChildren().GetCount() == 0) 
 408                 m_vAdjust
->value 
= m_vAdjust
->upper 
- m_vAdjust
->page_size
; 
 409                 m_yScrollPosition 
= (int)m_vAdjust
->value
; 
 414     m_xScrollLinesPerPage 
= (int)(m_hAdjust
->page_size 
+ 0.5); 
 415     m_yScrollLinesPerPage 
= (int)(m_vAdjust
->page_size 
+ 0.5); 
 417     gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" ); 
 418     gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" ); 
 422 // ---------------------------------------------------------------------------- 
 423 // target window handling 
 424 // ---------------------------------------------------------------------------- 
 426 void wxScrolledWindow::SetTargetWindow( wxWindow 
*target
, bool WXUNUSED(pushEventHandler
) ) 
 428     wxASSERT_MSG( target
, wxT("target window must not be NULL") ); 
 429     m_targetWindow 
= target
; 
 432 wxWindow 
*wxScrolledWindow::GetTargetWindow() 
 434     return m_targetWindow
; 
 437 // Override this function if you don't want to have wxScrolledWindow 
 438 // automatically change the origin according to the scroll position. 
 439 void wxScrolledWindow::PrepareDC(wxDC
& dc
) 
 441     dc
.SetDeviceOrigin( -m_xScrollPosition 
* m_xScrollPixelsPerLine
, 
 442                         -m_yScrollPosition 
* m_yScrollPixelsPerLine 
); 
 445 void wxScrolledWindow::SetScrollRate( int xstep
, int ystep 
) 
 447     int old_x 
= m_xScrollPixelsPerLine 
* m_xScrollPosition
; 
 448     int old_y 
= m_yScrollPixelsPerLine 
* m_yScrollPosition
; 
 450     m_xScrollPixelsPerLine 
= xstep
; 
 451     m_yScrollPixelsPerLine 
= ystep
; 
 453     int new_x 
= m_xScrollPixelsPerLine 
* m_xScrollPosition
; 
 454     int new_y 
= m_yScrollPixelsPerLine 
* m_yScrollPosition
; 
 456     m_targetWindow
->ScrollWindow( old_x 
- new_x
, old_y 
- new_y 
); 
 461 void wxScrolledWindow::GetScrollPixelsPerUnit (int *x_unit
, int *y_unit
) const 
 464         *x_unit 
= m_xScrollPixelsPerLine
; 
 466         *y_unit 
= m_yScrollPixelsPerLine
; 
 469 int wxScrolledWindow::GetScrollPageSize(int orient
) const 
 471     if ( orient 
== wxHORIZONTAL 
) 
 472         return m_xScrollLinesPerPage
; 
 474         return m_yScrollLinesPerPage
; 
 477 void wxScrolledWindow::SetScrollPageSize(int orient
, int pageSize
) 
 479     if ( orient 
== wxHORIZONTAL 
) 
 480         m_xScrollLinesPerPage 
= pageSize
; 
 482         m_yScrollLinesPerPage 
= pageSize
; 
 485 void wxScrolledWindow::OnScroll(wxScrollWinEvent
& event
) 
 487     int orient 
= event
.GetOrientation(); 
 489     int nScrollInc 
= CalcScrollInc(event
); 
 490     if (nScrollInc 
== 0) return; 
 492     if (orient 
== wxHORIZONTAL
) 
 494         int newPos 
= m_xScrollPosition 
+ nScrollInc
; 
 495         SetScrollPos(wxHORIZONTAL
, newPos
, TRUE 
); 
 499         int newPos 
= m_yScrollPosition 
+ nScrollInc
; 
 500         SetScrollPos(wxVERTICAL
, newPos
, TRUE 
); 
 503     if (orient 
== wxHORIZONTAL
) 
 505         m_xScrollPosition 
+= nScrollInc
; 
 509         m_yScrollPosition 
+= nScrollInc
; 
 512     if (orient 
== wxHORIZONTAL
) 
 514        if (m_xScrollingEnabled
) 
 515             m_targetWindow
->ScrollWindow(-m_xScrollPixelsPerLine 
* nScrollInc
, 0, (const wxRect 
*) NULL
); 
 517             m_targetWindow
->Refresh(); 
 521         if (m_yScrollingEnabled
) 
 522             m_targetWindow
->ScrollWindow(0, -m_yScrollPixelsPerLine 
* nScrollInc
, (const wxRect 
*) NULL
); 
 524             m_targetWindow
->Refresh(); 
 528 void wxScrolledWindow::Scroll( int x_pos
, int y_pos 
) 
 530     wxASSERT_MSG( m_targetWindow 
!= 0, _T("No target window") ); 
 532     if (((x_pos 
== -1) || (x_pos 
== m_xScrollPosition
)) && 
 533         ((y_pos 
== -1) || (y_pos 
== m_yScrollPosition
))) return; 
 535     if ((x_pos 
!= -1) && (m_xScrollPixelsPerLine
)) 
 537         int max 
= (int)(m_hAdjust
->upper 
- m_hAdjust
->page_size 
+ 0.5); 
 538         if (max 
< 0) max 
= 0; 
 539         if (x_pos 
> max
) x_pos 
= max
; 
 540         if (x_pos 
< 0) x_pos 
= 0; 
 542         int old_x 
= m_xScrollPosition
; 
 543         m_xScrollPosition 
= x_pos
; 
 544         m_hAdjust
->value 
= x_pos
; 
 546         m_targetWindow
->ScrollWindow( (old_x
-m_xScrollPosition
)*m_xScrollPixelsPerLine
, 0 ); 
 548         // Just update the scrollbar, don't send any wxWindows event 
 549         GtkHDisconnectEvent(); 
 550         gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "value_changed" ); 
 554     if ((y_pos 
!= -1) && (m_yScrollPixelsPerLine
)) 
 556         int max 
= (int)(m_vAdjust
->upper 
- m_vAdjust
->page_size 
+ 0.5); 
 557         if (max 
< 0) max 
= 0; 
 558         if (y_pos 
> max
) y_pos 
= max
; 
 559         if (y_pos 
< 0) y_pos 
= 0; 
 561         int old_y 
= m_yScrollPosition
; 
 562         m_yScrollPosition 
= y_pos
; 
 563         m_vAdjust
->value 
= y_pos
; 
 565         m_targetWindow
->ScrollWindow( 0, (old_y
-m_yScrollPosition
)*m_yScrollPixelsPerLine 
); 
 567         // Just update the scrollbar, don't send any wxWindows event 
 568         GtkVDisconnectEvent(); 
 569         gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "value_changed" ); 
 574 // TODO: [VH]Scroll functions should be combined 
 576 void wxScrolledWindow::GtkVScroll( float value
, unsigned int scroll_type 
) 
 578     wxASSERT_MSG( m_targetWindow 
!= 0, _T("No target window") ); 
 580     if (m_yScrollPixelsPerLine 
== 0) 
 583     int y_pos 
= (int)(value
+0.5); 
 585     if (y_pos 
== m_yScrollPosition
) 
 588     wxEventType command 
= GtkScrollWinTypeToWx(scroll_type
); 
 590     wxScrollWinEvent 
event( command
, y_pos
, wxVERTICAL 
); 
 591     event
.SetEventObject( this ); 
 592     GetEventHandler()->ProcessEvent( event 
); 
 595 void wxScrolledWindow::GtkHScroll( float value
, unsigned int scroll_type 
) 
 597     wxASSERT_MSG( m_targetWindow 
!= 0, _T("No target window") ); 
 599     if (m_xScrollPixelsPerLine 
== 0) 
 602     int x_pos 
= (int)(value
+0.5); 
 604     if (x_pos 
== m_xScrollPosition
) 
 607     wxEventType command 
= GtkScrollWinTypeToWx(scroll_type
); 
 609     wxScrollWinEvent 
event( command
, x_pos
, wxHORIZONTAL 
); 
 610     event
.SetEventObject( this ); 
 611     GetEventHandler()->ProcessEvent( event 
); 
 614 void wxScrolledWindow::EnableScrolling (bool x_scroll
, bool y_scroll
) 
 616     m_xScrollingEnabled 
= x_scroll
; 
 617     m_yScrollingEnabled 
= y_scroll
; 
 620 // Where the current view starts from 
 621 void wxScrolledWindow::GetViewStart (int *x
, int *y
) const 
 624         *x 
= m_xScrollPosition
; 
 626         *y 
= m_yScrollPosition
; 
 629 void wxScrolledWindow::DoCalcScrolledPosition(int x
, int y
, int *xx
, int *yy
) const 
 632         *xx 
= x 
- m_xScrollPosition 
* m_xScrollPixelsPerLine
; 
 634         *yy 
= y 
- m_yScrollPosition 
* m_yScrollPixelsPerLine
; 
 637 void wxScrolledWindow::DoCalcUnscrolledPosition(int x
, int y
, int *xx
, int *yy
) const 
 640         *xx 
= x 
+ m_xScrollPosition 
* m_xScrollPixelsPerLine
; 
 642         *yy 
= y 
+ m_yScrollPosition 
* m_yScrollPixelsPerLine
; 
 645 int wxScrolledWindow::CalcScrollInc(wxScrollWinEvent
& event
) 
 647     int pos 
= event
.GetPosition(); 
 648     int orient 
= event
.GetOrientation(); 
 651     if (event
.GetEventType() == wxEVT_SCROLLWIN_TOP
) 
 653             if (orient 
== wxHORIZONTAL
) 
 654                 nScrollInc 
= - m_xScrollPosition
; 
 656                 nScrollInc 
= - m_yScrollPosition
; 
 658     if (event
.GetEventType() == wxEVT_SCROLLWIN_BOTTOM
) 
 660             if (orient 
== wxHORIZONTAL
) 
 661                 nScrollInc 
= GetVirtualSize().GetWidth() / m_xScrollPixelsPerLine 
- m_xScrollPosition
; 
 663                 nScrollInc 
= GetVirtualSize().GetHeight() / m_yScrollPixelsPerLine 
- m_yScrollPosition
; 
 665     if (event
.GetEventType() == wxEVT_SCROLLWIN_LINEUP
) 
 669     if (event
.GetEventType() == wxEVT_SCROLLWIN_LINEDOWN
) 
 673     if (event
.GetEventType() == wxEVT_SCROLLWIN_PAGEUP
) 
 675             if (orient 
== wxHORIZONTAL
) 
 676                 nScrollInc 
= -GetScrollPageSize(wxHORIZONTAL
); 
 678                 nScrollInc 
= -GetScrollPageSize(wxVERTICAL
); 
 680     if (event
.GetEventType() == wxEVT_SCROLLWIN_PAGEDOWN
) 
 682             if (orient 
== wxHORIZONTAL
) 
 683                 nScrollInc 
= GetScrollPageSize(wxHORIZONTAL
); 
 685                 nScrollInc 
= GetScrollPageSize(wxVERTICAL
); 
 687     if ((event
.GetEventType() == wxEVT_SCROLLWIN_THUMBTRACK
) || 
 688         (event
.GetEventType() == wxEVT_SCROLLWIN_THUMBRELEASE
)) 
 690             if (orient 
== wxHORIZONTAL
) 
 691                 nScrollInc 
= pos 
- m_xScrollPosition
; 
 693                 nScrollInc 
= pos 
- m_yScrollPosition
; 
 696     if (orient 
== wxHORIZONTAL
) 
 698         if (m_xScrollPixelsPerLine 
> 0) 
 700             int max 
= (int)(m_hAdjust
->upper 
- m_hAdjust
->page_size 
+ 0.5); 
 701             if (max 
< 0) max 
= 0; 
 703             if ( (m_xScrollPosition 
+ nScrollInc
) < 0 ) 
 704                 nScrollInc 
= -m_xScrollPosition
; // As -ve as we can go 
 705             else if ( (m_xScrollPosition 
+ nScrollInc
) > max 
) 
 706                 nScrollInc 
= max 
- m_xScrollPosition
; // As +ve as we can go 
 709             m_targetWindow
->Refresh(); 
 713         if (m_yScrollPixelsPerLine 
> 0) 
 715             int max 
= (int)(m_vAdjust
->upper 
- m_vAdjust
->page_size 
+ 0.5); 
 716             if (max 
< 0) max 
= 0; 
 718             if ( (m_yScrollPosition 
+ nScrollInc
) < 0 ) 
 719                 nScrollInc 
= -m_yScrollPosition
; // As -ve as we can go 
 720             else if ( (m_yScrollPosition 
+ nScrollInc
) > max 
) 
 721                 nScrollInc 
= max 
- m_yScrollPosition
; // As +ve as we can go 
 724             m_targetWindow
->Refresh(); 
 730 void wxScrolledWindow::SetScrollPos( int orient
, int pos
, bool refresh 
) 
 732     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
 734     wxCHECK_RET( m_wxwindow 
!= NULL
, wxT("window needs client area for scrolling") ); 
 736     if (orient 
== wxHORIZONTAL
) 
 738         int max 
= (int)(m_hAdjust
->upper 
- m_hAdjust
->page_size 
+ 0.5); 
 739         if (max 
< 0) max 
= 0; 
 741         if (pos 
> max
) pos 
= 0; 
 742         if (pos 
< 0) pos 
= 0; 
 744         if (pos 
== (int)(m_hAdjust
->value
+0.5)) return; 
 745         m_hAdjust
->value 
= pos
; 
 749         int max 
= (int)(m_vAdjust
->upper 
- m_vAdjust
->page_size 
+ 0.5); 
 750         if (max 
< 0) max 
= 0; 
 752         if (pos 
> max
) pos 
= 0; 
 753         if (pos 
< 0) pos 
= 0; 
 755         if (pos 
== (int)(m_vAdjust
->value
+0.5)) return; 
 756         m_vAdjust
->value 
= pos
; 
 759     if (m_wxwindow
->window
) 
 761         if (orient 
== wxHORIZONTAL
) 
 763             // Just update the scrollbar, don't send any wxWindows event 
 764             GtkHDisconnectEvent(); 
 765             gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "value_changed" ); 
 770             // Just update the scrollbar, don't send any wxWindows event 
 771             GtkVDisconnectEvent(); 
 772             gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "value_changed" ); 
 778 void wxScrolledWindow::GtkVConnectEvent() 
 780     gtk_signal_connect( GTK_OBJECT(m_vAdjust
), "value_changed", 
 781           (GtkSignalFunc
) gtk_scrolled_window_vscroll_callback
, (gpointer
) this ); 
 784 void wxScrolledWindow::GtkHConnectEvent() 
 786     gtk_signal_connect( GTK_OBJECT(m_hAdjust
), "value_changed", 
 787           (GtkSignalFunc
) gtk_scrolled_window_hscroll_callback
, (gpointer
) this ); 
 790 void wxScrolledWindow::GtkHDisconnectEvent() 
 792     gtk_signal_disconnect_by_func( GTK_OBJECT(m_hAdjust
), 
 793         (GtkSignalFunc
) gtk_scrolled_window_hscroll_callback
, (gpointer
) this ); 
 796 void wxScrolledWindow::GtkVDisconnectEvent() 
 798     gtk_signal_disconnect_by_func( GTK_OBJECT(m_vAdjust
), 
 799         (GtkSignalFunc
) gtk_scrolled_window_vscroll_callback
, (gpointer
) this ); 
 802 bool wxScrolledWindow::Layout() 
 804     if (GetSizer() && m_targetWindow 
== this) 
 806         // If we're the scroll target, take into account the 
 807         // virtual size and scrolled position of the window. 
 810         CalcScrolledPosition(0,0, &x
,&y
); 
 811         GetVirtualSize(&w
, &h
); 
 812         GetSizer()->SetDimension(x
, y
, w
, h
); 
 816         return wxPanel::Layout();  // fall back to default for LayoutConstraints 
 819 // ---------------------------------------------------------------------------- 
 821 // ---------------------------------------------------------------------------- 
 823 // Default OnSize resets scrollbars, if any 
 824 void wxScrolledWindow::OnSize(wxSizeEvent
& WXUNUSED(event
)) 
 826     if( m_targetWindow 
!= this ) 
 827         m_targetWindow
->SetVirtualSize( m_targetWindow
->GetClientSize() ); 
 829     SetVirtualSize( GetClientSize() ); 
 832 // This calls OnDraw, having adjusted the origin according to the current 
 834 void wxScrolledWindow::OnPaint(wxPaintEvent
& WXUNUSED(event
)) 
 842 // kbd handling: notice that we use OnChar() and not OnKeyDown() for 
 843 // compatibility here - if we used OnKeyDown(), the programs which process 
 844 // arrows themselves in their OnChar() would never get the message and like 
 845 // this they always have the priority 
 846 void wxScrolledWindow::OnChar(wxKeyEvent
& event
) 
 848     int stx
, sty
,       // view origin 
 849         szx
, szy
,       // view size (total) 
 850         clix
, cliy
;     // view size (on screen) 
 852     ViewStart(&stx
, &sty
); 
 853     GetClientSize(&clix
, &cliy
); 
 854     GetVirtualSize(&szx
, &szy
); 
 856     if( m_xScrollPixelsPerLine 
) 
 858         clix 
/= m_xScrollPixelsPerLine
; 
 859         szx 
/= m_xScrollPixelsPerLine
; 
 866     if( m_yScrollPixelsPerLine 
) 
 868         cliy 
/= m_yScrollPixelsPerLine
; 
 869         szy 
/= m_yScrollPixelsPerLine
; 
 877     int xScrollOld 
= GetScrollPos(wxHORIZONTAL
), 
 878         yScrollOld 
= GetScrollPos(wxVERTICAL
); 
 881     switch ( event
.KeyCode() ) 
 885             dsty 
= sty 
- (5 * cliy 
/ 6); 
 886             Scroll(-1, (dsty 
== -1) ? 0 : dsty
); 
 891             Scroll(-1, sty 
+ (5 * cliy 
/ 6)); 
 895             Scroll(0, event
.ControlDown() ? 0 : -1); 
 899             Scroll(szx 
- clix
, event
.ControlDown() ? szy 
- cliy 
: -1); 
 924     int xScroll 
= GetScrollPos(wxHORIZONTAL
); 
 925     if ( xScroll 
!= xScrollOld 
) 
 927         wxScrollWinEvent 
event(wxEVT_SCROLLWIN_THUMBTRACK
, xScroll
, 
 929         event
.SetEventObject(this); 
 930         GetEventHandler()->ProcessEvent(event
); 
 933     int yScroll 
= GetScrollPos(wxVERTICAL
); 
 934     if ( yScroll 
!= yScrollOld 
) 
 936         wxScrollWinEvent 
event(wxEVT_SCROLLWIN_THUMBTRACK
, yScroll
, 
 938         event
.SetEventObject(this); 
 939         GetEventHandler()->ProcessEvent(event
);