1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/generic/splitter.cpp 
   3 // Purpose:     wxSplitterWindow implementation 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13     #pragma implementation "splitter.h" 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  25     #include "wx/string.h" 
  29     #include "wx/dcscreen.h" 
  31     #include "wx/window.h" 
  32     #include "wx/dialog.h" 
  35     #include "wx/settings.h" 
  38 #include "wx/splitter.h" 
  42 DEFINE_EVENT_TYPE(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
) 
  43 DEFINE_EVENT_TYPE(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
) 
  44 DEFINE_EVENT_TYPE(wxEVT_COMMAND_SPLITTER_DOUBLECLICKED
) 
  45 DEFINE_EVENT_TYPE(wxEVT_COMMAND_SPLITTER_UNSPLIT
) 
  47 IMPLEMENT_DYNAMIC_CLASS(wxSplitterWindow
, wxWindow
) 
  48 IMPLEMENT_DYNAMIC_CLASS(wxSplitterEvent
, wxNotifyEvent
) 
  50 BEGIN_EVENT_TABLE(wxSplitterWindow
, wxWindow
) 
  51     EVT_PAINT(wxSplitterWindow::OnPaint
) 
  52     EVT_SIZE(wxSplitterWindow::OnSize
) 
  53     EVT_IDLE(wxSplitterWindow::OnIdle
) 
  54     EVT_MOUSE_EVENTS(wxSplitterWindow::OnMouseEvent
) 
  56 #if defined( __WXMSW__ ) || defined( __WXMAC__) 
  57     EVT_SET_CURSOR(wxSplitterWindow::OnSetCursor
) 
  60     WX_EVENT_TABLE_CONTROL_CONTAINER(wxSplitterWindow
) 
  63 WX_DELEGATE_TO_CONTROL_CONTAINER(wxSplitterWindow
); 
  65 bool wxSplitterWindow::Create(wxWindow 
*parent
, wxWindowID id
, 
  71     // allow TABbing from one window to the other 
  72     style 
|= wxTAB_TRAVERSAL
; 
  74     if (!wxWindow::Create(parent
, id
, pos
, size
, style
, name
)) 
  77     m_permitUnsplitAlways 
= (style 
& wxSP_PERMIT_UNSPLIT
) != 0; 
  79     if ( style 
& wxSP_3DSASH 
) 
  84     if ( style 
& wxSP_3DBORDER 
) 
  86     else if ( style 
& wxSP_BORDER 
) 
  93     wxGetOsVersion( &major
, &minor 
); 
  95         m_windowStyle 
|= wxSP_SASH_AQUA
; 
 101 void wxSplitterWindow::Init() 
 103     m_container
.SetContainerWindow(this); 
 105     m_splitMode 
= wxSPLIT_VERTICAL
; 
 106     m_permitUnsplitAlways 
= TRUE
; 
 107     m_windowOne 
= (wxWindow 
*) NULL
; 
 108     m_windowTwo 
= (wxWindow 
*) NULL
; 
 109     m_dragMode 
= wxSPLIT_DRAG_NONE
; 
 116     m_sashPosition 
= m_requestedSashPosition 
= 0; 
 117     m_minimumPaneSize 
= 0; 
 118     m_sashCursorWE 
= wxCursor(wxCURSOR_SIZEWE
); 
 119     m_sashCursorNS 
= wxCursor(wxCURSOR_SIZENS
); 
 120     m_sashTrackerPen 
= new wxPen(*wxBLACK
, 2, wxSOLID
); 
 121     m_lightShadowPen 
= (wxPen 
*) NULL
; 
 122     m_mediumShadowPen 
= (wxPen 
*) NULL
; 
 123     m_darkShadowPen 
= (wxPen 
*) NULL
; 
 124     m_faceBrush 
= (wxBrush 
*) NULL
; 
 125     m_facePen 
= (wxPen 
*) NULL
; 
 126     m_hilightPen 
= (wxPen 
*) NULL
; 
 130     m_needUpdating 
= FALSE
; 
 133 wxSplitterWindow::~wxSplitterWindow() 
 135     delete m_sashTrackerPen
; 
 136     delete m_lightShadowPen
; 
 137     delete m_darkShadowPen
; 
 138     delete m_mediumShadowPen
; 
 144 void wxSplitterWindow::SetResizeCursor() 
 146     SetCursor(m_splitMode 
== wxSPLIT_VERTICAL 
? m_sashCursorWE
 
 150 void wxSplitterWindow::OnPaint(wxPaintEvent
& WXUNUSED(event
)) 
 154     if ( m_borderSize 
> 0 ) 
 159 void wxSplitterWindow::OnIdle(wxIdleEvent
& event
) 
 167 void wxSplitterWindow::OnMouseEvent(wxMouseEvent
& event
) 
 169     int x 
= (int)event
.GetX(), 
 170         y 
= (int)event
.GetY(); 
 172 #if defined(__WXMSW__) 
 173     // SetCursor(wxCursor());   // Is this required? 
 176     if (GetWindowStyle() & wxSP_NOSASH
) 
 179     // with wxSP_LIVE_UPDATE style the splitter windows are always resized 
 180     // following the mouse movement while it drags the sash, without it we only 
 181     // draw the sash at the new position but only resize the windows when the 
 182     // dragging is finished 
 183     bool isLive 
= (GetWindowStyleFlag() & wxSP_LIVE_UPDATE
) != 0; 
 185     if (event
.LeftDown()) 
 187         if ( SashHitTest(x
, y
) ) 
 189             // Start the drag now 
 190             m_dragMode 
= wxSPLIT_DRAG_DRAGGING
; 
 192             // Capture mouse and set the cursor 
 198                 // remember the initial sash position and draw the initial 
 200                 m_sashPositionCurrent 
= m_sashPosition
; 
 202                 DrawSashTracker(x
, y
); 
 212     else if (event
.LeftUp() && m_dragMode 
== wxSPLIT_DRAG_DRAGGING
) 
 214         // We can stop dragging now and see what we've got. 
 215         m_dragMode 
= wxSPLIT_DRAG_NONE
; 
 217         // Release mouse and unset the cursor 
 219         SetCursor(* wxSTANDARD_CURSOR
); 
 221         // exit if unsplit after doubleclick 
 230             DrawSashTracker(m_oldX
, m_oldY
); 
 233         // the position of the click doesn't exactly correspond to 
 234         // m_sashPosition, rather it changes it by the distance by which the 
 236         int diff 
= m_splitMode 
== wxSPLIT_VERTICAL 
? x 
- m_oldX 
: y 
- m_oldY
; 
 238         int posSashOld 
= isLive 
? m_sashPosition 
: m_sashPositionCurrent
; 
 239         int posSashNew 
= OnSashPositionChanging(posSashOld 
+ diff
); 
 240         if ( posSashNew 
== -1 ) 
 242             // change not allowed 
 246         if ( m_permitUnsplitAlways 
|| m_minimumPaneSize 
== 0 ) 
 248             // Deal with possible unsplit scenarios 
 249             if ( posSashNew 
== 0 ) 
 251                 // We remove the first window from the view 
 252                 wxWindow 
*removedWindow 
= m_windowOne
; 
 253                 m_windowOne 
= m_windowTwo
; 
 254                 m_windowTwo 
= (wxWindow 
*) NULL
; 
 255                 OnUnsplit(removedWindow
); 
 256                 wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_UNSPLIT
, this); 
 257                 event
.m_data
.win 
= removedWindow
; 
 258                 (void)DoSendEvent(event
); 
 259                 SetSashPositionAndNotify(0); 
 261             else if ( posSashNew 
== GetWindowSize() ) 
 263                 // We remove the second window from the view 
 264                 wxWindow 
*removedWindow 
= m_windowTwo
; 
 265                 m_windowTwo 
= (wxWindow 
*) NULL
; 
 266                 OnUnsplit(removedWindow
); 
 267                 wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_UNSPLIT
, this); 
 268                 event
