1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxSplitterWindow implementation 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13     #pragma implementation "splitter.h" 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  24     #include "wx/window.h" 
  25     #include "wx/dialog.h" 
  31 #include "wx/string.h" 
  32 #include "wx/splitter.h" 
  33 #include "wx/dcscreen.h" 
  34 #include "wx/settings.h" 
  36 #if !USE_SHARED_LIBRARY 
  37 IMPLEMENT_DYNAMIC_CLASS(wxSplitterWindow
, wxWindow
) 
  38 IMPLEMENT_DYNAMIC_CLASS(wxSplitterEvent
, wxCommandEvent
) 
  40 BEGIN_EVENT_TABLE(wxSplitterWindow
, wxWindow
) 
  41     EVT_PAINT(wxSplitterWindow::OnPaint
) 
  42     EVT_SIZE(wxSplitterWindow::OnSize
) 
  43     EVT_IDLE(wxSplitterWindow::OnIdle
) 
  44     EVT_MOUSE_EVENTS(wxSplitterWindow::OnMouseEvent
) 
  46     EVT_SPLITTER_SASH_POS_CHANGED(-1, wxSplitterWindow::OnSashPosChanged
) 
  47     // NB: we borrow OnSashPosChanged for purposes of 
  48     // EVT_SPLITTER_SASH_POS_CHANGING since default implementation is identical 
  49     EVT_SPLITTER_SASH_POS_CHANGING(-1, wxSplitterWindow::OnSashPosChanged
) 
  50     EVT_SPLITTER_DCLICK(-1,           wxSplitterWindow::OnDoubleClick
) 
  51     EVT_SPLITTER_UNSPLIT(-1,          wxSplitterWindow::OnUnsplitEvent
) 
  55 wxSplitterWindow::wxSplitterWindow() 
  57     m_splitMode 
= wxSPLIT_VERTICAL
; 
  58     m_permitUnsplitAlways 
= FALSE
; 
  59     m_windowOne 
= (wxWindow 
*) NULL
; 
  60     m_windowTwo 
= (wxWindow 
*) NULL
; 
  61     m_dragMode 
= wxSPLIT_DRAG_NONE
; 
  69     m_sashCursorWE 
= (wxCursor 
*) NULL
; 
  70     m_sashCursorNS 
= (wxCursor 
*) NULL
; 
  71     m_sashTrackerPen 
= (wxPen 
*) NULL
; 
  72     m_lightShadowPen 
= (wxPen 
*) NULL
; 
  73     m_mediumShadowPen 
= (wxPen 
*) NULL
; 
  74     m_darkShadowPen 
= (wxPen 
*) NULL
; 
  75     m_faceBrush 
= (wxBrush 
*) NULL
; 
  76     m_facePen 
= (wxPen 
*) NULL
; 
  77     m_hilightPen 
= (wxPen 
*) NULL
; 
  78     m_minimumPaneSize 
= 0; 
  79     m_needUpdating 
