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
)) ) 
 429 // Draw 3D effect borders 
 430 void wxSplitterWindow::DrawBorders(wxDC
& dc
) 
 433     GetClientSize(&w
, &h
); 
 435     if ( GetWindowStyleFlag() & wxSP_3D 
) 
 438         dc
.SetPen(*m_facePen
); 
 439         dc
.SetBrush(*m_faceBrush
); 
 440         dc
.DrawRectangle(1, 1 , w
-1, m_borderSize
-2 ); //high 
 441         dc
.DrawRectangle(1, m_borderSize
-2 , m_borderSize
-2, h
-1 ); // left 
 442         dc
.DrawRectangle(w
-m_borderSize
+2, m_borderSize
-2 , w
-1, h
-1 ); // right 
 443         dc
.DrawRectangle(m_borderSize
-2, h
-m_borderSize
+2 , w
-m_borderSize
+2, h
-1 ); //bottom 
 445         dc
.SetPen(*m_mediumShadowPen
); 
 446         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, w
-m_borderSize
+1, m_borderSize
-2); 
 447         dc
.DrawLine(m_borderSize
-2, m_borderSize
-2, m_borderSize
-2, h
-m_borderSize
+1); 
 449         dc
.SetPen(*m_darkShadowPen
); 
 450         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, w
-m_borderSize
, m_borderSize
-1); 
 451         dc
.DrawLine(m_borderSize
-1, m_borderSize
-1, m_borderSize
-1, h
-m_borderSize
); 
 453         dc
.SetPen(*m_hilightPen
); 
 454         dc
.DrawLine(m_borderSize 
- 2, h
-m_borderSize
+1, w
-m_borderSize
+1, h
-m_borderSize
+1); 
 455         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. 
 456                                      /// Anyway, h is required for MSW. 
 458         dc
.SetPen(*m_lightShadowPen
); 
 459         dc
.DrawLine(w
-m_borderSize
, m_borderSize
-1, w
-m_borderSize
, h
-m_borderSize
); // Right hand side 
 460         dc
.DrawLine(m_borderSize
-1, h
-m_borderSize
, w
-m_borderSize
+1, h
-m_borderSize
);     // Bottom 
 462     else if ( GetWindowStyleFlag() & wxSP_BORDER 
) 
 464         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 465         dc
.SetPen(*wxBLACK_PEN
); 
 466         dc
.DrawRectangle(0, 0, w
-1, h
-1); 
 469     dc
.SetPen(wxNullPen
); 
 470     dc
.SetBrush(wxNullBrush
); 
 474 void wxSplitterWindow::DrawSash(wxDC
& dc
) 
 476     if ( m_sashPosition 
== 0 || !m_windowTwo
) 
 480     GetClientSize(&w
, &h
); 
 482     if ( GetWindowStyleFlag() & wxSP_3D 
) 
 484         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 486             dc
.SetPen(*m_facePen
); 
 487             dc
.SetBrush(*m_faceBrush
); 
 488             dc
.DrawRectangle(m_sashPosition 
+ 2, 0 , m_sashSize 
- 4, h 
); 
 490             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 492             dc
.SetPen(*m_lightShadowPen
); 
 493                         int xShadow 
= m_borderSize 
? m_borderSize 
- 1 : 0 ; 
 494             dc
.DrawLine(m_sashPosition
, xShadow 
, m_sashPosition
, h
-m_borderSize
); 
 496             dc
.SetPen(*m_hilightPen
); 
 497             dc
.DrawLine(m_sashPosition
+1, m_borderSize 
- 2, m_sashPosition
+1, h 
- m_borderSize
+2); 
 499             dc
.SetPen(*m_mediumShadowPen
); 
 500                         int yMedium 
= m_borderSize 
? h
-m_borderSize
+1 : h 
; 
 501             dc
.DrawLine(m_sashPosition
+m_sashSize
-2, xShadow
, m_sashPosition
+m_sashSize
-2, yMedium
); 
 503             dc
.SetPen(*m_darkShadowPen
); 
 504             dc
.DrawLine(m_sashPosition
+m_sashSize
-1, m_borderSize
, m_sashPosition
+m_sashSize
-1, h
-m_borderSize 
); 
 508             dc
.SetPen(*m_facePen
); 
 509             dc
.SetBrush(*m_faceBrush
); 
 510             dc