.m_data
.win 
= removedWindow
; 
 269                 (void)DoSendEvent(event
); 
 270                 SetSashPositionAndNotify(0); 
 274                 SetSashPositionAndNotify(posSashNew
); 
 279             SetSashPositionAndNotify(posSashNew
); 
 283     }  // left up && dragging 
 284     else if (event
.Moving() && !event
.Dragging()) 
 286         // Just change the cursor as required 
 287         if ( SashHitTest(x
, y
) ) 
 290             SetCursor(* wxSTANDARD_CURSOR
); 
 292     else if (event
.Dragging() && (m_dragMode 
== wxSPLIT_DRAG_DRAGGING
)) 
 294         int diff 
= m_splitMode 
== wxSPLIT_VERTICAL 
? x 
- m_oldX 
: y 
- m_oldY
; 
 297             // nothing to do, mouse didn't really move far enough 
 301         int posSashOld 
= isLive 
? m_sashPosition 
: m_sashPositionCurrent
; 
 302         int posSashNew 
= OnSashPositionChanging(posSashOld 
+ diff
); 
 303         if ( posSashNew 
== -1 ) 
 305             // change not allowed 
 309         if ( posSashNew 
== m_sashPosition 
) 
 315             DrawSashTracker(m_oldX
, m_oldY
); 
 318         if (m_splitMode 
== wxSPLIT_VERTICAL
) 
 323         // Remember old positions 
 328         // As we captured the mouse, we may get the mouse events from outside 
 329         // our window - for example, negative values in x, y. This has a weird 
 330         // consequence under MSW where we use unsigned values sometimes and 
 331         // signed ones other times: the coordinates turn as big positive 
 332         // numbers and so the sash is drawn on the *right* side of the window 
 333         // instead of the left (or bottom instead of top). Correct this. 
 334         if ( (short)m_oldX 
< 0 ) 
 336         if ( (short)m_oldY 
< 0 ) 
 343             m_sashPositionCurrent 