= FALSE
; 
  82 wxSplitterWindow::wxSplitterWindow(wxWindow 
*parent
, wxWindowID id
, 
  87                 : wxWindow(parent
, id
, pos
, size
, style
, name
) 
  89     m_splitMode 
= wxSPLIT_VERTICAL
; 
  90     m_permitUnsplitAlways 
= (style 
& wxSP_PERMIT_UNSPLIT
) != 0; 
  91     m_windowOne 
= (wxWindow 
*) NULL
; 
  92     m_windowTwo 
= (wxWindow 
*) NULL
; 
  93     m_dragMode 
= wxSPLIT_DRAG_NONE
; 
 101     m_minimumPaneSize 
= 0; 
 102     m_sashCursorWE 
= new wxCursor(wxCURSOR_SIZEWE
); 
 103     m_sashCursorNS 
= new wxCursor(wxCURSOR_SIZENS
); 
 104     m_sashTrackerPen 
= new wxPen(*wxBLACK
, 2, wxSOLID
); 
 105     m_lightShadowPen 
= (wxPen 
*) NULL
; 
 106     m_mediumShadowPen 
= (wxPen 
*) NULL
; 
 107     m_darkShadowPen 
= (wxPen 
*) NULL
; 
 108     m_faceBrush 
= (wxBrush 
*) NULL
; 
 109     m_facePen 
= (wxPen 
*) NULL
; 
 110     m_hilightPen 
= (wxPen 
*) NULL
; 
 112     if ( style 
& wxSP_3D 
) 
 117     else if ( style 
& wxSP_BORDER 
) 
 128     // Eventually, we'll respond to colour change messages 
 131     // For debugging purposes, to see the background. 
 132 //    SetBackground(wxBLUE_BRUSH); 
 134     m_needUpdating 
= FALSE
; 
 137 wxSplitterWindow::~wxSplitterWindow() 
 139     delete m_sashCursorWE
; 
 140     delete m_sashCursorNS
; 
 141     delete m_sashTrackerPen
; 
 142     delete m_lightShadowPen
; 
 143     delete m_darkShadowPen
; 
 144     delete m_mediumShadowPen
; 
 150 void wxSplitterWindow::OnPaint(wxPaintEvent
& WXUNUSED(event
)) 
 154     if ( m_borderSize 
> 0 ) 
 159 void wxSplitterWindow::OnIdle(wxIdleEvent
& WXUNUSED(event
)) 
 165 void wxSplitterWindow::OnMouseEvent(wxMouseEvent
& event
) 
 167     long x 
= event
.GetX(); 
 168     long y 
= event
.GetY(); 
 172     SetCursor(* wxSTANDARD_CURSOR
); 
 175     SetCursor(wxCursor()); 
 179     if (event
.LeftDown()) 
 181         if ( SashHitTest(x
, y
) ) 
 185             m_dragMode 
= wxSPLIT_DRAG_DRAGGING
; 
 187             if ((GetWindowStyleFlag() & wxSP_LIVE_UPDATE
) == 0) 
 189                 DrawSashTracker(x
, y
); 
 197     else if (event
.LeftUp() && m_dragMode 
== wxSPLIT_DRAG_DRAGGING
) 
 199         // We can stop dragging now and see what we've got. 
 200         m_dragMode 
= wxSPLIT_DRAG_NONE
; 
 204         if ((GetWindowStyleFlag() & wxSP_LIVE_UPDATE
) == 0) 
 206             DrawSashTracker(m_oldX
, m_oldY
); 
 209         // Obtain window size. We are only interested in the dimension the sash 
 212         GetClientSize(&w
, &h
); 
 213         int window_size 
= (m_splitMode 
== wxSPLIT_VERTICAL 
? w 
: h 
); 
 214         int new_sash_position 
= 
 215             (int) ( m_splitMode 
== wxSPLIT_VERTICAL 
? x 
: y 
); 
 217         wxSplitterEvent 
eventSplitter(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
, 
 219         eventSplitter
.m_data
.pos 
= new_sash_position
; 
 220         if ( GetEventHandler()->ProcessEvent(eventSplitter
) ) 
 222             new_sash_position 
= eventSplitter
.GetSashPosition(); 
 223             if ( new_sash_position 
== -1 ) 
 225                 // change not allowed 
 230         if ( m_permitUnsplitAlways
 
 231                 || m_minimumPaneSize 
== 0 ) 
 233             // Deal with possible unsplit scenarios 
 234             if ( new_sash_position 
== 0 ) 
 236                 // We remove the first window from the view 
 237                 wxWindow 
*removedWindow 
= m_windowOne
; 
 238                 m_windowOne 
= m_windowTwo
; 
 239                 m_windowTwo 
= (wxWindow 
*) NULL
; 
 240                 SendUnsplitEvent(removedWindow
); 
 243             else if ( new_sash_position 
== window_size 
) 
 245                 // We remove the second window from the view 
 246                 wxWindow 
*removedWindow 
= m_windowTwo
; 
 247                 m_windowTwo 
= (wxWindow 
*) NULL
; 
 248                 SendUnsplitEvent(removedWindow
); 
 253                 m_sashPosition 
= new_sash_position
; 
 258             m_sashPosition 
= new_sash_position
; 
 262     }  // left up && dragging 
 263     else if (event
.Moving() && !event
.Dragging()) 
 265         // Just change the cursor if required 
 266         if ( SashHitTest(x
, y
) ) 
 268                 if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 270                     SetCursor(*m_sashCursorWE
); 
 274                     SetCursor(*m_sashCursorNS
); 
 280             // where else do we unset the cursor? 
 281             SetCursor(* wxSTANDARD_CURSOR
); 
 285     else if (event
.Dragging() && (m_dragMode 
== wxSPLIT_DRAG_DRAGGING
)) 
 287         // Obtain window size. We are only interested in the dimension the sash 
 289         int new_sash_position 
= 
 290             (int) ( m_splitMode 
== wxSPLIT_VERTICAL 
? x 
: y 
); 
 292         wxSplitterEvent 
eventSplitter(wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
, 
 294         eventSplitter
.m_data
.pos 
= new_sash_position
; 
 295         if ( GetEventHandler()->ProcessEvent(eventSplitter
) ) 
 297             new_sash_position 
= eventSplitter
.GetSashPosition(); 
 298             if ( new_sash_position 
== -1 ) 
 300                 // change not allowed 
 306         if ((GetWindowStyleFlag() & wxSP_LIVE_UPDATE
) == 0) 
 308             DrawSashTracker(m_oldX
, m_oldY
); 
 311         if (m_splitMode 
== wxSPLIT_VERTICAL
) 
 312             x 
= new_sash_position
; 
 314             y 
= new_sash_position
; 
 316         if (new_sash_position 
!= -1) 
 318             // Only modify if permitted 
 324         // As we captured the mouse, we may get the mouse events from outside 
 325         // our window - for example, negative values in x, y. This has a weird 
 326         // consequence under MSW where we use unsigned values sometimes and 
 327         // signed ones other times: the coordinates turn as big positive 
 328         // numbers and so the sash is drawn on the *right* side of the window 
 329         // instead of the left (or bottom instead of top). Correct this. 
 330         if ( (short)m_oldX 
< 0 ) 
 332         if ( (short)m_oldY 
< 0 ) 
 337         if ((GetWindowStyleFlag() & wxSP_LIVE_UPDATE
) == 0) 
 339             DrawSashTracker(m_oldX
, m_oldY
); 
 343             m_sashPosition 
= new_sash_position
; 
 344             m_needUpdating 
= TRUE
; 
 347     else if ( event
.LeftDClick() ) 
 349         wxSplitterEvent 
eventSplitter(wxEVT_COMMAND_SPLITTER_DOUBLECLICKED
, 
 351         eventSplitter
.m_data
.pt
.x 
= x
; 
 352         eventSplitter
.m_data
.pt
.y 
= y
; 
 354         (void)GetEventHandler()->ProcessEvent(eventSplitter
); 
 358 void wxSplitterWindow::OnSize(wxSizeEvent
& event
) 
 360     // only process this message if we're not iconized - otherwise iconizing 
 361     // and restoring a window containing the splitter has a funny side effect 
 362     // of changing the splitter position! 
 363     wxWindow 
*parent 
= GetParent(); 
 364     while ( parent 
&& !parent
->IsTopLevel() ) 
 366         parent 
= parent
->GetParent(); 
 369     bool iconized 
= FALSE
; 
 370     wxFrame 
*frame 
= wxDynamicCast(parent
, wxFrame
); 
 372         iconized 
= frame
->IsIconized(); 
 375         wxDialog 
*dialog 
= wxDynamicCast(parent
, wxDialog
); 
 377             iconized 
= dialog
->IsIconized(); 
 379             wxFAIL_MSG(wxT("should have a top level frame or dialog parent!")); 
 389         GetClientSize( &cw
, &ch 
); 
 392             if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 394                 if ( m_sashPosition 
>= (cw 
- 5) ) 
 395                     m_sashPosition 
= wxMax(10, cw 
- 40); 
 397             if ( m_splitMode 
== wxSPLIT_HORIZONTAL 
) 
 399                 if ( m_sashPosition 
>= (ch 
- 5) ) 
 400                     m_sashPosition 
= wxMax(10, ch 
- 40); 
 408 bool wxSplitterWindow::SashHitTest(int x
, int y
, int tolerance
) 
 410     if ( m_windowTwo 
== NULL 
|| m_sashPosition 
== 0) 
 411         return FALSE
; // No sash 
 413     if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 415         if ( (x 
>= m_sashPosition 
- tolerance
) && (x 
<= m_sashPosition 
+ m_sashSize 
+ tolerance
) ) 
 422         if ( (y 
>= (m_sashPosition
- tolerance
)) && (y 
<= (m_sashPosition 
+ m_sashSize 
+ tolerance
)) ) 
 431 // Draw 3D effect borders 
 432 void wxSplitterWindow::DrawBorders(wxDC
& dc
) 
 435     GetClientSize(&w
, &h
); 
 437     if ( GetWindowStyleFlag() & wxSP_3D 
) 
 440         dc
.SetPen(*m_facePen
); 
 441         dc
.SetBrush(*m_faceBrush
); 
 442         dc
.DrawRectangle(1, 1 , w
-1, m_borderSize
-2 ); //high 
 443         dc
.DrawRectangle(1, m_borderSize
-2 , m_borderSize
-2, h
-1 ); // left 
 444         dc
.DrawRectangle(w
-m_borderSize
+2, m_borderSize
-2 , w
-1, h
-1 ); // right 
 445         dc
.DrawRectangle(m_borderSize
-2, h
-m_borderSize
+2 , w
-m_borderSize
+2, h
-1 ); //bottom 
 447         dc
.SetPen(*m_mediumShadowPen
); 
 448         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, w
-m_borderSize
+1, m_borderSize
-2); 
 449         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, m_borderSize
-2, h
-m_borderSize
+1); 
 451         dc
.SetPen(*m_darkShadowPen
); 
 452         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, w
-m_borderSize
, m_borderSize
-1); 
 453         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, m_borderSize
