1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxSplitterWindow class 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 #ifndef __SPLITTERH_G__ 
  13 #define __SPLITTERH_G__ 
  15 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  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 WXUNUSED(width
)) { } 
 138     // Sets the border size 
 139     void SetBorderSize(int WXUNUSED(width
)) { } 
 141     // Gets the sash size 
 142     int GetSashSize() const; 
 144     // Gets the border size 
 145     int GetBorderSize() const; 
 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 OnInternalIdle(); 
 196     virtual void DrawSash(wxDC
& dc
); 
 198     // Draws the sash tracker (for whilst moving the sash) 
 199     virtual void DrawSashTracker(int x
, int y
); 
 201     // Tests for x, y over sash 
 202     virtual bool SashHitTest(int x
, int y
, int tolerance 
= 5); 
 204     // Resizes subwindows 
 205     virtual void SizeWindows(); 
 207     void SetNeedUpdating(bool needUpdating
) { m_needUpdating 
= needUpdating
; } 
 208     bool GetNeedUpdating() const { return m_needUpdating 
; } 
 212 #if defined(__WXMSW__) || defined(__WXMAC__) 
 213     void OnSetCursor(wxSetCursorEvent
& event
); 
 216     // send the given event, return FALSE if the event was processed and vetoed 
 218     inline bool DoSendEvent(wxSplitterEvent
& event
); 
 221     // common part of all ctors 
 224     // common part of SplitVertically() and SplitHorizontally() 
 225     bool DoSplit(wxSplitMode mode
, 
 226                  wxWindow 
*window1
, wxWindow 
*window2
, 
 229     // adjusts sash position with respect to min. pane and window sizes 
 230     int AdjustSashPosition(int sashPos
) const; 
 232     // get either width or height depending on the split mode 
 233     int GetWindowSize() const; 
 235     // convert the user specified sash position which may be > 0 (as is), < 0 
 236     // (specifying the size of the right pane) or 0 (use default) to the real 
 237     // position to be passed to DoSetSashPosition() 
 238     int ConvertSashPosition(int sashPos
) const; 
 240     // set the real sash position, sashPos here must be positive 
 242     // returns TRUE if the sash position has been changed, FALSE otherwise 
 243     bool DoSetSashPosition(int sashPos
); 
 245     // set the sash position and send an event about it having been changed 
 246     void SetSashPositionAndNotify(int sashPos
); 
 248     // callbacks executed when we detect that the mouse has entered or left 
 250     virtual void OnEnterSash(); 
 251     virtual void OnLeaveSash(); 
 253     // set the cursor appropriate for the current split mode 
 254     void SetResizeCursor(); 
 256     // redraw the splitter if its "hotness" changed if necessary 
 257     void RedrawIfHotSensitive(bool isHot
); 
 259     wxSplitMode m_splitMode
; 
 260     wxWindow
*   m_windowOne
; 
 261     wxWindow
*   m_windowTwo
; 
 265     int         m_sashPosition
; // Number of pixels from left or top 
 266     int         m_requestedSashPosition
; 
 267     int         m_sashPositionCurrent
; // while dragging 
 270     int         m_minimumPaneSize
; 
 271     wxCursor    m_sashCursorWE
; 
 272     wxCursor    m_sashCursorNS
; 
 273     wxPen      
*m_sashTrackerPen
; 
 275     // when in live mode, set this to TRUE to resize children in idle 
 276     bool        m_needUpdating
:1; 
 277     bool        m_permitUnsplitAlways
:1; 
 281     WX_DECLARE_CONTROL_CONTAINER(); 
 283     DECLARE_DYNAMIC_CLASS(wxSplitterWindow
) 
 284     DECLARE_EVENT_TABLE() 
 285     DECLARE_NO_COPY_CLASS(wxSplitterWindow
) 
 288 // ---------------------------------------------------------------------------- 
 289 // event class and macros 
 290 // ---------------------------------------------------------------------------- 
 292 // we reuse the same class for all splitter event types because this is the 
 293 // usual wxWin convention, but the three event types have different kind of 
 294 // data associated with them, so the accessors can be only used if the real 
 295 // event type matches with the one for which the accessors make sense 
 296 class WXDLLEXPORT wxSplitterEvent 
: public wxNotifyEvent
 
 299     wxSplitterEvent(wxEventType type 
= wxEVT_NULL
, 
 300                     wxSplitterWindow 
*splitter 
= (wxSplitterWindow 
*)NULL
) 
 301         : wxNotifyEvent(type
) 
 303         SetEventObject(splitter
); 
 304         if (splitter
) m_id 
= splitter
->GetId(); 
 307     // SASH_POS_CHANGED methods 
 309     // setting the sash position to -1 prevents the change from taking place at 
 311     void SetSashPosition(int pos
) 
 313         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
 
 314                 || GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
); 
 319     int GetSashPosition() const 
 321         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED
 
 322                 || GetEventType() == wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING
); 
 327     // UNSPLIT event methods 
 328     wxWindow 
*GetWindowBeingRemoved() const 
 330         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_UNSPLIT 
); 
 335     // DCLICK event methods 
 338         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_DOUBLECLICKED 
); 
 345         wxASSERT( GetEventType() == wxEVT_COMMAND_SPLITTER_DOUBLECLICKED 
); 
 351     friend class WXDLLEXPORT wxSplitterWindow
; 
 353     // data for the different types of event 
 356         int pos
;            // position for SASH_POS_CHANGED event 
 357         wxWindow 
*win
;      // window being removed for UNSPLIT event 
 361         } pt
;               // position of double click for DCLICK event 
 364     DECLARE_DYNAMIC_CLASS_NO_COPY(wxSplitterEvent
) 
 367 typedef void (wxEvtHandler::*wxSplitterEventFunction
)(wxSplitterEvent
&); 
 369 #define EVT_SPLITTER_SASH_POS_CHANGED(id, fn)                               \ 
 370   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 371     wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGED,                                \ 
 374     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 378 #define EVT_SPLITTER_SASH_POS_CHANGING(id, fn)                              \ 
 379   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 380     wxEVT_COMMAND_SPLITTER_SASH_POS_CHANGING,                               \ 
 383     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 387 #define EVT_SPLITTER_DCLICK(id, fn)                                         \ 
 388   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 389     wxEVT_COMMAND_SPLITTER_DOUBLECLICKED,                                   \ 
 392     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 396 #define EVT_SPLITTER_UNSPLIT(id, fn)                                        \ 
 397   DECLARE_EVENT_TABLE_ENTRY(                                                \ 
 398     wxEVT_COMMAND_SPLITTER_UNSPLIT,                                         \ 
 401     (wxObjectEventFunction)(wxEventFunction)(wxSplitterEventFunction) &fn,  \ 
 405 #endif // __SPLITTERH_G__