= posSashNew
; 
 345             DrawSashTracker(m_oldX
, m_oldY
); 
 349             SetSashPositionAndNotify(posSashNew
); 
 350             m_needUpdating 
= TRUE
; 
 353     else if ( event
.LeftDClick() && m_windowTwo 
) 
 355         OnDoubleClickSash(x
, y
); 
 359 void wxSplitterWindow::OnSize(wxSizeEvent
& event
) 
 361     // only process this message if we're not iconized - otherwise iconizing 
 362     // and restoring a window containing the splitter has a funny side effect 
 363     // of changing the splitter position! 
 364     wxWindow 
*parent 
= GetParent(); 
 365     while ( parent 
&& !parent
->IsTopLevel() ) 
 367         parent 
= parent
->GetParent(); 
 370     bool iconized 
= FALSE
; 
 372     // wxMotif doesn't yet have a wxTopLevelWindow implementation 
 374     wxFrame 
*winTop 
= wxDynamicCast(parent
, wxFrame
); 
 376     wxTopLevelWindow 
*winTop 
= wxDynamicCast(parent
, wxTopLevelWindow
); 
 380         iconized 
= winTop
->IsIconized(); 
 385         wxFAIL_MSG(wxT("should have a top level parent!")); 
 399     GetClientSize( &cw
, &ch 
); 
 402         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 404             if ( m_sashPosition 
>= (cw 
- 5) ) 
 405                 SetSashPositionAndNotify(wxMax(10, cw 
- 40)); 
 407         else // m_splitMode == wxSPLIT_HORIZONTAL 
 409             if ( m_sashPosition 
>= (ch 
- 5) ) 
 410                 SetSashPositionAndNotify(wxMax(10, ch 
- 40)); 
 417 bool wxSplitterWindow::SashHitTest(int x
, int y
, int tolerance
) 
 419     if ( m_windowTwo 
== NULL 
|| m_sashPosition 
== 0) 
 420         return FALSE
; // No sash 
 422     if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 424         if ( (x 
>= m_sashPosition 
- tolerance
) && (x 
<= m_sashPosition 
+ m_sashSize 
+ tolerance
) ) 
 431         if ( (y 
>= (m_sashPosition
- tolerance
)) && (y 
<= (m_sashPosition 
+ m_sashSize 
+ tolerance
)) ) 
 438 // Draw 3D effect borders 
 439 void wxSplitterWindow::DrawBorders(wxDC
& dc
) 
 442     GetClientSize(&w
, &h
); 
 444     if ( GetWindowStyleFlag() & wxSP_3DBORDER 
) 
 447         dc
.SetPen(*m_facePen
); 
 448         dc
.SetBrush(*m_faceBrush
); 
 449         dc
.DrawRectangle(1, 1 , w
-1, m_borderSize
-2 ); //high 
 450         dc
.DrawRectangle(1, m_borderSize
-2 , m_borderSize
-2, h
-1 ); // left 
 451         dc
.DrawRectangle(w
-m_borderSize
+2, m_borderSize
-2 , w
-1, h
-1 ); // right 
 452         dc
.DrawRectangle(m_borderSize
-2, h
-m_borderSize
+2 , w
-m_borderSize
+2, h
-1 ); //bottom 
 454         dc
.SetPen(*m_mediumShadowPen
); 
 455         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, w
-m_borderSize
+1, m_borderSize
-2); 
 456         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, m_borderSize
-2, h
-m_borderSize
+1); 
 458         dc
.SetPen(*m_darkShadowPen
); 
 459         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, w
-m_borderSize
, m_borderSize
-1); 
 460         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, m_borderSize