-1, h
-m_borderSize
); 
 455         dc
.SetPen(*m_hilightPen
); 
 456         dc
.DrawLine(m_borderSize 
- 2, h
-m_borderSize
+1, w
-m_borderSize
+1, h
-m_borderSize
+1); 
 457         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. 
 458                                      /// Anyway, h is required for MSW. 
 460         dc
.SetPen(*m_lightShadowPen
); 
 461         dc
.DrawLine(w
-m_borderSize
, m_borderSize
-1, w
-m_borderSize
, h
-m_borderSize
); // Right hand side 
 462         dc
.DrawLine(m_borderSize
-1, h
-m_borderSize
, w
-m_borderSize
+1, h
-m_borderSize
);     // Bottom 
 464     else if ( GetWindowStyleFlag() & wxSP_BORDER 
) 
 466         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 467         dc
.SetPen(*wxBLACK_PEN
); 
 468         dc
.DrawRectangle(0, 0, w
-1, h
-1); 
 471     dc
.SetPen(wxNullPen
); 
 472     dc
.SetBrush(wxNullBrush
); 
 476 void wxSplitterWindow::DrawSash(wxDC
& dc
) 
 478     if ( m_sashPosition 
== 0 || !m_windowTwo
) 
 482     GetClientSize(&w
, &h
); 
 484     if ( GetWindowStyleFlag() & wxSP_3D 
) 
 486         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 488             dc
