1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxSplitterWindow class 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 #ifndef __SPLITTERH_G__ 
  13 #define __SPLITTERH_G__ 
  15 #if defined(__GNUG__) && !defined(__APPLE__) 
  16     #pragma interface "splitter.h" 
  19 #include "wx/window.h"                      // base class declaration 
  20 #include "wx/containr.h"                    // wxControlContainer 
  22 class WXDLLEXPORT wxSplitterEvent
; 
  24 // --------------------------------------------------------------------------- 
  26 // --------------------------------------------------------------------------- 
  30     wxSPLIT_HORIZONTAL 
= 1, 
  37     wxSPLIT_DRAG_DRAGGING
, 
  38     wxSPLIT_DRAG_LEFT_DOWN
 
  41 // --------------------------------------------------------------------------- 
  42 // wxSplitterWindow maintains one or two panes, with 
  43 // an optional vertical or horizontal split which 
  44 // can be used with the mouse or programmatically. 
  45 // --------------------------------------------------------------------------- 
  48 // 1) Perhaps make the borders sensitive to dragging in order to create a split. 
  49 //    The MFC splitter window manages scrollbars as well so is able to 
  50 //    put sash buttons on the scrollbars, but we probably don't want to go down 
  52 // 2) for wxWindows 2.0, we must find a way to set the WS_CLIPCHILDREN style 
  53 //    to prevent flickering. (WS_CLIPCHILDREN doesn't work in all cases so can't be 
  56 class WXDLLEXPORT wxSplitterWindow
: public wxWindow
 
  60 //////////////////////////////////////////////////////////////////////////// 
  63     // Default constructor 
  70     wxSplitterWindow(wxWindow 
*parent
, wxWindowID id 
= -1, 
  71                      const wxPoint
& pos 
= wxDefaultPosition
, 
  72                      const wxSize
& size 
= wxDefaultSize
, 
  74                      const wxString
& name 
= wxT("splitter")) 
  77         Create(parent
, id
, pos
, size
, style
, name
); 
  80     virtual ~wxSplitterWindow(); 
  82     bool Create(wxWindow 
*parent
, wxWindowID id 
= -1, 
  83                      const wxPoint
& pos 
= wxDefaultPosition
, 
  84                      const wxSize
& size 
= wxDefaultSize
, 
  86                      const wxString
& name 
= wxT("splitter")); 
  88     // Gets the only or left/top pane 
  89     wxWindow 
*GetWindow1() const { return m_windowOne
; } 
  91     // Gets the right/bottom pane 
  92     wxWindow 
*GetWindow2() const { return m_windowTwo
; } 
  94     // Sets the split mode 
  95     void SetSplitMode(int mode
) 
  97         wxASSERT_MSG( mode 
== wxSPLIT_VERTICAL 
|| mode 
== wxSPLIT_HORIZONTAL
, 
  98                       _T("invalid split mode") ); 
 100         m_splitMode 
= (wxSplitMode
)mode
; 
 103     // Gets the split mode 
 104     wxSplitMode 