-1, h
-m_borderSize
); 
 462         dc
.SetPen(*m_hilightPen
); 
 463         dc
.DrawLine(m_borderSize 
- 2, h
-m_borderSize
+1, w
-m_borderSize
+1, h
-m_borderSize
+1); 
 464         dc
.DrawLine(w
-m_borderSize
+1, m_borderSize 
- 2, w
-m_borderSize
+1, h
-m_borderSize
+2); // Surely the maximum y pos. should be h - 1. 
 465                                      /// Anyway, h is required for MSW. 
 467         dc
.SetPen(*m_lightShadowPen
); 
 468         dc
.DrawLine(w
-m_borderSize
, m_borderSize
-1, w
-m_borderSize
, h
-m_borderSize
); // Right hand side 
 469         dc
.DrawLine(m_borderSize
-1, h
-m_borderSize
, w
-m_borderSize
+1, h
-m_borderSize
);     // Bottom 
 471     else if ( GetWindowStyleFlag() & wxSP_BORDER 
) 
 473         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 474         dc
.SetPen(*wxBLACK_PEN
); 
 475         dc
.DrawRectangle(0, 0, w
-1, h
-1); 
 478     dc
.SetPen(wxNullPen
); 
 479     dc
.SetBrush(wxNullBrush
); 
 483 void wxSplitterWindow::DrawSash(wxDC
& dc
) 
 485     if ( m_sashPosition 
== 0 || !m_windowTwo
) 
 487     if (GetWindowStyle() & wxSP_NOSASH
) 
 491     GetClientSize(&w
, &h
); 
 493     if ( GetWindowStyleFlag() & wxSP_3DSASH 
) 
 495         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 497             dc
.SetPen(*m_facePen
); 
 499             if (HasFlag( wxSP_SASH_AQUA 
)) 
 500                 dc
.SetBrush(*wxWHITE_BRUSH
); 
 502                 dc
.SetBrush(*m_faceBrush
); 
 503             dc
.DrawRectangle(m_sashPosition 
+ 2, 0 , m_sashSize 
- 4, h 
); 
 505             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 507             dc
.SetPen(*m_lightShadowPen
); 
 508             int xShadow 
= m_borderSize 
? m_borderSize 
- 1 : 0 ; 
 509             dc
.DrawLine(m_sashPosition
, xShadow 
, m_sashPosition
, h
-m_borderSize
); 
 511             dc
.SetPen(*m_hilightPen
); 
 512             dc
.DrawLine(m_sashPosition
+1, m_borderSize 
- 2, m_sashPosition
+1, h 
- m_borderSize
+2); 
 514             if (!HasFlag( wxSP_SASH_AQUA 
)) 
 515                 dc
.SetPen(*m_mediumShadowPen
); 
 517             int yMedium 
= m_borderSize 
? h
-m_borderSize
+1 : h 
; 
 518             dc
.DrawLine(m_sashPosition
+m_sashSize
-2, xShadow
, m_sashPosition
+m_sashSize
-2, yMedium
); 
 520             if (HasFlag( wxSP_SASH_AQUA 
)) 
 521                 dc
.SetPen(*m_lightShadowPen
); 
 523                 dc
.SetPen(*m_darkShadowPen
); 
 524             dc
.DrawLine(m_sashPosition
+m_sashSize
-1, m_borderSize
, m_sashPosition
+m_sashSize
-1, h
-m_borderSize 
); 
 526             // Draw the top and bottom edges of the sash, if requested 
 527             if (GetWindowStyle() & wxSP_FULLSASH
) 
 530                 dc
.SetPen(*m_hilightPen
); 
 531                 dc
.DrawLine(m_sashPosition
+1, m_borderSize
, m_sashPosition
+m_sashSize
-1, m_borderSize
); 
 534                 dc
.SetPen(*m_darkShadowPen
); 
 535                 dc
.DrawLine(m_sashPosition
+1, h
-m_borderSize
-1, m_sashPosition
+m_sashSize
-1, h
-m_borderSize
-1); 
 538         else // wxSPLIT_HORIZONTAL 
 540             dc
.SetPen(*m_facePen
); 
 541             if (HasFlag( wxSP_SASH_AQUA 
)) 
 542                 dc
.SetBrush(*wxWHITE_BRUSH
); 
 544                 dc
.SetBrush(*m_faceBrush
); 
 545             dc
.DrawRectangle( m_borderSize
-2, m_sashPosition 
+ 2, w
-m_borderSize
+2, m_sashSize 
- 4); 
 547             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 549             dc