.SetPen(*m_facePen
); 
 489             dc
.SetBrush(*m_faceBrush
); 
 490             dc
.DrawRectangle(m_sashPosition 
+ 2, 0 , m_sashSize 
- 4, h 
); 
 492             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 494             dc
.SetPen(*m_lightShadowPen
); 
 495                         int xShadow 
= m_borderSize 
? m_borderSize 
- 1 : 0 ; 
 496             dc
.DrawLine(m_sashPosition
, xShadow 
, m_sashPosition
, h
-m_borderSize
); 
 498             dc
.SetPen(*m_hilightPen
); 
 499             dc
.DrawLine(m_sashPosition
+1, m_borderSize 
- 2, m_sashPosition
+1, h 
- m_borderSize
+2); 
 501             dc
.SetPen(*m_mediumShadowPen
); 
 502                         int yMedium 
= m_borderSize 
? h
-m_borderSize
+1 : h 
; 
 503             dc
.DrawLine(m_sashPosition
+m_sashSize
-2, xShadow
, m_sashPosition
+m_sashSize
-2, yMedium
); 
 505             dc
.SetPen(*m_darkShadowPen
); 
 506             dc
.DrawLine(m_sashPosition
+m_sashSize
-1, m_borderSize
, m_sashPosition
+m_sashSize
-1, h
-m_borderSize 
); 
 510             dc
.SetPen(*m_facePen
); 
 511             dc
.SetBrush(*m_faceBrush
); 
 512             dc