GetSplitMode() const { return m_splitMode
; }; 
 106     // Initialize with one window 
 107     void Initialize(wxWindow 
*window
); 
 109     // Associates the given window with window 2, drawing the appropriate sash 
 110     // and changing the split mode. 
 111     // Does nothing and returns FALSE if the window is already split. 
 112     // A sashPosition of 0 means choose a default sash position, 
 113     // negative sashPosition specifies the size of right/lower pane as it's 
 114     // absolute value rather than the size of left/upper pane. 
 115     virtual bool SplitVertically(wxWindow 
*window1
, 
 117                                  int sashPosition 
= 0) 
 118         { return DoSplit(wxSPLIT_VERTICAL
, window1
, window2
, sashPosition
); } 
 119     virtual bool SplitHorizontally(wxWindow 
*window1
, 
 121                                    int sashPosition 
= 0) 
 122         { return DoSplit(wxSPLIT_HORIZONTAL
, window1
, window2
, sashPosition
); } 
 124     // Removes the specified (or second) window from the view 
 125     // Doesn't actually delete the window. 
 126     bool Unsplit(wxWindow 
*toRemove 
= (wxWindow 
*) NULL
); 
 128     // Replaces one of the windows with another one (neither old nor new 
 129     // parameter should be NULL) 
 130     bool ReplaceWindow(wxWindow 
*winOld
, wxWindow 
*winNew
); 
 132     // Is the window split? 
 133     bool IsSplit() const { return (m_windowTwo 
!= NULL
); } 
 135     // Sets the sash size 
 136     void SetSashSize(int width
) { m_sashSize 
= width
; } 
 138     // Sets the border size 
 139     void SetBorderSize(int width
) { m_borderSize 
= width
; } 
 141     // Gets the sash size 
 142     int GetSashSize() const { return m_sashSize
; } 
 144     // Gets the border size 
 145     int GetBorderSize() const { return m_borderSize
; } 
 147     // Set the sash position 
 148     void SetSashPosition(int position
, bool redraw 
= TRUE
); 
 150     // Gets the sash position 
 151     int GetSashPosition() const { return m_sashPosition
; } 
 153     // If this is zero, we can remove panes by dragging the sash. 
 154     void SetMinimumPaneSize(int min
); 
 155     int GetMinimumPaneSize() const { return m_minimumPaneSize
; } 
 157     // NB: the OnXXX() functions below are for backwards compatibility only, 
 158     //     don't use them in new code but handle the events instead! 
 160     // called when the sash position is about to change, may return a new value 
 161     // for the sash or -1 to prevent the change from happening at all 
 162     virtual int OnSashPositionChanging(int newSashPosition
); 
 164     // Called when the sash position is about to be changed, return 
 165     // FALSE from here to prevent the change from taking place. 
 166     // Repositions sash to minimum position if pane would be too small. 
 167     // newSashPosition here is always positive or zero. 
 168     virtual bool OnSashPositionChange(int newSashPosition
); 
 170     // If the sash is moved to an extreme position, a subwindow 
 171     // is removed from the splitter window, and the app is 
 172     // notified. The app should delete or hide the window. 
 173     virtual void OnUnsplit(wxWindow 
*removed
); 
 175     // Called when the sash is double-clicked. 
 176     // The default behaviour is to remove the sash if the 
 177     // minimum pane size is zero. 
 178     virtual void OnDoubleClickSash(int x
, int y
); 
 180 //////////////////////////////////////////////////////////////////////////// 
 183     // Paints the border and sash 
 184     void OnPaint(wxPaintEvent
& event
); 
 186     // Handles mouse events 
 187     void OnMouseEvent(wxMouseEvent
& ev
); 
 190     void OnSize(wxSizeEvent
& event
); 
 192     // In live mode, resize child windows in idle time 
 193     void OnIdle(wxIdleEvent
& event
); 
 196     virtual void DrawBorders(wxDC
& dc
); 
 199     virtual void DrawSash(wxDC
& dc
); 
 201     // Draws the sash tracker (for whilst moving the sash) 
 202     virtual void DrawSashTracker(int x
, int y
); 
 204     // Tests for x, y over sash 
 205     virtual bool SashHitTest(int x
, int y
, int tolerance 
= 2); 
 207     // Resizes subwindows 
 208     virtual void SizeWindows(); 
 210     // Initialize colours 
 213     void SetNeedUpdating(bool needUpdating
) { m_needUpdating 
= needUpdating
; } 
 214     bool GetNeedUpdating() const { return m_needUpdating 
; } 
 219     void OnSetCursor(wxSetCursorEvent
& event
); 
 222     // send the given event, return FALSE if the event was processed and vetoed 
 224     inline bool DoSendEvent(wxSplitterEvent
& event
); 
 227     // common part of all ctors 
 230     // common part of SplitVertically() and SplitHorizontally() 
 231     bool DoSplit(wxSplitMode mode
, 
 232                  wxWindow 
*window1
, wxWindow 
*window2
, 
 235     // adjusts sash position with respect to min. pane and window sizes 
 236     int AdjustSashPosition(int sashPos
) const; 
 238     // get either width or height depending on the split mode 
 239     int GetWindowSize() const; 
 241     // convert the user specified sash position which may be > 0 (as is), < 0 
 242     // (specifying the size of the right pane) or 0 (use default) to the real 
 243     // position to be passed to DoSetSashPosition() 
 244     int ConvertSashPosition(int sashPos
) const; 
 246     // set the real sash position, sashPos here must be positive 
 248     // returns TRUE if the sash position has been changed, FALSE otherwise 
 249     bool DoSetSashPosition(int sashPos
); 
 251     // set the sash position and send an event about it having been changed 
 252     void SetSashPositionAndNotify(int sashPos
); 
 254     // set the cursor appropriate for the current split mode 
 255     void SetResizeCursor(); 
 257     wxSplitMode m_splitMode
; 
 258     bool        m_permitUnsplitAlways
; 
 259     bool        m_needUpdating
; // when in live mode, set this to TRUE to resize children in idle 
 260     wxWindow
*   m_windowOne
; 
 261     wxWindow
*   m_windowTwo
; 
 266     int         m_sashSize
;     // Sash width or height 
 267     int         m_sashPosition
; // Number of pixels from left or top 
 268     int         m_requestedSashPosition
; 
 269     int         m_sashPositionCurrent
; // while dragging 
 272     int         m_minimumPaneSize
; 
 273     wxCursor    m_sashCursorWE
; 
 274     wxCursor    m_sashCursorNS
; 
 275     wxPen
*      m_sashTrackerPen
; 
 276     wxPen
*      m_lightShadowPen
; 
 277     wxPen
*      m_mediumShadowPen
; 
 278     wxPen
*      m_darkShadowPen
; 
 280     wxBrush
*    m_faceBrush
; 
 284     WX_DECLARE_CONTROL_CONTAINER(); 
 286     DECLARE_DYNAMIC_CLASS(wxSplitterWindow
) 
 287     DECLARE_EVENT_TABLE() 
 290 // ---------------------------------------------------------------------------- 
 291 // event class and macros 
 292 // ---------------------------------------------------------------------------- 
 294 // we reuse the same class for all splitter event types because this is the 
 295 // usual wxWin convention, but the three event types have different kind of 
 296 // data associated with them, so the accessors can be only used if the real 
 297 // event type matches with the one for which the accessors make sense 
 298 class WXDLLEXPORT wxSplitterEvent 
: public wxNotifyEvent
 
 301     wxSplitterEvent(wxEventType type 
= wxEVT_NULL
, 
 302                     wxSplitterWindow 
*splitter 
= (wxSplitterWindow 
*)NULL
) 
 303         : wxNotifyEvent(type
) 
 305         SetEventObject(splitter
); 
 306         if (splitter
) m_id 
= splitter
->GetId(); 
 309     // SASH_POS_CHANGED methods 
 311     // setting the sash position to -1 prevents the change from taking place at 
 313     void SetSashPosition(int pos
) 
 315         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
 
 316                 || GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
); 
 321     int GetSashPosition() const 
 323         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
 
 324                 || GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
); 
 329     // UNSPLIT event methods 
 330     wxWindow 