.SetPen(*m_lightShadowPen
); 
 550             dc
.DrawLine(m_borderSize
-1, m_sashPosition
, w
-m_borderSize
, m_sashPosition
); 
 552             dc
.SetPen(*m_hilightPen
); 
 553             dc
.DrawLine(m_borderSize
-2, m_sashPosition
+1, w
-m_borderSize
+1, m_sashPosition
+1); 
 555             if (!HasFlag( wxSP_SASH_AQUA 
)) 
 556                 dc
.SetPen(*m_mediumShadowPen
); 
 557             dc
.DrawLine(m_borderSize
-1, m_sashPosition
+m_sashSize
-2, w
-m_borderSize
+1, m_sashPosition
+m_sashSize
-2); 
 559             if (HasFlag( wxSP_SASH_AQUA 
)) 
 560                 dc
.SetPen(*m_lightShadowPen
); 
 562                 dc
.SetPen(*m_darkShadowPen
); 
 563             dc
.DrawLine(m_borderSize
, m_sashPosition
+m_sashSize
-1, w
-m_borderSize
, m_sashPosition
+m_sashSize
-1); 
 565             // Draw the left and right edges of the sash, if requested 
 566             if (GetWindowStyle() & wxSP_FULLSASH
) 
 569                 dc
.SetPen(*m_hilightPen
); 
 570                 dc
.DrawLine(m_borderSize
, m_sashPosition
, m_borderSize
, m_sashPosition
+m_sashSize
); 
 573                 dc
.SetPen(*m_darkShadowPen
); 
 574                 dc
.DrawLine(w
-m_borderSize
-1, m_sashPosition
+1, w
-m_borderSize
-1, m_sashPosition
+m_sashSize
-1); 
 580         dc
.SetPen(*wxTRANSPARENT_PEN
); 
 581         dc
.SetBrush(*m_faceBrush
); 
 582         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 586             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
&& (GetWindowStyleFlag() & wxSP_3DBORDER
) != wxSP_3DBORDER 
) 
 587                 h1 
+= 1; // Not sure why this is necessary... 
 588             if ( (GetWindowStyleFlag() & wxSP_3DBORDER
) == wxSP_3DBORDER
) 
 592             dc
.DrawRectangle(m_sashPosition
, y1
, m_sashSize
, h1
); 
 594         else // wxSPLIT_HORIZONTAL 
 598             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
&& (GetWindowStyleFlag() & wxSP_3DBORDER
) != wxSP_3DBORDER 
) 
 600             if ( (GetWindowStyleFlag() & wxSP_3DBORDER
) == wxSP_3DBORDER
) 
 604             dc
.DrawRectangle(x1
, m_sashPosition
, w1
, m_sashSize
); 
 608     dc
.SetPen(wxNullPen
); 
 609     dc
.SetBrush(wxNullBrush
); 
 612 // Draw the sash tracker (for whilst moving the sash) 
 613 void wxSplitterWindow::DrawSashTracker(int x
, int y
) 
 616     GetClientSize(&w
, &h
); 
 622     if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 653     ClientToScreen(&x1
, &y1
); 
 654     ClientToScreen(&x2
, &y2
); 
 656     screenDC
.SetLogicalFunction(wxINVERT
); 
 657     screenDC
.SetPen(*m_sashTrackerPen
); 
 658     screenDC
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 660     screenDC
.DrawLine(x1
, y1
, x2
, y2
); 
 662     screenDC
.SetLogicalFunction(wxCOPY
); 
 664     screenDC
.SetPen(wxNullPen
); 
 665     screenDC
.SetBrush(wxNullBrush
); 
 668 int wxSplitterWindow::GetWindowSize() const 
 670     wxSize size 
= GetClientSize(); 
 672     return m_splitMode 
== wxSPLIT_VERTICAL 
? size
.x 
: size
.y
; 
 675 int wxSplitterWindow::AdjustSashPosition(int sashPos
) const 
 677     int window_size 
= GetWindowSize(); 
 684         // the window shouldn't be smaller than its own minimal size nor 
 685         // smaller than the minimual pane size specified for this splitter 
 686         int minSize 
= m_splitMode 
== wxSPLIT_VERTICAL 
? win
->GetMinWidth() 
 687                                                       : win
->GetMinHeight(); 
 689         if ( minSize 
== -1 || m_minimumPaneSize 
> minSize 
) 
 690             minSize 
= m_minimumPaneSize
; 
 692         minSize 