.DrawRectangle( m_borderSize
-2, m_sashPosition 
+ 2, w
-m_borderSize
+2, m_sashSize 
- 4); 
 514             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 516             dc
.SetPen(*m_lightShadowPen
); 
 517             dc
.DrawLine(m_borderSize
-1, m_sashPosition
, w
-m_borderSize
, m_sashPosition
); 
 519             dc
.SetPen(*m_hilightPen
); 
 520             dc
.DrawLine(m_borderSize
-2, m_sashPosition
+1, w
-m_borderSize
+1, m_sashPosition
+1); 
 522             dc
.SetPen(*m_mediumShadowPen
); 
 523             dc
.DrawLine(m_borderSize
-1, m_sashPosition
+m_sashSize
-2, w
-m_borderSize
+1, m_sashPosition
+m_sashSize
-2); 
 525             dc
.SetPen(*m_darkShadowPen
); 
 526             dc
.DrawLine(m_borderSize
, m_sashPosition
+m_sashSize
-1, w
-m_borderSize
, m_sashPosition
+m_sashSize
-1); 
 531         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 533             dc
.SetPen(*wxBLACK_PEN
); 
 534             dc
.SetBrush(*wxBLACK_BRUSH
); 
 536             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
) 
 537                 h1 
+= 1; // Not sure why this is necessary... 
 538             dc
.DrawRectangle(m_sashPosition
, 0, m_sashSize
, h1
); 
 542             dc
.SetPen(*wxBLACK_PEN
); 
 543             dc
.SetBrush(*wxBLACK_BRUSH
); 
 545             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
) 
 548             dc
.DrawRectangle(0, m_sashPosition
, w1
, m_sashSize
); 
 553     dc
.SetPen(wxNullPen
); 
 554     dc
.SetBrush(wxNullBrush
); 
 557 // Draw the sash tracker (for whilst moving the sash) 
 558 void wxSplitterWindow::DrawSashTracker(int x
, int y
) 
 561     GetClientSize(&w
, &h
); 
 567     if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 598     ClientToScreen(&x1
, &y1
); 
 599     ClientToScreen(&x2
, &y2
); 
 601     screenDC
.SetLogicalFunction(wxINVERT
); 
 602     screenDC
.SetPen(*m_sashTrackerPen
); 
 603     screenDC
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 605     screenDC
.DrawLine(x1
, y1
, x2
, y2
); 
 607     screenDC
.SetLogicalFunction(wxCOPY
); 
 609     screenDC
.SetPen(wxNullPen
); 
 610     screenDC
.SetBrush(wxNullBrush
); 
 613 // Position and size subwindows. 
 614 // Note that the border size applies to each subwindow, not 
 615 // including the edges next to the sash. 
 616 void wxSplitterWindow::SizeWindows() 
 619     GetClientSize(&w
, &h
); 
 621     if ( m_windowOne 
&& !m_windowTwo 
) 
 623         m_windowOne
->SetSize(m_borderSize
, m_borderSize
, w 
- 2*m_borderSize
, h 
- 2*m_borderSize
); 
 626     else if ( m_windowOne 
&& m_windowTwo 
) 
 628         if (m_splitMode 
== wxSPLIT_VERTICAL
) 
 630             int x1 
= m_borderSize
; 
 631             int y1 
= m_borderSize
; 
 632             int w1 
= m_sashPosition 
- m_borderSize
; 
 633             int h1 
= h 
- 2*m_borderSize
; 
 635             int x2 
= m_sashPosition 
+ m_sashSize
; 
 636             int y2 
= m_borderSize
; 
 637             int w2 
= w 
- 2*m_borderSize 
- m_sashSize 
- w1
; 
 638             int h2 
= h 
- 2*m_borderSize
; 
 640             m_windowOne
->SetSize(x1
, y1
, w1
, h1
); 
 641             m_windowTwo
->SetSize(x2
, y2
, w2
, h2
); 
 646             m_windowOne
->SetSize(m_borderSize
, m_borderSize
, 
 647                 w 
- 2*m_borderSize
, m_sashPosition 
- m_borderSize
); 
 648             m_windowTwo