.DrawRectangle( m_borderSize
-2, m_sashPosition 
+ 2, w
-m_borderSize
+2, m_sashSize 
- 4); 
 512             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 514             dc
.SetPen(*m_lightShadowPen
); 
 515             dc
.DrawLine(m_borderSize
-1, m_sashPosition
, w
-m_borderSize
, m_sashPosition
); 
 517             dc
.SetPen(*m_hilightPen
); 
 518             dc
.DrawLine(m_borderSize
-2, m_sashPosition
+1, w
-m_borderSize
+1, m_sashPosition
+1); 
 520             dc
.SetPen(*m_mediumShadowPen
); 
 521             dc
.DrawLine(m_borderSize
-1, m_sashPosition
+m_sashSize
-2, w
-m_borderSize
+1, m_sashPosition
+m_sashSize
-2); 
 523             dc
.SetPen(*m_darkShadowPen
); 
 524             dc
.DrawLine(m_borderSize
, m_sashPosition
+m_sashSize
-1, w
-m_borderSize
, m_sashPosition
+m_sashSize
-1); 
 529         if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 531             dc
.SetPen(*wxBLACK_PEN
); 
 532             dc
.SetBrush(*wxBLACK_BRUSH
); 
 534             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
) 
 535                 h1 
+= 1; // Not sure why this is necessary... 
 536             dc
.DrawRectangle(m_sashPosition
, 0, m_sashSize
, h1
); 
 540             dc
.SetPen(*wxBLACK_PEN
); 
 541             dc
.SetBrush(*wxBLACK_BRUSH
); 
 543             if ( (GetWindowStyleFlag() & wxSP_BORDER
) != wxSP_BORDER 
) 
 546             dc
.DrawRectangle(0, m_sashPosition
, w1
, m_sashSize
); 
 551     dc
.SetPen(wxNullPen
); 
 552     dc
.SetBrush(wxNullBrush
); 
 555 // Draw the sash tracker (for whilst moving the sash) 
 556 void wxSplitterWindow::DrawSashTracker(int x
, int y
) 
 559     GetClientSize(&w
, &h
); 
 565     if ( m_splitMode 
== wxSPLIT_VERTICAL 
) 
 596     ClientToScreen(&x1
, &y1
); 
 597     ClientToScreen(&x2
, &y2
); 
 599     screenDC
.SetLogicalFunction(wxINVERT
); 
 600     screenDC
.SetPen(*m_sashTrackerPen
); 
 601     screenDC
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 603     screenDC
.DrawLine(x1
, y1
, x2
, y2
); 
 605     screenDC
.SetLogicalFunction(wxCOPY
); 
 607     screenDC
.SetPen(wxNullPen
); 
 608     screenDC
.SetBrush(wxNullBrush
); 
 611 // Position and size subwindows. 
 612 // Note that the border size applies to each subwindow, not 
 613 // including the edges next to the sash. 
 614 void wxSplitterWindow::SizeWindows() 
 617     GetClientSize(&w
, &h
); 
 619     if ( m_windowOne 
&& !m_windowTwo 
) 
 621         m_windowOne
->SetSize(m_borderSize
, m_borderSize
, w 
- 2*m_borderSize
, h 
- 2*m_borderSize
); 
 624     else if ( m_windowOne 
&& m_windowTwo 
) 
 626         if (m_splitMode 
== wxSPLIT_VERTICAL
) 
 628             int x1 
= m_borderSize
; 
 629             int y1 
= m_borderSize
; 
 630             int w1 
= m_sashPosition 
- m_borderSize
; 
 631             int h1 
= h 
- 2*m_borderSize
; 
 633             int x2 
= m_sashPosition 
+ m_sashSize
; 
 634             int y2 
= m_borderSize
; 
 635             int w2 
= w 
- 2*m_borderSize 
- m_sashSize 
- w1
; 
 636             int h2 
= h 
- 2*m_borderSize
; 
 638             m_windowOne
->SetSize(x1
, y1
, w1
, h1
); 
 639             m_windowTwo
->SetSize(x2
, y2
, w2
, h2
); 
 644             m_windowOne
->SetSize(m_borderSize
, m_borderSize
, 
 645                 w 
- 2*m_borderSize
, m_sashPosition 
- m_borderSize
); 
 646             m_windowTwo