+= GetBorderSize(); 
 694         if ( sashPos 
< minSize 
) 
 701         int minSize 
= m_splitMode 
== wxSPLIT_VERTICAL 
? win
->GetMinWidth() 
 702                                                       : win
->GetMinHeight(); 
 704         if ( minSize 
== -1 || m_minimumPaneSize 
> minSize 
) 
 705             minSize 
= m_minimumPaneSize
; 
 707         int maxSize 
= window_size 
- minSize 
- GetBorderSize(); 
 708         if ( sashPos 
> maxSize 
) 
 715 bool wxSplitterWindow::DoSetSashPosition(int sashPos
) 
 717     int newSashPosition 
= AdjustSashPosition(sashPos
); 
 719     if ( newSashPosition 
== m_sashPosition 
) 
 722     m_sashPosition 
= newSashPosition
; 
 727 void wxSplitterWindow::SetSashPositionAndNotify(int sashPos
) 
 729     if ( DoSetSashPosition(sashPos
) ) 
 731         wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
, this); 
 732         event
.m_data
.pos 
= m_sashPosition
; 
 734         (void)DoSendEvent(event
); 
 738 // Position and size subwindows. 
 739 // Note that the border size applies to each subwindow, not 
 740 // including the edges next to the sash. 
 741 void wxSplitterWindow::SizeWindows() 
 743     // check if we have delayed setting the real sash position 
 744     if ( m_requestedSashPosition 
!= INT_MAX 
) 
 746         int newSashPosition 
= ConvertSashPosition(m_requestedSashPosition
); 
 747         if ( newSashPosition 
!= m_sashPosition 
) 
 749             DoSetSashPosition(newSashPosition
); 
 752         if ( newSashPosition 
<= m_sashPosition
 
 753             && newSashPosition 
>= m_sashPosition 
- GetBorderSize() ) 
 755             // don't update it any more 
 756             m_requestedSashPosition 
= INT_MAX
; 
 761     GetClientSize(&w
, &h
); 
 763     if ( GetWindow1() && !GetWindow2() ) 
 765         GetWindow1()->SetSize(GetBorderSize(), GetBorderSize(), 
 766                               w 
- 2*GetBorderSize(), h 
- 2*GetBorderSize()); 
 768     else if ( GetWindow1() && GetWindow2() ) 
 770         if (GetSplitMode() == wxSPLIT_VERTICAL
) 
 772             int x1 
= GetBorderSize(); 
 773             int y1 
= GetBorderSize(); 
 774             int w1 
= GetSashPosition() - GetBorderSize(); 
 775             int h1 
= h 
- 2*GetBorderSize(); 
 777             int x2 
= GetSashPosition() + GetSashSize(); 
 778             int y2 
= GetBorderSize(); 
 779             int w2 
= w 
- 2*GetBorderSize() - GetSashSize() - w1
; 
 780             int h2 
= h 
- 2*GetBorderSize(); 
 782             GetWindow1()->SetSize(x1
, y1
, w1
, h1
); 
 783             GetWindow2()->SetSize(x2
, y2
, w2
, h2
); 
 787             GetWindow1()->SetSize(GetBorderSize(), GetBorderSize(), 
 788                 w 
- 2*GetBorderSize(), GetSashPosition() - GetBorderSize()); 
 789             GetWindow2()->SetSize(GetBorderSize(), GetSashPosition() + GetSashSize(), 
 790                 w 
- 2*GetBorderSize(), h 
- 2*GetBorderSize() - GetSashSize() - (GetSashPosition() - GetBorderSize())); 
 794     if ( GetBorderSize() > 0 ) 
 798     SetNeedUpdating(FALSE
); 
 801 // Set pane for unsplit window 
 802 void wxSplitterWindow::Initialize(wxWindow 
*window
) 
 804     wxASSERT_MSG( window 
&& window
->GetParent() == this, 
 805                   _T("windows in the splitter should have it as parent!") ); 
 807     m_windowOne 
= window
; 
 808     m_windowTwo 