->SetSize(m_borderSize
, m_sashPosition 
+ m_sashSize
, 
 649                 w 
- 2*m_borderSize
, h 
- 2*m_borderSize 
- m_sashSize 
- (m_sashPosition 
- m_borderSize
)); 
 654         if ( m_borderSize 
> 0 ) 
 659 // Set pane for unsplit window 
 660 void wxSplitterWindow::Initialize(wxWindow 
*window
) 
 662     m_windowOne 
= window
; 
 663     m_windowTwo 
= (wxWindow 
*) NULL
; 
 667 // Associates the given window with window 2, drawing the appropriate sash 
 668 // and changing the split mode. 
 669 // Does nothing and returns FALSE if the window is already split. 
 670 bool wxSplitterWindow::SplitVertically(wxWindow 
*window1
, wxWindow 
*window2
, int sashPosition
) 
 676     GetClientSize(&w
, &h
); 
 678     m_splitMode 
= wxSPLIT_VERTICAL
; 
 679     m_windowOne 
= window1
; 
 680     m_windowTwo 
= window2
; 
 681     if ( sashPosition 
> 0 ) 
 682         m_sashPosition 
= sashPosition
; 
 683     else if ( sashPosition 
< 0 ) 
 684         m_sashPosition 
= w 
- sashPosition
; 
 686         m_sashPosition 
= w
/2; 
 693 bool wxSplitterWindow::SplitHorizontally(wxWindow 
*window1
, wxWindow 
*window2
, int sashPosition
) 
 699     GetClientSize(&w
, &h
); 
 701     m_splitMode 
= wxSPLIT_HORIZONTAL
; 
 702     m_windowOne 
= window1
; 
 703     m_windowTwo 
= window2
; 
 704     if ( sashPosition 
> 0 ) 
 705         m_sashPosition 
= sashPosition
; 
 706     else if ( sashPosition 
< 0 ) 
 707         m_sashPosition 
= h 
- sashPosition
; 
 709         m_sashPosition 
= h
/2; 
 717 // Remove the specified (or second) window from the view 
 718 // Doesn't actually delete the window. 
 719 bool wxSplitterWindow::Unsplit(wxWindow 
*toRemove
) 
 724     wxWindow 
*win 
= NULL
; 
 725     if ( toRemove 
== NULL 
|| toRemove 
== m_windowTwo
) 
 728         m_windowTwo 
= (wxWindow 
*) NULL
; 
 730     else if ( toRemove 
== m_windowOne 
) 
 733         m_windowOne 
= m_windowTwo
; 
 734         m_windowTwo 
= (wxWindow 
*) NULL
; 
 738         wxFAIL_MSG(wxT("splitter: attempt to remove a non-existent window")); 
 743     SendUnsplitEvent(win
); 
 750 // Replace a window with another one 
 751 bool wxSplitterWindow::ReplaceWindow(wxWindow 
*winOld
, wxWindow 
*winNew
) 
 753     wxCHECK_MSG( winOld
, FALSE
, wxT("use one of Split() functions instead") ); 
 754     wxCHECK_MSG( winNew
, FALSE
, wxT("use Unsplit() functions instead") ); 
 756     if ( winOld 
== m_windowTwo 
) 
 758         m_windowTwo 
= winNew
; 
 760     else if ( winOld 
== m_windowOne 
) 
 762         m_windowOne 
= winNew
; 
 766         wxFAIL_MSG(wxT("splitter: attempt to replace a non-existent window")); 
 776 void wxSplitterWindow::SetSashPosition(int position
, bool redraw
) 
 778     m_sashPosition 
= position
; 
 786 // Initialize colours 
 787 void wxSplitterWindow::InitColours() 
 789     wxDELETE( m_facePen 
); 
 790     wxDELETE( m_faceBrush 
); 
 791     wxDELETE( m_mediumShadowPen 
); 
 792     wxDELETE( m_darkShadowPen 
); 
 793     wxDELETE( m_lightShadowPen 
); 
 794     wxDELETE( m_hilightPen 
); 
 797 #if defined(__WIN95__) 
 798     wxColour 
faceColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DFACE
)); 
 799     m_facePen 
= new wxPen(faceColour
, 1, wxSOLID
); 
 800     m_faceBrush 
= new wxBrush(faceColour
, wxSOLID
); 
 802     wxColour 
mediumShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DSHADOW
)); 
 803     m_mediumShadowPen 