->SetSize(m_borderSize
, m_sashPosition 
+ m_sashSize
, 
 647                 w 
- 2*m_borderSize
, h 
- 2*m_borderSize 
- m_sashSize 
- (m_sashPosition 
- m_borderSize
)); 
 652     if ( m_borderSize 
> 0 ) 
 657 // Set pane for unsplit window 
 658 void wxSplitterWindow::Initialize(wxWindow 
*window
) 
 660     m_windowOne 
= window
; 
 661     m_windowTwo 
= (wxWindow 
*) NULL
; 
 665 // Associates the given window with window 2, drawing the appropriate sash 
 666 // and changing the split mode. 
 667 // Does nothing and returns FALSE if the window is already split. 
 668 bool wxSplitterWindow::SplitVertically(wxWindow 
*window1
, wxWindow 
*window2
, int sashPosition
) 
 674     GetClientSize(&w
, &h
); 
 676     m_splitMode 
= wxSPLIT_VERTICAL
; 
 677     m_windowOne 
= window1
; 
 678     m_windowTwo 
= window2
; 
 679     if ( sashPosition 
> 0 ) 
 680         m_sashPosition 
= sashPosition
; 
 681     else if ( sashPosition 
< 0 ) 
 682         m_sashPosition 
= w 
- sashPosition
; 
 684         m_sashPosition 
= w
/2; 
 691 bool wxSplitterWindow::SplitHorizontally(wxWindow 
*window1
, wxWindow 
*window2
, int sashPosition
) 
 697     GetClientSize(&w
, &h
); 
 699     m_splitMode 
= wxSPLIT_HORIZONTAL
; 
 700     m_windowOne 
= window1
; 
 701     m_windowTwo 
= window2
; 
 702     if ( sashPosition 
> 0 ) 
 703         m_sashPosition 
= sashPosition
; 
 704     else if ( sashPosition 
< 0 ) 
 705         m_sashPosition 
= h 
- sashPosition
; 
 707         m_sashPosition 
= h
/2; 
 715 // Remove the specified (or second) window from the view 
 716 // Doesn't actually delete the window. 
 717 bool wxSplitterWindow::Unsplit(wxWindow 
*toRemove
) 
 722     wxWindow 
*win 
= NULL
; 
 723     if ( toRemove 
== NULL 
|| toRemove 
== m_windowTwo
) 
 726         m_windowTwo 
= (wxWindow 
*) NULL
; 
 728     else if ( toRemove 
== m_windowOne 
) 
 731         m_windowOne 
= m_windowTwo
; 
 732         m_windowTwo 
= (wxWindow 
*) NULL
; 
 736         wxFAIL_MSG(wxT("splitter: attempt to remove a non-existent window")); 
 741     SendUnsplitEvent(win
); 
 748 // Replace a window with another one 
 749 bool wxSplitterWindow::ReplaceWindow(wxWindow 
*winOld
, wxWindow 
*winNew
) 
 751     wxCHECK_MSG( winOld
, FALSE
, wxT("use one of Split() functions instead") ); 
 752     wxCHECK_MSG( winNew
, FALSE
, wxT("use Unsplit() functions instead") ); 
 754     if ( winOld 
== m_windowTwo 
) 
 756         m_windowTwo 
= winNew
; 
 758     else if ( winOld 
== m_windowOne 
) 
 760         m_windowOne 
= winNew
; 
 764         wxFAIL_MSG(wxT("splitter: attempt to replace a non-existent window")); 
 774 void wxSplitterWindow::SetSashPosition(int position
, bool redraw
) 
 776     m_sashPosition 
= position
; 
 784 // Initialize colours 
 785 void wxSplitterWindow::InitColours() 
 787     wxDELETE( m_facePen 
); 
 788     wxDELETE( m_faceBrush 
); 
 789     wxDELETE( m_mediumShadowPen 
); 
 790     wxDELETE( m_darkShadowPen 
); 
 791     wxDELETE( m_lightShadowPen 
); 
 792     wxDELETE( m_hilightPen 
); 
 795 #if defined(__WIN95__) 
 796     wxColour 
faceColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DFACE
)); 
 797     m_facePen 
= new wxPen(faceColour
, 1, wxSOLID
); 
 798     m_faceBrush 
= new wxBrush(faceColour
, wxSOLID
); 
 800     wxColour 
mediumShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DSHADOW
)); 
 801     m_mediumShadowPen 
= new wxPen(mediumShadowColour
, 1, wxSOLID
); 
 803     wxColour 
darkShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DDKSHADOW
)); 
 804     m_darkShadowPen 
= new wxPen(darkShadowColour
, 1, wxSOLID
); 
 806     wxColour 
lightShadowColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DLIGHT
)); 
 807     m_lightShadowPen 
= new wxPen(lightShadowColour
, 1, wxSOLID
); 
 809     wxColour 
hilightColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DHILIGHT
)); 
 810     m_hilightPen 
= new wxPen(hilightColour
, 1, wxSOLID
); 
 812     m_facePen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 813     m_faceBrush 
= new wxBrush("LIGHT GREY", wxSOLID
); 
 814     m_mediumShadowPen 
= new wxPen("GREY", 1, wxSOLID
); 
 815     m_darkShadowPen 
= new wxPen("BLACK", 1, wxSOLID
); 
 816     m_lightShadowPen 
= new wxPen("LIGHT GREY", 1, wxSOLID
); 
 817     m_hilightPen 
= new wxPen("WHITE", 1, wxSOLID
); 
 818 #endif // Win32/!Win32 
 821 void wxSplitterWindow::SendUnsplitEvent(wxWindow 
*winRemoved
) 
 823     wxSplitterEvent 
event(wxEVT_COMMAND_SPLITTER_UNSPLIT
, this); 
 824     event
.m_data
.win 
= winRemoved
; 
 826     (void)GetEventHandler()->ProcessEvent(event
); 
 829 // --------------------------------------------------------------------------- 
 830 // splitter event handlers 
 831 // --------------------------------------------------------------------------- 
 833 void wxSplitterWindow::OnSashPosChanged(wxSplitterEvent
& event
) 
 835     // If within UNSPLIT_THRESHOLD from edge, set to edge to cause closure. 
 836     const int UNSPLIT_THRESHOLD 
= 4; 
 838     int newSashPosition 
= event
.GetSashPosition(); 
 840     // Obtain relevant window dimension for bottom / right threshold check 
 842     GetClientSize(&w
, &h
); 
 843     int window_size 
= (m_splitMode 
== wxSPLIT_VERTICAL
) ? w 
: h 
; 
 845     bool unsplit_scenario 
= FALSE
; 
 846     if ( m_permitUnsplitAlways
 
 847             || m_minimumPaneSize 
== 0 ) 
 849         // Do edge detection if unsplit premitted 
 850         if ( newSashPosition 
<= UNSPLIT_THRESHOLD 
) 
 852             // threshold top / left check 
 854             unsplit_scenario 
= TRUE
; 
 856         if ( newSashPosition 
>= window_size 
- UNSPLIT_THRESHOLD 
) 
 858             // threshold bottom/right check 
 859             newSashPosition 
= window_size
; 
 860             unsplit_scenario 
= TRUE
; 
 864     if ( !unsplit_scenario 
) 
 866         // If resultant pane would be too small, enlarge it 
 867         if ( newSashPosition 
< m_minimumPaneSize 
) 
 868             newSashPosition 
= m_minimumPaneSize
; 
 869         if ( newSashPosition 
> window_size 
- m_minimumPaneSize 
) 
 870             newSashPosition 
= window_size 
- m_minimumPaneSize
; 
 873     // If the result is out of bounds it means minimum size is too big, 
 874     // so split window in half as best compromise. 
 875     if ( newSashPosition 
< 0 || newSashPosition 
> window_size 
) 
 876         newSashPosition 
= window_size 
/ 2; 
 878     // for compatibility, call the virtual function 
 879     if ( !OnSashPositionChange(newSashPosition
) ) 
 881         newSashPosition 
= -1; 
 884     event
.SetSashPosition(newSashPosition
); 
 887 // Called when the sash is double-clicked. The default behaviour is to remove 
 888 // the sash if the minimum pane size is zero. 
 889 void wxSplitterWindow::OnDoubleClick(wxSplitterEvent
& event
) 
 891     // for compatibility, call the virtual function 
 892     OnDoubleClickSash(event
.GetX(), event
.GetY()); 
 894     if ( GetMinimumPaneSize() == 0 || m_permitUnsplitAlways 
) 
 900 void wxSplitterWindow::OnUnsplitEvent(wxSplitterEvent
& event
) 
 902     wxWindow 
*win 
= event
.GetWindowBeingRemoved(); 
 904     // do it before calling OnUnsplit() which may delete the window 
 907     // for compatibility, call the virtual function