= (wxWindow 
*) NULL
; 
 809     DoSetSashPosition(0); 
 812 // Associates the given window with window 2, drawing the appropriate sash 
 813 // and changing the split mode. 
 814 // Does nothing and returns FALSE if the window is already split. 
 815 bool wxSplitterWindow::DoSplit(wxSplitMode mode
, 
 816                                wxWindow 
*window1
, wxWindow 
*window2
, 
 822     wxCHECK_MSG( window1 
&& window2
, FALSE
, 
 823                  _T("can not split with NULL window(s)") ); 
 825     wxCHECK_MSG( window1
->GetParent() == this && window2
->GetParent() == this, FALSE
, 
 826                   _T("windows in the splitter should have it as parent!") ); 
 829     m_windowOne 
= window1
; 
 830     m_windowTwo 
= window2
; 
 832     // remember the sash position we want to set for later if we can't set it 
 833     // right now (e.g. because the window is too small) 
 834     m_requestedSashPosition 
= sashPosition
; 
 836     DoSetSashPosition(ConvertSashPosition(sashPosition
)); 
 843 int wxSplitterWindow::ConvertSashPosition(int sashPosition
) const 
 845     if ( sashPosition 
> 0 ) 
 849     else if ( sashPosition 
< 0 ) 
 851         // It's negative so adding is subtracting 
 852         return GetWindowSize() + sashPosition
; 
 854     else // sashPosition == 0 
 856         // default, put it in the centre 
 857         return GetWindowSize() / 2; 
 861 // Remove the specified (or second) window from the view 
 862 // Doesn't actually delete the window. 
 863 bool wxSplitterWindow::Unsplit(wxWindow 
*toRemove
) 
 868     wxWindow 
*win 
= NULL
; 
 869     if ( toRemove 
== NULL 
|| toRemove 
== m_windowTwo
) 
 872         m_windowTwo 
= (wxWindow 
*) NULL
; 
 874     else if ( toRemove 
== m_windowOne 
) 
 877         m_windowOne 
= m_windowTwo
; 
 878         m_windowTwo 
= (wxWindow 
*) NULL
; 
 882         wxFAIL_MSG(wxT("splitter: attempt to remove a non-existent window")); 
 888     DoSetSashPosition(0); 
 894 // Replace a window with another one 
 895 bool wxSplitterWindow::ReplaceWindow(wxWindow 
*winOld
, wxWindow 
*winNew
) 
 897     wxCHECK_MSG( winOld
, FALSE
, wxT("use one of Split() functions instead") ); 
 898     wxCHECK_MSG( winNew
, FALSE
, wxT("use Unsplit() functions instead") ); 
 900     if ( winOld 
== m_windowTwo 
) 
 902         m_windowTwo 
= winNew
; 
 904     else if ( winOld 
== m_windowOne 
) 
 906         m_windowOne 
= winNew
; 
 910         wxFAIL_MSG(wxT("splitter: attempt to replace a non-existent window")); 
 920 void wxSplitterWindow::SetMinimumPaneSize(int min
) 
 922     m_minimumPaneSize 
= min
; 
 923     SetSashPosition(m_sashPosition
); // re-check limits 
 926 void wxSplitterWindow::SetSashPosition(int position
, bool redraw
) 
 928     DoSetSashPosition(position
); 
 936 // Initialize colours 
 937 void wxSplitterWindow::InitColours() 
 939     wxDELETE( m_facePen 
); 
 940     wxDELETE( m_faceBrush 
); 
 941     wxDELETE( m_mediumShadowPen 
); 
 942     wxDELETE( m_darkShadowPen 
); 
 943     wxDELETE( m_lightShadowPen 
); 
 944     wxDELETE( m_hilightPen 
); 
 948     wxColour 
faceColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
)); 
 949     m_facePen 
= new wxPen(faceColour
, 1, wxSOLID
); 
 950     m_faceBrush 
= new wxBrush(faceColour
, wxSOLID
); 
 952     wxColour 
mediumShadowColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW
)); 
 953     m_mediumShadowPen 
= new wxPen(mediumShadowColour
, 1, wxSOLID
); 
 955     wxColour 
darkShadowColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
)); 
 956     m_darkShadowPen 
= new wxPen(darkShadowColour
, 1, wxSOLID
); 
 958     wxColour 
lightShadowColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DLIGHT
)); 
 959     m_lightShadowPen 
= new wxPen(lightShadowColour
, 1, wxSOLID
); 
 961     wxColour 
hilightColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DHILIGHT
)); 
 962     m_hilightPen 
= new wxPen(hilightColour
, 1, wxSOLID
); 
 964     m_facePen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 965     m_faceBrush 
= new wxBrush("LIGHT GREY", wxSOLID
); 
 966     m_mediumShadowPen 
= new wxPen("GREY", 1, wxSOLID
); 
 967     m_darkShadowPen 
= new wxPen("BLACK", 1, wxSOLID
); 
 968     m_lightShadowPen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 969     m_hilightPen 