= new wxPen(mediumShadowColour
, 1, wxSOLID
); 
 805     wxColour 
darkShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DDKSHADOW
)); 
 806     m_darkShadowPen 
= new wxPen(darkShadowColour
, 1, wxSOLID
); 
 808     wxColour 
lightShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DLIGHT
)); 
 809     m_lightShadowPen 
= new wxPen(lightShadowColour
, 1, wxSOLID
); 
 811     wxColour 
hilightColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DHILIGHT
)); 
 812     m_hilightPen 
= new wxPen(hilightColour
, 1, wxSOLID
); 
 814     m_facePen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 815     m_faceBrush 
= new wxBrush("LIGHT GREY", wxSOLID
); 
 816     m_mediumShadowPen 
= new wxPen("GREY", 1, wxSOLID
); 
 817     m_darkShadowPen 
= new wxPen("BLACK", 1, wxSOLID
); 
 818     m_lightShadowPen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 819     m_hilightPen 
= new wxPen("WHITE", 1, wxSOLID
); 
 820 #endif // Win32/!Win32 
 823 void wxSplitterWindow::SendUnsplitEvent(wxWindow 
*winRemoved
) 
 825     wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_UNSPLIT
, this); 
 826     event
.m_data
.win 
= winRemoved
; 
 828     (void)GetEventHandler()->ProcessEvent(event
); 
 831 // --------------------------------------------------------------------------- 
 832 // splitter event handlers 
 833 // --------------------------------------------------------------------------- 
 835 void wxSplitterWindow::OnSashPosChanged(wxSplitterEvent
& event
) 
 837     // If within UNSPLIT_THRESHOLD from edge, set to edge to cause closure. 
 838     const int UNSPLIT_THRESHOLD 
= 4; 
 840     int newSashPosition 
= event
.GetSashPosition(); 
 842     // Obtain relevant window dimension for bottom / right threshold check 
 844     GetClientSize(&w
, &h
); 
 845     int window_size 
= (m_splitMode 
== wxSPLIT_VERTICAL
) ? w 
: h 
; 
 847     bool unsplit_scenario 
= FALSE
; 
 848     if ( m_permitUnsplitAlways
 
 849             || m_minimumPaneSize 
== 0 ) 
 851         // Do edge detection if unsplit premitted 
 852         if ( newSashPosition 
<= UNSPLIT_THRESHOLD 
) 
 854             // threshold top / left check 
 856             unsplit_scenario 
= TRUE
; 
 858         if ( newSashPosition 
>= window_size 
- UNSPLIT_THRESHOLD 
) 
 860             // threshold bottom/right check 
 861             newSashPosition 
= window_size
; 
 862             unsplit_scenario 
= TRUE
; 
 866     if ( !unsplit_scenario 
) 
 868         // If resultant pane would be too small, enlarge it 
 869         if ( newSashPosition 
< m_minimumPaneSize 
) 
 870             newSashPosition 
= m_minimumPaneSize
; 
 871         if ( newSashPosition 
> window_size 
- m_minimumPaneSize 
) 
 872             newSashPosition 
= window_size 
- m_minimumPaneSize
; 
 875     // If the result is out of bounds it means minimum size is too big, 
 876     // so split window in half as best compromise. 
 877     if ( newSashPosition 
< 0 || newSashPosition 
> window_size 
) 
 878         newSashPosition 
= window_size 
/ 2; 
 880     // for compatibility, call the virtual function 
 881     if ( !OnSashPositionChange(newSashPosition
) ) 
 883         newSashPosition 
= -1; 
 886     event
.SetSashPosition(newSashPosition
); 
 889 // Called when the sash is double-clicked. The default behaviour is to remove 
 890 // the sash if the minimum pane size is zero. 
 891 void wxSplitterWindow::OnDoubleClick(wxSplitterEvent
& event
) 
 893     // for compatibility, call the virtual function 
 894     OnDoubleClickSash(event
.GetX(), event
.GetY()); 
 896     if ( GetMinimumPaneSize() == 0 
 897         || m_permitUnsplitAlways
) 
 903 void wxSplitterWindow::OnUnsplitEvent(wxSplitterEvent
& event
) 
 905     wxWindow 
*win 
= event
.GetWindowBeingRemoved(); 
 907     // for compatibility, call the virtual function