*GetWindowBeingRemoved() const 
 332         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_UNSPLIT 
); 
 337     // DCLICK event methods 
 340         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_DOUBLECLICKED 
); 
 347         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_DOUBLECLICKED 
); 
 353     friend class WXDLLEXPORT wxSplitterWindow
; 
 355     // data for the different types of event 
 358         int pos
;            // position for SASH_POS_CHANGED event 
 359         wxWindow 
*win
;      // window being removed for UNSPLIT event 
 363         } pt
;               // position of double click for DCLICK event 
 366     DECLARE_DYNAMIC_CLASS(wxSplitterEvent
) 
 369 typedef void (wxEvtHandler::*wxSplitterEventFunction
)(wxSplitterEvent
&); 
 371 #define EVT_SPLITTER_SASH_POS_CHANGED(id, fn)                               \ 
 372   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 373     wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED,                                \ 
 376     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 380 #define EVT_SPLITTER_SASH_POS_CHANGING(id, fn)                              \ 
 381   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 382     wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING,                               \ 
 385     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 389 #define EVT_SPLITTER_DCLICK(id, fn)                                         \ 
 390   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 391     wxEVT_COMMAND_SPLITTER_DOUBLECLICKED,                                   \ 
 394     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 398 #define EVT_SPLITTER_UNSPLIT(id, fn)                                        \ 
 399   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 400     wxEVT_COMMAND_SPLITTER_UNSPLIT,                                         \ 
 403     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 407 #endif // __SPLITTERH_G__