= new wxPen("WHITE", 1, wxSOLID
); 
 973 bool wxSplitterWindow::DoSendEvent(wxSplitterEvent
& event
) 
 975     return !GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed(); 
 978 // --------------------------------------------------------------------------- 
 979 // wxSplitterWindow virtual functions: they now just generate the events 
 980 // --------------------------------------------------------------------------- 
 982 bool wxSplitterWindow::OnSashPositionChange(int WXUNUSED(newSashPosition
)) 
 984     // always allow by default 
 988 int wxSplitterWindow::OnSashPositionChanging(int newSashPosition
) 
 990     // If within UNSPLIT_THRESHOLD from edge, set to edge to cause closure. 
 991     const int UNSPLIT_THRESHOLD 
= 4; 
 993     // first of all, check if OnSashPositionChange() doesn't forbid this change 
 994     if ( !OnSashPositionChange(newSashPosition
) ) 
1000     // Obtain relevant window dimension for bottom / right threshold check 
1001     int window_size 
= GetWindowSize(); 
1003     bool unsplit_scenario 
= FALSE
; 
1004     if ( m_permitUnsplitAlways 
|| m_minimumPaneSize 
== 0 ) 
1006         // Do edge detection if unsplit premitted 
1007         if ( newSashPosition 
<= UNSPLIT_THRESHOLD 
) 
1009             // threshold top / left check 
1010             newSashPosition 
= 0; 
1011             unsplit_scenario 
= TRUE
; 
1013         if ( newSashPosition 
>= window_size 
- UNSPLIT_THRESHOLD 
) 
1015             // threshold bottom/right check 
1016             newSashPosition 
= window_size
; 
1017             unsplit_scenario 
= TRUE
; 
1021     if ( !unsplit_scenario 
) 
1023         // If resultant pane would be too small, enlarge it 
1024         newSashPosition 
= AdjustSashPosition(newSashPosition
); 
1027     // If the result is out of bounds it means minimum size is too big, 
1028     // so split window in half as best compromise. 
1029     if ( newSashPosition 
< 0 || newSashPosition 
> window_size 
) 
1030         newSashPosition 
= window_size 
/ 2; 
1032     // now let the event handler have it 
1034     // FIXME: shouldn't we do it before the adjustments above so as to ensure 
1035     //        that the sash position is always reasonable? 
1036     wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
, this); 
1037     event
.m_data
.pos 
= newSashPosition
; 
1039     if ( !DoSendEvent(event
) ) 
1041         // the event handler vetoed the change 
1042         newSashPosition 
= -1; 
1046         // it could have been changed by it 
1047         newSashPosition 
= event
.GetSashPosition(); 
1050     return newSashPosition
; 
1053 // Called when the sash is double-clicked. The default behaviour is to remove 
1054 // the sash if the minimum pane size is zero. 
1055 void wxSplitterWindow::OnDoubleClickSash(int x
, int y
) 
1057     wxCHECK_RET(m_windowTwo
, wxT("splitter: no window to remove")); 
1059     // new code should handle events instead of using the virtual functions 
1060     wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_DOUBLECLICKED
, this); 
1061     event
.m_data
.pt
.x 
= x
; 
1062     event
.m_data
.pt
.y 
= y
; 
1063     if ( DoSendEvent(event
) ) 
1065         if ( GetMinimumPaneSize() == 0 || m_permitUnsplitAlways 
) 
1067             wxWindow
* win 
= m_windowTwo
; 
1070                 wxSplitterEvent 
unsplitEvent(wxEVT_COMMAND_SPLITTER_UNSPLIT
, this); 
1071                 unsplitEvent
.m_data
.win 
= win
; 
1072                 (void)DoSendEvent(unsplitEvent
); 
1076     //else: blocked by user 
1079 void wxSplitterWindow::OnUnsplit(wxWindow 
*winRemoved
) 
1081     // call this before calling the event handler which may delete the window 
1082     winRemoved
->Show(FALSE
); 
1085 #if defined( __WXMSW__ ) || defined( __WXMAC__) 
1087 // this is currently called (and needed) under MSW only... 
1088 void wxSplitterWindow::OnSetCursor(wxSetCursorEvent
& event
) 
1090     // if we don't do it, the resizing cursor might be set for child window: 
1091     // and like this we explicitly say that our cursor should not be used for 
1092     // children windows which overlap us 
1094     if ( SashHitTest(event
.GetX(), event
.GetY()) ) 
1096         // default processing is ok 
1099     //else: do nothing, in particular, don't call Skip() 
1102 #endif // wxUSE_SPLITTER