# -*- coding: utf-8 -*- # --------------------------------------------------------------------------- # # AUI Library wxPython IMPLEMENTATION # # Original C++ Code From Kirix (wxAUI). You Can Find It At: # # License: wxWidgets license # # http:#www.kirix.com/en/community/opensource/wxaui/about_wxaui.html # # Current wxAUI Version Tracked: wxWidgets 2.9.4 SVN HEAD # # # Python Code By: # # Andrea Gavana, @ 23 Dec 2005 # Latest Revision: 17 Feb 2013, 21.00 GMT # # For All Kind Of Problems, Requests Of Enhancements And Bug Reports, Please # Write To Me At: # # andrea.gavana@gmail.com # andrea.gavana@maerskoil.com # # Or, Obviously, To The wxPython Mailing List!!! # # Tags: phoenix-port, unittest, documented, py3-port # # End Of Comments # --------------------------------------------------------------------------- # """ Description =========== `framemanager.py` is the central module of the AUI class framework. :class:`AuiManager` manages the panes associated with it for a particular :class:`Frame`, using a pane's :class:`AuiPaneInfo` information to determine each pane's docking and floating behavior. AuiManager uses wxPython' sizer mechanism to plan the layout of each frame. It uses a replaceable dock art class to do all drawing, so all drawing is localized in one area, and may be customized depending on an application's specific needs. AuiManager works as follows: the programmer adds panes to the class, or makes changes to existing pane properties (dock position, floating state, show state, etc...). To apply these changes, AuiManager's :meth:`AuiManager.Update() ` function is called. This batch processing can be used to avoid flicker, by modifying more than one pane at a time, and then "committing" all of the changes at once by calling `Update()`. Panes can be added quite easily:: text1 = wx.TextCtrl(self, -1) text2 = wx.TextCtrl(self, -1) self._mgr.AddPane(text1, AuiPaneInfo().Left().Caption("Pane Number One")) self._mgr.AddPane(text2, AuiPaneInfo().Bottom().Caption("Pane Number Two")) self._mgr.Update() Later on, the positions can be modified easily. The following will float an existing pane in a tool window:: self._mgr.GetPane(text1).Float() Layers, Rows and Directions, Positions ====================================== Inside AUI, the docking layout is figured out by checking several pane parameters. Four of these are important for determining where a pane will end up. **Direction** - Each docked pane has a direction, `Top`, `Bottom`, `Left`, `Right`, or `Center`. This is fairly self-explanatory. The pane will be placed in the location specified by this variable. **Position** - More than one pane can be placed inside of a "dock". Imagine two panes being docked on the left side of a window. One pane can be placed over another. In proportionally managed docks, the pane position indicates it's sequential position, starting with zero. So, in our scenario with two panes docked on the left side, the top pane in the dock would have position 0, and the second one would occupy position 1. **Row** - A row can allow for two docks to be placed next to each other. One of the most common places for this to happen is in the toolbar. Multiple toolbar rows are allowed, the first row being in row 0, and the second in row 1. Rows can also be used on vertically docked panes. **Layer** - A layer is akin to an onion. Layer 0 is the very center of the managed pane. Thus, if a pane is in layer 0, it will be closest to the center window (also sometimes known as the "content window"). Increasing layers "swallow up" all layers of a lower value. This can look very similar to multiple rows, but is different because all panes in a lower level yield to panes in higher levels. The best way to understand layers is by running the AUI sample (`AUI.py`). """ __author__ = "Andrea Gavana " __date__ = "31 March 2009" import wx # just for isinstance import wx.lib.sized_controls as sc import time import warnings import wx.lib.six as six from . import auibar from . import auibook from . import tabmdi from . import dockart from . import tabart from .aui_utilities import Clip, PaneCreateStippleBitmap, GetDockingImage, GetSlidingPoints from .aui_constants import * # Define this as a translation function _ = wx.GetTranslation _winxptheme = False if wx.Platform == "__WXMSW__": try: import winxptheme _winxptheme = True except ImportError: pass # wxPython version string _VERSION_STRING = wx.VERSION_STRING # AUI Events wxEVT_AUI_PANE_BUTTON = wx.NewEventType() wxEVT_AUI_PANE_CLOSE = wx.NewEventType() wxEVT_AUI_PANE_MAXIMIZE = wx.NewEventType() wxEVT_AUI_PANE_RESTORE = wx.NewEventType() wxEVT_AUI_RENDER = wx.NewEventType() wxEVT_AUI_FIND_MANAGER = wx.NewEventType() wxEVT_AUI_PANE_MINIMIZE = wx.NewEventType() wxEVT_AUI_PANE_MIN_RESTORE = wx.NewEventType() wxEVT_AUI_PANE_FLOATING = wx.NewEventType() wxEVT_AUI_PANE_FLOATED = wx.NewEventType() wxEVT_AUI_PANE_DOCKING = wx.NewEventType() wxEVT_AUI_PANE_DOCKED = wx.NewEventType() wxEVT_AUI_PANE_ACTIVATED = wx.NewEventType() wxEVT_AUI_PERSPECTIVE_CHANGED = wx.NewEventType() EVT_AUI_PANE_BUTTON = wx.PyEventBinder(wxEVT_AUI_PANE_BUTTON, 0) """ Fires an event when the user left-clicks on a pane button. """ EVT_AUI_PANE_CLOSE = wx.PyEventBinder(wxEVT_AUI_PANE_CLOSE, 0) """ A pane in `AuiManager` has been closed. """ EVT_AUI_PANE_MAXIMIZE = wx.PyEventBinder(wxEVT_AUI_PANE_MAXIMIZE, 0) """ A pane in `AuiManager` has been maximized. """ EVT_AUI_PANE_RESTORE = wx.PyEventBinder(wxEVT_AUI_PANE_RESTORE, 0) """ A pane in `AuiManager` has been restored from a maximized state. """ EVT_AUI_RENDER = wx.PyEventBinder(wxEVT_AUI_RENDER, 0) """ Fires an event every time the AUI frame is being repainted. """ EVT_AUI_FIND_MANAGER = wx.PyEventBinder(wxEVT_AUI_FIND_MANAGER, 0) """ Used to find which AUI manager is controlling a certain pane. """ EVT_AUI_PANE_MINIMIZE = wx.PyEventBinder(wxEVT_AUI_PANE_MINIMIZE, 0) """ A pane in `AuiManager` has been minimized. """ EVT_AUI_PANE_MIN_RESTORE = wx.PyEventBinder(wxEVT_AUI_PANE_MIN_RESTORE, 0) """ A pane in `AuiManager` has been restored from a minimized state. """ EVT_AUI_PANE_FLOATING = wx.PyEventBinder(wxEVT_AUI_PANE_FLOATING, 0) """ A pane in `AuiManager` is about to be floated. """ EVT_AUI_PANE_FLOATED = wx.PyEventBinder(wxEVT_AUI_PANE_FLOATED, 0) """ A pane in `AuiManager` has been floated. """ EVT_AUI_PANE_DOCKING = wx.PyEventBinder(wxEVT_AUI_PANE_DOCKING, 0) """ A pane in `AuiManager` is about to be docked. """ EVT_AUI_PANE_DOCKED = wx.PyEventBinder(wxEVT_AUI_PANE_DOCKED, 0) """ A pane in `AuiManager` has been docked. """ EVT_AUI_PANE_ACTIVATED = wx.PyEventBinder(wxEVT_AUI_PANE_ACTIVATED, 0) """ A pane in `AuiManager` has been activated. """ EVT_AUI_PERSPECTIVE_CHANGED = wx.PyEventBinder(wxEVT_AUI_PERSPECTIVE_CHANGED, 0) """ The layout in `AuiManager` has been changed. """ # ---------------------------------------------------------------------------- # class AuiDockInfo(object): """ A class to store all properties of a dock. """ def __init__(self): """ Default class constructor. Used internally, do not call it in your code! """ object.__init__(self) self.dock_direction = 0 self.dock_layer = 0 self.dock_row = 0 self.size = 0 self.min_size = 0 self.resizable = True self.fixed = False self.toolbar = False self.rect = wx.Rect() self.panes = [] def IsOk(self): """ Returns whether a dock is valid or not. In order to be valid, a dock needs to have a non-zero `dock_direction`. """ return self.dock_direction != 0 def IsHorizontal(self): """ Returns whether the dock is horizontal or not. """ return self.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_BOTTOM] def IsVertical(self): """ Returns whether the dock is vertical or not. """ return self.dock_direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT, AUI_DOCK_CENTER] # ---------------------------------------------------------------------------- # class AuiDockingGuideInfo(object): """ A class which holds information about VS2005 docking guide windows. """ def __init__(self, other=None): """ Default class constructor. Used internally, do not call it in your code! :param `other`: another instance of :class:`AuiDockingGuideInfo`. """ if other: self.Assign(other) else: # window representing the docking target self.host = None # dock direction (top, bottom, left, right, center) self.dock_direction = AUI_DOCK_NONE def Assign(self, other): """ Assigns the properties of the `other` :class:`AuiDockingGuideInfo` to `self`. :param `other`: another instance of :class:`AuiDockingGuideInfo`. """ self.host = other.host self.dock_direction = other.dock_direction def Host(self, h): """ Hosts a docking guide window. :param `h`: an instance of :class:`AuiDockingGuideWindow` or :class:`AuiDockingHintWindow`. """ self.host = h return self def Left(self): """ Sets the guide window to left docking. """ self.dock_direction = AUI_DOCK_LEFT return self def Right(self): """ Sets the guide window to right docking. """ self.dock_direction = AUI_DOCK_RIGHT return self def Top(self): """ Sets the guide window to top docking. """ self.dock_direction = AUI_DOCK_TOP return self def Bottom(self): """ Sets the guide window to bottom docking. """ self.dock_direction = AUI_DOCK_BOTTOM return self def Center(self): """ Sets the guide window to center docking. """ self.dock_direction = AUI_DOCK_CENTER return self def Centre(self): """ Sets the guide window to centre docking. """ self.dock_direction = AUI_DOCK_CENTRE return self # ---------------------------------------------------------------------------- # class AuiDockUIPart(object): """ A class which holds attributes for a UI part in the interface. """ typeCaption = 0 typeGripper = 1 typeDock = 2 typeDockSizer = 3 typePane = 4 typePaneSizer = 5 typeBackground = 6 typePaneBorder = 7 typePaneButton = 8 def __init__(self): """ Default class constructor. Used internally, do not call it in your code! """ self.orientation = wx.VERTICAL self.type = 0 self.rect = wx.Rect() # ---------------------------------------------------------------------------- # class AuiPaneButton(object): """ A simple class which describes the caption pane button attributes. """ def __init__(self, button_id): """ Default class constructor. Used internally, do not call it in your code! :param integer `button_id`: the pane button identifier. """ self.button_id = button_id # ---------------------------------------------------------------------------- # # event declarations/classes class AuiManagerEvent(wx.PyCommandEvent): """ A specialized command event class for events sent by :class:`AuiManager`. """ def __init__(self, eventType, id=1): """ Default class constructor. :param integer `eventType`: the event kind; :param integer `id`: the event identification number. """ wx.PyCommandEvent.__init__(self, eventType, id) self.manager = None self.pane = None self.button = 0 self.veto_flag = False self.canveto_flag = True self.dc = None def SetManager(self, mgr): """ Associates a :class:`AuiManager` to the current event. :param `mgr`: an instance of :class:`AuiManager`. """ self.manager = mgr def SetDC(self, pdc): """ Associates a :class:`DC` device context to this event. :param `pdc`: a :class:`DC` device context object. """ self.dc = pdc def SetPane(self, p): """ Associates a :class:`AuiPaneInfo` instance to this event. :param `p`: a :class:`AuiPaneInfo` instance. """ self.pane = p def SetButton(self, b): """ Associates a :class:`AuiPaneButton` instance to this event. :param `b`: a :class:`AuiPaneButton` instance. """ self.button = b def GetManager(self): """ Returns the associated :class:`AuiManager` (if any). """ return self.manager def GetDC(self): """ Returns the associated :class:`DC` device context (if any). """ return self.dc def GetPane(self): """ Returns the associated :class:`AuiPaneInfo` structure (if any). """ return self.pane def GetButton(self): """ Returns the associated :class:`AuiPaneButton` instance (if any). """ return self.button def Veto(self, veto=True): """ Prevents the change announced by this event from happening. It is in general a good idea to notify the user about the reasons for vetoing the change because otherwise the applications behaviour (which just refuses to do what the user wants) might be quite surprising. :param bool `veto`: ``True`` to veto the event, ``False`` otherwise. """ self.veto_flag = veto def GetVeto(self): """ Returns whether the event has been vetoed or not. """ return self.veto_flag def SetCanVeto(self, can_veto): """ Sets whether the event can be vetoed or not. :param bool `can_veto`: ``True`` if the event can be vetoed, ``False`` otherwise. """ self.canveto_flag = can_veto def CanVeto(self): """ Returns whether the event can be vetoed and has been vetoed. """ return self.canveto_flag and self.veto_flag # ---------------------------------------------------------------------------- # class AuiPaneInfo(object): """ AuiPaneInfo specifies all the parameters for a pane. These parameters specify where the pane is on the screen, whether it is docked or floating, or hidden. In addition, these parameters specify the pane's docked position, floating position, preferred size, minimum size, caption text among many other parameters. """ optionFloating = 2**0 optionHidden = 2**1 optionLeftDockable = 2**2 optionRightDockable = 2**3 optionTopDockable = 2**4 optionBottomDockable = 2**5 optionFloatable = 2**6 optionMovable = 2**7 optionResizable = 2**8 optionPaneBorder = 2**9 optionCaption = 2**10 optionGripper = 2**11 optionDestroyOnClose = 2**12 optionToolbar = 2**13 optionActive = 2**14 optionGripperTop = 2**15 optionMaximized = 2**16 optionDockFixed = 2**17 optionNotebookDockable = 2**18 optionMinimized = 2**19 optionLeftSnapped = 2**20 optionRightSnapped = 2**21 optionTopSnapped = 2**22 optionBottomSnapped = 2**23 optionFlyOut = 2**24 optionCaptionLeft = 2**25 buttonClose = 2**26 buttonMaximize = 2**27 buttonMinimize = 2**28 buttonPin = 2**29 buttonCustom1 = 2**30 buttonCustom2 = 2**31 buttonCustom3 = 2**32 savedHiddenState = 2**33 # used internally actionPane = 2**34 # used internally wasMaximized = 2**35 # used internally needsRestore = 2**36 # used internally def __init__(self): """ Default class constructor. """ self.window = None self.frame = None self.state = 0 self.dock_direction = AUI_DOCK_LEFT self.dock_layer = 0 self.dock_row = 0 self.dock_pos = 0 self.minimize_mode = AUI_MINIMIZE_POS_SMART self.floating_pos = wx.Point(-1, -1) self.floating_size = wx.Size(-1, -1) self.best_size = wx.Size(-1, -1) self.min_size = wx.Size(-1, -1) self.max_size = wx.Size(-1, -1) self.dock_proportion = 0 self.caption = "" self.buttons = [] self.name = "" self.icon = wx.NullIcon self.rect = wx.Rect() self.notebook_id = -1 self.transparent = 255 self.needsTransparency = False self.previousDockPos = None self.previousDockSize = 0 self.snapped = 0 self.minimize_target = None self.DefaultPane() def dock_direction_get(self): """ Getter for the `dock_direction`. :see: :meth:`~AuiPaneInfo.dock_direction_set` for a set of valid docking directions. """ if self.IsMaximized(): return AUI_DOCK_CENTER else: return self._dock_direction def dock_direction_set(self, value): """ Setter for the `dock_direction`. :param integer `value`: the docking direction. This can be one of the following bits: ============================ ======= ============================================= Dock Flag Value Description ============================ ======= ============================================= ``AUI_DOCK_NONE`` 0 No docking direction. ``AUI_DOCK_TOP`` 1 Top docking direction. ``AUI_DOCK_RIGHT`` 2 Right docking direction. ``AUI_DOCK_BOTTOM`` 3 Bottom docking direction. ``AUI_DOCK_LEFT`` 4 Left docking direction. ``AUI_DOCK_CENTER`` 5 Center docking direction. ``AUI_DOCK_CENTRE`` 5 Centre docking direction. ``AUI_DOCK_NOTEBOOK_PAGE`` 6 Automatic AuiNotebooks docking style. ============================ ======= ============================================= """ self._dock_direction = value dock_direction = property(dock_direction_get, dock_direction_set) def IsOk(self): """ Returns ``True`` if the :class:`AuiPaneInfo` structure is valid. :note: A pane structure is valid if it has an associated window. """ return self.window != None def IsMaximized(self): """ Returns ``True`` if the pane is maximized. """ return self.HasFlag(self.optionMaximized) def IsMinimized(self): """ Returns ``True`` if the pane is minimized. """ return self.HasFlag(self.optionMinimized) def IsFixed(self): """ Returns ``True`` if the pane cannot be resized. """ return not self.HasFlag(self.optionResizable) def IsResizeable(self): """ Returns ``True`` if the pane can be resized. """ return self.HasFlag(self.optionResizable) def IsShown(self): """ Returns ``True`` if the pane is currently shown. """ return not self.HasFlag(self.optionHidden) def IsFloating(self): """ Returns ``True`` if the pane is floating. """ return self.HasFlag(self.optionFloating) def IsDocked(self): """ Returns ``True`` if the pane is docked. """ return not self.HasFlag(self.optionFloating) def IsToolbar(self): """ Returns ``True`` if the pane contains a toolbar. """ return self.HasFlag(self.optionToolbar) def IsTopDockable(self): """ Returns ``True`` if the pane can be docked at the top of the managed frame. """ return self.HasFlag(self.optionTopDockable) def IsBottomDockable(self): """ Returns ``True`` if the pane can be docked at the bottom of the managed frame. """ return self.HasFlag(self.optionBottomDockable) def IsLeftDockable(self): """ Returns ``True`` if the pane can be docked at the left of the managed frame. """ return self.HasFlag(self.optionLeftDockable) def IsRightDockable(self): """ Returns ``True`` if the pane can be docked at the right of the managed frame. """ return self.HasFlag(self.optionRightDockable) def IsDockable(self): """ Returns ``True`` if the pane can be docked. """ return self.IsTopDockable() or self.IsBottomDockable() or self.IsLeftDockable() or \ self.IsRightDockable() or self.IsNotebookDockable() def IsFloatable(self): """ Returns ``True`` if the pane can be undocked and displayed as a floating window. """ return self.HasFlag(self.optionFloatable) def IsMovable(self): """ Returns ``True`` if the docked frame can be undocked or moved to another dock position. """ return self.HasFlag(self.optionMovable) def IsDestroyOnClose(self): """ Returns ``True`` if the pane should be destroyed when it is closed. Normally a pane is simply hidden when the close button is clicked. Calling :meth:`~AuiPaneInfo.DestroyOnClose` with a ``True`` input parameter will cause the window to be destroyed when the user clicks the pane's close button. """ return self.HasFlag(self.optionDestroyOnClose) def IsNotebookDockable(self): """ Returns ``True`` if a pane can be docked on top to another to create a :class:`~lib.agw.aui.auibook.AuiNotebook`. """ return self.HasFlag(self.optionNotebookDockable) def IsTopSnappable(self): """ Returns ``True`` if the pane can be snapped at the top of the managed frame. """ return self.HasFlag(self.optionTopSnapped) def IsBottomSnappable(self): """ Returns ``True`` if the pane can be snapped at the bottom of the managed frame. """ return self.HasFlag(self.optionBottomSnapped) def IsLeftSnappable(self): """ Returns ``True`` if the pane can be snapped on the left of the managed frame. """ return self.HasFlag(self.optionLeftSnapped) def IsRightSnappable(self): """ Returns ``True`` if the pane can be snapped on the right of the managed frame. """ return self.HasFlag(self.optionRightSnapped) def IsSnappable(self): """ Returns ``True`` if the pane can be snapped. """ return self.IsTopSnappable() or self.IsBottomSnappable() or self.IsLeftSnappable() or \ self.IsRightSnappable() def IsFlyOut(self): """ Returns ``True`` if the floating pane has a "fly-out" effect. """ return self.HasFlag(self.optionFlyOut) def HasCaption(self): """ Returns ``True`` if the pane displays a caption. """ return self.HasFlag(self.optionCaption) def HasCaptionLeft(self): """ Returns ``True`` if the pane displays a caption on the left (rotated by 90 degrees). """ return self.HasFlag(self.optionCaptionLeft) def HasGripper(self): """ Returns ``True`` if the pane displays a gripper. """ return self.HasFlag(self.optionGripper) def HasBorder(self): """ Returns ``True`` if the pane displays a border. """ return self.HasFlag(self.optionPaneBorder) def HasCloseButton(self): """ Returns ``True`` if the pane displays a button to close the pane. """ return self.HasFlag(self.buttonClose) def HasMaximizeButton(self): """ Returns ``True`` if the pane displays a button to maximize the pane. """ return self.HasFlag(self.buttonMaximize) def HasMinimizeButton(self): """ Returns ``True`` if the pane displays a button to minimize the pane. """ return self.HasFlag(self.buttonMinimize) def GetMinimizeMode(self): """ Returns the minimization style for this pane. Possible return values are: ============================== ========= ============================== Minimize Mode Flag Hex Value Description ============================== ========= ============================== ``AUI_MINIMIZE_POS_SMART`` 0x01 Minimizes the pane on the closest tool bar ``AUI_MINIMIZE_POS_TOP`` 0x02 Minimizes the pane on the top tool bar ``AUI_MINIMIZE_POS_LEFT`` 0x03 Minimizes the pane on its left tool bar ``AUI_MINIMIZE_POS_RIGHT`` 0x04 Minimizes the pane on its right tool bar ``AUI_MINIMIZE_POS_BOTTOM`` 0x05 Minimizes the pane on its bottom tool bar ``AUI_MINIMIZE_POS_TOOLBAR`` 0x06 Minimizes the pane on a target :class:`~lib.agw.aui.auibar.AuiToolBar` ``AUI_MINIMIZE_POS_MASK`` 0x17 Mask to filter the position flags ``AUI_MINIMIZE_CAPT_HIDE`` 0x0 Hides the caption of the minimized pane ``AUI_MINIMIZE_CAPT_SMART`` 0x08 Displays the caption in the best rotation (horizontal or clockwise) ``AUI_MINIMIZE_CAPT_HORZ`` 0x10 Displays the caption horizontally ``AUI_MINIMIZE_CAPT_MASK`` 0x18 Mask to filter the caption flags ============================== ========= ============================== The flags can be filtered with the following masks: ============================== ========= ============================== Minimize Mask Flag Hex Value Description ============================== ========= ============================== ``AUI_MINIMIZE_POS_MASK`` 0x17 Filters the position flags ``AUI_MINIMIZE_CAPT_MASK`` 0x18 Filters the caption flags ============================== ========= ============================== """ return self.minimize_mode def HasPinButton(self): """ Returns ``True`` if the pane displays a button to float the pane. """ return self.HasFlag(self.buttonPin) def HasGripperTop(self): """ Returns ``True`` if the pane displays a gripper at the top. """ return self.HasFlag(self.optionGripperTop) def Window(self, w): """ Associate a :class:`Window` derived window to this pane. This normally does not need to be specified, as the window pointer is automatically assigned to the :class:`AuiPaneInfo` structure as soon as it is added to the manager. :param `w`: a :class:`Window` derived window. """ self.window = w return self def Name(self, name): """ Sets the name of the pane so it can be referenced in lookup functions. If a name is not specified by the user, a random name is assigned to the pane when it is added to the manager. :param `name`: a string specifying the pane name. .. warning:: If you are using :meth:`AuiManager.SavePerspective` and :meth:`AuiManager.LoadPerspective`, you will have to specify a name for your pane using :meth:`~AuiPaneInfo.Name`, as perspectives containing randomly generated names can not be properly restored. """ self.name = name return self def Caption(self, caption): """ Sets the caption of the pane. :param string `caption`: a string specifying the pane caption. """ self.caption = caption return self def Left(self): """ Sets the pane dock position to the left side of the frame. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_LEFT`` as parameter. """ self.dock_direction = AUI_DOCK_LEFT return self def Right(self): """ Sets the pane dock position to the right side of the frame. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_RIGHT`` as parameter. """ self.dock_direction = AUI_DOCK_RIGHT return self def Top(self): """ Sets the pane dock position to the top of the frame. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_TOP`` as parameter. """ self.dock_direction = AUI_DOCK_TOP return self def Bottom(self): """ Sets the pane dock position to the bottom of the frame. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_BOTTOM`` as parameter. """ self.dock_direction = AUI_DOCK_BOTTOM return self def Center(self): """ Sets the pane to the center position of the frame. The centre pane is the space in the middle after all border panes (left, top, right, bottom) are subtracted from the layout. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_CENTER`` as parameter. """ self.dock_direction = AUI_DOCK_CENTER return self def Centre(self): """ Sets the pane to the center position of the frame. The centre pane is the space in the middle after all border panes (left, top, right, bottom) are subtracted from the layout. :note: This is the same thing as calling :meth:`~AuiPaneInfo.Direction` with ``AUI_DOCK_CENTRE`` as parameter. """ self.dock_direction = AUI_DOCK_CENTRE return self def Direction(self, direction): """ Determines the direction of the docked pane. It is functionally the same as calling :meth:`Left`, :meth:`Right`, :meth:`Top` or :meth:`Bottom`, except that docking direction may be specified programmatically via the parameter `direction`. :param integer `direction`: the direction of the docked pane. :see: :meth:`dock_direction_set` for a list of valid docking directions. """ self.dock_direction = direction return self def Layer(self, layer): """ Determines the layer of the docked pane. The dock layer is similar to an onion, the inner-most layer being layer 0. Each shell moving in the outward direction has a higher layer number. This allows for more complex docking layout formation. :param integer `layer`: the layer of the docked pane. """ self.dock_layer = layer return self def Row(self, row): """ Determines the row of the docked pane. :param integer `row`: the row of the docked pane. """ self.dock_row = row return self def Position(self, pos): """ Determines the position of the docked pane. :param integer `pos`: the position of the docked pane. """ self.dock_pos = pos return self def MinSize(self, arg1=None, arg2=None): """ Sets the minimum size of the pane. This method is split in 2 versions depending on the input type. If `arg1` is a :class:`Size` object, then :meth:`~AuiPaneInfo.MinSize1` is called. Otherwise, :meth:`~AuiPaneInfo.MinSize2` is called. :param `arg1`: a :class:`Size` object, a (x, y) tuple or or a `x` coordinate. :param `arg2`: a `y` coordinate (only if `arg1` is a `x` coordinate, otherwise unused). """ if isinstance(arg1, wx.Size): ret = self.MinSize1(arg1) elif isinstance(arg1, tuple): ret = self.MinSize1(wx.Size(*arg1)) elif isinstance(arg1, six.integer_types) and arg2 is not None: ret = self.MinSize2(arg1, arg2) else: raise Exception("Invalid argument passed to `MinSize`: arg1=%s, arg2=%s"%(repr(arg1), repr(arg2))) return ret def MinSize1(self, size): """ Sets the minimum size of the pane. :see: :meth:`MinSize` for an explanation of input parameters. """ self.min_size = size return self def MinSize2(self, x, y): """ Sets the minimum size of the pane. :see: :meth:`MinSize` for an explanation of input parameters. """ self.min_size = wx.Size(x, y) return self def MaxSize(self, arg1=None, arg2=None): """ Sets the maximum size of the pane. This method is split in 2 versions depending on the input type. If `arg1` is a :class:`Size` object, then :meth:`~AuiPaneInfo.MaxSize1` is called. Otherwise, :meth:`~AuiPaneInfo.MaxSize2` is called. :param `arg1`: a :class:`Size` object, a (x, y) tuple or a `x` coordinate. :param `arg2`: a `y` coordinate (only if `arg1` is a `x` coordinate, otherwise unused). """ if isinstance(arg1, wx.Size): ret = self.MaxSize1(arg1) elif isinstance(arg1, tuple): ret = self.MaxSize1(wx.Size(*arg1)) elif isinstance(arg1, six.integer_types) and arg2 is not None: ret = self.MaxSize2(arg1, arg2) else: raise Exception("Invalid argument passed to `MaxSize`: arg1=%s, arg2=%s"%(repr(arg1), repr(arg2))) return ret def MaxSize1(self, size): """ Sets the maximum size of the pane. :see: :meth:`MaxSize` for an explanation of input parameters. """ self.max_size = size return self def MaxSize2(self, x, y): """ Sets the maximum size of the pane. :see: :meth:`MaxSize` for an explanation of input parameters. """ self.max_size.Set(x,y) return self def BestSize(self, arg1=None, arg2=None): """ Sets the ideal size for the pane. The docking manager will attempt to use this size as much as possible when docking or floating the pane. This method is split in 2 versions depending on the input type. If `arg1` is a :class:`Size` object, then :meth:`BestSize1` is called. Otherwise, :meth:`BestSize2` is called. :param `arg1`: a :class:`Size` object, a (x, y) tuple or a `x` coordinate. :param `arg2`: a `y` coordinate (only if `arg1` is a `x` coordinate, otherwise unused). """ if isinstance(arg1, wx.Size): ret = self.BestSize1(arg1) elif isinstance(arg1, tuple): ret = self.BestSize1(wx.Size(*arg1)) elif isinstance(arg1, six.integer_types) and arg2 is not None: ret = self.BestSize2(arg1, arg2) else: raise Exception("Invalid argument passed to `BestSize`: arg1=%s, arg2=%s"%(repr(arg1), repr(arg2))) return ret def BestSize1(self, size): """ Sets the best size of the pane. :see: :meth:`BestSize` for an explanation of input parameters. """ self.best_size = size return self def BestSize2(self, x, y): """ Sets the best size of the pane. :see: :meth:`BestSize` for an explanation of input parameters. """ self.best_size.Set(x,y) return self def FloatingPosition(self, pos): """ Sets the position of the floating pane. :param `pos`: a :class:`Point` or a tuple indicating the pane floating position. """ self.floating_pos = wx.Point(*pos) return self def FloatingSize(self, size): """ Sets the size of the floating pane. :param `size`: a :class:`Size` or a tuple indicating the pane floating size. """ self.floating_size = wx.Size(*size) return self def Maximize(self): """ Makes the pane take up the full area.""" return self.SetFlag(self.optionMaximized, True) def Minimize(self): """ Makes the pane minimized in a :class:`~lib.agw.aui.auibar.AuiToolBar`. Clicking on the minimize button causes a new :class:`~lib.agw.aui.auibar.AuiToolBar` to be created and added to the frame manager, (currently the implementation is such that panes at West will have a toolbar at the right, panes at South will have toolbars at the bottom etc...) and the pane is hidden in the manager. Clicking on the restore button on the newly created toolbar will result in the toolbar being removed and the original pane being restored. """ return self.SetFlag(self.optionMinimized, True) def MinimizeMode(self, mode): """ Sets the expected minimized mode if the minimize button is visible. :param integer `mode`: the minimized pane can have a specific position in the work space: ============================== ========= ============================== Minimize Mode Flag Hex Value Description ============================== ========= ============================== ``AUI_MINIMIZE_POS_SMART`` 0x01 Minimizes the pane on the closest tool bar ``AUI_MINIMIZE_POS_TOP`` 0x02 Minimizes the pane on the top tool bar ``AUI_MINIMIZE_POS_LEFT`` 0x03 Minimizes the pane on its left tool bar ``AUI_MINIMIZE_POS_RIGHT`` 0x04 Minimizes the pane on its right tool bar ``AUI_MINIMIZE_POS_BOTTOM`` 0x05 Minimizes the pane on its bottom tool bar ``AUI_MINIMIZE_POS_TOOLBAR`` 0x06 Minimizes the pane on a target :class:`~lib.agw.aui.auibar.AuiToolBar` ============================== ========= ============================== The caption of the minimized pane can be displayed in different modes: ============================== ========= ============================== Caption Mode Flag Hex Value Description ============================== ========= ============================== ``AUI_MINIMIZE_CAPT_HIDE`` 0x0 Hides the caption of the minimized pane ``AUI_MINIMIZE_CAPT_SMART`` 0x08 Displays the caption in the best rotation (horizontal in the top and in the bottom tool bar or clockwise in the right and in the left tool bar) ``AUI_MINIMIZE_CAPT_HORZ`` 0x10 Displays the caption horizontally ============================== ========= ============================== .. note:: In order to use the ``AUI_MINIMIZE_POS_TOOLBAR`` flag, the instance of :class:`AuiManager` you pass as an input for :meth:`MinimizeTarget` **must** have a real name and not the randomly generated one. Remember to set the :meth:`Name` property of the toolbar pane before calling this method. """ self.minimize_mode = mode return self def MinimizeTarget(self, toolbarPane): """ Minimizes the panes using a :class:`AuiPaneInfo` as a target. As :class:`AuiPaneInfo` properties need to be copied back and forth every time the perspective has changed, we only store the toobar **name**. :param `toolbarPane`: an instance of :class:`AuiPaneInfo`, containing a :class:`~lib.agw.aui.auibar.AuiToolBar`. .. note:: In order to use this functionality (and with the ``AUI_MINIMIZE_POS_TOOLBAR`` flag set), the instance of :class:`AuiPaneInfo` you pass as an input **must** have a real name and not the randomly generated one. Remember to set the :meth:`Name` property of the toolbar pane before calling this method. """ self.minimize_target = toolbarPane.name return self def Restore(self): """ Is the reverse of :meth:`Maximize` and :meth:`Minimize`.""" return self.SetFlag(self.optionMaximized or self.optionMinimized, False) def Fixed(self): """ Forces a pane to be fixed size so that it cannot be resized. After calling :meth:`Fixed`, :meth:`IsFixed` will return ``True``. """ return self.SetFlag(self.optionResizable, False) def Resizable(self, resizable=True): """ Allows a pane to be resizable if `resizable` is ``True``, and forces it to be a fixed size if `resizeable` is ``False``. If `resizable` is ``False``, this is simply an antonym for :meth:`Fixed`. :param bool `resizable`: whether the pane will be resizeable or not. """ return self.SetFlag(self.optionResizable, resizable) def Transparent(self, alpha): """ Makes the pane transparent when floating. :param integer `alpha`: a value between 0 and 255 for pane transparency. """ if alpha < 0 or alpha > 255: raise Exception("Invalid transparency value (%s)"%repr(alpha)) self.transparent = alpha self.needsTransparency = True def Dock(self): """ Indicates that a pane should be docked. It is the opposite of :meth:`Float`. """ if self.IsNotebookPage(): self.notebook_id = -1 self.dock_direction = AUI_DOCK_NONE return self.SetFlag(self.optionFloating, False) def Float(self): """ Indicates that a pane should be floated. It is the opposite of :meth:`Dock`. """ if self.IsNotebookPage(): self.notebook_id = -1 self.dock_direction = AUI_DOCK_NONE return self.SetFlag(self.optionFloating, True) def Hide(self): """ Indicates that a pane should be hidden. Calling :meth:`Show(False) ` achieve the same effect. """ return self.SetFlag(self.optionHidden, True) def Show(self, show=True): """ Indicates that a pane should be shown. :param bool `show`: whether the pane should be shown or not. """ return self.SetFlag(self.optionHidden, not show) # By defaulting to 1000, the tab will get placed at the end def NotebookPage(self, id, tab_position=1000): """ Forces a pane to be a notebook page, so that the pane can be docked on top to another to create a :class:`~lib.agw.aui.auibook.AuiNotebook`. :param integer `id`: the notebook id; :param integer `tab_position`: the tab number of the pane once docked in a notebook. """ # Remove any floating frame self.Dock() self.notebook_id = id self.dock_pos = tab_position self.dock_row = 0 self.dock_layer = 0 self.dock_direction = AUI_DOCK_NOTEBOOK_PAGE return self def NotebookControl(self, id): """ Forces a pane to be a notebook control (:class:`~lib.agw.aui.auibook.AuiNotebook`). :param integer `id`: the notebook id. """ self.notebook_id = id self.window = None self.buttons = [] if self.dock_direction == AUI_DOCK_NOTEBOOK_PAGE: self.dock_direction = AUI_DOCK_NONE return self def HasNotebook(self): """ Returns whether a pane has a :class:`~lib.agw.aui.auibook.AuiNotebook` or not. """ return self.notebook_id >= 0 def IsNotebookPage(self): """ Returns whether the pane is a notebook page in a :class:`~lib.agw.aui.auibook.AuiNotebook`. """ return self.notebook_id >= 0 and self.dock_direction == AUI_DOCK_NOTEBOOK_PAGE def IsNotebookControl(self): """ Returns whether the pane is a notebook control (:class:`~lib.agw.aui.auibook.AuiNotebook`). """ return not self.IsNotebookPage() and self.HasNotebook() def SetNameFromNotebookId(self): """ Sets the pane name once docked in a :class:`~lib.agw.aui.auibook.AuiNotebook` using the notebook id. """ if self.notebook_id >= 0: self.name = "__notebook_%d"%self.notebook_id return self def CaptionVisible(self, visible=True, left=False): """ Indicates that a pane caption should be visible. If `visible` is ``False``, no pane caption is drawn. :param bool `visible`: whether the caption should be visible or not; :param bool `left`: whether the caption should be drawn on the left (rotated by 90 degrees) or not. """ if left: self.SetFlag(self.optionCaption, False) return self.SetFlag(self.optionCaptionLeft, visible) self.SetFlag(self.optionCaptionLeft, False) return self.SetFlag(self.optionCaption, visible) def PaneBorder(self, visible=True): """ Indicates that a border should be drawn for the pane. :param bool `visible`: whether the pane border should be visible or not. """ return self.SetFlag(self.optionPaneBorder, visible) def Gripper(self, visible=True): """ Indicates that a gripper should be drawn for the pane. :param bool `visible`: whether the gripper should be visible or not. """ return self.SetFlag(self.optionGripper, visible) def GripperTop(self, attop=True): """ Indicates that a gripper should be drawn at the top of the pane. :param bool `attop`: whether the gripper should be drawn at the top or not. """ return self.SetFlag(self.optionGripperTop, attop) def CloseButton(self, visible=True): """ Indicates that a close button should be drawn for the pane. :param bool `visible`: whether the close button should be visible or not. """ return self.SetFlag(self.buttonClose, visible) def MaximizeButton(self, visible=True): """ Indicates that a maximize button should be drawn for the pane. :param bool `visible`: whether the maximize button should be visible or not. """ return self.SetFlag(self.buttonMaximize, visible) def MinimizeButton(self, visible=True): """ Indicates that a minimize button should be drawn for the pane. :param bool `visible`: whether the minimize button should be visible or not. """ return self.SetFlag(self.buttonMinimize, visible) def PinButton(self, visible=True): """ Indicates that a pin button should be drawn for the pane. :param bool `visible`: whether the pin button should be visible or not. """ return self.SetFlag(self.buttonPin, visible) def DestroyOnClose(self, b=True): """ Indicates whether a pane should be destroyed when it is closed. Normally a pane is simply hidden when the close button is clicked. Setting `b` to ``True`` will cause the window to be destroyed when the user clicks the pane's close button. :param bool `b`: whether the pane should be destroyed when it is closed or not. """ return self.SetFlag(self.optionDestroyOnClose, b) def TopDockable(self, b=True): """ Indicates whether a pane can be docked at the top of the frame. :param bool `b`: whether the pane can be docked at the top or not. """ return self.SetFlag(self.optionTopDockable, b) def BottomDockable(self, b=True): """ Indicates whether a pane can be docked at the bottom of the frame. :param bool `b`: whether the pane can be docked at the bottom or not. """ return self.SetFlag(self.optionBottomDockable, b) def LeftDockable(self, b=True): """ Indicates whether a pane can be docked on the left of the frame. :param bool `b`: whether the pane can be docked at the left or not. """ return self.SetFlag(self.optionLeftDockable, b) def RightDockable(self, b=True): """ Indicates whether a pane can be docked on the right of the frame. :param bool `b`: whether the pane can be docked at the right or not. """ return self.SetFlag(self.optionRightDockable, b) def Floatable(self, b=True): """ Sets whether the user will be able to undock a pane and turn it into a floating window. :param bool `b`: whether the pane can be floated or not. """ return self.SetFlag(self.optionFloatable, b) def Movable(self, b=True): """ Indicates whether a pane can be moved. :param bool `b`: whether the pane can be moved or not. """ return self.SetFlag(self.optionMovable, b) def NotebookDockable(self, b=True): """ Indicates whether a pane can be docked in an automatic :class:`~lib.agw.aui.auibook.AuiNotebook`. :param bool `b`: whether the pane can be docked in a notebook or not. """ return self.SetFlag(self.optionNotebookDockable, b) def DockFixed(self, b=True): """ Causes the containing dock to have no resize sash. This is useful for creating panes that span the entire width or height of a dock, but should not be resizable in the other direction. :param bool `b`: whether the pane will have a resize sash or not. """ return self.SetFlag(self.optionDockFixed, b) def Dockable(self, b=True): """ Specifies whether a frame can be docked or not. It is the same as specifying :meth:`TopDockable` . :meth:`BottomDockable` . :meth:`LeftDockable` . :meth:`RightDockable` . :param bool `b`: whether the frame can be docked or not. """ return self.TopDockable(b).BottomDockable(b).LeftDockable(b).RightDockable(b) def TopSnappable(self, b=True): """ Indicates whether a pane can be snapped at the top of the main frame. :param bool `b`: whether the pane can be snapped at the top of the main frame or not. """ return self.SetFlag(self.optionTopSnapped, b) def BottomSnappable(self, b=True): """ Indicates whether a pane can be snapped at the bottom of the main frame. :param bool `b`: whether the pane can be snapped at the bottom of the main frame or not. """ return self.SetFlag(self.optionBottomSnapped, b) def LeftSnappable(self, b=True): """ Indicates whether a pane can be snapped on the left of the main frame. :param bool `b`: whether the pane can be snapped at the left of the main frame or not. """ return self.SetFlag(self.optionLeftSnapped, b) def RightSnappable(self, b=True): """ Indicates whether a pane can be snapped on the right of the main frame. :param bool `b`: whether the pane can be snapped at the right of the main frame or not. """ return self.SetFlag(self.optionRightSnapped, b) def Snappable(self, b=True): """ Indicates whether a pane can be snapped on the main frame. This is equivalent as calling :meth:`TopSnappable` . :meth:`BottomSnappable` . :meth:`LeftSnappable` . :meth:`RightSnappable` . :param bool `b`: whether the pane can be snapped on the main frame or not. """ return self.TopSnappable(b).BottomSnappable(b).LeftSnappable(b).RightSnappable(b) def FlyOut(self, b=True): """ Indicates whether a pane, when floating, has a "fly-out" effect (i.e., floating panes which only show themselves when moused over). :param bool `b`: whether the pane can be snapped on the main frame or not. """ return self.SetFlag(self.optionFlyOut, b) # Copy over the members that pertain to docking position def SetDockPos(self, source): """ Copies the `source` pane members that pertain to docking position to `self`. :param `source`: the source pane from where to copy the attributes, an instance of :class:`AuiPaneInfo`. """ self.dock_direction = source.dock_direction self.dock_layer = source.dock_layer self.dock_row = source.dock_row self.dock_pos = source.dock_pos self.dock_proportion = source.dock_proportion self.floating_pos = wx.Point(*source.floating_pos) self.floating_size = wx.Size(*source.floating_size) self.rect = wx.Rect(*source.rect) return self def DefaultPane(self): """ Specifies that the pane should adopt the default pane settings. """ state = self.state state |= self.optionTopDockable | self.optionBottomDockable | \ self.optionLeftDockable | self.optionRightDockable | \ self.optionNotebookDockable | \ self.optionFloatable | self.optionMovable | self.optionResizable | \ self.optionCaption | self.optionPaneBorder | self.buttonClose self.state = state return self def CentrePane(self): """ Specifies that the pane should adopt the default center pane settings. Centre panes usually do not have caption bars. This function provides an easy way of preparing a pane to be displayed in the center dock position. """ return self.CenterPane() def CenterPane(self): """ Specifies that the pane should adopt the default center pane settings. Centre panes usually do not have caption bars. This function provides an easy way of preparing a pane to be displayed in the center dock position. """ self.state = 0 return self.Center().PaneBorder().Resizable() def ToolbarPane(self): """ Specifies that the pane should adopt the default toolbar pane settings. """ self.DefaultPane() state = self.state state |= (self.optionToolbar | self.optionGripper) state &= ~(self.optionResizable | self.optionCaption | self.optionCaptionLeft) if self.dock_layer == 0: self.dock_layer = 10 self.state = state return self def Icon(self, icon): """ Specifies whether an icon is drawn on the left of the caption text when the pane is docked. If `icon` is ``None`` or :class:`NullIcon`, no icon is drawn on the caption space. :param icon: an icon to draw on the caption space, or ``None``. :type `icon`: :class:`Icon` or ``None`` """ if icon is None: icon = wx.NullIcon self.icon = icon return self def SetFlag(self, flag, option_state): """ Turns the property given by `flag` on or off with the `option_state` parameter. :param integer `flag`: the property to set; :param bool `option_state`: either ``True`` or ``False``. """ state = self.state if option_state: state |= flag else: state &= ~flag self.state = state if flag in [self.buttonClose, self.buttonMaximize, self.buttonMinimize, self.buttonPin]: self.ResetButtons() return self def HasFlag(self, flag): """ Returns ``True`` if the the property specified by flag is active for the pane. :param integer `flag`: the property to check for activity. """ return (self.state & flag and [True] or [False])[0] def ResetButtons(self): """ Resets all the buttons and recreates them from scratch depending on the :class:`AuiManager` flags. """ floating = self.HasFlag(self.optionFloating) self.buttons = [] if not floating and self.HasMinimizeButton(): button = AuiPaneButton(AUI_BUTTON_MINIMIZE) self.buttons.append(button) if not floating and self.HasMaximizeButton(): button = AuiPaneButton(AUI_BUTTON_MAXIMIZE_RESTORE) self.buttons.append(button) if not floating and self.HasPinButton(): button = AuiPaneButton(AUI_BUTTON_PIN) self.buttons.append(button) if self.HasCloseButton(): button = AuiPaneButton(AUI_BUTTON_CLOSE) self.buttons.append(button) def CountButtons(self): """ Returns the number of visible buttons in the docked pane. """ n = 0 if self.HasCaption() or self.HasCaptionLeft(): if isinstance(wx.GetTopLevelParent(self.window), AuiFloatingFrame): return 1 if self.HasCloseButton(): n += 1 if self.HasMaximizeButton(): n += 1 if self.HasMinimizeButton(): n += 1 if self.HasPinButton(): n += 1 return n def IsHorizontal(self): """ Returns ``True`` if the pane `dock_direction` is horizontal. """ return self.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_BOTTOM] def IsVertical(self): """ Returns ``True`` if the pane `dock_direction` is vertical. """ return self.dock_direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT] # Null AuiPaneInfo reference NonePaneInfo = AuiPaneInfo() """ Null :class:`AuiPaneInfo` reference, an invalid instance of :class:`AuiPaneInfo`. """ # ---------------------------------------------------------------------------- # class AuiDockingGuide(wx.Frame): """ Base class for :class:`AuiSingleDockingGuide` and :class:`AuiCenterDockingGuide`.""" def __init__(self, parent, id=wx.ID_ANY, title="", pos=wx.DefaultPosition, size=wx.DefaultSize, style=wx.FRAME_TOOL_WINDOW | wx.STAY_ON_TOP | wx.FRAME_NO_TASKBAR | wx.NO_BORDER, name="AuiDockingGuide"): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent; :param integer `id`: the window identifier. It may take a value of -1 to indicate a default value. :param string `title`: the caption to be displayed on the frame's title bar. :param Point `pos`: the window position. A value of (-1, -1) indicates a default position, chosen by either the windowing system or wxPython, depending on platform. :param Size `size`: the window size. A value of (-1, -1) indicates a default size, chosen by either the windowing system or wxPython, depending on platform. :param integer `style`: the window style. :param string `name`: the name of the window. This parameter is used to associate a name with the item, allowing the application user to set Motif resource values for individual windows. """ wx.Frame.__init__(self, parent, id, title, pos, size, style, name=name) def HitTest(self, x, y): """ To be overridden by parent classes. :param integer `x`: the `x` mouse position; :param integer `y`: the `y` mouse position. """ return 0 def ValidateNotebookDocking(self, valid): """ To be overridden by parent classes. :param bool `valid`: whether a pane can be docked on top to another to form an automatic :class:`~lib.agw.aui.auibook.AuiNotebook`. """ return 0 # ============================================================================ # implementation # ============================================================================ # --------------------------------------------------------------------------- # AuiDockingGuideWindow # --------------------------------------------------------------------------- class AuiDockingGuideWindow(wx.Window): """ Target class for :class:`AuiDockingGuide` and :class:`AuiCenterDockingGuide`. """ def __init__(self, parent, rect, direction=0, center=False, useAero=False): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent; :param Rect `rect`: the window rect; :param integer `direction`: one of ``wx.TOP``, ``wx.BOTTOM``, ``wx.LEFT``, ``wx.RIGHT``, ``wx.CENTER``; :param bool `center`: whether the calling class is a :class:`AuiCenterDockingGuide`; :param bool `useAero`: whether to use the new Aero-style bitmaps or Whidbey-style bitmaps for the docking guide. """ wx.Window.__init__(self, parent, -1, rect.GetPosition(), rect.GetSize(), wx.NO_BORDER) self._direction = direction self._center = center self._valid = True self._useAero = useAero self._bmp_unfocus, self._bmp_focus = GetDockingImage(direction, useAero, center) self._currentImage = self._bmp_unfocus self.SetBackgroundStyle(wx.BG_STYLE_CUSTOM) self.Bind(wx.EVT_ERASE_BACKGROUND, self.OnEraseBackground) self.Bind(wx.EVT_PAINT, self.OnPaint) def SetValid(self, valid): """ Sets the docking direction as valid or invalid. :param bool `valid`: whether the docking direction is allowed or not. """ self._valid = valid def IsValid(self): """ Returns whether the docking direction is valid. """ return self._valid def OnEraseBackground(self, event): """ Handles the ``wx.EVT_ERASE_BACKGROUND`` event for :class:`AuiDockingGuideWindow`. :param `event`: a :class:`EraseEvent` to be processed. :note: This is intentionally empty to reduce flickering while drawing. """ pass def DrawBackground(self, dc): """ Draws the docking guide background. :param `dc`: a :class:`DC` device context object. """ rect = self.GetClientRect() dc.SetPen(wx.TRANSPARENT_PEN) dc.SetBrush(wx.Brush(colourTargetBackground)) dc.DrawRectangle(rect) dc.SetPen(wx.Pen(colourTargetBorder)) left = rect.GetLeft() top = rect.GetTop() right = rect.GetRight() bottom = rect.GetBottom() if self._direction != wx.CENTER: if not self._center or self._direction != wx.BOTTOM: dc.DrawLine(left, top, right+1, top) if not self._center or self._direction != wx.RIGHT: dc.DrawLine(left, top, left, bottom+1) if not self._center or self._direction != wx.LEFT: dc.DrawLine(right, top, right, bottom+1) if not self._center or self._direction != wx.TOP: dc.DrawLine(left, bottom, right+1, bottom) dc.SetPen(wx.Pen(colourTargetShade)) if self._direction != wx.RIGHT: dc.DrawLine(left + 1, top + 1, left + 1, bottom) if self._direction != wx.BOTTOM: dc.DrawLine(left + 1, top + 1, right, top + 1) def DrawDottedLine(self, dc, point, length, vertical): """ Draws a dotted line (not used if the docking guide images are ok). :param `dc`: a :class:`DC` device context object; :param `point`: a :class:`Point` where to start drawing the dotted line; :param integer `length`: the length of the dotted line; :param bool `vertical`: whether it is a vertical docking guide window or not. """ for i in range(0, length, 2): dc.DrawPoint(point.x, point.y) if vertical: point.y += 2 else: point.x += 2 def DrawIcon(self, dc): """ Draws the docking guide icon (not used if the docking guide images are ok). :param `dc`: a :class:`DC` device context object. """ rect = wx.Rect(*self.GetClientRect()) point = wx.Point() length = 0 rect.Deflate(4, 4) dc.SetPen(wx.Pen(colourIconBorder)) dc.SetBrush(wx.Brush(colourIconBackground)) dc.DrawRectangle(rect) right1 = rect.GetRight() + 1 bottom1 = rect.GetBottom() + 1 dc.SetPen(wx.Pen(colourIconShadow)) dc.DrawLine(rect.x + 1, bottom1, right1 + 1, bottom1) dc.DrawLine(right1, rect.y + 1, right1, bottom1 + 1) rect.Deflate(1, 1) if self._direction == wx.TOP: rect.height -= rect.height / 2 point = rect.GetBottomLeft() length = rect.width elif self._direction == wx.LEFT: rect.width -= rect.width / 2 point = rect.GetTopRight() length = rect.height elif self._direction == wx.RIGHT: rect.x += rect.width / 2 rect.width -= rect.width / 2 point = rect.GetTopLeft() length = rect.height elif self._direction == wx.BOTTOM: rect.y += rect.height / 2 rect.height -= rect.height / 2 point = rect.GetTopLeft() length = rect.width elif self._direction == wx.CENTER: rect.Deflate(1, 1) point = rect.GetTopLeft() length = rect.width dc.GradientFillLinear(rect, colourIconDockingPart1, colourIconDockingPart2, self._direction) dc.SetPen(wx.Pen(colourIconBorder)) if self._direction == wx.CENTER: self.DrawDottedLine(dc, rect.GetTopLeft(), rect.width, False) self.DrawDottedLine(dc, rect.GetTopLeft(), rect.height, True) self.DrawDottedLine(dc, rect.GetBottomLeft(), rect.width, False) self.DrawDottedLine(dc, rect.GetTopRight(), rect.height, True) elif self._direction in [wx.TOP, wx.BOTTOM]: self.DrawDottedLine(dc, point, length, False) else: self.DrawDottedLine(dc, point, length, True) def DrawArrow(self, dc): """ Draws the docking guide arrow icon (not used if the docking guide images are ok). :param `dc`: a :class:`DC` device context object. """ rect = self.GetClientRect() point = wx.Point() point.x = (rect.GetLeft() + rect.GetRight()) / 2 point.y = (rect.GetTop() + rect.GetBottom()) / 2 rx, ry = wx.Size(), wx.Size() if self._direction == wx.TOP: rx = wx.Size(1, 0) ry = wx.Size(0, 1) elif self._direction == wx.LEFT: rx = wx.Size(0, -1) ry = wx.Size(1, 0) elif self._direction == wx.RIGHT: rx = wx.Size(0, 1) ry = wx.Size(-1, 0) elif self._direction == wx.BOTTOM: rx = wx.Size(-1, 0) ry = wx.Size(0, -1) point.x += ry.x*3 point.y += ry.y*3 dc.SetPen(wx.Pen(colourIconArrow)) for i in range(4): pt1 = wx.Point(point.x - rx.x*i, point.y - rx.y*i) pt2 = wx.Point(point.x + rx.x*(i+1), point.y + rx.y*(i+1)) dc.DrawLine(pt1, pt2) point.x += ry.x point.y += ry.y def OnPaint(self, event): """ Handles the ``wx.EVT_PAINT`` event for :class:`AuiDockingGuideWindow`. :param `event`: a :class:`PaintEvent` to be processed. """ dc = wx.AutoBufferedPaintDC(self) if self._currentImage.IsOk() and self._valid: dc.DrawBitmap(self._currentImage, 0, 0, True) else: self.Draw(dc) def Draw(self, dc): """ Draws the whole docking guide window (not used if the docking guide images are ok). :param `dc`: a :class:`DC` device context object. """ self.DrawBackground(dc) if self._valid: self.DrawIcon(dc) self.DrawArrow(dc) def UpdateDockGuide(self, pos): """ Updates the docking guide images depending on the mouse position, using focused images if the mouse is inside the docking guide or unfocused images if it is outside. :param `pos`: a :class:`Point` mouse position. """ inside = self.GetScreenRect().Contains(pos) if inside: image = self._bmp_focus else: image = self._bmp_unfocus if image != self._currentImage: self._currentImage = image self.Refresh() self.Update() # --------------------------------------------------------------------------- # AuiSingleDockingGuide # --------------------------------------------------------------------------- class AuiSingleDockingGuide(AuiDockingGuide): """ A docking guide window for single docking hint (not diamond-shaped HUD). """ def __init__(self, parent, direction=0): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent; :param integer `direction`: one of ``wx.TOP``, ``wx.BOTTOM``, ``wx.LEFT``, ``wx.RIGHT``. """ self._direction = direction style = wx.FRAME_TOOL_WINDOW | wx.STAY_ON_TOP | \ wx.FRAME_NO_TASKBAR | wx.NO_BORDER # Use of FRAME_SHAPED on wxMac causes the frame to be visible # breaking the docking hints. if wx.Platform != '__WXMAC__': style |= wx.FRAME_SHAPED AuiDockingGuide.__init__(self, parent, style=style, name="auiSingleDockTarget") self.Hide() useAero = GetManager(self.GetParent()).GetAGWFlags() & AUI_MGR_AERO_DOCKING_GUIDES useWhidbey = GetManager(self.GetParent()).GetAGWFlags() & AUI_MGR_WHIDBEY_DOCKING_GUIDES self._useAero = useAero or useWhidbey self._valid = True if useAero: sizeX, sizeY = aeroguideSizeX, aeroguideSizeY elif useWhidbey: sizeX, sizeY = whidbeySizeX, whidbeySizeY else: sizeX, sizeY = guideSizeX, guideSizeY if direction not in [wx.TOP, wx.BOTTOM]: sizeX, sizeY = sizeY, sizeX if self._useAero: self.CreateShapesWithStyle(useWhidbey) if wx.Platform == "__WXGTK__": self.Bind(wx.EVT_WINDOW_CREATE, self.SetGuideShape) else: self.SetGuideShape() self.SetSize(self.region.GetBox().GetSize()) else: self.SetSize((sizeX, sizeY)) self.rect = wx.Rect(0, 0, sizeX, sizeY) if self._useAero: useAero = (useWhidbey and [2] or [1])[0] else: useAero = 0 self.target = AuiDockingGuideWindow(self, self.rect, direction, False, useAero) def CreateShapesWithStyle(self, useWhidbey): """ Creates the docking guide window shape based on which docking bitmaps are used. :param bool `useWhidbey`: if ``True``, use Whidbey-style bitmaps; if ``False``, use the Aero-style bitmaps. """ sizeX, sizeY = aeroguideSizeX, aeroguideSizeY if useWhidbey: sizeX, sizeY = whidbeySizeX, whidbeySizeY if self._direction not in [wx.TOP, wx.BOTTOM]: sizeX, sizeY = sizeY, sizeX useAero = (useWhidbey and [2] or [1])[0] bmp, dummy = GetDockingImage(self._direction, useAero, False) region = wx.Region(bmp) self.region = region def AeroMove(self, pos): """ Moves the docking window to the new position. Overridden in children classes. :param Point `pos`: the new docking guide position. """ pass def SetGuideShape(self, event=None): """ Sets the correct shape for the docking guide window. :param `event`: on wxGTK, a :class:`WindowCreateEvent` event to process. """ self.SetShape(self.region) if event is not None: # Skip the event on wxGTK event.Skip() wx.CallAfter(wx.SafeYield, self, True) def SetShape(self, region): """ If the platform supports it, sets the shape of the window to that depicted by `region`. The system will not display or respond to any mouse event for the pixels that lie outside of the region. To reset the window to the normal rectangular shape simply call :meth:`SetShape` again with an empty region. :param Region `region`: the shape of the frame. :note: Overridden for wxMAC. """ if wx.Platform == '__WXMAC__': # HACK so we don't crash when SetShape is called return else: super(AuiSingleDockingGuide, self).SetShape(region) def SetValid(self, valid): """ Sets the docking direction as valid or invalid. :param bool `valid`: whether the docking direction is allowed or not. """ self._valid = valid def IsValid(self): """ Returns whether the docking direction is valid. """ return self._valid def UpdateDockGuide(self, pos): """ Updates the docking guide images depending on the mouse position, using focused images if the mouse is inside the docking guide or unfocused images if it is outside. :param Point `pos`: the mouse position. """ self.target.UpdateDockGuide(pos) def HitTest(self, x, y): """ Checks if the mouse position is inside the target window rect. :param integer `x`: the `x` mouse position; :param integer `y`: the `y` mouse position. """ if self.target.GetScreenRect().Contains((x, y)): return wx.ALL return -1 # --------------------------------------------------------------------------- # AuiCenterDockingGuide # --------------------------------------------------------------------------- class AuiCenterDockingGuide(AuiDockingGuide): """ A docking guide window for multiple docking hint (diamond-shaped HUD). """ def __init__(self, parent): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent. """ AuiDockingGuide.__init__(self, parent, style=wx.FRAME_TOOL_WINDOW | wx.STAY_ON_TOP | wx.FRAME_NO_TASKBAR | wx.NO_BORDER | wx.FRAME_SHAPED, name="auiCenterDockTarget") self.Hide() self.CreateShapesWithStyle() self.SetBackgroundStyle(wx.BG_STYLE_CUSTOM) if wx.Platform == "__WXGTK__": self.Bind(wx.EVT_WINDOW_CREATE, self.SetGuideShape) else: self.SetGuideShape() self.SetSize(self.region.GetBox().GetSize()) self.Bind(wx.EVT_ERASE_BACKGROUND, self.OnEraseBackground) self.Bind(wx.EVT_PAINT, self.OnPaint) def CreateShapesWithStyle(self): """ Creates the docking guide window shape based on which docking bitmaps are used. """ useAero = (GetManager(self.GetParent()).GetAGWFlags() & AUI_MGR_AERO_DOCKING_GUIDES) != 0 useWhidbey = (GetManager(self.GetParent()).GetAGWFlags() & AUI_MGR_WHIDBEY_DOCKING_GUIDES) != 0 self._useAero = 0 if useAero: self._useAero = 1 elif useWhidbey: self._useAero = 2 if useAero: sizeX, sizeY = aeroguideSizeX, aeroguideSizeY elif useWhidbey: sizeX, sizeY = whidbeySizeX, whidbeySizeY else: sizeX, sizeY = guideSizeX, guideSizeY rectLeft = wx.Rect(0, sizeY, sizeY, sizeX) rectTop = wx.Rect(sizeY, 0, sizeX, sizeY) rectRight = wx.Rect(sizeY+sizeX, sizeY, sizeY, sizeX) rectBottom = wx.Rect(sizeY, sizeX + sizeY, sizeX, sizeY) rectCenter = wx.Rect(sizeY, sizeY, sizeX, sizeX) if not self._useAero: self.targetLeft = AuiDockingGuideWindow(self, rectLeft, wx.LEFT, True, useAero) self.targetTop = AuiDockingGuideWindow(self, rectTop, wx.TOP, True, useAero) self.targetRight = AuiDockingGuideWindow(self, rectRight, wx.RIGHT, True, useAero) self.targetBottom = AuiDockingGuideWindow(self, rectBottom, wx.BOTTOM, True, useAero) self.targetCenter = AuiDockingGuideWindow(self, rectCenter, wx.CENTER, True, useAero) # top-left diamond tld = [wx.Point(rectTop.x, rectTop.y+rectTop.height-8), wx.Point(rectLeft.x+rectLeft.width-8, rectLeft.y), rectTop.GetBottomLeft()] # bottom-left diamond bld = [wx.Point(rectLeft.x+rectLeft.width-8, rectLeft.y+rectLeft.height), wx.Point(rectBottom.x, rectBottom.y+8), rectBottom.GetTopLeft()] # top-right diamond trd = [wx.Point(rectTop.x+rectTop.width, rectTop.y+rectTop.height-8), wx.Point(rectRight.x+8, rectRight.y), rectRight.GetTopLeft()] # bottom-right diamond brd = [wx.Point(rectRight.x+8, rectRight.y+rectRight.height), wx.Point(rectBottom.x+rectBottom.width, rectBottom.y+8), rectBottom.GetTopRight()] self._triangles = [tld[0:2], bld[0:2], [wx.Point(rectTop.x+rectTop.width-1, rectTop.y+rectTop.height-8), wx.Point(rectRight.x+7, rectRight.y)], [wx.Point(rectRight.x+7, rectRight.y+rectRight.height), wx.Point(rectBottom.x+rectBottom.width-1, rectBottom.y+8)]] region = wx.Region() region.Union(rectLeft) region.Union(rectTop) region.Union(rectRight) region.Union(rectBottom) region.Union(rectCenter) region.Union(wx.Region(tld)) region.Union(wx.Region(bld)) region.Union(wx.Region(trd)) region.Union(wx.Region(brd)) elif useAero: self._aeroBmp = aero_dock_pane.GetBitmap() region = wx.Region(self._aeroBmp) self._allAeroBmps = [aero_dock_pane_left.GetBitmap(), aero_dock_pane_top.GetBitmap(), aero_dock_pane_right.GetBitmap(), aero_dock_pane_bottom.GetBitmap(), aero_dock_pane_center.GetBitmap(), aero_dock_pane.GetBitmap()] self._deniedBitmap = aero_denied.GetBitmap() self._aeroRects = [rectLeft, rectTop, rectRight, rectBottom, rectCenter] self._valid = True elif useWhidbey: self._aeroBmp = whidbey_dock_pane.GetBitmap() region = wx.Region(self._aeroBmp) self._allAeroBmps = [whidbey_dock_pane_left.GetBitmap(), whidbey_dock_pane_top.GetBitmap(), whidbey_dock_pane_right.GetBitmap(), whidbey_dock_pane_bottom.GetBitmap(), whidbey_dock_pane_center.GetBitmap(), whidbey_dock_pane.GetBitmap()] self._deniedBitmap = whidbey_denied.GetBitmap() self._aeroRects = [rectLeft, rectTop, rectRight, rectBottom, rectCenter] self._valid = True self.region = region def SetGuideShape(self, event=None): """ Sets the correct shape for the docking guide window. :param `event`: on wxGTK, a :class:`WindowCreateEvent` event to process. """ self.SetShape(self.region) if event is not None: # Skip the event on wxGTK event.Skip() wx.CallAfter(wx.SafeYield, self, True) def UpdateDockGuide(self, pos): """ Updates the docking guides images depending on the mouse position, using focused images if the mouse is inside the docking guide or unfocused images if it is outside. :param Point `pos`: the mouse position. """ if not self._useAero: for target in self.GetChildren(): target.UpdateDockGuide(pos) else: lenRects = len(self._aeroRects) for indx, rect in enumerate(self._aeroRects): if rect.Contains(pos): if self._allAeroBmps[indx] != self._aeroBmp: if indx < lenRects - 1 or (indx == lenRects - 1 and self._valid): self._aeroBmp = self._allAeroBmps[indx] self.Refresh() else: self._aeroBmp = self._allAeroBmps[-1] self.Refresh() return if self._aeroBmp != self._allAeroBmps[-1]: self._aeroBmp = self._allAeroBmps[-1] self.Refresh() def HitTest(self, x, y): """ Checks if the mouse position is inside the target windows rect. :param integer `x`: the `x` mouse position; :param integer `y`: the `y` mouse position. """ if not self._useAero: if self.targetLeft.GetScreenRect().Contains((x, y)): return wx.LEFT if self.targetTop.GetScreenRect().Contains((x, y)): return wx.UP if self.targetRight.GetScreenRect().Contains((x, y)): return wx.RIGHT if self.targetBottom.GetScreenRect().Contains((x, y)): return wx.DOWN if self.targetCenter.IsValid() and self.targetCenter.GetScreenRect().Contains((x, y)): return wx.CENTER else: constants = [wx.LEFT, wx.UP, wx.RIGHT, wx.DOWN, wx.CENTER] lenRects = len(self._aeroRects) for indx, rect in enumerate(self._aeroRects): if rect.Contains((x, y)): if indx < lenRects or (indx == lenRects-1 and self._valid): return constants[indx] return -1 def ValidateNotebookDocking(self, valid): """ Sets whether a pane can be docked on top of another to create an automatic :class:`~lib.agw.aui.auibook.AuiNotebook`. :param bool `valid`: whether a pane can be docked on top to another to form an automatic :class:`~lib.agw.aui.auibook.AuiNotebook`. """ if not self._useAero: if self.targetCenter.IsValid() != valid: self.targetCenter.SetValid(valid) self.targetCenter.Refresh() else: if self._valid != valid: self._valid = valid self.Refresh() def AeroMove(self, pos): """ Moves the docking guide window to the new position. :param Point `pos`: the new docking guide position. """ if not self._useAero: return useWhidbey = (GetManager(self.GetParent()).GetAGWFlags() & AUI_MGR_WHIDBEY_DOCKING_GUIDES) != 0 if useWhidbey: sizeX, sizeY = whidbeySizeX, whidbeySizeY else: sizeX, sizeY = aeroguideSizeX, aeroguideSizeY size = self.GetSize() leftRect, topRect, rightRect, bottomRect, centerRect = self._aeroRects thePos = pos + wx.Point((size.x-sizeY)/2, (size.y-sizeX)/2) centerRect.SetTopLeft(thePos) leftRect.SetTopLeft(thePos + wx.Point(-sizeY, 0)) topRect.SetTopLeft(thePos + wx.Point(0, -sizeY)) rightRect.SetTopLeft(thePos + wx.Point(sizeX, 0)) bottomRect.SetTopLeft(thePos + wx.Point(0, sizeX)) def OnEraseBackground(self, event): """ Handles the ``wx.EVT_ERASE_BACKGROUND`` event for :class:`AuiCenterDockingGuide`. :param `event`: :class:`EraseEvent` to be processed. :note: This is intentionally empty to reduce flickering while drawing. """ pass def OnPaint(self, event): """ Handles the ``wx.EVT_PAINT`` event for :class:`AuiCenterDockingGuide`. :param `event`: a :class:`PaintEvent` to be processed. """ dc = wx.AutoBufferedPaintDC(self) if self._useAero: dc.SetBrush(wx.TRANSPARENT_BRUSH) dc.SetPen(wx.TRANSPARENT_PEN) else: dc.SetBrush(wx.Brush(colourTargetBackground)) dc.SetPen(wx.Pen(colourTargetBorder)) rect = self.GetClientRect() dc.DrawRectangle(rect.x, rect.y, rect.width, rect.height) if self._useAero: dc.DrawBitmap(self._aeroBmp, 0, 0, True) if not self._valid: diff = (self._useAero == 2 and [1] or [0])[0] bmpX, bmpY = self._deniedBitmap.GetWidth(), self._deniedBitmap.GetHeight() xPos, yPos = (rect.x + (rect.width)/2 - bmpX/2), (rect.y + (rect.height)/2 - bmpY/2) dc.DrawBitmap(self._deniedBitmap, xPos+1, yPos+diff, True) return dc.SetPen(wx.Pen(colourTargetBorder, 2)) for pts in self._triangles: dc.DrawLine(pts[0], pts[1]) # ---------------------------------------------------------------------------- # AuiDockingHintWindow # ---------------------------------------------------------------------------- class AuiDockingHintWindow(wx.Frame): """ The original wxAUI docking window hint. """ def __init__(self, parent, id=wx.ID_ANY, title="", pos=wx.DefaultPosition, size=wx.Size(1, 1), style=wx.FRAME_TOOL_WINDOW | wx.FRAME_FLOAT_ON_PARENT | wx.FRAME_NO_TASKBAR | wx.NO_BORDER | wx.FRAME_SHAPED, name="auiHintWindow"): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent; :param integer `id`: the window identifier. It may take a value of -1 to indicate a default value. :param string `title`: the caption to be displayed on the frame's title bar; :param Point `pos`: the window position. A value of (-1, -1) indicates a default position, chosen by either the windowing system or wxPython, depending on platform; :param Size `size`: the window size. A value of (-1, -1) indicates a default size, chosen by either the windowing system or wxPython, depending on platform; :param integer `style`: the window style; :param string `name`: the name of the window. This parameter is used to associate a name with the item, allowing the application user to set Motif resource values for individual windows. """ if wx.Platform == '__WXMAC__' and style & wx.FRAME_SHAPED: # Having the shaped frame causes the frame to not be visible # with the transparent style hints. style -= wx.FRAME_SHAPED wx.Frame.__init__(self, parent, id, title, pos, size, style, name=name) self._blindMode = False self._art = parent.GetEventHandler().GetArtProvider() background = self._art.GetColour(AUI_DOCKART_HINT_WINDOW_COLOUR) self.SetBackgroundColour(background) # Can't set background colour on a frame on wxMac # so add a panel to set the colour on. if wx.Platform == '__WXMAC__': sizer = wx.BoxSizer(wx.HORIZONTAL) self.panel = wx.Panel(self) sizer.Add(self.panel, 1, wx.EXPAND) self.SetSizer(sizer) self.panel.SetBackgroundColour(background) else: self.Bind(wx.EVT_PAINT, self.OnPaint) self.Bind(wx.EVT_SIZE, self.OnSize) def MakeVenetianBlinds(self): """ Creates the "venetian blind" effect if :class:`AuiManager` has the ``AUI_MGR_VENETIAN_BLINDS_HINT`` flag set. """ amount = 128 size = self.GetClientSize() region = wx.Region(0, 0, size.x, 1) for y in range(size.y): # Reverse the order of the bottom 4 bits j = (y & 8 and [1] or [0])[0] | (y & 4 and [2] or [0])[0] | \ (y & 2 and [4] or [0])[0] | (y & 1 and [8] or [0])[0] if 16*j+8 < amount: region.Union(0, y, size.x, 1) self.SetShape(region) def SetBlindMode(self, agwFlags): """ Sets whether venetian blinds or transparent hints will be shown as docking hint. This depends on the :class:`AuiManager` flags. :param integer `agwFlags`: the :class:`AuiManager` flags. """ self._blindMode = (agwFlags & AUI_MGR_VENETIAN_BLINDS_HINT) != 0 if self._blindMode or not self.CanSetTransparent(): self.MakeVenetianBlinds() self.SetTransparent(255) else: self.SetShape(wx.Region()) if agwFlags & AUI_MGR_HINT_FADE == 0: self.SetTransparent(80) else: self.SetTransparent(0) def SetShape(self, region): """ If the platform supports it, sets the shape of the window to that depicted by `region`. The system will not display or respond to any mouse event for the pixels that lie outside of the region. To reset the window to the normal rectangular shape simply call :meth:`SetShape` again with an empty region. :param Region `region`: the shape of the frame. :note: Overridden for wxMAC. """ if wx.Platform == '__WXMAC__': # HACK so we don't crash when SetShape is called return else: super(AuiDockingHintWindow, self).SetShape(region) def Show(self, show=True): """ Show the hint window. :param bool `show`: whether to show or hide the hint docking window. """ background = self._art.GetColour(AUI_DOCKART_HINT_WINDOW_COLOUR) if wx.Platform == '__WXMAC__': self.panel.SetBackgroundColour(background) else: self.SetBackgroundColour(background) super(AuiDockingHintWindow, self).Show(show) self.Refresh() if wx.Platform == '__WXMAC__': # Need to manually do layout since its a borderless frame. self.Layout() def OnSize(self, event): """ Handles the ``wx.EVT_SIZE`` event for :class:`AuiDockingHintWindow`. :param `event`: a :class:`SizeEvent` to be processed. """ if self._blindMode or not self.CanSetTransparent(): self.MakeVenetianBlinds() self.Refresh() def OnPaint(self, event): """ Handles the ``wx.EVT_PAINT`` event for :class:`AuiDockingHintWindow`. :param `event`: an instance of :class:`PaintEvent` to be processed. """ rect = wx.Rect(wx.Point(0, 0), self.GetSize()) dc = wx.PaintDC(self) event.Skip() dc.SetBrush(wx.TRANSPARENT_BRUSH) dc.SetPen(wx.Pen(wx.Colour(60, 60, 60), 5)) rect.Deflate(1, 1) dc.DrawRectangle(rect) # ---------------------------------------------------------------------------- # # -- AuiFloatingFrame class implementation -- class AuiFloatingFrame(wx.MiniFrame): """ AuiFloatingFrame is the frame class that holds floating panes. """ def __init__(self, parent, owner_mgr, pane=None, id=wx.ID_ANY, title="", style=wx.FRAME_TOOL_WINDOW | wx.FRAME_FLOAT_ON_PARENT | wx.FRAME_NO_TASKBAR | wx.CLIP_CHILDREN): """ Default class constructor. Used internally, do not call it in your code! :param `parent`: the :class:`AuiManager` parent; :param `owner_mgr`: the :class:`AuiManager` that manages the floating pane; :param `pane`: the :class:`AuiPaneInfo` pane that is about to float; :param integer `id`: the window identifier. It may take a value of -1 to indicate a default value. :param string `title`: the caption to be displayed on the frame's title bar. :param integer `style`: the window style. """ if pane and pane.IsResizeable(): style += wx.RESIZE_BORDER if pane: self._is_toolbar = pane.IsToolbar() self._useNativeMiniframes = False if AuiManager_UseNativeMiniframes(owner_mgr): # On wxMac we always use native miniframes self._useNativeMiniframes = True style += wx.CAPTION + wx.SYSTEM_MENU if pane.HasCloseButton(): style += wx.CLOSE_BOX if pane.HasMaximizeButton(): style += wx.MAXIMIZE_BOX if pane.HasMinimizeButton(): style += wx.MINIMIZE_BOX wx.MiniFrame.__init__(self, parent, id, title, pos=pane.floating_pos, size=pane.floating_size, style=style, name="auiFloatingFrame") self._fly_timer = wx.Timer(self, wx.ID_ANY) self._check_fly_timer = wx.Timer(self, wx.ID_ANY) self.Bind(wx.EVT_CLOSE, self.OnClose) self.Bind(wx.EVT_SIZE, self.OnSize) self.Bind(wx.EVT_ACTIVATE, self.OnActivate) self.Bind(wx.EVT_TIMER, self.OnCheckFlyTimer, self._check_fly_timer) self.Bind(wx.EVT_TIMER, self.OnFlyTimer, self._fly_timer) self.Bind(EVT_AUI_FIND_MANAGER, self.OnFindManager) if self._useNativeMiniframes: self.Bind(wx.EVT_MOVE, self.OnMoveEvent) self.Bind(wx.EVT_MOVING, self.OnMoveEvent) self.Bind(wx.EVT_IDLE, self.OnIdle) self._useNativeMiniframes = True self.SetExtraStyle(wx.WS_EX_PROCESS_IDLE) else: self.Bind(wx.EVT_MOVE, self.OnMove) self._fly = False self._send_size = True self._alpha_amount = 255 self._owner_mgr = owner_mgr self._moving = False self._lastDirection = None self._transparent = 255 self._last_rect = wx.Rect() self._last2_rect = wx.Rect() self._last3_rect = wx.Rect() self._mgr = AuiManager() self._mgr.SetManagedWindow(self) self._mgr.SetArtProvider(owner_mgr.GetArtProvider()) self._mgr.SetAGWFlags(owner_mgr.GetAGWFlags()) def CopyAttributes(self, pane): """ Copies all the attributes of the input `pane` into another :class:`AuiPaneInfo`. :param `pane`: the source :class:`AuiPaneInfo` from where to copy attributes. """ contained_pane = AuiPaneInfo() contained_pane.name = pane.name contained_pane.caption = pane.caption contained_pane.window = pane.window contained_pane.frame = pane.frame contained_pane.state = pane.state contained_pane.dock_direction = pane.dock_direction contained_pane.dock_layer = pane.dock_layer contained_pane.dock_row = pane.dock_row contained_pane.dock_pos = pane.dock_pos contained_pane.best_size = wx.Size(*pane.best_size) contained_pane.min_size = wx.Size(*pane.min_size) contained_pane.max_size = wx.Size(*pane.max_size) contained_pane.floating_pos = wx.Point(*pane.floating_pos) contained_pane.floating_size = wx.Size(*pane.floating_size) contained_pane.dock_proportion = pane.dock_proportion contained_pane.buttons = pane.buttons contained_pane.rect = wx.Rect(*pane.rect) contained_pane.icon = pane.icon contained_pane.notebook_id = pane.notebook_id contained_pane.transparent = pane.transparent contained_pane.snapped = pane.snapped contained_pane.minimize_mode = pane.minimize_mode contained_pane.minimize_target = pane.minimize_target return contained_pane def SetPaneWindow(self, pane): """ Sets all the properties of a pane. :param `pane`: the :class:`AuiPaneInfo` to analyze. """ self._is_toolbar = pane.IsToolbar() self._pane_window = pane.window if isinstance(pane.window, auibar.AuiToolBar): pane.window.SetAuiManager(self._mgr) self._pane_window.Reparent(self) contained_pane = self.CopyAttributes(pane) contained_pane.Dock().Center().Show(). \ CaptionVisible(False). \ PaneBorder(False). \ Layer(0).Row(0).Position(0) if not contained_pane.HasGripper() and not self._useNativeMiniframes: contained_pane.CaptionVisible(True) indx = self._owner_mgr._panes.index(pane) # Carry over the minimum size pane_min_size = pane.window.GetMinSize() # if the best size is smaller than the min size # then set the min size to the best size as well pane_best_size = contained_pane.best_size if pane_best_size.IsFullySpecified() and (pane_best_size.x < pane_min_size.x or \ pane_best_size.y < pane_min_size.y): pane_min_size = pane_best_size self._pane_window.SetMinSize(pane_min_size) # if the frame window's max size is greater than the min size # then set the max size to the min size as well cur_max_size = self.GetMaxSize() if cur_max_size.IsFullySpecified() and (cur_max_size.x < pane_min_size.x or \ cur_max_size.y < pane_min_size.y): self.SetMaxSize(pane_min_size) art_provider = self._mgr.GetArtProvider() caption_size = art_provider.GetMetric(AUI_DOCKART_CAPTION_SIZE) button_size = art_provider.GetMetric(AUI_DOCKART_PANE_BUTTON_SIZE) + \ 4*art_provider.GetMetric(AUI_DOCKART_PANE_BORDER_SIZE) min_size = pane.window.GetMinSize() if min_size.y < caption_size or min_size.x < button_size: new_x, new_y = min_size.x, min_size.y if min_size.y < caption_size: new_y = (pane.IsResizeable() and [2*wx.SystemSettings.GetMetric(wx.SYS_EDGE_Y)+caption_size] or [1])[0] if min_size.x < button_size: new_x = (pane.IsResizeable() and [2*wx.SystemSettings.GetMetric(wx.SYS_EDGE_X)+button_size] or [1])[0] self.SetMinSize((new_x, new_y)) else: self.SetMinSize(min_size) self._mgr.AddPane(self._pane_window, contained_pane) self._mgr.Update() if pane.min_size.IsFullySpecified(): # because SetSizeHints() calls Fit() too (which sets the window # size to its minimum allowed), we keep the size before calling # SetSizeHints() and reset it afterwards... tmp = self.GetSize() self.GetSizer().SetSizeHints(self) self.SetSize(tmp) self.SetTitle(pane.caption) if pane.floating_size != wx.Size(-1, -1): self.SetSize(pane.floating_size) else: size = pane.best_size if size == wx.Size(-1, -1): size = pane.min_size if size == wx.Size(-1, -1): size = self._pane_window.GetSize() if self._owner_mgr and pane.HasGripper(): if pane.HasGripperTop(): size.y += self._owner_mgr._art.GetMetric(AUI_DOCKART_GRIPPER_SIZE) else: size.x += self._owner_mgr._art.GetMetric(AUI_DOCKART_GRIPPER_SIZE) if not self._useNativeMiniframes: size.y += self._owner_mgr._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) pane.floating_size = size self.SetClientSize(size) self._owner_mgr._panes[indx] = pane self._fly_step = abs(pane.floating_size.y - \ (caption_size + 2*wx.SystemSettings.GetMetric(wx.SYS_EDGE_Y)))/10 self._floating_size = wx.Size(*self.GetSize()) if pane.IsFlyOut(): self._check_fly_timer.Start(50) def GetOwnerManager(self): """ Returns the :class:`AuiManager` that manages the pane. """ return self._owner_mgr def OnSize(self, event): """ Handles the ``wx.EVT_SIZE`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`SizeEvent` to be processed. """ if self._owner_mgr and self._send_size: self._owner_mgr.OnFloatingPaneResized(self._pane_window, event.GetSize()) def OnClose(self, event): """ Handles the ``wx.EVT_CLOSE`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`CloseEvent` to be processed. """ if self._owner_mgr: self._owner_mgr.OnFloatingPaneClosed(self._pane_window, event) if not event.GetVeto(): self._mgr.DetachPane(self._pane_window) if isinstance(self._pane_window, auibar.AuiToolBar): self._pane_window.SetAuiManager(self._owner_mgr) # if we do not do this, then we can crash... if self._owner_mgr and self._owner_mgr._action_window == self: self._owner_mgr._action_window = None self._mgr.UnInit() self.Destroy() def OnActivate(self, event): """ Handles the ``wx.EVT_ACTIVATE`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`ActivateEvent` to be processed. """ if self._owner_mgr and event.GetActive(): self._owner_mgr.OnFloatingPaneActivated(self._pane_window) def OnMove(self, event): """ Handles the ``wx.EVT_MOVE`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`MoveEvent` to be processed. .. note:: This event is not processed on wxMAC or if :class:`AuiManager` is not using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ if self._owner_mgr: self._owner_mgr.OnFloatingPaneMoved(self._pane_window, event) def OnMoveEvent(self, event): """ Handles the ``wx.EVT_MOVE`` and ``wx.EVT_MOVING`` events for :class:`AuiFloatingFrame`. :param `event`: a :class:`MoveEvent` to be processed. .. note:: This event is only processed on wxMAC or if :class:`AuiManager` is using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ win_rect = self.GetRect() if win_rect == self._last_rect: return # skip the first move event if self._last_rect.IsEmpty(): self._last_rect = wx.Rect(*win_rect) return # As on OSX moving windows are not getting all move events, only sporadically, this difference # is almost always big on OSX, so avoid this early exit opportunity if wx.Platform != '__WXMAC__': # skip if moving too fast to avoid massive redraws and # jumping hint windows if abs(win_rect.x - self._last_rect.x) > 3 or abs(win_rect.y - self._last_rect.y) > 3: self._last3_rect = wx.Rect(*self._last2_rect) self._last2_rect = wx.Rect(*self._last_rect) self._last_rect = wx.Rect(*win_rect) # However still update the internally stored position to avoid # snapping back to the old one later. if self._owner_mgr: self._owner_mgr.GetPane(self._pane_window).floating_pos = win_rect.GetPosition() return # prevent frame redocking during resize if self._last_rect.GetSize() != win_rect.GetSize(): self._last3_rect = wx.Rect(*self._last2_rect) self._last2_rect = wx.Rect(*self._last_rect) self._last_rect = wx.Rect(*win_rect) return dir = wx.ALL horiz_dist = abs(win_rect.x - self._last3_rect.x) vert_dist = abs(win_rect.y - self._last3_rect.y) if vert_dist >= horiz_dist: if win_rect.y < self._last3_rect.y: dir = wx.NORTH else: dir = wx.SOUTH else: if win_rect.x < self._last3_rect.x: dir = wx.WEST else: dir = wx.EAST self._last3_rect = wx.Rect(*self._last2_rect) self._last2_rect = wx.Rect(*self._last_rect) self._last_rect = wx.Rect(*win_rect) if _VERSION_STRING < "2.9": leftDown = wx.GetMouseState().LeftDown() else: leftDown = wx.GetMouseState().LeftIsDown() if not leftDown: return if not self._moving: self.OnMoveStart(event) self._moving = True if self._last3_rect.IsEmpty(): return if event.GetEventType() == wx.wxEVT_MOVING: self.OnMoving(event.GetRect(), dir) else: self.OnMoving(wx.Rect(event.GetPosition(), self.GetSize()), dir) def OnIdle(self, event): """ Handles the ``wx.EVT_IDLE`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`IdleEvent` event to be processed. .. note:: This event is only processed on wxMAC if :class:`AuiManager` is using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ if self._moving: if _VERSION_STRING < "2.9": leftDown = wx.GetMouseState().LeftDown() else: leftDown = wx.GetMouseState().LeftIsDown() if not leftDown: self._moving = False self.OnMoveFinished() else: event.RequestMore() def OnMoveStart(self, event): """ The user has just started moving the floating pane. :param `event`: an instance of :class:`MouseEvent`. .. note:: This event is only processed on wxMAC if :class:`AuiManager` is using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ # notify the owner manager that the pane has started to move if self._owner_mgr: if self._owner_mgr._from_move: return self._owner_mgr._action_window = self._pane_window point = wx.GetMousePosition() action_offset = point - self.GetPosition() if self._is_toolbar: self._owner_mgr._toolbar_action_offset = action_offset self._owner_mgr.OnMotion_DragToolbarPane(point) else: self._owner_mgr._action_offset = action_offset self._owner_mgr.OnMotion_DragFloatingPane(point) def OnMoving(self, rect, direction): """ The user is moving the floating pane. :param Rect `rect`: the pane client rectangle; :param integer `direction`: the direction in which the pane is moving, can be one of ``wx.NORTH``, ``wx.SOUTH``, ``wx.EAST`` or ``wx.WEST``. .. note:: This event is only processed on wxMAC if :class:`AuiManager` is using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ # notify the owner manager that the pane is moving self.OnMoveStart(None) self._lastDirection = direction def OnMoveFinished(self): """ The user has just finished moving the floating pane. .. note:: This method is used only on wxMAC if :class:`AuiManager` is using the ``AUI_MGR_USE_NATIVE_MINIFRAMES`` style. """ # notify the owner manager that the pane has finished moving if self._owner_mgr: self._owner_mgr._action_window = self._pane_window point = wx.GetMousePosition() if self._is_toolbar: self._owner_mgr.OnLeftUp_DragToolbarPane(point) else: self._owner_mgr.OnLeftUp_DragFloatingPane(point) self._owner_mgr.OnFloatingPaneMoved(self._pane_window, point) def OnCheckFlyTimer(self, event): """ Handles the ``wx.EVT_TIMER`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`TimerEvent` to be processed. :note: This is used solely for "fly-out" panes. """ if self._owner_mgr: pane = self._mgr.GetPane(self._pane_window) if pane.IsFlyOut(): if self.IsShownOnScreen(): self.FlyOut() def OnFindManager(self, event): """ Handles the ``EVT_AUI_FIND_MANAGER`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`AuiManagerEvent` event to be processed. """ event.SetManager(self._owner_mgr) def FlyOut(self): """ Starts the flying in and out of a floating pane. """ if self._fly_timer.IsRunning(): return if _VERSION_STRING < "2.9": leftDown = wx.GetMouseState().LeftDown() else: leftDown = wx.GetMouseState().LeftIsDown() if leftDown: return rect = wx.Rect(*self.GetScreenRect()) rect.Inflate(10, 10) if rect.Contains(wx.GetMousePosition()): if not self._fly: return self._send_size = False self._fly_timer.Start(5) else: if self._fly: return self._send_size = False self._fly_timer.Start(5) def OnFlyTimer(self, event): """ Handles the ``wx.EVT_TIMER`` event for :class:`AuiFloatingFrame`. :param `event`: a :class:`TimerEvent` to be processed. """ current_size = self.GetClientSize() floating_size = wx.Size(*self._owner_mgr.GetPane(self._pane_window).floating_size) if floating_size.y == -1: floating_size = self._floating_size if not self._fly: min_size = self._mgr.GetArtProvider().GetMetric(AUI_DOCKART_CAPTION_SIZE) if wx.Platform != "__WXMSW__": min_size += 2*wx.SystemSettings.GetMetric(wx.SYS_EDGE_Y) if current_size.y - self._fly_step <= min_size: self.SetClientSize((current_size.x, min_size)) self._fly = True self._fly_timer.Stop() self._send_size = True else: self.SetClientSize((current_size.x, current_size.y-self._fly_step)) else: if current_size.y + self._fly_step >= floating_size.y: self.SetClientSize((current_size.x, floating_size.y)) self._fly = False self._fly_timer.Stop() self._send_size = True else: self.SetClientSize((current_size.x, current_size.y+self._fly_step)) self.Update() self.Refresh() def FadeOut(self): """ Actually starts the fading out of the floating pane. """ while 1: self._alpha_amount -= 10 if self._alpha_amount <= 0: self._alpha_amount = 255 return self.SetTransparent(self._alpha_amount) wx.SafeYield() wx.MilliSleep(15) # -- static utility functions -- def DrawResizeHint(dc, rect): """ Draws a resize hint while a sash is dragged. :param Rect `rect`: a rectangle which specifies the sash dimensions. """ if wx.Platform == "__WXMSW__" and wx.App.GetComCtl32Version() >= 600: if wx.GetOsVersion()[1] > 5: # Windows Vista dc.SetPen(wx.Pen("black", 2, wx.SOLID)) dc.SetBrush(wx.TRANSPARENT_BRUSH) else: # Draw the nice XP style splitter dc.SetPen(wx.TRANSPARENT_PEN) dc.SetBrush(wx.BLACK_BRUSH) dc.SetLogicalFunction(wx.INVERT) dc.DrawRectangle(rect) dc.SetLogicalFunction(wx.COPY) else: stipple = PaneCreateStippleBitmap() brush = wx.Brush(stipple) dc.SetBrush(brush) dc.SetPen(wx.TRANSPARENT_PEN) dc.SetLogicalFunction(wx.XOR) dc.DrawRectangle(rect) def CopyDocksAndPanes(src_docks, src_panes): """ This utility function creates shallow copies of the dock and pane info. :class:`AuiManager` usually contain pointers to :class:`AuiPaneInfo` classes, thus this function is necessary to reliably reconstruct that relationship in the new dock info and pane info arrays. :param `src_docks`: a list of :class:`AuiDockInfo` classes; :param `src_panes`: a list of :class:`AuiPaneInfo` classes. """ dest_docks = src_docks dest_panes = src_panes for ii in range(len(dest_docks)): dock = dest_docks[ii] for jj in range(len(dock.panes)): for kk in range(len(src_panes)): if dock.panes[jj] == src_panes[kk]: dock.panes[jj] = dest_panes[kk] return dest_docks, dest_panes def CopyDocksAndPanes2(src_docks, src_panes): """ This utility function creates full copies of the dock and pane info. :class:`AuiManager` usually contain pointers to :class:`AuiPaneInfo` classes, thus this function is necessary to reliably reconstruct that relationship in the new dock info and pane info arrays. :param `src_docks`: a list of :class:`AuiDockInfo` classes; :param `src_panes`: a list of :class:`AuiPaneInfo` classes. """ dest_docks = [] for ii in range(len(src_docks)): dest_docks.append(AuiDockInfo()) dest_docks[ii].dock_direction = src_docks[ii].dock_direction dest_docks[ii].dock_layer = src_docks[ii].dock_layer dest_docks[ii].dock_row = src_docks[ii].dock_row dest_docks[ii].size = src_docks[ii].size dest_docks[ii].min_size = src_docks[ii].min_size dest_docks[ii].resizable = src_docks[ii].resizable dest_docks[ii].fixed = src_docks[ii].fixed dest_docks[ii].toolbar = src_docks[ii].toolbar dest_docks[ii].panes = src_docks[ii].panes dest_docks[ii].rect = wx.Rect(*src_docks[ii].rect) dest_panes = [] for ii in range(len(src_panes)): dest_panes.append(AuiPaneInfo()) dest_panes[ii].name = src_panes[ii].name dest_panes[ii].caption = src_panes[ii].caption dest_panes[ii].window = src_panes[ii].window dest_panes[ii].frame = src_panes[ii].frame dest_panes[ii].state = src_panes[ii].state dest_panes[ii].dock_direction = src_panes[ii].dock_direction dest_panes[ii].dock_layer = src_panes[ii].dock_layer dest_panes[ii].dock_row = src_panes[ii].dock_row dest_panes[ii].dock_pos = src_panes[ii].dock_pos dest_panes[ii].best_size = wx.Size(*src_panes[ii].best_size) dest_panes[ii].min_size = wx.Size(*src_panes[ii].min_size) dest_panes[ii].max_size = wx.Size(*src_panes[ii].max_size) dest_panes[ii].floating_pos = wx.Point(*src_panes[ii].floating_pos) dest_panes[ii].floating_size = wx.Size(*src_panes[ii].floating_size) dest_panes[ii].dock_proportion = src_panes[ii].dock_proportion dest_panes[ii].buttons = src_panes[ii].buttons dest_panes[ii].rect = wx.Rect(*src_panes[ii].rect) dest_panes[ii].icon = src_panes[ii].icon dest_panes[ii].notebook_id = src_panes[ii].notebook_id dest_panes[ii].transparent = src_panes[ii].transparent dest_panes[ii].snapped = src_panes[ii].snapped dest_panes[ii].minimize_mode = src_panes[ii].minimize_mode dest_panes[ii].minimize_target = src_panes[ii].minimize_target for ii in range(len(dest_docks)): dock = dest_docks[ii] for jj in range(len(dock.panes)): for kk in range(len(src_panes)): if dock.panes[jj] == src_panes[kk]: dock.panes[jj] = dest_panes[kk] dest_docks[ii] = dock return dest_docks, dest_panes def GetMaxLayer(docks, dock_direction): """ This is an internal function which returns the highest layer inside the specified dock. :param `docks`: a list of :class:`AuiDockInfo`; :param `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze. """ max_layer = 0 for dock in docks: if dock.dock_direction == dock_direction and dock.dock_layer > max_layer and not dock.fixed: max_layer = dock.dock_layer return max_layer def GetMaxRow(panes, dock_direction, dock_layer): """ This is an internal function which returns the highest layer inside the specified dock. :param `panes`: a list of :class:`AuiPaneInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; :param integer `dock_layer`: the :class:`AuiDockInfo` layer to analyze. """ max_row = 0 for pane in panes: if pane.dock_direction == dock_direction and pane.dock_layer == dock_layer and \ pane.dock_row > max_row: max_row = pane.dock_row return max_row def DoInsertDockLayer(panes, dock_direction, dock_layer): """ This is an internal function that inserts a new dock layer by incrementing all existing dock layer values by one. :param `panes`: a list of :class:`AuiPaneInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; :param integer `dock_layer`: the :class:`AuiDockInfo` layer to analyze. """ for ii in range(len(panes)): pane = panes[ii] if not pane.IsFloating() and pane.dock_direction == dock_direction and pane.dock_layer >= dock_layer: pane.dock_layer = pane.dock_layer + 1 panes[ii] = pane return panes def DoInsertDockRow(panes, dock_direction, dock_layer, dock_row): """ This is an internal function that inserts a new dock row by incrementing all existing dock row values by one. :param `panes`: a list of :class:`AuiPaneInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; :param integer `dock_layer`: the :class:`AuiDockInfo` layer to analyze; :param integer `dock_row`: the :class:`AuiDockInfo` row to analyze. """ for pane in panes: if not pane.IsFloating() and pane.dock_direction == dock_direction and \ pane.dock_layer == dock_layer and pane.dock_row >= dock_row: pane.dock_row += 1 return panes def DoInsertPane(panes, dock_direction, dock_layer, dock_row, dock_pos): """ This is an internal function that inserts a new pane by incrementing all existing dock position values by one. :param `panes`: a list of :class:`AuiPaneInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; :param integer `dock_layer`: the :class:`AuiDockInfo` layer to analyze. :param integer `dock_row`: the :class:`AuiDockInfo` row to analyze; :param integer `dock_pos`: the :class:`AuiDockInfo` position to analyze. """ for ii in range(len(panes)): pane = panes[ii] if not pane.IsFloating() and pane.dock_direction == dock_direction and \ pane.dock_layer == dock_layer and pane.dock_row == dock_row and \ pane.dock_pos >= dock_pos: pane.dock_pos = pane.dock_pos + 1 panes[ii] = pane return panes def FindDocks(docks, dock_direction, dock_layer=-1, dock_row=-1, reverse=False): """ This is an internal function that returns a list of docks which meet the specified conditions in the parameters and returns a sorted array (sorted by layer and then row). :param `docks`: a list of :class:`AuiDockInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; :param integer `dock_layer`: the :class:`AuiDockInfo` layer to analyze. :param integer `dock_row`: the :class:`AuiDockInfo` row to analyze; """ matchDocks = [(d.dock_layer, d.dock_row, d.dock_direction, d) for d in docks if \ (dock_direction == -1 or dock_direction == d.dock_direction) and \ ((dock_layer == -1 or dock_layer == d.dock_layer) and \ (dock_row == -1 or dock_row == d.dock_row))] arr = [x[-1] for x in sorted(matchDocks, reverse=reverse)] return arr def FindOppositeDocks(docks, dock_direction): """ This is an internal function that returns a list of docks which is related to the opposite direction. :param `docks`: a list of :class:`AuiDockInfo`; :param integer `dock_direction`: the :class:`AuiDockInfo` docking direction to analyze; """ if dock_direction == AUI_DOCK_LEFT: arr = FindDocks(docks, AUI_DOCK_RIGHT, -1, -1) elif dock_direction == AUI_DOCK_TOP: arr = FindDocks(docks, AUI_DOCK_BOTTOM, -1, -1) elif dock_direction == AUI_DOCK_RIGHT: arr = FindDocks(docks, AUI_DOCK_LEFT, -1, -1) elif dock_direction == AUI_DOCK_BOTTOM: arr = FindDocks(docks, AUI_DOCK_TOP, -1, -1) return arr def FindPaneInDock(dock, window): """ This method looks up a specified window pointer inside a dock. If found, the corresponding :class:`AuiDockInfo` pointer is returned, otherwise ``None``. :param `dock`: a :class:`AuiDockInfo` structure; :param Window `window`: the window associated to the pane we are seeking. """ for p in dock.panes: if p.window == window: return p return None def GetToolBarDockOffsets(docks): """ Returns the toolbar dock offsets (top-left and bottom-right). :param `docks`: a list of :class:`AuiDockInfo` to analyze. """ top_left = wx.Size(0, 0) bottom_right = wx.Size(0, 0) for dock in docks: if dock.toolbar: dock_direction = dock.dock_direction if dock_direction == AUI_DOCK_LEFT: top_left.x += dock.rect.width bottom_right.x += dock.rect.width elif dock_direction == AUI_DOCK_TOP: top_left.y += dock.rect.height bottom_right.y += dock.rect.height elif dock_direction == AUI_DOCK_RIGHT: bottom_right.x += dock.rect.width elif dock_direction == AUI_DOCK_BOTTOM: bottom_right.y += dock.rect.height return top_left, bottom_right def GetInternalFrameRect(window, docks): """ Returns the window rectangle excluding toolbars. :param `window`: a :class:`Window` derived window; :param `docks`: a list of :class:`AuiDockInfo` structures. """ frameRect = wx.Rect() frameRect.SetTopLeft(window.ClientToScreen(window.GetClientAreaOrigin())) frameRect.SetSize(window.GetClientSize()) top_left, bottom_right = GetToolBarDockOffsets(docks) # make adjustments for toolbars frameRect.x += top_left.x frameRect.y += top_left.y frameRect.width -= bottom_right.x frameRect.height -= bottom_right.y return frameRect def CheckOutOfWindow(window, pt): """ Checks if a point is outside the window rectangle. :param `window`: a :class:`Window` derived window; :param `pt`: a :class:`Point` object. """ auiWindowMargin = 30 marginRect = wx.Rect(*window.GetClientRect()) marginRect.Inflate(auiWindowMargin, auiWindowMargin) return not marginRect.Contains(pt) def CheckEdgeDrop(window, docks, pt): """ Checks on which edge of a window the drop action has taken place. :param `window`: a :class:`Window` derived window; :param `docks`: a list of :class:`AuiDockInfo` structures; :param `pt`: a :class:`Point` object. """ screenPt = window.ClientToScreen(pt) clientSize = window.GetClientSize() frameRect = GetInternalFrameRect(window, docks) if screenPt.y >= frameRect.GetTop() and screenPt.y < frameRect.GetBottom(): if pt.x < auiLayerInsertOffset and pt.x > auiLayerInsertOffset - auiLayerInsertPixels: return wx.LEFT if pt.x >= clientSize.x - auiLayerInsertOffset and \ pt.x < clientSize.x - auiLayerInsertOffset + auiLayerInsertPixels: return wx.RIGHT if screenPt.x >= frameRect.GetLeft() and screenPt.x < frameRect.GetRight(): if pt.y < auiLayerInsertOffset and pt.y > auiLayerInsertOffset - auiLayerInsertPixels: return wx.TOP if pt.y >= clientSize.y - auiLayerInsertOffset and \ pt.y < clientSize.y - auiLayerInsertOffset + auiLayerInsertPixels: return wx.BOTTOM return -1 def RemovePaneFromDocks(docks, pane, exc=None): """ Removes a pane window from all docks with a possible exception specified by parameter `exc`. :param `docks`: a list of :class:`AuiDockInfo` structures; :param `pane`: the pane to be removed, an instance of :class:`AuiPaneInfo`; :param `exc`: the possible pane exception, an instance of :class:`AuiPaneInfo`. """ for ii in range(len(docks)): d = docks[ii] if d == exc: continue pi = FindPaneInDock(d, pane.window) if pi: d.panes.remove(pi) docks[ii] = d return docks def RenumberDockRows(docks): """ Takes a dock and assigns sequential numbers to existing rows. Basically it takes out the gaps so if a dock has rows with numbers 0, 2, 5, they will become 0, 1, 2. :param `docks`: a list of :class:`AuiDockInfo` structures. """ for ii in range(len(docks)): dock = docks[ii] dock.dock_row = ii for jj in range(len(dock.panes)): dock.panes[jj].dock_row = ii docks[ii] = dock return docks def SetActivePane(panes, active_pane): """ Sets the active pane, as well as cycles through every other pane and makes sure that all others' active flags are turned off. :param `panes`: a list of :class:`AuiPaneInfo` structures; :param `active_pane`: the pane to be made active (if found), an instance of :class:`AuiPaneInfo`. """ for pane in panes: pane.state &= ~AuiPaneInfo.optionActive for pane in panes: if pane.window == active_pane and not pane.IsNotebookPage(): pane.state |= AuiPaneInfo.optionActive return True, panes return False, panes def ShowDockingGuides(guides, show): """ Shows or hide the docking guide windows. :param `guides`: a list of :class:`AuiDockingGuide` classes; :param bool `show`: whether to show or hide the docking guide windows. """ for target in guides: if show and not target.host.IsShown(): target.host.Show() target.host.Update() elif not show and target.host.IsShown(): target.host.Hide() def RefreshDockingGuides(guides): """ Refreshes the docking guide windows. :param `guides`: a list of :class:`AuiDockingGuide` classes; """ for target in guides: if target.host.IsShown(): target.host.Refresh() def PaneSortFunc(p1, p2): """ This function is used to sort panes by dock position. :param `p1`: the first pane instance to compare, an instance of :class:`AuiPaneInfo`; :param `p2`: the second pane instance to compare, an instance of :class:`AuiPaneInfo`. """ return (p1.dock_pos < p2.dock_pos and [-1] or [1])[0] def GetNotebookRoot(panes, notebook_id): """ Returns the :class:`~lib.agw.aui.auibook.AuiNotebook` which has the specified `notebook_id`. :param `panes`: a list of :class:`AuiPaneInfo` instances; :param integer `notebook_id`: the target notebook id. """ for paneInfo in panes: if paneInfo.IsNotebookControl() and paneInfo.notebook_id == notebook_id: return paneInfo return None def EscapeDelimiters(s): """ Changes ``;`` into ``\`` and ``|`` into ``\|`` in the input string. :param string `s`: the string to be analyzed. :note: This is an internal functions which is used for saving perspectives. """ result = s.replace(";", "\\") result = result.replace("|", "|\\") return result def IsDifferentDockingPosition(pane1, pane2): """ Returns whether `pane1` and `pane2` are in a different docking position based on pane status, docking direction, docking layer and docking row. :param `pane1`: a :class:`AuiPaneInfo` instance; :param `pane2`: another :class:`AuiPaneInfo` instance. """ return pane1.IsFloating() != pane2.IsFloating() or \ pane1.dock_direction != pane2.dock_direction or \ pane1.dock_layer != pane2.dock_layer or \ pane1.dock_row != pane2.dock_row # Convenience function def AuiManager_HasLiveResize(manager): """ Static function which returns if the input `manager` should have "live resize" behaviour. :param `manager`: an instance of :class:`AuiManager`. .. note:: This method always returns ``True`` on wxMAC as this platform doesn't have the ability to use :class:`ScreenDC` to draw sashes. """ # With Core Graphics on Mac, it's not possible to show sash feedback, # so we'll always use live update instead. if wx.Platform == "__WXMAC__": return True else: return (manager.GetAGWFlags() & AUI_MGR_LIVE_RESIZE) == AUI_MGR_LIVE_RESIZE # Convenience function def AuiManager_UseNativeMiniframes(manager): """ Static function which returns if the input `manager` should use native :class:`MiniFrame` as floating panes. :param `manager`: an instance of :class:`AuiManager`. .. note:: This method always returns ``True`` on wxMAC as this platform doesn't have the ability to use custom drawn miniframes. """ # With Core Graphics on Mac, it's not possible to show sash feedback, # so we'll always use live update instead. if wx.Platform == "__WXMAC__": return True else: return (manager.GetAGWFlags() & AUI_MGR_USE_NATIVE_MINIFRAMES) == AUI_MGR_USE_NATIVE_MINIFRAMES def GetManager(window): """ This function will return the aui manager for a given window. :param Window `window`: this parameter should be any child window or grand-child window (and so on) of the frame/window managed by :class:`AuiManager`. The window does not need to be managed by the manager itself, nor does it even need to be a child or sub-child of a managed window. It must however be inside the window hierarchy underneath the managed window. """ if not isinstance(wx.GetTopLevelParent(window), AuiFloatingFrame): if isinstance(window, auibar.AuiToolBar): return window.GetAuiManager() evt = AuiManagerEvent(wxEVT_AUI_FIND_MANAGER) evt.SetManager(None) evt.ResumePropagation(wx.EVENT_PROPAGATE_MAX) if not window.GetEventHandler().ProcessEvent(evt): return None return evt.GetManager() # ---------------------------------------------------------------------------- # class AuiManager(wx.EvtHandler): """ AuiManager manages the panes associated with it for a particular :class:`Frame`, using a pane's :class:`AuiManager` information to determine each pane's docking and floating behavior. :class:`AuiManager` uses wxPython's sizer mechanism to plan the layout of each frame. It uses a replaceable dock art class to do all drawing, so all drawing is localized in one area, and may be customized depending on an applications' specific needs. :class:`AuiManager` works as follows: the programmer adds panes to the class, or makes changes to existing pane properties (dock position, floating state, show state, etc...). To apply these changes, the :meth:`AuiManager.Update() ` function is called. This batch processing can be used to avoid flicker, by modifying more than one pane at a time, and then "committing" all of the changes at once by calling `Update()`. Panes can be added quite easily:: text1 = wx.TextCtrl(self, -1) text2 = wx.TextCtrl(self, -1) self._mgr.AddPane(text1, AuiPaneInfo().Left().Caption("Pane Number One")) self._mgr.AddPane(text2, AuiPaneInfo().Bottom().Caption("Pane Number Two")) self._mgr.Update() Later on, the positions can be modified easily. The following will float an existing pane in a tool window:: self._mgr.GetPane(text1).Float() **Layers, Rows and Directions, Positions:** Inside AUI, the docking layout is figured out by checking several pane parameters. Four of these are important for determining where a pane will end up. **Direction** - Each docked pane has a direction, `Top`, `Bottom`, `Left`, `Right`, or `Center`. This is fairly self-explanatory. The pane will be placed in the location specified by this variable. **Position** - More than one pane can be placed inside of a "dock". Imagine two panes being docked on the left side of a window. One pane can be placed over another. In proportionally managed docks, the pane position indicates it's sequential position, starting with zero. So, in our scenario with two panes docked on the left side, the top pane in the dock would have position 0, and the second one would occupy position 1. **Row** - A row can allow for two docks to be placed next to each other. One of the most common places for this to happen is in the toolbar. Multiple toolbar rows are allowed, the first row being in row 0, and the second in row 1. Rows can also be used on vertically docked panes. **Layer** - A layer is akin to an onion. Layer 0 is the very center of the managed pane. Thus, if a pane is in layer 0, it will be closest to the center window (also sometimes known as the "content window"). Increasing layers "swallow up" all layers of a lower value. This can look very similar to multiple rows, but is different because all panes in a lower level yield to panes in higher levels. The best way to understand layers is by running the AUI sample (`AUI.py`). """ def __init__(self, managed_window=None, agwFlags=None): """ Default class constructor. :param Window `managed_window`: specifies the window which should be managed; :param integer `agwFlags`: specifies options which allow the frame management behavior to be modified. `agwFlags` can be a combination of the following style bits: ==================================== ================================== Flag name Description ==================================== ================================== ``AUI_MGR_ALLOW_FLOATING`` Allow floating of panes ``AUI_MGR_ALLOW_ACTIVE_PANE`` If a pane becomes active, "highlight" it in the interface ``AUI_MGR_TRANSPARENT_DRAG`` If the platform supports it, set transparency on a floating pane while it is dragged by the user ``AUI_MGR_TRANSPARENT_HINT`` If the platform supports it, show a transparent hint window when the user is about to dock a floating pane ``AUI_MGR_VENETIAN_BLINDS_HINT`` Show a "venetian blind" effect when the user is about to dock a floating pane ``AUI_MGR_RECTANGLE_HINT`` Show a rectangle hint effect when the user is about to dock a floating pane ``AUI_MGR_HINT_FADE`` If the platform supports it, the hint window will fade in and out ``AUI_MGR_NO_VENETIAN_BLINDS_FADE`` Disables the "venetian blind" fade in and out ``AUI_MGR_LIVE_RESIZE`` Live resize when the user drag a sash ``AUI_MGR_ANIMATE_FRAMES`` Fade-out floating panes when they are closed (all platforms which support frames transparency) and show a moving rectangle when they are docked (Windows < Vista and GTK only) ``AUI_MGR_AERO_DOCKING_GUIDES`` Use the new Aero-style bitmaps as docking guides ``AUI_MGR_PREVIEW_MINIMIZED_PANES`` Slide in and out minimized panes to preview them ``AUI_MGR_WHIDBEY_DOCKING_GUIDES`` Use the new Whidbey-style bitmaps as docking guides ``AUI_MGR_SMOOTH_DOCKING`` Performs a "smooth" docking of panes (a la PyQT) ``AUI_MGR_USE_NATIVE_MINIFRAMES`` Use miniframes with native caption bar as floating panes instead or custom drawn caption bars (forced on wxMAC) ``AUI_MGR_AUTONB_NO_CAPTION`` Panes that merge into an automatic notebook will not have the pane caption visible ==================================== ================================== Default value for `agwFlags` is: ``AUI_MGR_DEFAULT`` = ``AUI_MGR_ALLOW_FLOATING`` | ``AUI_MGR_TRANSPARENT_HINT`` | ``AUI_MGR_HINT_FADE`` | ``AUI_MGR_NO_VENETIAN_BLINDS_FADE`` .. note:: If using the ``AUI_MGR_USE_NATIVE_MINIFRAMES``, double-clicking on a floating pane caption will not re-dock the pane, but simply maximize it (if :meth:`AuiPaneInfo.MaximizeButton` has been set to ``True``) or do nothing. """ wx.EvtHandler.__init__(self) self._action = actionNone self._action_window = None self._hover_button = None self._art = dockart.AuiDefaultDockArt() self._hint_window = None self._active_pane = None self._has_maximized = False self._has_minimized = False self._frame = None self._dock_constraint_x = 0.3 self._dock_constraint_y = 0.3 self._reserved = None self._panes = [] self._docks = [] self._uiparts = [] self._guides = [] self._notebooks = [] self._masterManager = None self._currentDragItem = -1 self._lastknowndocks = {} self._hint_fadetimer = wx.Timer(self, wx.ID_ANY) self._hint_fademax = 50 self._last_hint = wx.Rect() self._from_move = False self._last_rect = wx.Rect() if agwFlags is None: agwFlags = AUI_MGR_DEFAULT self._agwFlags = agwFlags self._is_docked = (False, wx.RIGHT, wx.TOP, 0) self._snap_limits = (15, 15) if wx.Platform == "__WXMSW__": self._animation_step = 30.0 else: self._animation_step = 5.0 self._hint_rect = wx.Rect() self._preview_timer = wx.Timer(self, wx.ID_ANY) self._sliding_frame = None self._autoNBTabArt = tabart.AuiDefaultTabArt() self._autoNBStyle = AUI_NB_DEFAULT_STYLE | AUI_NB_BOTTOM | \ AUI_NB_SUB_NOTEBOOK | AUI_NB_TAB_EXTERNAL_MOVE self._autoNBStyle -= AUI_NB_DRAW_DND_TAB if managed_window: self.SetManagedWindow(managed_window) self.Bind(wx.EVT_PAINT, self.OnPaint) self.Bind(wx.EVT_ERASE_BACKGROUND, self.OnEraseBackground) self.Bind(wx.EVT_SIZE, self.OnSize) self.Bind(wx.EVT_SET_CURSOR, self.OnSetCursor) self.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown) self.Bind(wx.EVT_LEFT_DCLICK, self.OnLeftDClick) self.Bind(wx.EVT_LEFT_UP, self.OnLeftUp) self.Bind(wx.EVT_MOTION, self.OnMotion) self.Bind(wx.EVT_LEAVE_WINDOW, self.OnLeaveWindow) self.Bind(wx.EVT_CHILD_FOCUS, self.OnChildFocus) self.Bind(wx.EVT_MOUSE_CAPTURE_LOST, self.OnCaptureLost) self.Bind(wx.EVT_TIMER, self.OnHintFadeTimer, self._hint_fadetimer) self.Bind(wx.EVT_TIMER, self.SlideIn, self._preview_timer) self.Bind(wx.EVT_WINDOW_DESTROY, self.OnDestroy) self.Bind(wx.EVT_MOVE, self.OnMove) self.Bind(wx.EVT_SYS_COLOUR_CHANGED, self.OnSysColourChanged) self.Bind(EVT_AUI_PANE_BUTTON, self.OnPaneButton) self.Bind(EVT_AUI_RENDER, self.OnRender) self.Bind(EVT_AUI_FIND_MANAGER, self.OnFindManager) self.Bind(EVT_AUI_PANE_MIN_RESTORE, self.OnRestoreMinimizedPane) self.Bind(EVT_AUI_PANE_DOCKED, self.OnPaneDocked) self.Bind(auibook.EVT_AUINOTEBOOK_BEGIN_DRAG, self.OnTabBeginDrag) self.Bind(auibook.EVT_AUINOTEBOOK_PAGE_CLOSE, self.OnTabPageClose) self.Bind(auibook.EVT_AUINOTEBOOK_PAGE_CHANGED, self.OnTabSelected) def CreateFloatingFrame(self, parent, pane_info): """ Creates a floating frame for the windows. :param Window `parent`: the floating frame parent; :param `pane_info`: the :class:`AuiPaneInfo` class with all the pane's information. """ return AuiFloatingFrame(parent, self, pane_info) def CanDockPanel(self, p): """ Returns whether a pane can be docked or not. :param `p`: the :class:`AuiPaneInfo` class with all the pane's information. """ # is the pane dockable? if not p.IsDockable(): return False # if a key modifier is pressed while dragging the frame, # don't dock the window return not (wx.GetKeyState(wx.WXK_CONTROL) or wx.GetKeyState(wx.WXK_ALT)) def GetPaneByWidget(self, window): """ This version of :meth:`GetPane` looks up a pane based on a 'pane window'. :param `window`: a :class:`Window` derived window. :see: :meth:`~AuiManager.GetPane` """ for p in self._panes: if p.window == window: return p return NonePaneInfo def GetPaneByName(self, name): """ This version of :meth:`GetPane` looks up a pane based on a 'pane name'. :param string `name`: the pane name. :see: :meth:`GetPane` """ for p in self._panes: if p.name == name: return p return NonePaneInfo def GetPane(self, item): """ Looks up a :class:`AuiPaneInfo` structure based on the supplied window pointer. Upon failure, :meth:`GetPane` returns an empty :class:`AuiPaneInfo`, a condition which can be checked by calling :meth:`AuiPaneInfo.IsOk() `. The pane info's structure may then be modified. Once a pane's info is modified, :meth:`Update` must be called to realize the changes in the UI. :param `item`: either a pane name or a :class:`Window`. """ if isinstance(item, six.string_types): return self.GetPaneByName(item) else: return self.GetPaneByWidget(item) def GetAllPanes(self): """ Returns a reference to all the pane info structures. """ return self._panes def ShowPane(self, window, show): """ Shows or hides a pane based on the window passed as input. :param Window `window`: any subclass or derivation of :class:`Window`; :param bool `show`: ``True`` to show the pane, ``False`` otherwise. """ p = self.GetPane(window) if p.IsOk(): if p.IsNotebookPage(): if show: notebook = self._notebooks[p.notebook_id] id = notebook.GetPageIndex(p.window) if id >= 0: notebook.SetSelection(id) self.ShowPane(notebook, True) else: p.Show(show) if p.frame: p.frame.Raise() self.Update() def HitTest(self, x, y): """ This is an internal function which determines which UI item the specified coordinates are over. :param integer `x`: specifies a x position in client coordinates; :param integer `y`: specifies a y position in client coordinates. """ result = None for item in self._uiparts: # we are not interested in typeDock, because this space # isn't used to draw anything, just for measurements # besides, the entire dock area is covered with other # rectangles, which we are interested in. if item.type == AuiDockUIPart.typeDock: continue # if we already have a hit on a more specific item, we are not # interested in a pane hit. If, however, we don't already have # a hit, returning a pane hit is necessary for some operations if item.type in [AuiDockUIPart.typePane, AuiDockUIPart.typePaneBorder] and result: continue # if the point is inside the rectangle, we have a hit if item.rect.Contains((x, y)): result = item return result def PaneHitTest(self, panes, pt): """ Similar to :meth:`HitTest`, but it checks in which :class:`AuiManager` rectangle the input point belongs to. :param `panes`: a list of :class:`AuiPaneInfo` instances; :param Point `pt`: the mouse position. """ for paneInfo in panes: if paneInfo.IsDocked() and paneInfo.IsShown() and paneInfo.rect.Contains(pt): return paneInfo return NonePaneInfo # SetAGWFlags() and GetAGWFlags() allow the owner to set various # options which are global to AuiManager def SetAGWFlags(self, agwFlags): """ This method is used to specify :class:`AuiManager` 's settings flags. :param integer `agwFlags`: specifies options which allow the frame management behavior to be modified. `agwFlags` can be one of the following style bits: ==================================== ================================== Flag name Description ==================================== ================================== ``AUI_MGR_ALLOW_FLOATING`` Allow floating of panes ``AUI_MGR_ALLOW_ACTIVE_PANE`` If a pane becomes active, "highlight" it in the interface ``AUI_MGR_TRANSPARENT_DRAG`` If the platform supports it, set transparency on a floating pane while it is dragged by the user ``AUI_MGR_TRANSPARENT_HINT`` If the platform supports it, show a transparent hint window when the user is about to dock a floating pane ``AUI_MGR_VENETIAN_BLINDS_HINT`` Show a "venetian blind" effect when the user is about to dock a floating pane ``AUI_MGR_RECTANGLE_HINT`` Show a rectangle hint effect when the user is about to dock a floating pane ``AUI_MGR_HINT_FADE`` If the platform supports it, the hint window will fade in and out ``AUI_MGR_NO_VENETIAN_BLINDS_FADE`` Disables the "venetian blind" fade in and out ``AUI_MGR_LIVE_RESIZE`` Live resize when the user drag a sash ``AUI_MGR_ANIMATE_FRAMES`` Fade-out floating panes when they are closed (all platforms which support frames transparency) and show a moving rectangle when they are docked (Windows < Vista and GTK only) ``AUI_MGR_AERO_DOCKING_GUIDES`` Use the new Aero-style bitmaps as docking guides ``AUI_MGR_PREVIEW_MINIMIZED_PANES`` Slide in and out minimized panes to preview them ``AUI_MGR_WHIDBEY_DOCKING_GUIDES`` Use the new Whidbey-style bitmaps as docking guides ``AUI_MGR_SMOOTH_DOCKING`` Performs a "smooth" docking of panes (a la PyQT) ``AUI_MGR_USE_NATIVE_MINIFRAMES`` Use miniframes with native caption bar as floating panes instead or custom drawn caption bars (forced on wxMAC) ``AUI_MGR_AUTONB_NO_CAPTION`` Panes that merge into an automatic notebook will not have the pane caption visible ==================================== ================================== .. note:: If using the ``AUI_MGR_USE_NATIVE_MINIFRAMES``, double-clicking on a floating pane caption will not re-dock the pane, but simply maximize it (if :meth:`AuiPaneInfo.MaximizeButton` has been set to ``True``) or do nothing. """ self._agwFlags = agwFlags if len(self._guides) > 0: self.CreateGuideWindows() if self._hint_window and agwFlags & AUI_MGR_RECTANGLE_HINT == 0: self.CreateHintWindow() def GetAGWFlags(self): """ Returns the current manager's flags. :see: :meth:`SetAGWFlags` for a list of possible :class:`AuiManager` flags. """ return self._agwFlags def SetManagedWindow(self, managed_window): """ Called to specify the frame or window which is to be managed by :class:`AuiManager`. Frame management is not restricted to just frames. Child windows or custom controls are also allowed. :param Window `managed_window`: specifies the window which should be managed by the AUI manager. """ if not managed_window: raise Exception("Specified managed window must be non-null. ") if isinstance(managed_window, sc.SizedParent): raise Exception("Do not use wx.lib.sized_control for managed window. ") self.UnInit() self._frame = managed_window self._frame.PushEventHandler(self) # if the owner is going to manage an MDI parent frame, # we need to add the MDI client window as the default # center pane if isinstance(self._frame, wx.MDIParentFrame): mdi_frame = self._frame client_window = mdi_frame.GetClientWindow() if not client_window: raise Exception("Client window is None!") self.AddPane(client_window, AuiPaneInfo().Name("mdiclient"). CenterPane().PaneBorder(False)) elif isinstance(self._frame, tabmdi.AuiMDIParentFrame): mdi_frame = self._frame client_window = mdi_frame.GetClientWindow() if not client_window: raise Exception("Client window is None!") self.AddPane(client_window, AuiPaneInfo().Name("mdiclient"). CenterPane().PaneBorder(False)) def GetManagedWindow(self): """ Returns the window being managed by :class:`AuiManager`. """ return self._frame def SetFrame(self, managed_window): """ Called to specify the frame or window which is to be managed by :class:`AuiManager`. Frame management is not restricted to just frames. Child windows or custom controls are also allowed. :param Window `managed_window`: specifies the window which should be managed by the AUI manager. .. deprecated:: 0.6 This method is now deprecated, use :meth:`SetManagedWindow` instead. """ DeprecationWarning("This method is deprecated, use SetManagedWindow instead.") return self.SetManagedWindow(managed_window) def GetFrame(self): """ Returns the window being managed by :class:`AuiManager`. .. deprecated:: 0.6 This method is now deprecated, use :meth:`GetManagedWindow` instead. """ DeprecationWarning("This method is deprecated, use GetManagedWindow instead.") return self._frame def CreateGuideWindows(self): """ Creates the VS2005 HUD guide windows. """ self.DestroyGuideWindows() self._guides.append(AuiDockingGuideInfo().Left(). Host(AuiSingleDockingGuide(self._frame, wx.LEFT))) self._guides.append(AuiDockingGuideInfo().Top(). Host(AuiSingleDockingGuide(self._frame, wx.TOP))) self._guides.append(AuiDockingGuideInfo().Right(). Host(AuiSingleDockingGuide(self._frame, wx.RIGHT))) self._guides.append(AuiDockingGuideInfo().Bottom(). Host(AuiSingleDockingGuide(self._frame, wx.BOTTOM))) self._guides.append(AuiDockingGuideInfo().Centre(). Host(AuiCenterDockingGuide(self._frame))) def DestroyGuideWindows(self): """ Destroys the VS2005 HUD guide windows. """ for guide in self._guides: if guide.host: guide.host.Destroy() self._guides = [] def CreateHintWindow(self): """ Creates the standard wxAUI hint window. """ self.DestroyHintWindow() self._hint_window = AuiDockingHintWindow(self._frame) self._hint_window.SetBlindMode(self._agwFlags) def DestroyHintWindow(self): """ Destroys the standard wxAUI hint window. """ if self._hint_window: self._hint_window.Destroy() self._hint_window = None def UnInit(self): """ Uninitializes the framework and should be called before a managed frame or window is destroyed. :meth:`UnInit` is usually called in the managed :class:`Frame` / :class:`Window` destructor. It is necessary to call this function before the managed frame or window is destroyed, otherwise the manager cannot remove its custom event handlers from a window. """ if not self._frame: return for klass in [self._frame] + list(self._frame.GetChildren()): handler = klass.GetEventHandler() if klass is not handler: if isinstance(handler, AuiManager): klass.RemoveEventHandler(handler) def OnDestroy(self, event) : if self._frame == event.GetEventObject(): self.UnInit(); def GetArtProvider(self): """ Returns the current art provider being used. """ return self._art def ProcessMgrEvent(self, event): """ Process the AUI events sent to the manager. :param `event`: the event to process, an instance of :class:`AuiManagerEvent`. """ # first, give the owner frame a chance to override if self._frame: if self._frame.GetEventHandler().ProcessEvent(event): return self.ProcessEvent(event) def FireEvent(self, evtType, pane, canVeto=False): """ Fires one of the ``EVT_AUI_PANE_FLOATED`` / ``FLOATING`` / ``DOCKING`` / ``DOCKED`` / ``ACTIVATED`` event. :param integer `evtType`: one of the aforementioned events; :param `pane`: the :class:`AuiPaneInfo` instance associated to this event; :param bool `canVeto`: whether the event can be vetoed or not. """ event = AuiManagerEvent(evtType) event.SetPane(pane) event.SetCanVeto(canVeto) self.ProcessMgrEvent(event) return event def CanUseModernDockArt(self): """ Returns whether :class:`dockart` can be used (Windows XP / Vista / 7 only, requires Mark Hammonds's `pywin32 `_ package). """ if not _winxptheme: return False # Get the size of a small close button (themed) hwnd = self._frame.GetHandle() hTheme = winxptheme.OpenThemeData(hwnd, "Window") if not hTheme: return False return True def SetArtProvider(self, art_provider): """ Instructs :class:`AuiManager` to use art provider specified by the parameter `art_provider` for all drawing calls. This allows plugable look-and-feel features. :param `art_provider`: a AUI dock art provider. :note: The previous art provider object, if any, will be deleted by :class:`AuiManager`. """ # delete the last art provider, if any del self._art # assign the new art provider self._art = art_provider for pane in self.GetAllPanes(): if pane.IsFloating() and pane.frame: pane.frame._mgr.SetArtProvider(art_provider) pane.frame._mgr.Update() def AddPane(self, window, arg1=None, arg2=None, target=None): """ Tells the frame manager to start managing a child window. There are four versions of this function. The first verison allows the full spectrum of pane parameter possibilities (:meth:`AddPane1`). The second version is used for simpler user interfaces which do not require as much configuration (:meth:`AddPane2`). The :meth:`AddPane3` version allows a drop position to be specified, which will determine where the pane will be added. The :meth:`AddPane4` version allows to turn the target :class:`AuiPaneInfo` pane into a notebook and the added pane into a page. In your code, simply call :meth:`AddPane`. :param Window `window`: the child window to manage; :param `arg1`: a :class:`AuiPaneInfo` or an integer value (direction); :param `arg2`: a :class:`AuiPaneInfo` or a :class:`Point` (drop position); :param `target`: a :class:`AuiPaneInfo` to be turned into a notebook and new pane added to it as a page. (additionally, target can be any pane in an existing notebook) """ if target in self._panes: return self.AddPane4(window, arg1, target) if isinstance(arg1, int): # This Is Addpane2 if arg1 is None: arg1 = wx.LEFT if arg2 is None: arg2 = "" return self.AddPane2(window, arg1, arg2) else: if isinstance(arg2, wx.Point): return self.AddPane3(window, arg1, arg2) else: return self.AddPane1(window, arg1) def AddPane1(self, window, pane_info): """ See comments on :meth:`AddPane`. """ # check if the pane has a valid window if not window: return False # check if the pane already exists if self.GetPane(pane_info.window).IsOk(): return False # check if the pane name already exists, this could reveal a # bug in the library user's application already_exists = False if pane_info.name != "" and self.GetPane(pane_info.name).IsOk(): warnings.warn("A pane with the name '%s' already exists in the manager!"%pane_info.name) already_exists = True # if the new pane is docked then we should undo maximize if pane_info.IsDocked(): self.RestoreMaximizedPane() self._panes.append(pane_info) pinfo = self._panes[-1] # set the pane window pinfo.window = window # if the pane's name identifier is blank, create a random string if pinfo.name == "" or already_exists: pinfo.name = ("%s%08x%08x%08x") % (pinfo.window.GetName(), int(time.time()), int(time.clock()), len(self._panes)) # set initial proportion (if not already set) if pinfo.dock_proportion == 0: pinfo.dock_proportion = 100000 floating = isinstance(self._frame, AuiFloatingFrame) pinfo.buttons = [] if not floating and pinfo.HasMinimizeButton(): button = AuiPaneButton(AUI_BUTTON_MINIMIZE) pinfo.buttons.append(button) if not floating and pinfo.HasMaximizeButton(): button = AuiPaneButton(AUI_BUTTON_MAXIMIZE_RESTORE) pinfo.buttons.append(button) if not floating and pinfo.HasPinButton(): button = AuiPaneButton(AUI_BUTTON_PIN) pinfo.buttons.append(button) if pinfo.HasCloseButton(): button = AuiPaneButton(AUI_BUTTON_CLOSE) pinfo.buttons.append(button) if pinfo.HasGripper(): if isinstance(pinfo.window, auibar.AuiToolBar): # prevent duplicate gripper -- both AuiManager and AuiToolBar # have a gripper control. The toolbar's built-in gripper # meshes better with the look and feel of the control than ours, # so turn AuiManager's gripper off, and the toolbar's on. tb = pinfo.window pinfo.SetFlag(AuiPaneInfo.optionGripper, False) tb.SetGripperVisible(True) if pinfo.window: if pinfo.best_size == wx.Size(-1, -1): pinfo.best_size = pinfo.window.GetClientSize() if isinstance(pinfo.window, wx.ToolBar): # GetClientSize() doesn't get the best size for # a toolbar under some newer versions of wxWidgets, # so use GetBestSize() pinfo.best_size = pinfo.window.GetBestSize() # this is needed for Win2000 to correctly fill toolbar backround # it should probably be repeated once system colour change happens if wx.Platform == "__WXMSW__" and pinfo.window.UseBgCol(): pinfo.window.SetBackgroundColour(self.GetArtProvider().GetColour(AUI_DOCKART_BACKGROUND_COLOUR)) if pinfo.min_size != wx.Size(-1, -1): if pinfo.best_size.x < pinfo.min_size.x: pinfo.best_size.x = pinfo.min_size.x if pinfo.best_size.y < pinfo.min_size.y: pinfo.best_size.y = pinfo.min_size.y self._panes[-1] = pinfo if isinstance(window, auibar.AuiToolBar): window.SetAuiManager(self) return True def AddPane2(self, window, direction, caption): """ See comments on :meth:`AddPane`. """ pinfo = AuiPaneInfo() pinfo.Caption(caption) if direction == wx.TOP: pinfo.Top() elif direction == wx.BOTTOM: pinfo.Bottom() elif direction == wx.LEFT: pinfo.Left() elif direction == wx.RIGHT: pinfo.Right() elif direction == wx.CENTER: pinfo.CenterPane() return self.AddPane(window, pinfo) def AddPane3(self, window, pane_info, drop_pos): """ See comments on :meth:`AddPane`. """ if not self.AddPane(window, pane_info): return False pane = self.GetPane(window) indx = self._panes.index(pane) ret, pane = self.DoDrop(self._docks, self._panes, pane, drop_pos, wx.Point(0, 0)) self._panes[indx] = pane return True def AddPane4(self, window, pane_info, target): """ See comments on :meth:`AddPane`. """ if not self.AddPane(window, pane_info): return False paneInfo = self.GetPane(window) if not paneInfo.IsNotebookDockable(): return self.AddPane1(window, pane_info) if not target.IsNotebookDockable() and not target.IsNotebookControl(): return self.AddPane1(window, pane_info) if not target.HasNotebook(): self.CreateNotebookBase(self._panes, target) # Add new item to notebook paneInfo.NotebookPage(target.notebook_id) # we also want to remove our captions sometimes self.RemoveAutoNBCaption(paneInfo) self.UpdateNotebook() return True def InsertPane(self, window, pane_info, insert_level=AUI_INSERT_PANE): """ This method is used to insert either a previously unmanaged pane window into the frame manager, or to insert a currently managed pane somewhere else. :meth:`InsertPane` will push all panes, rows, or docks aside and insert the window into the position specified by `pane_info`. Because `pane_info` can specify either a pane, dock row, or dock layer, the `insert_level` parameter is used to disambiguate this. The parameter `insert_level` can take a value of ``AUI_INSERT_PANE``, ``AUI_INSERT_ROW`` or ``AUI_INSERT_DOCK``. :param Window `window`: the window to be inserted and managed; :param `pane_info`: the insert location for the new window; :param integer `insert_level`: the insertion level of the new pane. """ if not window: raise Exception("Invalid window passed to InsertPane.") # shift the panes around, depending on the insert level if insert_level == AUI_INSERT_PANE: self._panes = DoInsertPane(self._panes, pane_info.dock_direction, pane_info.dock_layer, pane_info.dock_row, pane_info.dock_pos) elif insert_level == AUI_INSERT_ROW: self._panes = DoInsertDockRow(self._panes, pane_info.dock_direction, pane_info.dock_layer, pane_info.dock_row) elif insert_level == AUI_INSERT_DOCK: self._panes = DoInsertDockLayer(self._panes, pane_info.dock_direction, pane_info.dock_layer) # if the window already exists, we are basically just moving/inserting the # existing window. If it doesn't exist, we need to add it and insert it existing_pane = self.GetPane(window) indx = self._panes.index(existing_pane) if not existing_pane.IsOk(): return self.AddPane(window, pane_info) else: if pane_info.IsFloating(): existing_pane.Float() if pane_info.floating_pos != wx.Point(-1, -1): existing_pane.FloatingPosition(pane_info.floating_pos) if pane_info.floating_size != wx.Size(-1, -1): existing_pane.FloatingSize(pane_info.floating_size) else: # if the new pane is docked then we should undo maximize self.RestoreMaximizedPane() existing_pane.Direction(pane_info.dock_direction) existing_pane.Layer(pane_info.dock_layer) existing_pane.Row(pane_info.dock_row) existing_pane.Position(pane_info.dock_pos) self._panes[indx] = existing_pane return True def DetachPane(self, window): """ Tells the :class:`AuiManager` to stop managing the pane specified by `window`. The window, if in a floated frame, is reparented to the frame managed by :class:`AuiManager`. :param Window `window`: the window to be un-managed. """ for p in self._panes: if p.window == window: if p.frame: # we have a floating frame which is being detached. We need to # reparent it to self._frame and destroy the floating frame # reduce flicker p.window.SetSize((1, 1)) if p.frame.IsShown(): p.frame.Show(False) if self._action_window == p.frame: self._action_window = None # reparent to self._frame and destroy the pane p.window.Reparent(self._frame) p.frame.SetSizer(None) p.frame.Destroy() p.frame = None elif p.IsNotebookPage(): notebook = self._notebooks[p.notebook_id] id = notebook.GetPageIndex(p.window) notebook.RemovePage(id) p.window.Reparent(self._frame) # make sure there are no references to this pane in our uiparts, # just in case the caller doesn't call Update() immediately after # the DetachPane() call. This prevets obscure crashes which would # happen at window repaint if the caller forgets to call Update() counter = 0 for pi in range(len(self._uiparts)): part = self._uiparts[counter] if part.pane == p: self._uiparts.pop(counter) counter -= 1 counter += 1 self._panes.remove(p) return True return False def ClosePane(self, pane_info): """ Destroys or hides the pane depending on its flags. :param `pane_info`: a :class:`AuiPaneInfo` instance. """ # if we were maximized, restore if pane_info.IsMaximized(): self.RestorePane(pane_info) if pane_info.frame: if self._agwFlags & AUI_MGR_ANIMATE_FRAMES: pane_info.frame.FadeOut() # first, hide the window if pane_info.window and pane_info.window.IsShown(): pane_info.window.Show(False) # make sure that we are the parent of this window if pane_info.window and pane_info.window.GetParent() != self._frame: pane_info.window.Reparent(self._frame) # if we have a frame, destroy it if pane_info.frame: pane_info.frame.Destroy() pane_info.frame = None elif pane_info.IsNotebookPage(): # if we are a notebook page, remove ourselves... # the code would index out of bounds # if the last page of a sub-notebook was closed # because the notebook would be deleted, before this # code is executed. # This code just prevents an out-of bounds error. if self._notebooks: nid = pane_info.notebook_id if nid >= 0 and nid < len(self._notebooks): notebook = self._notebooks[nid] page_idx = notebook.GetPageIndex(pane_info.window) if page_idx >= 0: notebook.RemovePage(page_idx) # now we need to either destroy or hide the pane to_destroy = 0 if pane_info.IsDestroyOnClose(): to_destroy = pane_info.window self.DetachPane(to_destroy) else: if isinstance(pane_info.window, auibar.AuiToolBar) and pane_info.IsFloating(): tb = pane_info.window if pane_info.dock_direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT]: tb.SetAGWWindowStyleFlag(tb.GetAGWWindowStyleFlag() | AUI_TB_VERTICAL) pane_info.Dock().Hide() if pane_info.IsNotebookControl(): notebook = self._notebooks[pane_info.notebook_id] while notebook.GetPageCount(): window = notebook.GetPage(0) notebook.RemovePage(0) info = self.GetPane(window) if info.IsOk(): info.notebook_id = -1 info.dock_direction = AUI_DOCK_NONE # Note: this could change our paneInfo reference ... self.ClosePane(info) if to_destroy: to_destroy.Destroy() def MaximizePane(self, pane_info, savesizes=True): """ Maximizes the input pane. :param `pane_info`: a :class:`AuiPaneInfo` instance. :param bool `savesizes`: whether to save previous dock sizes. """ if savesizes: self.SavePreviousDockSizes(pane_info) for p in self._panes: # save hidden state p.SetFlag(p.savedHiddenState, p.HasFlag(p.optionHidden)) if not p.IsToolbar() and not p.IsFloating(): p.Restore() # hide the pane, because only the newly # maximized pane should show p.Hide() pane_info.previousDockPos = pane_info.dock_pos # mark ourselves maximized pane_info.Maximize() pane_info.Show() self._has_maximized = True # last, show the window if pane_info.window and not pane_info.window.IsShown(): pane_info.window.Show(True) def SavePreviousDockSizes(self, pane_info): """ Stores the previous dock sizes, to be used in a "restore" action later. :param `pane_info`: a :class:`AuiPaneInfo` instance. """ for d in self._docks: if not d.toolbar: for p in d.panes: p.previousDockSize = d.size if pane_info is not p: p.SetFlag(p.needsRestore, True) def RestorePane(self, pane_info): """ Restores the input pane from a previous maximized or minimized state. :param `pane_info`: a :class:`AuiPaneInfo` instance. """ # restore all the panes for p in self._panes: if not p.IsToolbar(): p.SetFlag(p.optionHidden, p.HasFlag(p.savedHiddenState)) pane_info.SetFlag(pane_info.needsRestore, True) # mark ourselves non-maximized pane_info.Restore() self._has_maximized = False self._has_minimized = False # last, show the window if pane_info.window and not pane_info.window.IsShown(): pane_info.window.Show(True) def RestoreMaximizedPane(self): """ Restores the current maximized pane (if any). """ # restore all the panes for p in self._panes: if p.IsMaximized(): self.RestorePane(p) break def ActivatePane(self, window): """ Activates the pane to which `window` is associated. :param `window`: a :class:`Window` derived window. """ if self.GetAGWFlags() & AUI_MGR_ALLOW_ACTIVE_PANE: while window: ret, self._panes = SetActivePane(self._panes, window) if ret: break window = window.GetParent() self.RefreshCaptions() self.FireEvent(wxEVT_AUI_PANE_ACTIVATED, window, canVeto=False) def CreateNotebook(self): """ Creates an automatic :class:`~lib.agw.aui.auibook.AuiNotebook` when a pane is docked on top of another pane. """ notebook = auibook.AuiNotebook(self._frame, -1, wx.Point(0, 0), wx.Size(0, 0), agwStyle=self._autoNBStyle) # This is so we can get the tab-drag event. notebook.GetAuiManager().SetMasterManager(self) notebook.SetArtProvider(self._autoNBTabArt.Clone()) self._notebooks.append(notebook) return notebook def SetAutoNotebookTabArt(self, art): """ Sets the default tab art provider for automatic notebooks. :param `art`: a tab art provider. """ for nb in self._notebooks: nb.SetArtProvider(art.Clone()) nb.Refresh() nb.Update() self._autoNBTabArt = art def GetAutoNotebookTabArt(self): """ Returns the default tab art provider for automatic notebooks. """ return self._autoNBTabArt def SetAutoNotebookStyle(self, agwStyle): """ Sets the default AGW-specific window style for automatic notebooks. :param integer `agwStyle`: the underlying :class:`~lib.agw.aui.auibook.AuiNotebook` window style. This can be a combination of the following bits: ==================================== ================================== Flag name Description ==================================== ================================== ``AUI_NB_TOP`` With this style, tabs are drawn along the top of the notebook ``AUI_NB_LEFT`` With this style, tabs are drawn along the left of the notebook. Not implemented yet. ``AUI_NB_RIGHT`` With this style, tabs are drawn along the right of the notebook. Not implemented yet. ``AUI_NB_BOTTOM`` With this style, tabs are drawn along the bottom of the notebook ``AUI_NB_TAB_SPLIT`` Allows the tab control to be split by dragging a tab ``AUI_NB_TAB_MOVE`` Allows a tab to be moved horizontally by dragging ``AUI_NB_TAB_EXTERNAL_MOVE`` Allows a tab to be moved to another tab control ``AUI_NB_TAB_FIXED_WIDTH`` With this style, all tabs have the same width ``AUI_NB_SCROLL_BUTTONS`` With this style, left and right scroll buttons are displayed ``AUI_NB_WINDOWLIST_BUTTON`` With this style, a drop-down list of windows is available ``AUI_NB_CLOSE_BUTTON`` With this style, a close button is available on the tab bar ``AUI_NB_CLOSE_ON_ACTIVE_TAB`` With this style, a close button is available on the active tab ``AUI_NB_CLOSE_ON_ALL_TABS`` With this style, a close button is available on all tabs ``AUI_NB_MIDDLE_CLICK_CLOSE`` Allows to close :class:`~lib.agw.aui.auibook.AuiNotebook` tabs by mouse middle button click ``AUI_NB_SUB_NOTEBOOK`` This style is used by :class:`AuiManager` to create automatic AuiNotebooks ``AUI_NB_HIDE_ON_SINGLE_TAB`` Hides the tab window if only one tab is present ``AUI_NB_SMART_TABS`` Use Smart Tabbing, like ``Alt`` + ``Tab`` on Windows ``AUI_NB_USE_IMAGES_DROPDOWN`` Uses images on dropdown window list menu instead of check items ``AUI_NB_CLOSE_ON_TAB_LEFT`` Draws the tab close button on the left instead of on the right (a la Camino browser) ``AUI_NB_TAB_FLOAT`` Allows the floating of single tabs. Known limitation: when the notebook is more or less full screen, tabs cannot be dragged far enough outside of the notebook to become floating pages ``AUI_NB_DRAW_DND_TAB`` Draws an image representation of a tab while dragging (on by default) ``AUI_NB_ORDER_BY_ACCESS`` Tab navigation order by last access time for the tabs ``AUI_NB_NO_TAB_FOCUS`` Don't draw tab focus rectangle ==================================== ================================== """ for nb in self._notebooks: nb.SetAGWWindowStyleFlag(agwStyle) nb.Refresh() nb.Update() self._autoNBStyle = agwStyle def GetAutoNotebookStyle(self): """ Returns the default AGW-specific window style for automatic notebooks. :see: :meth:`SetAutoNotebookStyle` method for a list of possible styles. """ return self._autoNBStyle def SavePaneInfo(self, pane): """ This method is similar to :meth:`SavePerspective`, with the exception that it only saves information about a single pane. It is used in combination with :meth:`LoadPaneInfo`. :param `pane`: a :class:`AuiPaneInfo` instance to save. """ result = "name=" + EscapeDelimiters(pane.name) + ";" result += "caption=" + EscapeDelimiters(pane.caption) + ";" result += "state=%u;"%pane.state result += "dir=%d;"%pane.dock_direction result += "layer=%d;"%pane.dock_layer result += "row=%d;"%pane.dock_row result += "pos=%d;"%pane.dock_pos result += "prop=%d;"%pane.dock_proportion result += "bestw=%d;"%pane.best_size.x result += "besth=%d;"%pane.best_size.y result += "minw=%d;"%pane.min_size.x result += "minh=%d;"%pane.min_size.y result += "maxw=%d;"%pane.max_size.x result += "maxh=%d;"%pane.max_size.y result += "floatx=%d;"%pane.floating_pos.x result += "floaty=%d;"%pane.floating_pos.y result += "floatw=%d;"%pane.floating_size.x result += "floath=%d;"%pane.floating_size.y result += "notebookid=%d;"%pane.notebook_id result += "transparent=%d"%pane.transparent return result def LoadPaneInfo(self, pane_part, pane): """ This method is similar to to :meth:`LoadPerspective`, with the exception that it only loads information about a single pane. It is used in combination with :meth:`SavePaneInfo`. :param string `pane_part`: the string to analyze; :param `pane`: the :class:`AuiPaneInfo` structure in which to load `pane_part`. """ # replace escaped characters so we can # split up the string easily pane_part = pane_part.replace("\\|", "\a") pane_part = pane_part.replace("\\;", "\b") options = pane_part.split(";") for items in options: val_name, value = items.split("=") val_name = val_name.strip() if val_name == "name": pane.name = value elif val_name == "caption": pane.caption = value elif val_name == "state": pane.state = int(value) elif val_name == "dir": pane.dock_direction = int(value) elif val_name == "layer": pane.dock_layer = int(value) elif val_name == "row": pane.dock_row = int(value) elif val_name == "pos": pane.dock_pos = int(value) elif val_name == "prop": pane.dock_proportion = int(value) elif val_name == "bestw": pane.best_size.x = int(value) elif val_name == "besth": pane.best_size.y = int(value) pane.best_size = wx.Size(pane.best_size.x, pane.best_size.y) elif val_name == "minw": pane.min_size.x = int(value) elif val_name == "minh": pane.min_size.y = int(value) pane.min_size = wx.Size(pane.min_size.x, pane.min_size.y) elif val_name == "maxw": pane.max_size.x = int(value) elif val_name == "maxh": pane.max_size.y = int(value) pane.max_size = wx.Size(pane.max_size.x, pane.max_size.y) elif val_name == "floatx": pane.floating_pos.x = int(value) elif val_name == "floaty": pane.floating_pos.y = int(value) pane.floating_pos = wx.Point(pane.floating_pos.x, pane.floating_pos.y) elif val_name == "floatw": pane.floating_size.x = int(value) elif val_name == "floath": pane.floating_size.y = int(value) pane.floating_size = wx.Size(pane.floating_size.x, pane.floating_size.y) elif val_name == "notebookid": pane.notebook_id = int(value) elif val_name == "transparent": pane.transparent = int(value) else: raise Exception("Bad perspective string") # replace escaped characters so we can # split up the string easily pane.name = pane.name.replace("\a", "|") pane.name = pane.name.replace("\b", ";") pane.caption = pane.caption.replace("\a", "|") pane.caption = pane.caption.replace("\b", ";") pane_part = pane_part.replace("\a", "|") pane_part = pane_part.replace("\b", ";") return pane def SavePerspective(self): """ Saves the entire user interface layout into an encoded string, which can then be stored by the application (probably using :class:`Config `). When a perspective is restored using :meth:`LoadPerspective`, the entire user interface will return to the state it was when the perspective was saved. """ result = "layout2|" for pane in self._panes: result += self.SavePaneInfo(pane) + "|" for dock in self._docks: result = result + ("dock_size(%d,%d,%d)=%d|")%(dock.dock_direction, dock.dock_layer, dock.dock_row, dock.size) return result def LoadPerspective(self, layout, update=True, restorecaption=False): """ Loads a layout which was saved with :meth:`SavePerspective`. If the `update` flag parameter is ``True``, :meth:`Update` will be automatically invoked, thus realizing the saved perspective on screen. :param string `layout`: a string which contains a saved AUI layout; :param bool `update`: whether to update immediately the window or not; :param bool `restorecaption`: ``False``, restore from persist storage, otherwise use the caption defined in code. """ input = layout # check layout string version # 'layout1' = wxAUI 0.9.0 - wxAUI 0.9.2 # 'layout2' = wxAUI 0.9.2 (wxWidgets 2.8) index = input.find("|") part = input[0:index].strip() input = input[index+1:] if part != "layout2": return False # mark all panes currently managed as docked and hidden saveCapt = {} # see restorecaption param for pane in self._panes: pane.Dock().Hide() saveCapt[pane.name] = pane.caption # clear out the dock array; this will be reconstructed self._docks = [] # replace escaped characters so we can # split up the string easily input = input.replace("\\|", "\a") input = input.replace("\\;", "\b") while 1: pane = AuiPaneInfo() index = input.find("|") pane_part = input[0:index].strip() input = input[index+1:] # if the string is empty, we're done parsing if pane_part == "": break if pane_part[0:9] == "dock_size": index = pane_part.find("=") val_name = pane_part[0:index] value = pane_part[index+1:] index = val_name.find("(") piece = val_name[index+1:] index = piece.find(")") piece = piece[0:index] vals = piece.split(",") dir = int(vals[0]) layer = int(vals[1]) row = int(vals[2]) size = int(value) dock = AuiDockInfo() dock.dock_direction = dir dock.dock_layer = layer dock.dock_row = row dock.size = size self._docks.append(dock) continue # Undo our escaping as LoadPaneInfo needs to take an unescaped # name so it can be called by external callers pane_part = pane_part.replace("\a", "|") pane_part = pane_part.replace("\b", ";") pane = self.LoadPaneInfo(pane_part, pane) p = self.GetPane(pane.name) # restore pane caption from code if restorecaption: if pane.name in saveCapt: pane.Caption(saveCapt[pane.name]) if not p.IsOk(): if pane.IsNotebookControl(): # notebook controls - auto add... self._panes.append(pane) indx = self._panes.index(pane) else: # the pane window couldn't be found # in the existing layout -- skip it continue else: indx = self._panes.index(p) pane.window = p.window pane.frame = p.frame pane.buttons = p.buttons self._panes[indx] = pane if isinstance(pane.window, auibar.AuiToolBar) and (pane.IsFloatable() or pane.IsDockable()): pane.window.SetGripperVisible(True) for p in self._panes: if p.IsMinimized(): self.MinimizePane(p, False) if update: self.Update() return True def GetPanePositionsAndSizes(self, dock): """ Returns all the panes positions and sizes in a dock. :param `dock`: a :class:`AuiDockInfo` instance. """ caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) pane_border_size = self._art.GetMetric(AUI_DOCKART_PANE_BORDER_SIZE) gripper_size = self._art.GetMetric(AUI_DOCKART_GRIPPER_SIZE) positions = [] sizes = [] action_pane = -1 pane_count = len(dock.panes) # find the pane marked as our action pane for pane_i in range(pane_count): pane = dock.panes[pane_i] if pane.HasFlag(AuiPaneInfo.actionPane): if action_pane != -1: raise Exception("Too many action panes!") action_pane = pane_i # set up each panes default position, and # determine the size (width or height, depending # on the dock's orientation) of each pane for pane in dock.panes: positions.append(pane.dock_pos) size = 0 if pane.HasBorder(): size += pane_border_size*2 if dock.IsHorizontal(): if pane.HasGripper() and not pane.HasGripperTop(): size += gripper_size if pane.HasCaptionLeft(): size += caption_size size += pane.best_size.x else: if pane.HasGripper() and pane.HasGripperTop(): size += gripper_size if pane.HasCaption() and not pane.HasCaptionLeft(): size += caption_size size += pane.best_size.y sizes.append(size) # if there is no action pane, just return the default # positions (as specified in pane.pane_pos) if action_pane == -1: return positions, sizes offset = 0 for pane_i in range(action_pane-1, -1, -1): amount = positions[pane_i+1] - (positions[pane_i] + sizes[pane_i]) if amount >= 0: offset += amount else: positions[pane_i] -= -amount offset += sizes[pane_i] # if the dock mode is fixed, make sure none of the panes # overlap we will bump panes that overlap offset = 0 for pane_i in range(action_pane, pane_count): amount = positions[pane_i] - offset if amount >= 0: offset += amount else: positions[pane_i] += -amount offset += sizes[pane_i] return positions, sizes def LayoutAddPane(self, cont, dock, pane, uiparts, spacer_only): """ Adds a pane into the existing layout (in an existing dock). :param `cont`: a :class:`Sizer` object; :param `dock`: the :class:`AuiDockInfo` structure in which to add the pane; :param `pane`: the :class:`AuiPaneInfo` instance to add to the dock; :param `uiparts`: a list of UI parts in the interface; :param bool `spacer_only`: whether to add a simple spacer or a real window. """ #sizer_item = wx.SizerItem() caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) gripper_size = self._art.GetMetric(AUI_DOCKART_GRIPPER_SIZE) pane_border_size = self._art.GetMetric(AUI_DOCKART_PANE_BORDER_SIZE) pane_button_size = self._art.GetMetric(AUI_DOCKART_PANE_BUTTON_SIZE) # find out the orientation of the item (orientation for panes # is the same as the dock's orientation) if dock.IsHorizontal(): orientation = wx.HORIZONTAL else: orientation = wx.VERTICAL # this variable will store the proportion # value that the pane will receive pane_proportion = pane.dock_proportion horz_pane_sizer = wx.BoxSizer(wx.HORIZONTAL) vert_pane_sizer = wx.BoxSizer(wx.VERTICAL) if pane.HasGripper(): part = AuiDockUIPart() if pane.HasGripperTop(): sizer_item = vert_pane_sizer.Add((1, gripper_size), 0, wx.EXPAND) else: sizer_item = horz_pane_sizer.Add((gripper_size, 1), 0, wx.EXPAND) part.type = AuiDockUIPart.typeGripper part.dock = dock part.pane = pane part.button = None part.orientation = orientation part.cont_sizer = horz_pane_sizer part.sizer_item = sizer_item uiparts.append(part) button_count = len(pane.buttons) button_width_total = button_count*pane_button_size if button_count >= 1: button_width_total += 3 caption, captionLeft = pane.HasCaption(), pane.HasCaptionLeft() button_count = len(pane.buttons) if captionLeft: caption_sizer = wx.BoxSizer(wx.VERTICAL) # add pane buttons to the caption dummy_parts = [] for btn_id in range(len(pane.buttons)-1, -1, -1): sizer_item = caption_sizer.Add((caption_size, pane_button_size), 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typePaneButton part.dock = dock part.pane = pane part.button = pane.buttons[btn_id] part.orientation = orientation part.cont_sizer = caption_sizer part.sizer_item = sizer_item dummy_parts.append(part) sizer_item = caption_sizer.Add((caption_size, 1), 1, wx.EXPAND) vert_pane_sizer = wx.BoxSizer(wx.HORIZONTAL) # create the caption sizer part = AuiDockUIPart() part.type = AuiDockUIPart.typeCaption part.dock = dock part.pane = pane part.button = None part.orientation = orientation part.cont_sizer = vert_pane_sizer part.sizer_item = sizer_item caption_part_idx = len(uiparts) uiparts.append(part) uiparts.extend(dummy_parts) elif caption: caption_sizer = wx.BoxSizer(wx.HORIZONTAL) sizer_item = caption_sizer.Add((1, caption_size), 1, wx.EXPAND) # create the caption sizer part = AuiDockUIPart() part.type = AuiDockUIPart.typeCaption part.dock = dock part.pane = pane part.button = None part.orientation = orientation part.cont_sizer = vert_pane_sizer part.sizer_item = sizer_item caption_part_idx = len(uiparts) uiparts.append(part) # add pane buttons to the caption for button in pane.buttons: sizer_item = caption_sizer.Add((pane_button_size, caption_size), 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typePaneButton part.dock = dock part.pane = pane part.button = button part.orientation = orientation part.cont_sizer = caption_sizer part.sizer_item = sizer_item uiparts.append(part) if caption or captionLeft: # if we have buttons, add a little space to the right # of them to ease visual crowding if button_count >= 1: if captionLeft: caption_sizer.Add((caption_size, 3), 0, wx.EXPAND) else: caption_sizer.Add((3, caption_size), 0, wx.EXPAND) # add the caption sizer sizer_item = vert_pane_sizer.Add(caption_sizer, 0, wx.EXPAND) uiparts[caption_part_idx].sizer_item = sizer_item # add the pane window itself if spacer_only or not pane.window: sizer_item = vert_pane_sizer.Add((1, 1), 1, wx.EXPAND) else: sizer_item = vert_pane_sizer.Add(pane.window, 1, wx.EXPAND) vert_pane_sizer.SetItemMinSize(pane.window, (1, 1)) part = AuiDockUIPart() part.type = AuiDockUIPart.typePane part.dock = dock part.pane = pane part.button = None part.orientation = orientation part.cont_sizer = vert_pane_sizer part.sizer_item = sizer_item uiparts.append(part) # determine if the pane should have a minimum size if the pane is # non-resizable (fixed) then we must set a minimum size. Alternatively, # if the pane.min_size is set, we must use that value as well min_size = pane.min_size if pane.IsFixed(): if min_size == wx.Size(-1, -1): min_size = pane.best_size pane_proportion = 0 if min_size != wx.Size(-1, -1): vert_pane_sizer.SetItemMinSize(len(vert_pane_sizer.GetChildren())-1, (min_size.x, min_size.y)) # add the vertical/horizontal sizer (caption, pane window) to the # horizontal sizer (gripper, vertical sizer) horz_pane_sizer.Add(vert_pane_sizer, 1, wx.EXPAND) # finally, add the pane sizer to the dock sizer if pane.HasBorder(): # allowing space for the pane's border sizer_item = cont.Add(horz_pane_sizer, pane_proportion, wx.EXPAND | wx.ALL, pane_border_size) part = AuiDockUIPart() part.type = AuiDockUIPart.typePaneBorder part.dock = dock part.pane = pane part.button = None part.orientation = orientation part.cont_sizer = cont part.sizer_item = sizer_item uiparts.append(part) else: sizer_item = cont.Add(horz_pane_sizer, pane_proportion, wx.EXPAND) return uiparts def LayoutAddDock(self, cont, dock, uiparts, spacer_only): """ Adds a dock into the existing layout. :param `cont`: a :class:`Sizer` object; :param `dock`: the :class:`AuiDockInfo` structure to add to the layout; :param `uiparts`: a list of UI parts in the interface; :param bool `spacer_only`: whether to add a simple spacer or a real window. """ # sizer_item = wx.SizerItem() part = AuiDockUIPart() sash_size = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) orientation = (dock.IsHorizontal() and [wx.HORIZONTAL] or [wx.VERTICAL])[0] # resizable bottom and right docks have a sash before them if not self._has_maximized and not dock.fixed and \ dock.dock_direction in [AUI_DOCK_BOTTOM, AUI_DOCK_RIGHT]: sizer_item = cont.Add((sash_size, sash_size), 0, wx.EXPAND) part.type = AuiDockUIPart.typeDockSizer part.orientation = orientation part.dock = dock part.pane = None part.button = None part.cont_sizer = cont part.sizer_item = sizer_item uiparts.append(part) # create the sizer for the dock dock_sizer = wx.BoxSizer(orientation) # add each pane to the dock has_maximized_pane = False pane_count = len(dock.panes) if dock.fixed: # figure out the real pane positions we will # use, without modifying the each pane's pane_pos member pane_positions, pane_sizes = self.GetPanePositionsAndSizes(dock) offset = 0 for pane_i in range(pane_count): pane = dock.panes[pane_i] pane_pos = pane_positions[pane_i] if pane.IsMaximized(): has_maximized_pane = True amount = pane_pos - offset if amount > 0: if dock.IsVertical(): sizer_item = dock_sizer.Add((1, amount), 0, wx.EXPAND) else: sizer_item = dock_sizer.Add((amount, 1), 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeBackground part.dock = dock part.pane = None part.button = None part.orientation = (orientation==wx.HORIZONTAL and \ [wx.VERTICAL] or [wx.HORIZONTAL])[0] part.cont_sizer = dock_sizer part.sizer_item = sizer_item uiparts.append(part) offset = offset + amount uiparts = self.LayoutAddPane(dock_sizer, dock, pane, uiparts, spacer_only) offset = offset + pane_sizes[pane_i] # at the end add a very small stretchable background area sizer_item = dock_sizer.Add((0, 0), 1, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeBackground part.dock = dock part.pane = None part.button = None part.orientation = orientation part.cont_sizer = dock_sizer part.sizer_item = sizer_item uiparts.append(part) else: for pane_i in range(pane_count): pane = dock.panes[pane_i] if pane.IsMaximized(): has_maximized_pane = True # if this is not the first pane being added, # we need to add a pane sizer if not self._has_maximized and pane_i > 0: sizer_item = dock_sizer.Add((sash_size, sash_size), 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typePaneSizer part.dock = dock part.pane = dock.panes[pane_i-1] part.button = None part.orientation = (orientation==wx.HORIZONTAL and \ [wx.VERTICAL] or [wx.HORIZONTAL])[0] part.cont_sizer = dock_sizer part.sizer_item = sizer_item uiparts.append(part) uiparts = self.LayoutAddPane(dock_sizer, dock, pane, uiparts, spacer_only) if dock.dock_direction == AUI_DOCK_CENTER or has_maximized_pane: sizer_item = cont.Add(dock_sizer, 1, wx.EXPAND) else: sizer_item = cont.Add(dock_sizer, 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeDock part.dock = dock part.pane = None part.button = None part.orientation = orientation part.cont_sizer = cont part.sizer_item = sizer_item uiparts.append(part) if dock.IsHorizontal(): cont.SetItemMinSize(dock_sizer, (0, dock.size)) else: cont.SetItemMinSize(dock_sizer, (dock.size, 0)) # top and left docks have a sash after them if not self._has_maximized and not dock.fixed and \ dock.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_LEFT]: sizer_item = cont.Add((sash_size, sash_size), 0, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeDockSizer part.dock = dock part.pane = None part.button = None part.orientation = orientation part.cont_sizer = cont part.sizer_item = sizer_item uiparts.append(part) return uiparts def LayoutAll(self, panes, docks, uiparts, spacer_only=False, oncheck=True): """ Layouts all the UI structures in the interface. :param `panes`: a list of :class:`AuiPaneInfo` instances; :param `docks`: a list of :class:`AuiDockInfo` classes; :param `uiparts`: a list of UI parts in the interface; :param bool `spacer_only`: whether to add a simple spacer or a real window; :param bool `oncheck`: whether to store the results in a class member or not. """ container = wx.BoxSizer(wx.VERTICAL) pane_border_size = self._art.GetMetric(AUI_DOCKART_PANE_BORDER_SIZE) caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) cli_size = self._frame.GetClientSize() # empty all docks out for dock in docks: dock.panes = [] if dock.fixed: # always reset fixed docks' sizes, because # the contained windows may have been resized dock.size = 0 dock_count = len(docks) # iterate through all known panes, filing each # of them into the appropriate dock. If the # pane does not exist in the dock, add it for p in panes: # don't layout hidden panes. if p.IsShown(): # find any docks with the same dock direction, dock layer, and # dock row as the pane we are working on arr = FindDocks(docks, p.dock_direction, p.dock_layer, p.dock_row) if arr: dock = arr[0] else: # dock was not found, so we need to create a new one d = AuiDockInfo() d.dock_direction = p.dock_direction d.dock_layer = p.dock_layer d.dock_row = p.dock_row docks.append(d) dock = docks[-1] if p.HasFlag(p.needsRestore) and not p.HasFlag(p.wasMaximized): isHor = dock.IsHorizontal() sashSize = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) # get the sizes of any docks that might # overlap with our restored dock # make list of widths or heights from the size in the dock rects sizes = [d.rect[2:][isHor] for \ d in docks if d.IsOk() and \ (d.IsHorizontal() == isHor) and \ not d.toolbar and \ d.dock_direction != AUI_DOCK_CENTER] frameRect = GetInternalFrameRect(self._frame, self._docks) # set max size allowing for sashes and absolute minimum maxsize = frameRect[2:][isHor] - sum(sizes) - (len(sizes)*10) - (sashSize*len(sizes)) dock.size = min(p.previousDockSize,maxsize) else: dock.size = 0 if p.HasFlag(p.wasMaximized): self.MaximizePane(p, savesizes=False) p.SetFlag(p.wasMaximized, False) if p.HasFlag(p.needsRestore): if p.previousDockPos is not None: DoInsertPane(dock.panes, dock.dock_direction, dock.dock_layer, dock.dock_row, p.previousDockPos) p.dock_pos = p.previousDockPos p.previousDockPos = None p.SetFlag(p.needsRestore, False) if p.IsDocked(): # remove the pane from any existing docks except this one docks = RemovePaneFromDocks(docks, p, dock) # pane needs to be added to the dock, # if it doesn't already exist if not FindPaneInDock(dock, p.window): dock.panes.append(p) else: # remove the pane from any existing docks docks = RemovePaneFromDocks(docks, p) # remove any empty docks docks = [dock for dock in docks if dock.panes] dock_count = len(docks) # configure the docks further for ii, dock in enumerate(docks): # sort the dock pane array by the pane's # dock position (dock_pos), in ascending order dock_pane_count = len(dock.panes) if dock_pane_count > 1: #~ dock.panes.sort(PaneSortFunc) dock.panes.sort(key = lambda pane: pane.dock_pos) # for newly created docks, set up their initial size if dock.size == 0: size = 0 for pane in dock.panes: pane_size = pane.best_size if pane_size == wx.Size(-1, -1): pane_size = pane.min_size if pane_size == wx.Size(-1, -1) and pane.window: pane_size = pane.window.GetSize() if dock.IsHorizontal(): size = max(pane_size.y, size) else: size = max(pane_size.x, size) # add space for the border (two times), but only # if at least one pane inside the dock has a pane border for pane in dock.panes: if pane.HasBorder(): size = size + pane_border_size*2 break # if pane is on the top or bottom, add the caption height, # but only if at least one pane inside the dock has a caption if dock.IsHorizontal(): for pane in dock.panes: if pane.HasCaption() and not pane.HasCaptionLeft(): size = size + caption_size break else: for pane in dock.panes: if pane.HasCaptionLeft() and not pane.HasCaption(): size = size + caption_size break # new dock's size may not be more than the dock constraint # parameter specifies. See SetDockSizeConstraint() max_dock_x_size = int(self._dock_constraint_x*float(cli_size.x)) max_dock_y_size = int(self._dock_constraint_y*float(cli_size.y)) if tuple(cli_size) <= tuple(wx.Size(20, 20)): max_dock_x_size = 10000 max_dock_y_size = 10000 if dock.IsHorizontal(): size = min(size, max_dock_y_size) else: size = min(size, max_dock_x_size) # absolute minimum size for a dock is 10 pixels if size < 10: size = 10 dock.size = size # determine the dock's minimum size plus_border = False plus_caption = False plus_caption_left = False dock_min_size = 0 for pane in dock.panes: if pane.min_size != wx.Size(-1, -1): if pane.HasBorder(): plus_border = True if pane.HasCaption(): plus_caption = True if pane.HasCaptionLeft(): plus_caption_left = True if dock.IsHorizontal(): if pane.min_size.y > dock_min_size: dock_min_size = pane.min_size.y else: if pane.min_size.x > dock_min_size: dock_min_size = pane.min_size.x if plus_border: dock_min_size += pane_border_size*2 if plus_caption and dock.IsHorizontal(): dock_min_size += caption_size if plus_caption_left and dock.IsVertical(): dock_min_size += caption_size dock.min_size = dock_min_size # if the pane's current size is less than it's # minimum, increase the dock's size to it's minimum if dock.size < dock.min_size: dock.size = dock.min_size # determine the dock's mode (fixed or proportional) # determine whether the dock has only toolbars action_pane_marked = False dock.fixed = True dock.toolbar = True for pane in dock.panes: if not pane.IsFixed(): dock.fixed = False if not pane.IsToolbar(): dock.toolbar = False if pane.HasFlag(AuiPaneInfo.optionDockFixed): dock.fixed = True if pane.HasFlag(AuiPaneInfo.actionPane): action_pane_marked = True # if the dock mode is proportional and not fixed-pixel, # reassign the dock_pos to the sequential 0, 1, 2, 3 # e.g. remove gaps like 1, 2, 30, 500 if not dock.fixed: for jj in range(dock_pane_count): pane = dock.panes[jj] pane.dock_pos = jj # if the dock mode is fixed, and none of the panes # are being moved right now, make sure the panes # do not overlap each other. If they do, we will # adjust the panes' positions if dock.fixed and not action_pane_marked: pane_positions, pane_sizes = self.GetPanePositionsAndSizes(dock) offset = 0 for jj in range(dock_pane_count): pane = dock.panes[jj] pane.dock_pos = pane_positions[jj] amount = pane.dock_pos - offset if amount >= 0: offset += amount else: pane.dock_pos += -amount offset += pane_sizes[jj] dock.panes[jj] = pane if oncheck: self._docks[ii] = dock # shrink docks if needed ## docks = self.SmartShrink(docks, AUI_DOCK_TOP) ## docks = self.SmartShrink(docks, AUI_DOCK_LEFT) if oncheck: self._docks = docks # discover the maximum dock layer max_layer = 0 dock_count = len(docks) for ii in range(dock_count): max_layer = max(max_layer, docks[ii].dock_layer) # clear out uiparts uiparts = [] # create a bunch of box sizers, # from the innermost level outwards. cont = None middle = None if oncheck: docks = self._docks for layer in range(max_layer+1): # find any docks in this layer arr = FindDocks(docks, -1, layer, -1) # if there aren't any, skip to the next layer if not arr: continue old_cont = cont # create a container which will hold this layer's # docks (top, bottom, left, right) cont = wx.BoxSizer(wx.VERTICAL) # find any top docks in this layer arr = FindDocks(docks, AUI_DOCK_TOP, layer, -1) for row in arr: uiparts = self.LayoutAddDock(cont, row, uiparts, spacer_only) # fill out the middle layer (which consists # of left docks, content area and right docks) middle = wx.BoxSizer(wx.HORIZONTAL) # find any left docks in this layer arr = FindDocks(docks, AUI_DOCK_LEFT, layer, -1) for row in arr: uiparts = self.LayoutAddDock(middle, row, uiparts, spacer_only) # add content dock (or previous layer's sizer # to the middle if not old_cont: # find any center docks arr = FindDocks(docks, AUI_DOCK_CENTER, -1, -1) if arr: for row in arr: uiparts = self.LayoutAddDock(middle, row, uiparts, spacer_only) elif not self._has_maximized: # there are no center docks, add a background area sizer_item = middle.Add((1, 1), 1, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeBackground part.pane = None part.dock = None part.button = None part.cont_sizer = middle part.sizer_item = sizer_item uiparts.append(part) else: middle.Add(old_cont, 1, wx.EXPAND) # find any right docks in this layer arr = FindDocks(docks, AUI_DOCK_RIGHT, layer, -1, reverse=True) for row in arr: uiparts = self.LayoutAddDock(middle, row, uiparts, spacer_only) if len(middle.GetChildren()) > 0: cont.Add(middle, 1, wx.EXPAND) # find any bottom docks in this layer arr = FindDocks(docks, AUI_DOCK_BOTTOM, layer, -1, reverse=True) for row in arr: uiparts = self.LayoutAddDock(cont, row, uiparts, spacer_only) if not cont: # no sizer available, because there are no docks, # therefore we will create a simple background area cont = wx.BoxSizer(wx.VERTICAL) sizer_item = cont.Add((1, 1), 1, wx.EXPAND) part = AuiDockUIPart() part.type = AuiDockUIPart.typeBackground part.pane = None part.dock = None part.button = None part.cont_sizer = middle part.sizer_item = sizer_item uiparts.append(part) if oncheck: self._uiparts = uiparts self._docks = docks container.Add(cont, 1, wx.EXPAND) if oncheck: return container else: return container, panes, docks, uiparts def SetDockSizeConstraint(self, width_pct, height_pct): """ When a user creates a new dock by dragging a window into a docked position, often times the large size of the window will create a dock that is unwieldly large. :class:`AuiManager` by default limits the size of any new dock to 1/3 of the window size. For horizontal docks, this would be 1/3 of the window height. For vertical docks, 1/3 of the width. Calling this function will adjust this constraint value. The numbers must be between 0.0 and 1.0. For instance, calling :meth:`SetDockSizeConstraint` with (0.5, 0.5) will cause new docks to be limited to half of the size of the entire managed window. :param float `width_pct`: a number representing the `x` dock size constraint; :param float `width_pct`: a number representing the `y` dock size constraint. """ self._dock_constraint_x = max(0.0, min(1.0, width_pct)) self._dock_constraint_y = max(0.0, min(1.0, height_pct)) def GetDockSizeConstraint(self): """ Returns the current dock constraint values. :see: :meth:`SetDockSizeConstraint` """ return self._dock_constraint_x, self._dock_constraint_y def Update(self): """ This method is called after any number of changes are made to any of the managed panes. :meth:`Update` must be invoked after :meth:`AddPane` or :meth:`InsertPane` are called in order to "realize" or "commit" the changes. In addition, any number of changes may be made to :class:`AuiManager` structures (retrieved with :meth:`GetPane`), but to realize the changes, :meth:`Update` must be called. This construction allows pane flicker to be avoided by updating the whole layout at one time. """ self._hover_button = None self._action_part = None # destroy floating panes which have been # redocked or are becoming non-floating for p in self._panes: if p.IsFloating() or not p.frame: continue # because the pane is no longer in a floating, we need to # reparent it to self._frame and destroy the floating frame # reduce flicker p.window.SetSize((1, 1)) # the following block is a workaround for bug #1531361 # (see wxWidgets sourceforge page). On wxGTK (only), when # a frame is shown/hidden, a move event unfortunately # also gets fired. Because we may be dragging around # a pane, we need to cancel that action here to prevent # a spurious crash. if self._action_window == p.frame: if self._frame.HasCapture(): self._frame.ReleaseMouse() self._action = actionNone self._action_window = None # hide the frame if p.frame.IsShown(): p.frame.Show(False) if self._action_window == p.frame: self._action_window = None # reparent to self._frame and destroy the pane p.window.Reparent(self._frame) if isinstance(p.window, auibar.AuiToolBar): p.window.SetAuiManager(self) if p.frame: p.frame.SetSizer(None) p.frame.Destroy() p.frame = None # Only the master manager should create/destroy notebooks... if not self._masterManager: self.UpdateNotebook() # delete old sizer first self._frame.SetSizer(None) # create a layout for all of the panes sizer = self.LayoutAll(self._panes, self._docks, self._uiparts, False) # hide or show panes as necessary, # and float panes as necessary pane_count = len(self._panes) for ii in range(pane_count): p = self._panes[ii] pFrame = p.frame if p.IsFloating(): if pFrame is None: # we need to create a frame for this # pane, which has recently been floated frame = self.CreateFloatingFrame(self._frame, p) # on MSW and Mac, if the owner desires transparent dragging, and # the dragging is happening right now, then the floating # window should have this style by default if self._action in [actionDragFloatingPane, actionDragToolbarPane] and \ self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: frame.SetTransparent(150) if p.IsToolbar(): bar = p.window if isinstance(bar, auibar.AuiToolBar): bar.SetGripperVisible(False) agwStyle = bar.GetAGWWindowStyleFlag() bar.SetAGWWindowStyleFlag(agwStyle & ~AUI_TB_VERTICAL) bar.Realize() s = p.window.GetMinSize() p.BestSize(s) p.FloatingSize(wx.DefaultSize) frame.SetPaneWindow(p) p.needsTransparency = True p.frame = pFrame = frame if p.IsShown() and not frame.IsShown(): frame.Show() frame.Update() else: # frame already exists, make sure it's position # and size reflect the information in AuiPaneInfo if pFrame.GetPosition() != p.floating_pos or pFrame.GetSize() != p.floating_size: pFrame.SetSize(p.floating_pos.x, p.floating_pos.y, p.floating_size.x, p.floating_size.y, wx.SIZE_USE_EXISTING) # update whether the pane is resizable or not style = p.frame.GetWindowStyleFlag() if p.IsFixed(): style &= ~wx.RESIZE_BORDER else: style |= wx.RESIZE_BORDER p.frame.SetWindowStyleFlag(style) if pFrame.IsShown() != p.IsShown(): p.needsTransparency = True pFrame.Show(p.IsShown()) if pFrame.GetTitle() != p.caption: pFrame.SetTitle(p.caption) if p.icon.IsOk(): pFrame.SetIcon(wx.Icon(p.icon)) else: if p.IsToolbar(): # self.SwitchToolBarOrientation(p) p.best_size = p.window.GetBestSize() if p.window and not p.IsNotebookPage() and p.window.IsShown() != p.IsShown(): p.window.Show(p.IsShown()) if pFrame and p.needsTransparency: if pFrame.IsShown() and pFrame._transparent != p.transparent: pFrame.SetTransparent(p.transparent) pFrame._transparent = p.transparent p.needsTransparency = False # if "active panes" are no longer allowed, clear # any optionActive values from the pane states if self._agwFlags & AUI_MGR_ALLOW_ACTIVE_PANE == 0: p.state &= ~AuiPaneInfo.optionActive self._panes[ii] = p old_pane_rects = [] pane_count = len(self._panes) for p in self._panes: r = wx.Rect() if p.window and p.IsShown() and p.IsDocked(): r = p.rect old_pane_rects.append(r) # apply the new sizer self._frame.SetSizer(sizer) self._frame.SetAutoLayout(False) self.DoFrameLayout() # now that the frame layout is done, we need to check # the new pane rectangles against the old rectangles that # we saved a few lines above here. If the rectangles have # changed, the corresponding panes must also be updated for ii in range(pane_count): p = self._panes[ii] if p.window and p.IsShown() and p.IsDocked(): if p.rect != old_pane_rects[ii]: p.window.Refresh() p.window.Update() if wx.Platform == "__WXMAC__": self._frame.Refresh() else: self.Repaint() if not self._masterManager: e = self.FireEvent(wxEVT_AUI_PERSPECTIVE_CHANGED, None, canVeto=False) def UpdateNotebook(self): """ Updates the automatic :class:`~lib.agw.aui.auibook.AuiNotebook` in the layout (if any exists). """ # Workout how many notebooks we need. max_notebook = -1 # destroy floating panes which have been # redocked or are becoming non-floating for paneInfo in self._panes: if max_notebook < paneInfo.notebook_id: max_notebook = paneInfo.notebook_id # We are the master of our domain extra_notebook = len(self._notebooks) max_notebook += 1 for i in range(extra_notebook, max_notebook): self.CreateNotebook() # Remove pages from notebooks that no-longer belong there ... for nb, notebook in enumerate(self._notebooks): pages = notebook.GetPageCount() pageCounter, allPages = 0, pages # Check each tab ... for page in range(pages): if page >= allPages: break window = notebook.GetPage(pageCounter) paneInfo = self.GetPane(window) if paneInfo.IsOk() and paneInfo.notebook_id != nb: notebook.RemovePage(pageCounter) window.Hide() window.Reparent(self._frame) pageCounter -= 1 allPages -= 1 pageCounter += 1 notebook.DoSizing() # Add notebook pages that aren't there already... for paneInfo in self._panes: if paneInfo.IsNotebookPage(): title = (paneInfo.caption == "" and [paneInfo.name] or [paneInfo.caption])[0] notebook = self._notebooks[paneInfo.notebook_id] page_id = notebook.GetPageIndex(paneInfo.window) if page_id < 0: paneInfo.window.Reparent(notebook) notebook.AddPage(paneInfo.window, title, True, paneInfo.icon) # Update title and icon ... else: notebook.SetPageText(page_id, title) notebook.SetPageBitmap(page_id, paneInfo.icon) notebook.DoSizing() # Wire-up newly created notebooks elif paneInfo.IsNotebookControl() and not paneInfo.window: paneInfo.window = self._notebooks[paneInfo.notebook_id] # Delete empty notebooks, and convert notebooks with 1 page to # normal panes... remap_ids = [-1]*len(self._notebooks) nb_idx = 0 for nb, notebook in enumerate(self._notebooks): if notebook.GetPageCount() == 1: # Convert notebook page to pane... window = notebook.GetPage(0) child_pane = self.GetPane(window) notebook_pane = self.GetPane(notebook) if child_pane.IsOk() and notebook_pane.IsOk(): child_pane.SetDockPos(notebook_pane) child_pane.window.Hide() child_pane.window.Reparent(self._frame) child_pane.frame = None child_pane.notebook_id = -1 if notebook_pane.IsFloating(): child_pane.Float() self.DetachPane(notebook) notebook.RemovePage(0) notebook.Destroy() else: raise Exception("Odd notebook docking") elif notebook.GetPageCount() == 0: self.DetachPane(notebook) notebook.Destroy() else: # Correct page ordering. The original wxPython code # for this did not work properly, and would misplace # windows causing errors. notebook.Freeze() self._notebooks[nb_idx] = notebook pages = notebook.GetPageCount() selected = notebook.GetPage(notebook.GetSelection()) # Take each page out of the notebook, group it with # its current pane, and sort the list by pane.dock_pos # order pages_and_panes = [] for idx in reversed(list(range(pages))): page = notebook.GetPage(idx) pane = self.GetPane(page) pages_and_panes.append((page, pane)) notebook.RemovePage(idx) sorted_pnp = sorted(pages_and_panes, key=lambda tup: tup[1].dock_pos) # Grab the attributes from the panes which are ordered # correctly, and copy those attributes to the original # panes. (This avoids having to change the ordering # of self._panes) Then, add the page back into the notebook sorted_attributes = [self.GetAttributes(tup[1]) for tup in sorted_pnp] for attrs, tup in zip(sorted_attributes, pages_and_panes): pane = tup[1] self.SetAttributes(pane, attrs) notebook.AddPage(pane.window, pane.caption) notebook.SetSelection(notebook.GetPageIndex(selected), True) notebook.DoSizing() notebook.Thaw() # It's a keeper. remap_ids[nb] = nb_idx nb_idx += 1 # Apply remap... nb_count = len(self._notebooks) if nb_count != nb_idx: self._notebooks = self._notebooks[0:nb_idx] for p in self._panes: if p.notebook_id >= 0: p.notebook_id = remap_ids[p.notebook_id] if p.IsNotebookControl(): p.SetNameFromNotebookId() # Make sure buttons are correct ... for notebook in self._notebooks: want_max = True want_min = True want_close = True pages = notebook.GetPageCount() for page in range(pages): win = notebook.GetPage(page) pane = self.GetPane(win) if pane.IsOk(): if not pane.HasCloseButton(): want_close = False if not pane.HasMaximizeButton(): want_max = False if not pane.HasMinimizeButton(): want_min = False notebook_pane = self.GetPane(notebook) if notebook_pane.IsOk(): if notebook_pane.HasMinimizeButton() != want_min: if want_min: button = AuiPaneButton(AUI_BUTTON_MINIMIZE) notebook_pane.state |= AuiPaneInfo.buttonMinimize notebook_pane.buttons.append(button) # todo: remove min/max if notebook_pane.HasMaximizeButton() != want_max: if want_max: button = AuiPaneButton(AUI_BUTTON_MAXIMIZE_RESTORE) notebook_pane.state |= AuiPaneInfo.buttonMaximize notebook_pane.buttons.append(button) # todo: remove min/max if notebook_pane.HasCloseButton() != want_close: if want_close: button = AuiPaneButton(AUI_BUTTON_CLOSE) notebook_pane.state |= AuiPaneInfo.buttonClose notebook_pane.buttons.append(button) # todo: remove close def SmartShrink(self, docks, direction): """ Used to intelligently shrink the docks' size (if needed). :param `docks`: a list of :class:`AuiDockInfo` instances; :param integer `direction`: the direction in which to shrink. """ sashSize = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) clientSize = self._frame.GetClientSize() ourDocks = FindDocks(docks, direction, -1, -1) oppositeDocks = FindOppositeDocks(docks, direction) oppositeSize = self.GetOppositeDockTotalSize(docks, direction) ourSize = 0 for dock in ourDocks: ourSize += dock.size if not dock.toolbar: ourSize += sashSize shrinkSize = ourSize + oppositeSize if direction == AUI_DOCK_TOP or direction == AUI_DOCK_BOTTOM: shrinkSize -= clientSize.y else: shrinkSize -= clientSize.x if shrinkSize <= 0: return docks # Combine arrays for dock in oppositeDocks: ourDocks.append(dock) oppositeDocks = [] for dock in ourDocks: if dock.toolbar or not dock.resizable: continue dockRange = dock.size - dock.min_size if dock.min_size == 0: dockRange -= sashSize if direction == AUI_DOCK_TOP or direction == AUI_DOCK_BOTTOM: dockRange -= caption_size if dockRange >= shrinkSize: dock.size -= shrinkSize return docks else: dock.size -= dockRange shrinkSize -= dockRange return docks def UpdateDockingGuides(self, paneInfo): """ Updates the docking guide windows positions and appearance. :param `paneInfo`: a :class:`AuiPaneInfo` instance. """ if len(self._guides) == 0: self.CreateGuideWindows() captionSize = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) frameRect = GetInternalFrameRect(self._frame, self._docks) mousePos = wx.GetMousePosition() for indx, guide in enumerate(self._guides): pt = wx.Point() guide_size = guide.host.GetSize() if not guide.host: raise Exception("Invalid docking host") direction = guide.dock_direction if direction == AUI_DOCK_LEFT: pt.x = frameRect.x + guide_size.x / 2 + 16 pt.y = frameRect.y + frameRect.height / 2 elif direction == AUI_DOCK_TOP: pt.x = frameRect.x + frameRect.width / 2 pt.y = frameRect.y + guide_size.y / 2 + 16 elif direction == AUI_DOCK_RIGHT: pt.x = frameRect.x + frameRect.width - guide_size.x / 2 - 16 pt.y = frameRect.y + frameRect.height / 2 elif direction == AUI_DOCK_BOTTOM: pt.x = frameRect.x + frameRect.width / 2 pt.y = frameRect.y + frameRect.height - guide_size.y / 2 - 16 elif direction == AUI_DOCK_CENTER: rc = paneInfo.window.GetScreenRect() pt.x = rc.x + rc.width / 2 pt.y = rc.y + rc.height / 2 if paneInfo.HasCaption(): pt.y -= captionSize / 2 elif paneInfo.HasCaptionLeft(): pt.x -= captionSize / 2 # guide will be centered around point 'pt' targetPosition = wx.Point(pt.x - guide_size.x / 2, pt.y - guide_size.y / 2) if guide.host.GetPosition() != targetPosition: guide.host.Move(targetPosition) guide.host.AeroMove(targetPosition) if guide.dock_direction == AUI_DOCK_CENTER: guide.host.ValidateNotebookDocking(paneInfo.IsNotebookDockable()) guide.host.UpdateDockGuide(mousePos) paneInfo.window.Lower() def DoFrameLayout(self): """ This is an internal function which invokes :meth:`Sizer.Layout() ` on the frame's main sizer, then measures all the various UI items and updates their internal rectangles. :note: This should always be called instead of calling `self._managed_window.Layout()` directly. """ self._frame.Layout() for part in self._uiparts: # get the rectangle of the UI part # originally, this code looked like this: # part.rect = wx.Rect(part.sizer_item.GetPosition(), # part.sizer_item.GetSize()) # this worked quite well, with one exception: the mdi # client window had a "deferred" size variable # that returned the wrong size. It looks like # a bug in wx, because the former size of the window # was being returned. So, we will retrieve the part's # rectangle via other means part.rect = part.sizer_item.GetRect() flag = part.sizer_item.GetFlag() border = part.sizer_item.GetBorder() if flag & wx.TOP: part.rect.y -= border part.rect.height += border if flag & wx.LEFT: part.rect.x -= border part.rect.width += border if flag & wx.BOTTOM: part.rect.height += border if flag & wx.RIGHT: part.rect.width += border if part.type == AuiDockUIPart.typeDock: part.dock.rect = part.rect if part.type == AuiDockUIPart.typePane: part.pane.rect = part.rect def GetPanePart(self, wnd): """ Looks up the pane border UI part of the pane specified. This allows the caller to get the exact rectangle of the pane in question, including decorations like caption and border. :param Window `wnd`: the window to which the pane border belongs to. """ for part in self._uiparts: if part.type == AuiDockUIPart.typePaneBorder and \ part.pane and part.pane.window == wnd: return part for part in self._uiparts: if part.type == AuiDockUIPart.typePane and \ part.pane and part.pane.window == wnd: return part return None def GetDockPixelOffset(self, test): """ This is an internal function which returns a dock's offset in pixels from the left side of the window (for horizontal docks) or from the top of the window (for vertical docks). This value is necessary for calculating fixed-pane/toolbar offsets when they are dragged. :param `test`: a fake :class:`AuiPaneInfo` for testing purposes. """ # the only way to accurately calculate the dock's # offset is to actually run a theoretical layout docks, panes = CopyDocksAndPanes2(self._docks, self._panes) panes.append(test) sizer, panes, docks, uiparts = self.LayoutAll(panes, docks, [], True, False) client_size = self._frame.GetClientSize() sizer.SetDimension(0, 0, client_size.x, client_size.y) sizer.Layout() for part in uiparts: pos = part.sizer_item.GetPosition() size = part.sizer_item.GetSize() part.rect = wx.Rect(pos, size) if part.type == AuiDockUIPart.typeDock: part.dock.rect = part.rect sizer.Destroy() for dock in docks: if test.dock_direction == dock.dock_direction and \ test.dock_layer == dock.dock_layer and \ test.dock_row == dock.dock_row: if dock.IsVertical(): return dock.rect.y else: return dock.rect.x return 0 def GetPartnerDock(self, dock): """ Returns the partner dock for the input dock. :param `dock`: a :class:`AuiDockInfo` instance. """ for layer in range(dock.dock_layer, -1, -1): bestDock = None for tmpDock in self._docks: if tmpDock.dock_layer != layer: continue if tmpDock.dock_direction != dock.dock_direction: continue if tmpDock.dock_layer < dock.dock_layer: if not bestDock or tmpDock.dock_row < bestDock.dock_row: bestDock = tmpDock elif tmpDock.dock_row > dock.dock_row: if not bestDock or tmpDock.dock_row > bestDock.dock_row: bestDock = tmpDock if bestDock: return bestDock return None def GetPartnerPane(self, dock, pane): """ Returns the partner pane for the input pane. They both need to live in the same :class:`AuiDockInfo`. :param `dock`: a :class:`AuiDockInfo` instance; :param `pane`: a :class:`AuiPaneInfo` class. """ panePosition = -1 for i, tmpPane in enumerate(dock.panes): if tmpPane.window == pane.window: panePosition = i elif not tmpPane.IsFixed() and panePosition != -1: return tmpPane return None def GetTotalPixSizeAndProportion(self, dock): """ Returns the dimensions and proportion of the input dock. :param `dock`: the :class:`AuiDockInfo` structure to analyze. """ totalPixsize = 0 totalProportion = 0 # determine the total proportion of all resizable panes, # and the total size of the dock minus the size of all # the fixed panes for tmpPane in dock.panes: if tmpPane.IsFixed(): continue totalProportion += tmpPane.dock_proportion if dock.IsHorizontal(): totalPixsize += tmpPane.rect.width else: totalPixsize += tmpPane.rect.height ## if tmpPane.min_size.IsFullySpecified(): ## ## if dock.IsHorizontal(): ## totalPixsize -= tmpPane.min_size.x ## else: ## totalPixsize -= tmpPane.min_size.y return totalPixsize, totalProportion def GetOppositeDockTotalSize(self, docks, direction): """ Returns the dimensions of the dock which lives opposite of the input dock. :param `docks`: a list of :class:`AuiDockInfo` structures to analyze; :param integer `direction`: the direction in which to look for the opposite dock. """ sash_size = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) pane_border_size = self._art.GetMetric(AUI_DOCKART_PANE_BORDER_SIZE) minSizeMax = 0 result = sash_size vertical = False if direction in [AUI_DOCK_TOP, AUI_DOCK_BOTTOM]: vertical = True # Get minimum size of the most inner area for tmpDock in docks: if tmpDock.dock_layer != 0: continue if tmpDock.dock_direction != AUI_DOCK_CENTER and tmpDock.IsVertical() != vertical: continue for tmpPane in tmpDock.panes: minSize = pane_border_size*2 - sash_size if vertical: minSize += tmpPane.min_size.y + caption_size else: minSize += tmpPane.min_size.x if minSize > minSizeMax: minSizeMax = minSize result += minSizeMax # Get opposite docks oppositeDocks = FindOppositeDocks(docks, direction) # Sum size of the opposite docks and their sashes for dock in oppositeDocks: result += dock.size # if it's not a toolbar add the sash_size too if not dock.toolbar: result += sash_size return result def CalculateDockSizerLimits(self, dock): """ Calculates the minimum and maximum sizes allowed for the input dock. :param `dock`: the :class:`AuiDockInfo` structure to analyze. """ docks, panes = CopyDocksAndPanes2(self._docks, self._panes) sash_size = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) caption_size = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) opposite_size = self.GetOppositeDockTotalSize(docks, dock.dock_direction) for tmpDock in docks: if tmpDock.dock_direction == dock.dock_direction and \ tmpDock.dock_layer == dock.dock_layer and \ tmpDock.dock_row == dock.dock_row: tmpDock.size = 1 break sizer, panes, docks, uiparts = self.LayoutAll(panes, docks, [], True, False) client_size = self._frame.GetClientSize() sizer.SetDimension(0, 0, client_size.x, client_size.y) sizer.Layout() for part in uiparts: part.rect = wx.Rect(part.sizer_item.GetPosition(), part.sizer_item.GetSize()) if part.type == AuiDockUIPart.typeDock: part.dock.rect = part.rect sizer.Destroy() new_dock = None for tmpDock in docks: if tmpDock.dock_direction == dock.dock_direction and \ tmpDock.dock_layer == dock.dock_layer and \ tmpDock.dock_row == dock.dock_row: new_dock = tmpDock break partnerDock = self.GetPartnerDock(dock) if partnerDock: partnerRange = partnerDock.size - partnerDock.min_size if partnerDock.min_size == 0: partnerRange -= sash_size if dock.IsHorizontal(): partnerRange -= caption_size direction = dock.dock_direction if direction == AUI_DOCK_LEFT: minPix = new_dock.rect.x + new_dock.rect.width maxPix = dock.rect.x + dock.rect.width maxPix += partnerRange elif direction == AUI_DOCK_TOP: minPix = new_dock.rect.y + new_dock.rect.height maxPix = dock.rect.y + dock.rect.height maxPix += partnerRange elif direction == AUI_DOCK_RIGHT: minPix = dock.rect.x - partnerRange - sash_size maxPix = new_dock.rect.x - sash_size elif direction == AUI_DOCK_BOTTOM: minPix = dock.rect.y - partnerRange - sash_size maxPix = new_dock.rect.y - sash_size return minPix, maxPix direction = new_dock.dock_direction if direction == AUI_DOCK_LEFT: minPix = new_dock.rect.x + new_dock.rect.width maxPix = client_size.x - opposite_size - sash_size elif direction == AUI_DOCK_TOP: minPix = new_dock.rect.y + new_dock.rect.height maxPix = client_size.y - opposite_size - sash_size elif direction == AUI_DOCK_RIGHT: minPix = opposite_size maxPix = new_dock.rect.x - sash_size elif direction == AUI_DOCK_BOTTOM: minPix = opposite_size maxPix = new_dock.rect.y - sash_size return minPix, maxPix def CalculatePaneSizerLimits(self, dock, pane): """ Calculates the minimum and maximum sizes allowed for the input pane. :param `dock`: the :class:`AuiDockInfo` structure to which `pane` belongs to; :param `pane`: a :class:`AuiPaneInfo` class for which calculation are requested. """ if pane.IsFixed(): if dock.IsHorizontal(): minPix = maxPix = pane.rect.x + 1 + pane.rect.width else: minPix = maxPix = pane.rect.y + 1 + pane.rect.height return minPix, maxPix totalPixsize, totalProportion = self.GetTotalPixSizeAndProportion(dock) partnerPane = self.GetPartnerPane(dock, pane) if dock.IsHorizontal(): minPix = pane.rect.x + 1 maxPix = pane.rect.x + 1 + pane.rect.width if pane.min_size.IsFullySpecified(): minPix += pane.min_size.x else: minPix += 1 if partnerPane: maxPix += partnerPane.rect.width if partnerPane.min_size.IsFullySpecified(): maxPix -= partnerPane.min_size.x - 1 else: minPix = maxPix else: minPix = pane.rect.y + 1 maxPix = pane.rect.y + 1 + pane.rect.height if pane.min_size.IsFullySpecified(): minPix += pane.min_size.y else: minPix += 1 if partnerPane: maxPix += partnerPane.rect.height if partnerPane.min_size.IsFullySpecified(): maxPix -= partnerPane.min_size.y - 1 else: minPix = maxPix return minPix, maxPix def CheckMovableSizer(self, part): """ Checks if a UI part can be actually resized. :param `part`: a UI part, an instance of :class:`AuiDockUIPart`. """ # a dock may not be resized if it has a single # pane which is not resizable if part.type == AuiDockUIPart.typeDockSizer and part.dock and \ len(part.dock.panes) == 1 and part.dock.panes[0].IsFixed(): return False if part.pane: # panes that may not be resized should be ignored here minPix, maxPix = self.CalculatePaneSizerLimits(part.dock, part.pane) if minPix == maxPix: return False return True def PaneFromTabEvent(self, event): """ Returns a :class:`AuiPaneInfo` from a :class:`~lib.agw.aui.auibook.AuiNotebook` event. :param `event`: a :class:`~lib.agw.aui.auibook.AuiNotebookEvent` event. """ obj = event.GetEventObject() if obj and isinstance(obj, auibook.AuiTabCtrl): page_idx = obj.GetActivePage() if page_idx >= 0: page = obj.GetPage(page_idx) window = page.window if window: return self.GetPane(window) elif obj and isinstance(obj, auibook.AuiNotebook): page_idx = event.GetSelection() if page_idx >= 0: window = obj.GetPage(page_idx) if window: return self.GetPane(window) return NonePaneInfo def OnTabBeginDrag(self, event): """ Handles the ``EVT_AUINOTEBOOK_BEGIN_DRAG`` event. :param `event`: a :class:`~lib.agw.aui.auibook.AuiNotebookEvent` event to be processed. """ if self._masterManager: self._masterManager.OnTabBeginDrag(event) else: paneInfo = self.PaneFromTabEvent(event) if paneInfo.IsOk(): # It's one of ours! self._action = actionDragFloatingPane mouse = wx.GetMousePosition() # set initial float position - may have to think about this # offset a bit more later ... self._action_offset = wx.Point(20, 10) self._toolbar_action_offset = wx.Point(20, 10) paneInfo.floating_pos = mouse - self._action_offset paneInfo.dock_pos = AUI_DOCK_NONE paneInfo.notebook_id = -1 tab = event.GetEventObject() if tab.HasCapture(): tab.ReleaseMouse() # float the window if paneInfo.IsMaximized(): self.RestorePane(paneInfo) paneInfo.Float() self.Update() self._action_window = paneInfo.window self._frame.CaptureMouse() event.SetDispatched(True) else: # not our window event.Skip() def OnTabPageClose(self, event): """ Handles the ``EVT_AUINOTEBOOK_PAGE_CLOSE`` event. :param `event`: a :class:`~lib.agw.aui.auibook.AuiNotebookEvent` event to be processed. """ if self._masterManager: self._masterManager.OnTabPageClose(event) else: p = self.PaneFromTabEvent(event) if p.IsOk(): # veto it because we will call "RemovePage" ourselves event.Veto() # Now ask the app if they really want to close... # fire pane close event e = AuiManagerEvent(wxEVT_AUI_PANE_CLOSE) e.SetPane(p) e.SetCanVeto(True) self.ProcessMgrEvent(e) if e.GetVeto(): return self.ClosePane(p) self.Update() else: event.Skip() def OnTabSelected(self, event): """ Handles the ``EVT_AUINOTEBOOK_PAGE_CHANGED`` event. :param `event`: a :class:`~lib.agw.aui.auibook.AuiNotebookEvent` event to be processed. """ if self._masterManager: self._masterManager.OnTabSelected(event) return obj = event.GetEventObject() if obj and isinstance(obj, auibook.AuiNotebook): notebook = obj page = notebook.GetPage(event.GetSelection()) paneInfo = self.GetPane(page) if paneInfo.IsOk(): notebookRoot = GetNotebookRoot(self._panes, paneInfo.notebook_id) if notebookRoot: notebookRoot.Caption(paneInfo.caption) self.RefreshCaptions() event.Skip() def GetNotebooks(self): """ Returns all the automatic :class:`~lib.agw.aui.auibook.AuiNotebook` in the :class:`AuiManager`. """ if self._masterManager: return self._masterManager.GetNotebooks() return self._notebooks def SetMasterManager(self, manager): """ Sets the master manager for an automatic :class:`~lib.agw.aui.auibook.AuiNotebook`. :param `manager`: an instance of :class:`AuiManager`. """ self._masterManager = manager def ProcessDockResult(self, target, new_pos): """ This is a utility function used by :meth:`DoDrop` - it checks if a dock operation is allowed, the new dock position is copied into the target info. If the operation was allowed, the function returns ``True``. :param `target`: the :class:`AuiPaneInfo` instance to be docked; :param integer `new_pos`: the new docking position if the docking operation is allowed. """ allowed = False direction = new_pos.dock_direction if direction == AUI_DOCK_TOP: allowed = target.IsTopDockable() elif direction == AUI_DOCK_BOTTOM: allowed = target.IsBottomDockable() elif direction == AUI_DOCK_LEFT: allowed = target.IsLeftDockable() elif direction == AUI_DOCK_RIGHT: allowed = target.IsRightDockable() if allowed: target = new_pos if target.IsToolbar(): self.SwitchToolBarOrientation(target) return allowed, target def SwitchToolBarOrientation(self, pane): """ Switches the toolbar orientation from vertical to horizontal and vice-versa. This is especially useful for vertical docked toolbars once they float. :param `pane`: an instance of :class:`AuiPaneInfo`, which may have a :class:`~lib.agw.aui.auibar.AuiToolBar` window associated with it. """ if not isinstance(pane.window, auibar.AuiToolBar): return pane if pane.IsFloating(): return pane toolBar = pane.window direction = pane.dock_direction vertical = direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT] agwStyle = toolBar.GetAGWWindowStyleFlag() new_agwStyle = agwStyle if vertical: new_agwStyle |= AUI_TB_VERTICAL else: new_agwStyle &= ~(AUI_TB_VERTICAL) if agwStyle != new_agwStyle: toolBar.SetAGWWindowStyleFlag(new_agwStyle) if not toolBar.GetGripperVisible(): toolBar.SetGripperVisible(True) s = pane.window.GetMinSize() pane.BestSize(s) if new_agwStyle != agwStyle: toolBar.Realize() return pane def DoDrop(self, docks, panes, target, pt, offset=wx.Point(0, 0)): """ This is an important function. It basically takes a mouse position, and determines where the panes new position would be. If the pane is to be dropped, it performs the drop operation using the specified dock and pane arrays. By specifying copy dock and pane arrays when calling, a "what-if" scenario can be performed, giving precise coordinates for drop hints. :param `docks`: a list of :class:`AuiDockInfo` classes; :param `panes`: a list of :class:`AuiPaneInfo` instances; :param Point `pt`: a mouse position to check for a drop operation; :param Point `offset`: a possible offset from the input point `pt`. """ if target.IsToolbar(): return self.DoDropToolbar(docks, panes, target, pt, offset) elif target.IsFloating(): return self.DoDropFloatingPane(docks, panes, target, pt) else: return self.DoDropNonFloatingPane(docks, panes, target, pt) def CopyTarget(self, target): """ Copies all the attributes of the input `target` into another :class:`AuiPaneInfo`. :param `target`: the source :class:`AuiPaneInfo` from where to copy attributes. """ drop = AuiPaneInfo() drop.name = target.name drop.caption = target.caption drop.window = target.window drop.frame = target.frame drop.state = target.state drop.dock_direction = target.dock_direction drop.dock_layer = target.dock_layer drop.dock_row = target.dock_row drop.dock_pos = target.dock_pos drop.best_size = wx.Size(*target.best_size) drop.min_size = wx.Size(*target.min_size) drop.max_size = wx.Size(*target.max_size) drop.floating_pos = wx.Point(*target.floating_pos) drop.floating_size = wx.Size(*target.floating_size) drop.dock_proportion = target.dock_proportion drop.buttons = target.buttons drop.rect = wx.Rect(*target.rect) drop.icon = target.icon drop.notebook_id = target.notebook_id drop.transparent = target.transparent drop.snapped = target.snapped drop.minimize_mode = target.minimize_mode drop.minimize_target = target.minimize_target return drop def DoDropToolbar(self, docks, panes, target, pt, offset): """ Handles the situation in which the dropped pane contains a toolbar. :param `docks`: a list of :class:`AuiDockInfo` classes; :param `panes`: a list of :class:`AuiPaneInfo` instances; :param `target`: the target pane containing the toolbar, an instance of :class:`AuiPaneInfo`; :param Point `pt`: a mouse position to check for a drop operation; :param Point `offset`: a possible offset from the input point `pt`. """ drop = self.CopyTarget(target) # The result should always be shown drop.Show() # Check to see if the toolbar has been dragged out of the window if CheckOutOfWindow(self._frame, pt): if self._agwFlags & AUI_MGR_ALLOW_FLOATING and drop.IsFloatable(): drop.Float() return self.ProcessDockResult(target, drop) # Allow directional change when the cursor leaves this rect safeRect = wx.Rect(*target.rect) if target.IsHorizontal(): safeRect.Inflate(100, 50) else: safeRect.Inflate(50, 100) # Check to see if the toolbar has been dragged to edge of the frame dropDir = CheckEdgeDrop(self._frame, docks, pt) if dropDir != -1: if dropDir == wx.LEFT: drop.Dock().Left().Layer(auiToolBarLayer).Row(0). \ Position(pt.y - self.GetDockPixelOffset(drop) - offset.y) elif dropDir == wx.RIGHT: drop.Dock().Right().Layer(auiToolBarLayer).Row(0). \ Position(pt.y - self.GetDockPixelOffset(drop) - offset.y) elif dropDir == wx.TOP: drop.Dock().Top().Layer(auiToolBarLayer).Row(0). \ Position(pt.x - self.GetDockPixelOffset(drop) - offset.x) elif dropDir == wx.BOTTOM: drop.Dock().Bottom().Layer(auiToolBarLayer).Row(0). \ Position(pt.x - self.GetDockPixelOffset(drop) - offset.x) if not target.IsFloating() and safeRect.Contains(pt) and \ target.dock_direction != drop.dock_direction: return False, target return self.ProcessDockResult(target, drop) # If the windows is floating and out of the client area, do nothing if drop.IsFloating() and not self._frame.GetClientRect().Contains(pt): return False, target # Ok, can't drop on edge - check internals ... clientSize = self._frame.GetClientSize() x = Clip(pt.x, 0, clientSize.x - 1) y = Clip(pt.y, 0, clientSize.y - 1) part = self.HitTest(x, y) if not part or not part.dock: return False, target dock = part.dock # toolbars may only be moved in and to fixed-pane docks, # otherwise we will try to float the pane. Also, the pane # should float if being dragged over center pane windows if not dock.fixed or dock.dock_direction == AUI_DOCK_CENTER: if (self._agwFlags & AUI_MGR_ALLOW_FLOATING and drop.IsFloatable()) or \ dock.dock_direction not in [AUI_DOCK_CENTER, AUI_DOCK_NONE]: if drop.IsFloatable(): drop.Float() return self.ProcessDockResult(target, drop) # calculate the offset from where the dock begins # to the point where the user dropped the pane dockDropOffset = 0 if dock.IsHorizontal(): dockDropOffset = pt.x - dock.rect.x - offset.x else: dockDropOffset = pt.y - dock.rect.y - offset.y drop.Dock().Direction(dock.dock_direction).Layer(dock.dock_layer). \ Row(dock.dock_row).Position(dockDropOffset) if (pt.y <= dock.rect.GetTop() + 2 and dock.IsHorizontal()) or \ (pt.x <= dock.rect.GetLeft() + 2 and dock.IsVertical()): if dock.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_LEFT]: row = drop.dock_row panes = DoInsertDockRow(panes, dock.dock_direction, dock.dock_layer, dock.dock_row) drop.dock_row = row else: panes = DoInsertDockRow(panes, dock.dock_direction, dock.dock_layer, dock.dock_row+1) drop.dock_row = dock.dock_row + 1 if (pt.y >= dock.rect.GetBottom() - 2 and dock.IsHorizontal()) or \ (pt.x >= dock.rect.GetRight() - 2 and dock.IsVertical()): if dock.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_LEFT]: panes = DoInsertDockRow(panes, dock.dock_direction, dock.dock_layer, dock.dock_row+1) drop.dock_row = dock.dock_row+1 else: row = drop.dock_row panes = DoInsertDockRow(panes, dock.dock_direction, dock.dock_layer, dock.dock_row) drop.dock_row = row if not target.IsFloating() and safeRect.Contains(pt) and \ target.dock_direction != drop.dock_direction: return False, target return self.ProcessDockResult(target, drop) def DoDropFloatingPane(self, docks, panes, target, pt): """ Handles the situation in which the dropped pane contains a normal window. :param `docks`: a list of :class:`AuiDockInfo` classes; :param `panes`: a list of :class:`AuiPaneInfo` instances; :param `target`: the target pane containing the window, an instance of :class:`AuiPaneInfo`; :param Point `pt`: a mouse position to check for a drop operation. """ screenPt = self._frame.ClientToScreen(pt) paneInfo = self.PaneHitTest(panes, pt) if paneInfo.IsMaximized(): return False, target if paneInfo.window is None: return False, target # search the dock guides. # reverse order to handle the center first. for i in range(len(self._guides)-1, -1, -1): guide = self._guides[i] # do hit testing on the guide dir = guide.host.HitTest(screenPt.x, screenPt.y) if dir == -1: # point was outside of the dock guide continue if dir == wx.ALL: # target is a single dock guide return self.DoDropLayer(docks, target, guide.dock_direction) elif dir == wx.CENTER: if not target.IsNotebookDockable(): continue if not paneInfo.IsNotebookDockable() and not paneInfo.IsNotebookControl(): continue if not paneInfo.HasNotebook(): # Add a new notebook pane with the original as a tab... self.CreateNotebookBase(panes, paneInfo) # Add new item to notebook target.NotebookPage(paneInfo.notebook_id) else: drop_pane = False drop_row = False insert_dir = paneInfo.dock_direction insert_layer = paneInfo.dock_layer insert_row = paneInfo.dock_row insert_pos = paneInfo.dock_pos if insert_dir == AUI_DOCK_CENTER: insert_layer = 0 if dir == wx.LEFT: insert_dir = AUI_DOCK_LEFT elif dir == wx.UP: insert_dir = AUI_DOCK_TOP elif dir == wx.RIGHT: insert_dir = AUI_DOCK_RIGHT elif dir == wx.DOWN: insert_dir = AUI_DOCK_BOTTOM if insert_dir == AUI_DOCK_LEFT: drop_pane = (dir == wx.UP or dir == wx.DOWN) drop_row = (dir == wx.LEFT or dir == wx.RIGHT) if dir == wx.RIGHT: insert_row += 1 elif dir == wx.DOWN: insert_pos += 1 elif insert_dir == AUI_DOCK_RIGHT: drop_pane = (dir == wx.UP or dir == wx.DOWN) drop_row = (dir == wx.LEFT or dir == wx.RIGHT) if dir == wx.LEFT: insert_row += 1 elif dir == wx.DOWN: insert_pos += 1 elif insert_dir == AUI_DOCK_TOP: drop_pane = (dir == wx.LEFT or dir == wx.RIGHT) drop_row = (dir == wx.UP or dir == wx.DOWN) if dir == wx.DOWN: insert_row += 1 elif dir == wx.RIGHT: insert_pos += 1 elif insert_dir == AUI_DOCK_BOTTOM: drop_pane = (dir == wx.LEFT or dir == wx.RIGHT) drop_row = (dir == wx.UP or dir == wx.DOWN) if dir == wx.UP: insert_row += 1 elif dir == wx.RIGHT: insert_pos += 1 if paneInfo.dock_direction == AUI_DOCK_CENTER: insert_row = GetMaxRow(panes, insert_dir, insert_layer) + 1 if drop_pane: return self.DoDropPane(panes, target, insert_dir, insert_layer, insert_row, insert_pos) if drop_row: return self.DoDropRow(panes, target, insert_dir, insert_layer, insert_row) return True, target return False, target def DoDropNonFloatingPane(self, docks, panes, target, pt): """ Handles the situation in which the dropped pane is not floating. :param `docks`: a list of :class:`AuiDockInfo` classes; :param `panes`: a list of :class:`AuiPaneInfo` instances; :param `target`: the target pane containing the toolbar, an instance of :class:`AuiPaneInfo`; :param Point `pt`: a mouse position to check for a drop operation. """ screenPt = self._frame.ClientToScreen(pt) clientSize = self._frame.GetClientSize() frameRect = GetInternalFrameRect(self._frame, self._docks) drop = self.CopyTarget(target) # The result should always be shown drop.Show() part = self.HitTest(pt.x, pt.y) if not part: return False, target if part.type == AuiDockUIPart.typeDockSizer: if len(part.dock.panes) != 1: return False, target part = self.GetPanePart(part.dock.panes[0].window) if not part: return False, target if not part.pane: return False, target part = self.GetPanePart(part.pane.window) if not part: return False, target insert_dock_row = False insert_row = part.pane.dock_row insert_dir = part.pane.dock_direction insert_layer = part.pane.dock_layer direction = part.pane.dock_direction if direction == AUI_DOCK_TOP: if pt.y >= part.rect.y and pt.y < part.rect.y+auiInsertRowPixels: insert_dock_row = True elif direction == AUI_DOCK_BOTTOM: if pt.y > part.rect.y+part.rect.height-auiInsertRowPixels and \ pt.y <= part.rect.y + part.rect.height: insert_dock_row = True elif direction == AUI_DOCK_LEFT: if pt.x >= part.rect.x and pt.x < part.rect.x+auiInsertRowPixels: insert_dock_row = True elif direction == AUI_DOCK_RIGHT: if pt.x > part.rect.x+part.rect.width-auiInsertRowPixels and \ pt.x <= part.rect.x+part.rect.width: insert_dock_row = True elif direction == AUI_DOCK_CENTER: # "new row pixels" will be set to the default, but # must never exceed 20% of the window size new_row_pixels_x = auiNewRowPixels new_row_pixels_y = auiNewRowPixels if new_row_pixels_x > (part.rect.width*20)/100: new_row_pixels_x = (part.rect.width*20)/100 if new_row_pixels_y > (part.rect.height*20)/100: new_row_pixels_y = (part.rect.height*20)/100 # determine if the mouse pointer is in a location that # will cause a new row to be inserted. The hot spot positions # are along the borders of the center pane insert_layer = 0 insert_dock_row = True pr = part.rect if pt.x >= pr.x and pt.x < pr.x + new_row_pixels_x: insert_dir = AUI_DOCK_LEFT elif pt.y >= pr.y and pt.y < pr.y + new_row_pixels_y: insert_dir = AUI_DOCK_TOP elif pt.x >= pr.x + pr.width - new_row_pixels_x and pt.x < pr.x + pr.width: insert_dir = AUI_DOCK_RIGHT elif pt.y >= pr.y+ pr.height - new_row_pixels_y and pt.y < pr.y + pr.height: insert_dir = AUI_DOCK_BOTTOM else: return False, target insert_row = GetMaxRow(panes, insert_dir, insert_layer) + 1 if insert_dock_row: panes = DoInsertDockRow(panes, insert_dir, insert_layer, insert_row) drop.Dock().Direction(insert_dir).Layer(insert_layer). \ Row(insert_row).Position(0) return self.ProcessDockResult(target, drop) # determine the mouse offset and the pane size, both in the # direction of the dock itself, and perpendicular to the dock if part.orientation == wx.VERTICAL: offset = pt.y - part.rect.y size = part.rect.GetHeight() else: offset = pt.x - part.rect.x size = part.rect.GetWidth() drop_position = part.pane.dock_pos # if we are in the top/left part of the pane, # insert the pane before the pane being hovered over if offset <= size/2: drop_position = part.pane.dock_pos panes = DoInsertPane(panes, part.pane.dock_direction, part.pane.dock_layer, part.pane.dock_row, part.pane.dock_pos) # if we are in the bottom/right part of the pane, # insert the pane before the pane being hovered over if offset > size/2: drop_position = part.pane.dock_pos+1 panes = DoInsertPane(panes, part.pane.dock_direction, part.pane.dock_layer, part.pane.dock_row, part.pane.dock_pos+1) drop.Dock(). \ Direction(part.dock.dock_direction). \ Layer(part.dock.dock_layer).Row(part.dock.dock_row). \ Position(drop_position) return self.ProcessDockResult(target, drop) def DoDropLayer(self, docks, target, dock_direction): """ Handles the situation in which `target` is a single dock guide. :param `docks`: a list of :class:`AuiDockInfo` classes; :param `target`: the target pane, an instance of :class:`AuiPaneInfo`; :param integer `dock_direction`: the docking direction. """ drop = self.CopyTarget(target) if dock_direction == AUI_DOCK_LEFT: drop.Dock().Left() drop_new_layer = max(max(GetMaxLayer(docks, AUI_DOCK_LEFT), GetMaxLayer(docks, AUI_DOCK_BOTTOM)), GetMaxLayer(docks, AUI_DOCK_TOP)) + 1 elif dock_direction == AUI_DOCK_TOP: drop.Dock().Top() drop_new_layer = max(max(GetMaxLayer(docks, AUI_DOCK_TOP), GetMaxLayer(docks, AUI_DOCK_LEFT)), GetMaxLayer(docks, AUI_DOCK_RIGHT)) + 1 elif dock_direction == AUI_DOCK_RIGHT: drop.Dock().Right() drop_new_layer = max(max(GetMaxLayer(docks, AUI_DOCK_RIGHT), GetMaxLayer(docks, AUI_DOCK_TOP)), GetMaxLayer(docks, AUI_DOCK_BOTTOM)) + 1 elif dock_direction == AUI_DOCK_BOTTOM: drop.Dock().Bottom() drop_new_layer = max(max(GetMaxLayer(docks, AUI_DOCK_BOTTOM), GetMaxLayer(docks, AUI_DOCK_LEFT)), GetMaxLayer(docks, AUI_DOCK_RIGHT)) + 1 else: return False, target drop.Dock().Layer(drop_new_layer) return self.ProcessDockResult(target, drop) def DoDropPane(self, panes, target, dock_direction, dock_layer, dock_row, dock_pos): """ Drop a pane in the interface. :param `panes`: a list of :class:`AuiPaneInfo` classes; :param `target`: the target pane, an instance of :class:`AuiPaneInfo`; :param integer `dock_direction`: the docking direction; :param integer `dock_layer`: the docking layer; :param integer `dock_row`: the docking row; :param integer `dock_pos`: the docking position. """ drop = self.CopyTarget(target) panes = DoInsertPane(panes, dock_direction, dock_layer, dock_row, dock_pos) drop.Dock().Direction(dock_direction).Layer(dock_layer).Row(dock_row).Position(dock_pos) return self.ProcessDockResult(target, drop) def DoDropRow(self, panes, target, dock_direction, dock_layer, dock_row): """ Insert a row in the interface before dropping. :param `panes`: a list of :class:`AuiPaneInfo` classes; :param `target`: the target pane, an instance of :class:`AuiPaneInfo`; :param integer `dock_direction`: the docking direction; :param integer `dock_layer`: the docking layer; :param integer `dock_row`: the docking row. """ drop = self.CopyTarget(target) panes = DoInsertDockRow(panes, dock_direction, dock_layer, dock_row) drop.Dock().Direction(dock_direction).Layer(dock_layer).Row(dock_row).Position(0) return self.ProcessDockResult(target, drop) def ShowHint(self, rect): """ Shows the AUI hint window. :param Rect `rect`: the hint rect calculated in advance. """ if rect == self._last_hint: return if self._agwFlags & AUI_MGR_RECTANGLE_HINT and wx.Platform != "__WXMAC__": if self._last_hint != rect: # remove the last hint rectangle self._last_hint = wx.Rect(*rect) self._frame.Refresh() self._frame.Update() screendc = wx.ScreenDC() clip = wx.Region(1, 1, 10000, 10000) # clip all floating windows, so we don't draw over them for pane in self._panes: if pane.IsFloating() and pane.frame.IsShown(): rect2 = wx.Rect(*pane.frame.GetRect()) if wx.Platform == "__WXGTK__": # wxGTK returns the client size, not the whole frame size rect2.width += 15 rect2.height += 35 rect2.Inflate(5, 5) clip.Subtract(rect2) # As we can only hide the hint by redrawing the managed window, we # need to clip the region to the managed window too or we get # nasty redrawn problems. clip.Intersect(self._frame.GetRect()) screendc.SetDeviceClippingRegion(clip) stipple = PaneCreateStippleBitmap() brush = wx.Brush(stipple) screendc.SetBrush(brush) screendc.SetPen(wx.TRANSPARENT_PEN) screendc.DrawRectangle(rect.x, rect.y, 5, rect.height) screendc.DrawRectangle(rect.x+5, rect.y, rect.width-10, 5) screendc.DrawRectangle(rect.x+rect.width-5, rect.y, 5, rect.height) screendc.DrawRectangle(rect.x+5, rect.y+rect.height-5, rect.width-10, 5) RefreshDockingGuides(self._guides) return if not self._hint_window: self.CreateHintWindow() if self._hint_window: self._hint_window.SetRect(rect) self._hint_window.Show() self._hint_fadeamt = self._hint_fademax if self._agwFlags & AUI_MGR_HINT_FADE: self._hint_fadeamt = 0 self._hint_window.SetTransparent(self._hint_fadeamt) if self._action == actionDragFloatingPane and self._action_window: self._action_window.SetFocus() if self._hint_fadeamt != self._hint_fademax: # Only fade if we need to # start fade in timer self._hint_fadetimer.Start(5) self._last_hint = wx.Rect(*rect) def HideHint(self): """ Hides a transparent window hint if there is one. """ # hides a transparent window hint if there is one if self._hint_window: self._hint_window.Hide() self._hint_fadetimer.Stop() self._last_hint = wx.Rect() def IsPaneButtonVisible(self, part): """ Returns whether a pane button in the pane caption is visible. :param `part`: the UI part to analyze, an instance of :class:`AuiDockUIPart`. """ captionRect = wx.Rect() for temp_part in self._uiparts: if temp_part.pane == part.pane and \ temp_part.type == AuiDockUIPart.typeCaption: captionRect = temp_part.rect break return captionRect.Contains(part.rect) def DrawPaneButton(self, dc, part, pt): """ Draws a pane button in the caption (convenience function). :param `dc`: a :class:`DC` device context object; :param `part`: the UI part to analyze, an instance of :class:`AuiDockUIPart`; :param Point `pt`: the mouse location. """ if not self.IsPaneButtonVisible(part): return state = AUI_BUTTON_STATE_NORMAL if part.rect.Contains(pt): if _VERSION_STRING < "2.9": leftDown = wx.GetMouseState().LeftDown() else: leftDown = wx.GetMouseState().LeftIsDown() if leftDown: state = AUI_BUTTON_STATE_PRESSED else: state = AUI_BUTTON_STATE_HOVER self._art.DrawPaneButton(dc, self._frame, part.button.button_id, state, part.rect, part.pane) def RefreshButton(self, part): """ Refreshes a pane button in the caption. :param `part`: the UI part to analyze, an instance of :class:`AuiDockUIPart`. """ rect = wx.Rect(*part.rect) rect.Inflate(2, 2) self._frame.Refresh(True, rect) self._frame.Update() def RefreshCaptions(self): """ Refreshes all pane captions. """ for part in self._uiparts: if part.type == AuiDockUIPart.typeCaption: self._frame.Refresh(True, part.rect) self._frame.Update() def CalculateHintRect(self, pane_window, pt, offset): """ Calculates the drop hint rectangle. The method first calls :meth:`DoDrop` to determine the exact position the pane would be at were if dropped. If the pane would indeed become docked at the specified drop point, the the rectangle hint will be returned in screen coordinates. Otherwise, an empty rectangle is returned. :param Window `pane_window`: it is the window pointer of the pane being dragged; :param Point `pt`: is the mouse position, in client coordinates; :param Point `offset`: describes the offset that the mouse is from the upper-left corner of the item being dragged. """ # we need to paint a hint rectangle to find out the exact hint rectangle, # we will create a new temporary layout and then measure the resulting # rectangle we will create a copy of the docking structures (self._docks) # so that we don't modify the real thing on screen rect = wx.Rect() pane = self.GetPane(pane_window) attrs = self.GetAttributes(pane) hint = AuiPaneInfo() hint = self.SetAttributes(hint, attrs) if hint.name != "__HINT__": self._oldname = hint.name hint.name = "__HINT__" hint.PaneBorder(True) hint.Show() if not hint.IsOk(): hint.name = self._oldname return rect docks, panes = CopyDocksAndPanes2(self._docks, self._panes) # remove any pane already there which bears the same window # this happens when you are moving a pane around in a dock for ii in range(len(panes)): if panes[ii].window == pane_window: docks = RemovePaneFromDocks(docks, panes[ii]) panes.pop(ii) break # find out where the new pane would be allow, hint = self.DoDrop(docks, panes, hint, pt, offset) if not allow: return rect panes.append(hint) sizer, panes, docks, uiparts = self.LayoutAll(panes, docks, [], True, False) client_size = self._frame.GetClientSize() sizer.SetDimension(0, 0, client_size.x, client_size.y) sizer.Layout() sought = "__HINT__" # For a notebook page, actually look for the notebook itself. if hint.IsNotebookPage(): id = hint.notebook_id for pane in panes: if pane.IsNotebookControl() and pane.notebook_id==id: sought = pane.name break for part in uiparts: if part.pane and part.pane.name == sought: rect.Union(wx.Rect(part.sizer_item.GetPosition(), part.sizer_item.GetSize())) sizer.Destroy() # check for floating frame ... if rect.IsEmpty(): for p in panes: if p.name == sought and p.IsFloating(): return wx.Rect(p.floating_pos, p.floating_size) if rect.IsEmpty(): return rect # actually show the hint rectangle on the screen rect.x, rect.y = self._frame.ClientToScreen((rect.x, rect.y)) if self._frame.GetLayoutDirection() == wx.Layout_RightToLeft: # Mirror rectangle in RTL mode rect.x -= rect.GetWidth() return rect def DrawHintRect(self, pane_window, pt, offset): """ Calculates the hint rectangle by calling :meth:`CalculateHintRect`. If there is a rectangle, it shows it by calling :meth:`ShowHint`, otherwise it hides any hint rectangle currently shown. :param Window `pane_window`: it is the window pointer of the pane being dragged; :param Point `pt`: is the mouse position, in client coordinates; :param Point `offset`: describes the offset that the mouse is from the upper-left corner of the item being dragged. """ rect = self.CalculateHintRect(pane_window, pt, offset) if rect.IsEmpty(): self.HideHint() self._hint_rect = wx.Rect() else: self.ShowHint(rect) self._hint_rect = wx.Rect(*rect) def GetPartSizerRect(self, uiparts): """ Returns the rectangle surrounding the specified UI parts. :param list `uiparts`: list of :class:`AuiDockUIPart` parts. """ rect = wx.Rect() for part in self._uiparts: if part.pane and part.pane.name == "__HINT__": rect.Union(wx.Rect(part.sizer_item.GetPosition(), part.sizer_item.GetSize())) return rect def GetAttributes(self, pane): """ Returns all the attributes of a :class:`AuiPaneInfo`. :param `pane`: a :class:`AuiPaneInfo` instance. """ attrs = [] attrs.extend([pane.window, pane.frame, pane.state, pane.dock_direction, pane.dock_layer, pane.dock_pos, pane.dock_row, pane.dock_proportion, pane.floating_pos, pane.floating_size, pane.best_size, pane.min_size, pane.max_size, pane.caption, pane.name, pane.buttons, pane.rect, pane.icon, pane.notebook_id, pane.transparent, pane.snapped, pane.minimize_mode, pane.minimize_target]) return attrs def SetAttributes(self, pane, attrs): """ Sets all the attributes contained in `attrs` to a :class:`AuiPaneInfo`. :param `pane`: a :class:`AuiPaneInfo` instance; :param list `attrs`: a list of attributes. """ pane.window = attrs[0] pane.frame = attrs[1] pane.state = attrs[2] pane.dock_direction = attrs[3] pane.dock_layer = attrs[4] pane.dock_pos = attrs[5] pane.dock_row = attrs[6] pane.dock_proportion = attrs[7] pane.floating_pos = attrs[8] pane.floating_size = attrs[9] pane.best_size = attrs[10] pane.min_size = attrs[11] pane.max_size = attrs[12] pane.caption = attrs[13] pane.name = attrs[14] pane.buttons = attrs[15] pane.rect = attrs[16] pane.icon = attrs[17] pane.notebook_id = attrs[18] pane.transparent = attrs[19] pane.snapped = attrs[20] pane.minimize_mode = attrs[21] pane.minimize_target = attrs[22] return pane def OnFloatingPaneResized(self, wnd, size): """ Handles the resizing of a floating pane. :param Window `wnd`: the window managed by the pane; :param Size `size`: the new pane floating size. """ # try to find the pane pane = self.GetPane(wnd) if not pane.IsOk(): raise Exception("Pane window not found") if pane.frame: indx = self._panes.index(pane) pane.floating_pos = pane.frame.GetPosition() pane.floating_size = size self._panes[indx] = pane if pane.IsSnappable(): self.SnapPane(pane, pane.floating_pos, pane.floating_size, True) def OnFloatingPaneClosed(self, wnd, event): """ Handles the close event of a floating pane. :param Window `wnd`: the window managed by the pane; :param `event`: a :class:`CloseEvent` to be processed. """ # try to find the pane pane = self.GetPane(wnd) if not pane.IsOk(): raise Exception("Pane window not found") # fire pane close event e = AuiManagerEvent(wxEVT_AUI_PANE_CLOSE) e.SetPane(pane) e.SetCanVeto(event.CanVeto()) self.ProcessMgrEvent(e) if e.GetVeto(): event.Veto() return else: # close the pane, but check that it # still exists in our pane array first # (the event handler above might have removed it) check = self.GetPane(wnd) if check.IsOk(): self.ClosePane(pane) def OnFloatingPaneActivated(self, wnd): """ Handles the activation event of a floating pane. :param Window `wnd`: the window managed by the pane. """ pane = self.GetPane(wnd) if not pane.IsOk(): raise Exception("Pane window not found") if self.GetAGWFlags() & AUI_MGR_ALLOW_ACTIVE_PANE: ret, self._panes = SetActivePane(self._panes, wnd) self.RefreshCaptions() self.FireEvent(wxEVT_AUI_PANE_ACTIVATED, wnd, canVeto=False) def OnFloatingPaneMoved(self, wnd, eventOrPt): """ Handles the move event of a floating pane. :param Window `wnd`: the window managed by the pane; :param `eventOrPt`: a :class:`MoveEvent` to be processed or an instance of :class:`Point`. """ pane = self.GetPane(wnd) if not pane.IsOk(): raise Exception("Pane window not found") if not pane.IsSnappable(): return if isinstance(eventOrPt, wx.Point): pane_pos = wx.Point(*eventOrPt) else: pane_pos = eventOrPt.GetPosition() pane_size = pane.floating_size self.SnapPane(pane, pane_pos, pane_size, False) def SnapPane(self, pane, pane_pos, pane_size, toSnap=False): """ Snaps a floating pane to one of the main frame sides. :param `pane`: a :class:`AuiPaneInfo` instance; :param Point `pane_pos`: the new pane floating position; :param Size `pane_size`: the new pane floating size; :param bool `toSnap`: a bool variable to check if :meth:`SnapPane` was called from a move event. """ if self._from_move: return managed_window = self.GetManagedWindow() wnd_pos = managed_window.GetPosition() wnd_size = managed_window.GetSize() snapX, snapY = self._snap_limits if not toSnap: pane.snapped = 0 if pane.IsLeftSnappable(): # Check if we can snap to the left diff = wnd_pos.x - (pane_pos.x + pane_size.x) if -snapX <= diff <= snapX: pane.snapped = wx.LEFT pane.floating_pos = wx.Point(wnd_pos.x-pane_size.x, pane_pos.y) elif pane.IsTopSnappable(): # Check if we can snap to the top diff = wnd_pos.y - (pane_pos.y + pane_size.y) if -snapY <= diff <= snapY: pane.snapped = wx.TOP pane.floating_pos = wx.Point(pane_pos.x, wnd_pos.y-pane_size.y) elif pane.IsRightSnappable(): # Check if we can snap to the right diff = pane_pos.x - (wnd_pos.x + wnd_size.x) if -snapX <= diff <= snapX: pane.snapped = wx.RIGHT pane.floating_pos = wx.Point(wnd_pos.x + wnd_size.x, pane_pos.y) elif pane.IsBottomSnappable(): # Check if we can snap to the bottom diff = pane_pos.y - (wnd_pos.y + wnd_size.y) if -snapY <= diff <= snapY: pane.snapped = wx.BOTTOM pane.floating_pos = wx.Point(pane_pos.x, wnd_pos.y + wnd_size.y) self.RepositionPane(pane, wnd_pos, wnd_size) def RepositionPane(self, pane, wnd_pos, wnd_size): """ Repositions a pane after the main frame has been moved/resized. :param `pane`: a :class:`AuiPaneInfo` instance; :param Point `wnd_pos`: the main frame position; :param Size `wnd_size`: the main frame size. """ pane_pos = pane.floating_pos pane_size = pane.floating_size snap = pane.snapped if snap == wx.LEFT: floating_pos = wx.Point(wnd_pos.x - pane_size.x, pane_pos.y) elif snap == wx.TOP: floating_pos = wx.Point(pane_pos.x, wnd_pos.y - pane_size.y) elif snap == wx.RIGHT: floating_pos = wx.Point(wnd_pos.x + wnd_size.x, pane_pos.y) elif snap == wx.BOTTOM: floating_pos = wx.Point(pane_pos.x, wnd_pos.y + wnd_size.y) if snap: if pane_pos != floating_pos: pane.floating_pos = floating_pos self._from_move = True pane.frame.SetPosition(pane.floating_pos) self._from_move = False def OnGripperClicked(self, pane_window, start, offset): """ Handles the mouse click on the pane gripper. :param Window `pane_window`: the window managed by the pane; :param Point `start`: the mouse-click position; :param Point `offset`: an offset point from the `start` position. """ # try to find the pane paneInfo = self.GetPane(pane_window) if not paneInfo.IsOk(): raise Exception("Pane window not found") if self.GetAGWFlags() & AUI_MGR_ALLOW_ACTIVE_PANE: # set the caption as active ret, self._panes = SetActivePane(self._panes, pane_window) self.RefreshCaptions() self.FireEvent(wxEVT_AUI_PANE_ACTIVATED, pane_window, canVeto=False) self._action_part = None self._action_pane = paneInfo self._action_window = pane_window self._action_start = start self._action_offset = offset self._toolbar_action_offset = wx.Point(*self._action_offset) self._frame.CaptureMouse() if paneInfo.IsDocked(): self._action = actionClickCaption else: if paneInfo.IsToolbar(): self._action = actionDragToolbarPane else: self._action = actionDragFloatingPane if paneInfo.frame: windowPt = paneInfo.frame.GetRect().GetTopLeft() originPt = paneInfo.frame.ClientToScreen(wx.Point()) self._action_offset += originPt - windowPt self._toolbar_action_offset = wx.Point(*self._action_offset) if self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: paneInfo.frame.SetTransparent(150) if paneInfo.IsToolbar(): self._frame.SetCursor(wx.Cursor(wx.CURSOR_SIZING)) def OnRender(self, event): """ Draws all of the pane captions, sashes, backgrounds, captions, grippers, pane borders and buttons. It renders the entire user interface. It binds the ``EVT_AUI_RENDER`` event. :param `event`: an instance of :class:`AuiManagerEvent`. """ # if the frame is about to be deleted, don't bother if not self._frame or self._frame.IsBeingDeleted(): return if not self._frame.GetSizer(): return mouse = wx.GetMouseState() mousePos = wx.Point(mouse.GetX(), mouse.GetY()) point = self._frame.ScreenToClient(mousePos) art = self._art dc = event.GetDC() for part in self._uiparts: # don't draw hidden pane items or items that aren't windows if part.sizer_item and ((not part.sizer_item.IsWindow() and \ not part.sizer_item.IsSpacer() and \ not part.sizer_item.IsSizer()) or \ not part.sizer_item.IsShown()): continue ptype = part.type if ptype in [AuiDockUIPart.typeDockSizer, AuiDockUIPart.typePaneSizer]: art.DrawSash(dc, self._frame, part.orientation, part.rect) elif ptype == AuiDockUIPart.typeBackground: art.DrawBackground(dc, self._frame, part.orientation, part.rect) elif ptype == AuiDockUIPart.typeCaption: art.DrawCaption(dc, self._frame, part.pane.caption, part.rect, part.pane) elif ptype == AuiDockUIPart.typeGripper: art.DrawGripper(dc, self._frame, part.rect, part.pane) elif ptype == AuiDockUIPart.typePaneBorder: art.DrawBorder(dc, self._frame, part.rect, part.pane) elif ptype == AuiDockUIPart.typePaneButton: self.DrawPaneButton(dc, part, point) def Repaint(self, dc=None): """ Repaints the entire frame decorations (sashes, borders, buttons and so on). It renders the entire user interface. :param `dc`: if not ``None``, an instance of :class:`PaintDC`. """ w, h = self._frame.GetClientSize() # Figure out which dc to use; if one # has been specified, use it, otherwise # make a client dc if dc is None: client_dc = wx.ClientDC(self._frame) dc = client_dc # If the frame has a toolbar, the client area # origin will not be (0, 0). pt = self._frame.GetClientAreaOrigin() if pt.x != 0 or pt.y != 0: dc.SetDeviceOrigin(pt.x, pt.y) # Render all the items self.Render(dc) def Render(self, dc): """ Fires a render event, which is normally handled by :meth:`OnRender`. This allows the render function to be overridden via the render event. This can be useful for painting custom graphics in the main window. Default behavior can be invoked in the overridden function by calling :meth:`OnRender`. :param `dc`: a :class:`DC` device context object. """ e = AuiManagerEvent(wxEVT_AUI_RENDER) e.SetManager(self) e.SetDC(dc) self.ProcessMgrEvent(e) def OnCaptionDoubleClicked(self, pane_window): """ Handles the mouse double click on the pane caption. :param Window `pane_window`: the window managed by the pane. """ # try to find the pane paneInfo = self.GetPane(pane_window) if not paneInfo.IsOk(): raise Exception("Pane window not found") if not paneInfo.IsFloatable() or not paneInfo.IsDockable() or \ self._agwFlags & AUI_MGR_ALLOW_FLOATING == 0: return indx = self._panes.index(paneInfo) win_rect = None if paneInfo.IsFloating(): if paneInfo.name.startswith("__floating__"): # It's a floating tab from a AuiNotebook notebook = paneInfo.window.__aui_notebook__ notebook.ReDockPage(paneInfo) self.Update() return else: e = self.FireEvent(wxEVT_AUI_PANE_DOCKING, paneInfo, canVeto=True) if e.GetVeto(): self.HideHint() ShowDockingGuides(self._guides, False) return win_rect = paneInfo.frame.GetRect() paneInfo.Dock() if paneInfo.IsToolbar(): paneInfo = self.SwitchToolBarOrientation(paneInfo) e = self.FireEvent(wxEVT_AUI_PANE_DOCKED, paneInfo, canVeto=False) else: e = self.FireEvent(wxEVT_AUI_PANE_FLOATING, paneInfo, canVeto=True) if e.GetVeto(): return # float the window if paneInfo.IsMaximized(): self.RestorePane(paneInfo) if paneInfo.floating_pos == wx.Point(-1, -1): captionSize = self._art.GetMetric(AUI_DOCKART_CAPTION_SIZE) paneInfo.floating_pos = pane_window.GetScreenPosition() paneInfo.floating_pos.y -= captionSize paneInfo.Float() e = self.FireEvent(wxEVT_AUI_PANE_FLOATED, paneInfo, canVeto=False) self._panes[indx] = paneInfo self.Update() if win_rect and self._agwFlags & AUI_MGR_ANIMATE_FRAMES: paneInfo = self.GetPane(pane_window) pane_rect = paneInfo.window.GetScreenRect() self.AnimateDocking(win_rect, pane_rect) def OnPaint(self, event): """ Handles the ``wx.EVT_PAINT`` event for :class:`AuiManager`. :param `event`: an instance of :class:`PaintEvent` to be processed. """ dc = wx.PaintDC(self._frame) self.Repaint(dc) def OnEraseBackground(self, event): """ Handles the ``wx.EVT_ERASE_BACKGROUND`` event for :class:`AuiManager`. :param `event`: :class:`EraseEvent` to be processed. :note: This is intentionally empty (excluding wxMAC) to reduce flickering while drawing. """ if wx.Platform == "__WXMAC__": event.Skip() def OnSize(self, event): """ Handles the ``wx.EVT_SIZE`` event for :class:`AuiManager`. :param `event`: a :class:`SizeEvent` to be processed. """ skipped = False if isinstance(self._frame, AuiFloatingFrame) and self._frame.IsShownOnScreen(): skipped = True event.Skip() if self._frame: self.DoFrameLayout() if wx.Platform == "__WXMAC__": self._frame.Refresh() else: self.Repaint() if isinstance(self._frame, wx.MDIParentFrame) or isinstance(self._frame, tabmdi.AuiMDIClientWindow) \ or isinstance(self._frame, tabmdi.AuiMDIParentFrame): # for MDI parent frames, this event must not # be "skipped". In other words, the parent frame # must not be allowed to resize the client window # after we are finished processing sizing changes return if not skipped: event.Skip() # For the snap to screen... self.OnMove(None) def OnFindManager(self, event): """ Handles the ``EVT_AUI_FIND_MANAGER`` event for :class:`AuiManager`. :param `event`: a :class:`AuiManagerEvent` event to be processed. """ # Initialize to None event.SetManager(None) if not self._frame: return # See it this window wants to overwrite self._frame.ProcessEvent(event) # if no, it must be us if not event.GetManager(): event.SetManager(self) def OnSetCursor(self, event): """ Handles the ``wx.EVT_SET_CURSOR`` event for :class:`AuiManager`. :param `event`: a :class:`SetCursorEvent` to be processed. """ # determine cursor part = self.HitTest(event.GetX(), event.GetY()) cursor = wx.NullCursor if part: if part.type in [AuiDockUIPart.typeDockSizer, AuiDockUIPart.typePaneSizer]: if not self.CheckMovableSizer(part): return if part.orientation == wx.VERTICAL: cursor = wx.Cursor(wx.CURSOR_SIZEWE) else: cursor = wx.Cursor(wx.CURSOR_SIZENS) elif part.type == AuiDockUIPart.typeGripper: cursor = wx.Cursor(wx.CURSOR_SIZING) event.SetCursor(cursor) def UpdateButtonOnScreen(self, button_ui_part, event): """ Updates/redraws the UI part containing a pane button. :param `button_ui_part`: the UI part the button belongs to, an instance of :class:`AuiDockUIPart`.; :param `event`: a :class:`MouseEvent` to be processed. """ hit_test = self.HitTest(*event.GetPosition()) if not hit_test or not button_ui_part: return state = AUI_BUTTON_STATE_NORMAL if hit_test == button_ui_part: if event.LeftDown(): state = AUI_BUTTON_STATE_PRESSED else: state = AUI_BUTTON_STATE_HOVER else: if event.LeftDown(): state = AUI_BUTTON_STATE_HOVER # now repaint the button with hover state cdc = wx.ClientDC(self._frame) # if the frame has a toolbar, the client area # origin will not be (0,0). pt = self._frame.GetClientAreaOrigin() if pt.x != 0 or pt.y != 0: cdc.SetDeviceOrigin(pt.x, pt.y) if hit_test.pane: self._art.DrawPaneButton(cdc, self._frame, button_ui_part.button.button_id, state, button_ui_part.rect, hit_test.pane) def OnLeftDown(self, event): """ Handles the ``wx.EVT_LEFT_DOWN`` event for :class:`AuiManager`. :param `event`: a :class:`MouseEvent` to be processed. """ part = self.HitTest(*event.GetPosition()) if not part: event.Skip() return self._currentDragItem = -1 if part.type in [AuiDockUIPart.typeDockSizer, AuiDockUIPart.typePaneSizer]: if not self.CheckMovableSizer(part): return self._action = actionResize self._action_part = part self._action_pane = None self._action_rect = wx.Rect() self._action_start = wx.Point(event.GetX(), event.GetY()) self._action_offset = wx.Point(event.GetX() - part.rect.x, event.GetY() - part.rect.y) # draw the resize hint rect = wx.Rect(self._frame.ClientToScreen(part.rect.GetPosition()), part.rect.GetSize()) self._action_rect = wx.Rect(*rect) if not AuiManager_HasLiveResize(self): if wx.Platform == "__WXMAC__": dc = wx.ClientDC(self._frame) else: dc = wx.ScreenDC() DrawResizeHint(dc, rect) self._frame.CaptureMouse() elif part.type == AuiDockUIPart.typePaneButton: if self.IsPaneButtonVisible(part): self._action = actionClickButton self._action_part = part self._action_pane = None self._action_start = wx.Point(*event.GetPosition()) self._frame.CaptureMouse() self.RefreshButton(part) elif part.type in [AuiDockUIPart.typeCaption, AuiDockUIPart.typeGripper]: # if we are managing a AuiFloatingFrame window, then # we are an embedded AuiManager inside the AuiFloatingFrame. # We want to initiate a toolbar drag in our owner manager if isinstance(part.pane.window.GetParent(), AuiFloatingFrame): rootManager = GetManager(part.pane.window) else: rootManager = self offset = wx.Point(event.GetX() - part.rect.x, event.GetY() - part.rect.y) rootManager.OnGripperClicked(part.pane.window, event.GetPosition(), offset) if wx.Platform != "__WXMAC__": event.Skip() def OnLeftDClick(self, event): """ Handles the ``wx.EVT_LEFT_DCLICK`` event for :class:`AuiManager`. :param `event`: a :class:`MouseEvent` to be processed. """ part = self.HitTest(event.GetX(), event.GetY()) if part and part.type == AuiDockUIPart.typeCaption: if isinstance(part.pane.window.GetParent(), AuiFloatingFrame): rootManager = GetManager(part.pane.window) else: rootManager = self rootManager.OnCaptionDoubleClicked(part.pane.window) elif part and part.type in [AuiDockUIPart.typeDockSizer, AuiDockUIPart.typePaneSizer]: # Handles double click on AuiNotebook sashes to unsplit sash_size = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) for child in part.cont_sizer.GetChildren(): if child.IsSizer(): win = child.GetSizer().GetContainingWindow() if isinstance(win, auibook.AuiNotebook): win.UnsplitDClick(part, sash_size, event.GetPosition()) break event.Skip() def DoEndResizeAction(self, event): """ Ends a resize action, or for live update, resizes the sash. :param `event`: a :class:`MouseEvent` to be processed. """ clientPt = event.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) return self.RestrictResize(clientPt, screenPt, createDC=False) def RestrictResize(self, clientPt, screenPt, createDC): """ Common method between :meth:`DoEndResizeAction` and :meth:`OnLeftUp_Resize`. """ dock = self._action_part.dock pane = self._action_part.pane if createDC: if wx.Platform == "__WXMAC__": dc = wx.ClientDC(self._frame) else: dc = wx.ScreenDC() DrawResizeHint(dc, self._action_rect) self._action_rect = wx.Rect() newPos = clientPt - self._action_offset if self._action_part.type == AuiDockUIPart.typeDockSizer: minPix, maxPix = self.CalculateDockSizerLimits(dock) else: if not self._action_part.pane: return minPix, maxPix = self.CalculatePaneSizerLimits(dock, pane) if self._action_part.orientation == wx.HORIZONTAL: newPos.y = Clip(newPos.y, minPix, maxPix) else: newPos.x = Clip(newPos.x, minPix, maxPix) if self._action_part.type == AuiDockUIPart.typeDockSizer: partnerDock = self.GetPartnerDock(dock) sash_size = self._art.GetMetric(AUI_DOCKART_SASH_SIZE) new_dock_size = 0 direction = dock.dock_direction if direction == AUI_DOCK_LEFT: new_dock_size = newPos.x - dock.rect.x elif direction == AUI_DOCK_TOP: new_dock_size = newPos.y - dock.rect.y elif direction == AUI_DOCK_RIGHT: new_dock_size = dock.rect.x + dock.rect.width - newPos.x - sash_size elif direction == AUI_DOCK_BOTTOM: new_dock_size = dock.rect.y + dock.rect.height - newPos.y - sash_size deltaDockSize = new_dock_size - dock.size if partnerDock: if deltaDockSize > partnerDock.size - sash_size: deltaDockSize = partnerDock.size - sash_size partnerDock.size -= deltaDockSize dock.size += deltaDockSize self.Update() else: # determine the new pixel size that the user wants # this will help us recalculate the pane's proportion if dock.IsHorizontal(): oldPixsize = pane.rect.width newPixsize = oldPixsize + newPos.x - self._action_part.rect.x else: oldPixsize = pane.rect.height newPixsize = oldPixsize + newPos.y - self._action_part.rect.y totalPixsize, totalProportion = self.GetTotalPixSizeAndProportion(dock) partnerPane = self.GetPartnerPane(dock, pane) # prevent division by zero if totalPixsize <= 0 or totalProportion <= 0 or not partnerPane: return # adjust for the surplus while (oldPixsize > 0 and totalPixsize > 10 and \ oldPixsize*totalProportion/totalPixsize < pane.dock_proportion): totalPixsize -= 1 # calculate the new proportion of the pane newProportion = newPixsize*totalProportion/totalPixsize newProportion = Clip(newProportion, 1, totalProportion) deltaProp = newProportion - pane.dock_proportion if partnerPane.dock_proportion - deltaProp < 1: deltaProp = partnerPane.dock_proportion - 1 newProportion = pane.dock_proportion + deltaProp # borrow the space from our neighbor pane to the # right or bottom (depending on orientation) partnerPane.dock_proportion -= deltaProp pane.dock_proportion = newProportion self.Update() return True def OnLeftUp(self, event): """ Handles the ``wx.EVT_LEFT_UP`` event for :class:`AuiManager`. :param `event`: a :class:`MouseEvent` to be processed. """ if self._action == actionResize: ## self._frame.Freeze() self.OnLeftUp_Resize(event) ## self._frame.Thaw() elif self._action == actionClickButton: self.OnLeftUp_ClickButton(event) elif self._action == actionDragFloatingPane: self.OnLeftUp_DragFloatingPane(event) elif self._action == actionDragToolbarPane: self.OnLeftUp_DragToolbarPane(event) elif self._action == actionDragMovablePane: self.OnLeftUp_DragMovablePane(event) else: event.Skip() try: if self._frame.HasCapture(): self._frame.ReleaseMouse() except RuntimeError: pass self._action = actionNone def OnMotion(self, event): """ Handles the ``wx.EVT_MOTION`` event for :class:`AuiManager`. :param `event`: a :class:`MouseEvent` to be processed. """ if self._action == actionResize: self.OnMotion_Resize(event) elif self._action == actionClickCaption: self.OnMotion_ClickCaption(event) elif self._action == actionDragFloatingPane: self.OnMotion_DragFloatingPane(event) elif self._action == actionDragToolbarPane: self.OnMotion_DragToolbarPane(event) elif self._action == actionDragMovablePane: self.OnMotion_DragMovablePane(event) else: self.OnMotion_Other(event) def OnLeaveWindow(self, event): """ Handles the ``wx.EVT_LEAVE_WINDOW`` event for :class:`AuiManager`. :param `event`: a :class:`MouseEvent` to be processed. """ if self._hover_button: self.RefreshButton(self._hover_button) self._hover_button = None def OnCaptureLost(self, event): """ Handles the ``wx.EVT_MOUSE_CAPTURE_LOST`` event for :class:`AuiManager`. :param `event`: a :class:`MouseCaptureLostEvent` to be processed. """ # cancel the operation in progress, if any if self._action != actionNone: self._action = actionNone self.HideHint() def OnHintFadeTimer(self, event): """ Handles the ``wx.EVT_TIMER`` event for :class:`AuiManager`. :param `event`: a :class:`TimerEvent` to be processed. """ if not self._hint_window or self._hint_fadeamt >= self._hint_fademax: self._hint_fadetimer.Stop() return self._hint_fadeamt += 4 self._hint_window.SetTransparent(self._hint_fadeamt) def OnMove(self, event): """ Handles the ``wx.EVT_MOVE`` event for :class:`AuiManager`. :param `event`: a :class:`MoveEvent` to be processed. """ if event is not None: event.Skip() if isinstance(self._frame, AuiFloatingFrame) and self._frame.IsShownOnScreen(): return docked, hAlign, vAlign, monitor = self._is_docked if docked: self.Snap() for pane in self._panes: if pane.IsSnappable(): if pane.IsFloating() and pane.IsShown(): self.SnapPane(pane, pane.floating_pos, pane.floating_size, True) def OnSysColourChanged(self, event): """ Handles the ``wx.EVT_SYS_COLOUR_CHANGED`` event for :class:`AuiManager`. :param `event`: a :class:`SysColourChangedEvent` to be processed. """ # This event is probably triggered by a theme change # so we have to re-init the art provider. if self._art: self._art.Init() if self._frame: self.Update() self._frame.Refresh() def OnChildFocus(self, event): """ Handles the ``wx.EVT_CHILD_FOCUS`` event for :class:`AuiManager`. :param `event`: a :class:`ChildFocusEvent` to be processed. """ # when a child pane has it's focus set, we should change the # pane's active state to reflect this. (this is only true if # active panes are allowed by the owner) window = event.GetWindow() if isinstance(window, wx.Dialog): # Ignore EVT_CHILD_FOCUS events originating from dialogs not # managed by AUI rootManager = None elif isinstance(window.GetParent(), AuiFloatingFrame): rootManager = GetManager(window) else: rootManager = self if rootManager: rootManager.ActivatePane(window) event.Skip() def OnMotion_ClickCaption(self, event): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ clientPt = event.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) drag_x_threshold = wx.SystemSettings.GetMetric(wx.SYS_DRAG_X) drag_y_threshold = wx.SystemSettings.GetMetric(wx.SYS_DRAG_Y) if not self._action_pane: return # we need to check if the mouse is now being dragged if not (abs(clientPt.x - self._action_start.x) > drag_x_threshold or \ abs(clientPt.y - self._action_start.y) > drag_y_threshold): return # dragged -- we need to change the mouse action to 'drag' if self._action_pane.IsToolbar(): self._action = actionDragToolbarPane self._action_window = self._action_pane.window elif self._action_pane.IsFloatable() and self._agwFlags & AUI_MGR_ALLOW_FLOATING: e = self.FireEvent(wxEVT_AUI_PANE_FLOATING, self._action_pane, canVeto=True) if e.GetVeto(): return self._action = actionDragFloatingPane # set initial float position self._action_pane.floating_pos = screenPt - self._action_offset # float the window if self._action_pane.IsMaximized(): self.RestorePane(self._action_pane) self._action_pane.Hide() self._action_pane.Float() if wx.Platform == "__WXGTK__": self._action_pane.Show() e = self.FireEvent(wxEVT_AUI_PANE_FLOATED, self._action_pane, canVeto=False) if not self._action_pane.frame: self.Update() self._action_window = self._action_pane.window # adjust action offset for window frame windowPt = self._action_pane.frame.GetRect().GetTopLeft() originPt = self._action_pane.frame.ClientToScreen(wx.Point()) self._toolbar_action_offset = originPt - windowPt if self._agwFlags & AUI_MGR_USE_NATIVE_MINIFRAMES: originPt = windowPt + wx.Point(3, 3) self._action_offset += originPt - windowPt # action offset is used here to make it feel "natural" to the user # to drag a docked pane and suddenly have it become a floating frame. # Sometimes, however, the offset where the user clicked on the docked # caption is bigger than the width of the floating frame itself, so # in that case we need to set the action offset to a sensible value frame_size = self._action_pane.frame.GetSize() if self._action_offset.x > frame_size.x * 2 / 3: self._action_offset.x = frame_size.x / 2 if self._action_offset.y > frame_size.y * 2 / 3: self._action_offset.y = frame_size.y / 2 self.OnMotion_DragFloatingPane(event) if wx.Platform != "__WXGTK__": self._action_pane.Show() self.Update() elif self._action_pane.IsMovable(): self._action = actionDragMovablePane self._action_window = self._action_pane.window def OnMotion_Resize(self, event): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ if AuiManager_HasLiveResize(self): if self._currentDragItem != -1: self._action_part = self._uiparts[self._currentDragItem] else: self._currentDragItem = self._uiparts.index(self._action_part) if self._frame.HasCapture(): self._frame.ReleaseMouse() self.DoEndResizeAction(event) self._frame.CaptureMouse() return if not self._action_part or not self._action_part.dock or not self._action_part.orientation: return clientPt = event.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) dock = self._action_part.dock pos = self._action_part.rect.GetPosition() if self._action_part.type == AuiDockUIPart.typeDockSizer: minPix, maxPix = self.CalculateDockSizerLimits(dock) else: if not self._action_part.pane: return pane = self._action_part.pane minPix, maxPix = self.CalculatePaneSizerLimits(dock, pane) if self._action_part.orientation == wx.HORIZONTAL: pos.y = Clip(clientPt.y - self._action_offset.y, minPix, maxPix) else: pos.x = Clip(clientPt.x - self._action_offset.x, minPix, maxPix) hintrect = wx.Rect(self._frame.ClientToScreen(pos), self._action_part.rect.GetSize()) if hintrect != self._action_rect: if wx.Platform == "__WXMAC__": dc = wx.ClientDC(self._frame) else: dc = wx.ScreenDC() DrawResizeHint(dc, self._action_rect) DrawResizeHint(dc, hintrect) self._action_rect = wx.Rect(*hintrect) def OnLeftUp_Resize(self, event): """ Sub-handler for the :meth:`OnLeftUp` event. :param `event`: a :class:`MouseEvent` to be processed. """ if self._currentDragItem != -1 and AuiManager_HasLiveResize(self): self._action_part = self._uiparts[self._currentDragItem] if self._frame.HasCapture(): self._frame.ReleaseMouse() self.DoEndResizeAction(event) self._currentDragItem = -1 return if not self._action_part or not self._action_part.dock: return clientPt = event.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) return self.RestrictResize(clientPt, screenPt, createDC=True) def OnLeftUp_ClickButton(self, event): """ Sub-handler for the :meth:`OnLeftUp` event. :param `event`: a :class:`MouseEvent` to be processed. """ self._hover_button = None if self._action_part: self.RefreshButton(self._action_part) # make sure we're still over the item that was originally clicked if self._action_part == self.HitTest(*event.GetPosition()): # fire button-click event e = AuiManagerEvent(wxEVT_AUI_PANE_BUTTON) e.SetManager(self) e.SetPane(self._action_part.pane) e.SetButton(self._action_part.button.button_id) self.ProcessMgrEvent(e) def CheckPaneMove(self, pane): """ Checks if a pane has moved by a visible amount. :param `pane`: an instance of :class:`AuiPaneInfo`. """ win_rect = pane.frame.GetRect() win_rect.x, win_rect.y = pane.floating_pos if win_rect == self._last_rect: return False # skip the first move event if self._last_rect.IsEmpty(): self._last_rect = wx.Rect(*win_rect) return False # skip if moving too fast to avoid massive redraws and # jumping hint windows if abs(win_rect.x - self._last_rect.x) > 10 or \ abs(win_rect.y - self._last_rect.y) > 10: self._last_rect = wx.Rect(*win_rect) return False return True def OnMotion_DragFloatingPane(self, eventOrPt): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ isPoint = False if isinstance(eventOrPt, wx.Point): clientPt = self._frame.ScreenToClient(eventOrPt) screenPt = wx.Point(*eventOrPt) isPoint = True else: clientPt = eventOrPt.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) framePos = wx.Point() # try to find the pane pane = self.GetPane(self._action_window) if not pane.IsOk(): raise Exception("Pane window not found") # update floating position if pane.IsFloating(): diff = pane.floating_pos - (screenPt - self._action_offset) pane.floating_pos = screenPt - self._action_offset framePos = pane.floating_pos # Move the pane window if pane.frame: if diff.x != 0 or diff.y != 0: if wx.Platform == "__WXMSW__" and (self._agwFlags & AUI_MGR_TRANSPARENT_DRAG) == 0: # and not self.CheckPaneMove(pane): # return # HACK: Terrible hack on wxMSW (!) pane.frame.SetTransparent(254) self._from_move = True pane.frame.Move(pane.floating_pos) self._from_move = False if self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: pane.frame.SetTransparent(150) # calculate the offset from the upper left-hand corner # of the frame to the mouse pointer action_offset = screenPt - framePos # is the pane dockable? if not self.CanDockPanel(pane): self.HideHint() ShowDockingGuides(self._guides, False) return for paneInfo in self._panes: if not paneInfo.IsDocked() or not paneInfo.IsShown(): continue if paneInfo.IsToolbar() or paneInfo.IsNotebookControl(): continue if paneInfo.IsMaximized(): continue if paneInfo.IsNotebookPage(): notebookRoot = GetNotebookRoot(self._panes, paneInfo.notebook_id) if not notebookRoot or not notebookRoot.IsDocked(): continue rc = paneInfo.window.GetScreenRect() if rc.Contains(screenPt): if rc.height < 20 or rc.width < 20: return self.UpdateDockingGuides(paneInfo) ShowDockingGuides(self._guides, True) break self.DrawHintRect(pane.window, clientPt, action_offset) def OnMotion_DragMovablePane(self, eventOrPt): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ # Try to find the pane. pane = self.GetPane(self._action_window) if not pane.IsOk(): raise Exception("Pane window not found") # Draw a hint for where the window will be moved. if isinstance(eventOrPt, wx.Point): pt = wx.Point(*eventOrPt) else: pt = eventOrPt.GetPosition() self.DrawHintRect(self._action_window, pt, wx.Point(0, 0)) # Reduces flicker. self._frame.Update() def OnLeftUp_DragFloatingPane(self, eventOrPt): """ Sub-handler for the :meth:`OnLeftUp` event. :param `event`: a :class:`MouseEvent` to be processed. """ if isinstance(eventOrPt, wx.Point): clientPt = self._frame.ScreenToClient(eventOrPt) screenPt = wx.Point(*eventOrPt) else: clientPt = eventOrPt.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) # try to find the pane paneInfo = self.GetPane(self._action_window) if not paneInfo.IsOk(): raise Exception("Pane window not found") ret = False if paneInfo.frame: # calculate the offset from the upper left-hand corner # of the frame to the mouse pointer framePos = paneInfo.frame.GetPosition() action_offset = screenPt - framePos # is the pane dockable? if self.CanDockPanel(paneInfo): # do the drop calculation indx = self._panes.index(paneInfo) ret, paneInfo = self.DoDrop(self._docks, self._panes, paneInfo, clientPt, action_offset) if ret: e = self.FireEvent(wxEVT_AUI_PANE_DOCKING, paneInfo, canVeto=True) if e.GetVeto(): self.HideHint() ShowDockingGuides(self._guides, False) return e = self.FireEvent(wxEVT_AUI_PANE_DOCKED, paneInfo, canVeto=False) if self._agwFlags & AUI_MGR_SMOOTH_DOCKING: self.SmoothDock(paneInfo) self._panes[indx] = paneInfo # if the pane is still floating, update it's floating # position (that we store) if paneInfo.IsFloating(): paneInfo.floating_pos = paneInfo.frame.GetPosition() if paneInfo.frame._transparent != paneInfo.transparent or self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: paneInfo.frame.SetTransparent(paneInfo.transparent) paneInfo.frame._transparent = paneInfo.transparent elif self._has_maximized: self.RestoreMaximizedPane() # reorder for dropping to a new notebook # (caution: this code breaks the reference!) tempPaneInfo = self.CopyTarget(paneInfo) self._panes.remove(paneInfo) self._panes.append(tempPaneInfo) if ret: self.Update() if tempPaneInfo.IsFloating(): self.SnapPane(tempPaneInfo, tempPaneInfo.floating_pos, tempPaneInfo.floating_size, False) self.HideHint() ShowDockingGuides(self._guides, False) def OnLeftUp_DragMovablePane(self, event): """ Sub-handler for the :meth:`OnLeftUp` event. :param `event`: a :class:`MouseEvent` to be processed. """ # Try to find the pane. paneInfo = self.GetPane(self._action_window) if not paneInfo.IsOk(): raise Exception("Pane window not found") # Hide the hint as it is no longer needed. self.HideHint() # is the pane dockable? if self.CanDockPanel(paneInfo): # Move the pane to new position. pt = event.GetPosition() # do the drop calculation indx = self._panes.index(paneInfo) ret, paneInfo = self.DoDrop(self._docks, self._panes, paneInfo, pt, wx.Point(0,0)) if ret: e = self.FireEvent(wxEVT_AUI_PANE_DOCKING, paneInfo, canVeto=True) if e.GetVeto(): self.HideHint() ShowDockingGuides(self._guides, False) return e = self.FireEvent(wxEVT_AUI_PANE_DOCKED, paneInfo, canVeto=False) if self._agwFlags & AUI_MGR_SMOOTH_DOCKING: self.SmoothDock(paneInfo) self._panes[indx] = paneInfo if ret: # Update the layout to realize new position and e.g. form notebooks if needed. self.Update() if self.GetAGWFlags() & AUI_MGR_ALLOW_ACTIVE_PANE: # Ensure active before doing actual display. ret, self._panes = SetActivePane(self._panes, paneInfo.window) # Make changes visible to user. self.Repaint() # Cancel the action and release the mouse. self._action = actionNone self._frame.ReleaseMouse() self._action_window = None def OnMotion_DragToolbarPane(self, eventOrPt): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ isPoint = False if isinstance(eventOrPt, wx.Point): clientPt = self._frame.ScreenToClient(eventOrPt) screenPt = wx.Point(*eventOrPt) isPoint = True else: clientPt = eventOrPt.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) pane = self.GetPane(self._action_window) if not pane.IsOk(): raise Exception("Pane window not found") pane.state |= AuiPaneInfo.actionPane indx = self._panes.index(pane) ret = False wasFloating = pane.IsFloating() # is the pane dockable? if self.CanDockPanel(pane): # do the drop calculation ret, pane = self.DoDrop(self._docks, self._panes, pane, clientPt, self._action_offset) # update floating position if pane.IsFloating(): pane.floating_pos = screenPt - self._toolbar_action_offset # move the pane window if pane.frame: if wx.Platform == "__WXMSW__" and (self._agwFlags & AUI_MGR_TRANSPARENT_DRAG) == 0: # and not self.CheckPaneMove(pane): # return # HACK: Terrible hack on wxMSW (!) pane.frame.SetTransparent(254) self._from_move = True pane.frame.Move(pane.floating_pos) self._from_move = False if self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: pane.frame.SetTransparent(150) self._panes[indx] = pane if ret and wasFloating != pane.IsFloating() or (ret and not wasFloating): wx.CallAfter(self.Update) # when release the button out of the window. # TODO: a better fix is needed. if _VERSION_STRING < "2.9": leftDown = wx.GetMouseState().LeftDown() else: leftDown = wx.GetMouseState().LeftIsDown() if not leftDown: self._action = actionNone self.OnLeftUp_DragToolbarPane(eventOrPt) def OnMotion_Other(self, event): """ Sub-handler for the :meth:`OnMotion` event. :param `event`: a :class:`MouseEvent` to be processed. """ part = self.HitTest(*event.GetPosition()) if part and part.type == AuiDockUIPart.typePaneButton \ and self.IsPaneButtonVisible(part): if part != self._hover_button: if self._hover_button: self.RefreshButton(self._hover_button) self._hover_button = part self.RefreshButton(part) else: if self._hover_button: self.RefreshButton(self._hover_button) else: event.Skip() self._hover_button = None def OnLeftUp_DragToolbarPane(self, eventOrPt): """ Sub-handler for the :meth:`OnLeftUp` event. :param `event`: a :class:`MouseEvent` to be processed. """ isPoint = False if isinstance(eventOrPt, wx.Point): clientPt = self._frame.ScreenToClient(eventOrPt) screenPt = wx.Point(*eventOrPt) isPoint = True else: clientPt = eventOrPt.GetPosition() screenPt = self._frame.ClientToScreen(clientPt) # try to find the pane pane = self.GetPane(self._action_window) if not pane.IsOk(): raise Exception("Pane window not found") if pane.IsFloating(): pane.floating_pos = pane.frame.GetPosition() if pane.frame._transparent != pane.transparent or self._agwFlags & AUI_MGR_TRANSPARENT_DRAG: pane.frame.SetTransparent(pane.transparent) pane.frame._transparent = pane.transparent # save the new positions docks = FindDocks(self._docks, pane.dock_direction, pane.dock_layer, pane.dock_row) if len(docks) == 1: dock = docks[0] pane_positions, pane_sizes = self.GetPanePositionsAndSizes(dock) for i in range(len(dock.panes)): dock.panes[i].dock_pos = pane_positions[i] pane.state &= ~AuiPaneInfo.actionPane self.Update() def OnPaneButton(self, event): """ Handles the ``EVT_AUI_PANE_BUTTON`` event for :class:`AuiManager`. :param `event`: a :class:`AuiManagerEvent` event to be processed. """ if not event.pane: raise Exception("Pane Info passed to AuiManager.OnPaneButton must be non-null") pane = event.pane if event.button == AUI_BUTTON_CLOSE: if isinstance(pane.window.GetParent(), AuiFloatingFrame): rootManager = GetManager(pane.window) else: rootManager = self if rootManager != self: self._frame.Close() return # fire pane close event e = AuiManagerEvent(wxEVT_AUI_PANE_CLOSE) e.SetManager(self) e.SetPane(event.pane) self.ProcessMgrEvent(e) if not e.GetVeto(): # close the pane, but check that it # still exists in our pane array first # (the event handler above might have removed it) check = self.GetPane(pane.window) if check.IsOk(): self.ClosePane(pane) self.Update() # mn this performs the minimizing of a pane elif event.button == AUI_BUTTON_MINIMIZE: e = AuiManagerEvent(wxEVT_AUI_PANE_MINIMIZE) e.SetManager(self) e.SetPane(event.pane) self.ProcessMgrEvent(e) if not e.GetVeto(): self.MinimizePane(pane) elif event.button == AUI_BUTTON_MAXIMIZE_RESTORE and not pane.IsMaximized(): # fire pane close event e = AuiManagerEvent(wxEVT_AUI_PANE_MAXIMIZE) e.SetManager(self) e.SetPane(event.pane) self.ProcessMgrEvent(e) if not e.GetVeto(): self.MaximizePane(pane) self.Update() elif event.button == AUI_BUTTON_MAXIMIZE_RESTORE and pane.IsMaximized(): # fire pane close event e = AuiManagerEvent(wxEVT_AUI_PANE_RESTORE) e.SetManager(self) e.SetPane(event.pane) self.ProcessMgrEvent(e) if not e.GetVeto(): self.RestorePane(pane) self.Update() elif event.button == AUI_BUTTON_PIN: if self._agwFlags & AUI_MGR_ALLOW_FLOATING and pane.IsFloatable(): e = self.FireEvent(wxEVT_AUI_PANE_FLOATING, pane, canVeto=True) if e.GetVeto(): return pane.Float() e = self.FireEvent(wxEVT_AUI_PANE_FLOATED, pane, canVeto=False) self.Update() def MinimizePane(self, paneInfo, mgrUpdate=True): """ Minimizes a pane in a newly and automatically created :class:`~lib.agw.aui.auibar.AuiToolBar`. Clicking on the minimize button causes a new :class:`~lib.agw.aui.auibar.AuiToolBar` to be created and added to the frame manager (currently the implementation is such that panes at West will have a toolbar at the right, panes at South will have toolbars at the bottom etc...) and the pane is hidden in the manager. Clicking on the restore button on the newly created toolbar will result in the toolbar being removed and the original pane being restored. :param `paneInfo`: a :class:`AuiPaneInfo` instance for the pane to be minimized; :param bool `mgrUpdate`: ``True`` to call :meth:`Update` to realize the new layout, ``False`` otherwise. .. note:: The `mgrUpdate` parameter is currently only used while loading perspectives using :meth:`LoadPerspective`, as minimized panes were not correctly taken into account before. """ if not paneInfo.IsToolbar(): if paneInfo.IsMinimized() and mgrUpdate: # We are already minimized return # Basically the idea is this. # # 1) create a toolbar, with a restore button # # 2) place the new toolbar in the toolbar area representative of the location of the pane # (NORTH/SOUTH/EAST/WEST, central area always to the right) # # 3) Hide the minimizing pane # personalize the toolbar style tbStyle = AUI_TB_DEFAULT_STYLE posMask = paneInfo.minimize_mode & AUI_MINIMIZE_POS_MASK captMask = paneInfo.minimize_mode & AUI_MINIMIZE_CAPT_MASK dockDirection = paneInfo.dock_direction if captMask != 0: tbStyle |= AUI_TB_TEXT if posMask == AUI_MINIMIZE_POS_TOOLBAR: minimize_toolbar = self.GetPane(paneInfo.minimize_target) if not minimize_toolbar.IsOk(): posMask = AUI_MINIMIZE_POS_SMART if paneInfo.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_BOTTOM]: tbStyle |= AUI_TB_HORZ_LAYOUT elif paneInfo.dock_direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT, AUI_DOCK_CENTER]: tbStyle |= AUI_TB_VERTICAL if captMask == AUI_MINIMIZE_CAPT_SMART: tbStyle |= AUI_TB_CLOCKWISE else: minimize_toolbar = minimize_toolbar.window elif posMask == AUI_MINIMIZE_POS_SMART: if paneInfo.dock_direction in [AUI_DOCK_TOP, AUI_DOCK_BOTTOM]: tbStyle |= AUI_TB_HORZ_LAYOUT elif paneInfo.dock_direction in [AUI_DOCK_LEFT, AUI_DOCK_RIGHT, AUI_DOCK_CENTER]: tbStyle |= AUI_TB_VERTICAL if captMask == AUI_MINIMIZE_CAPT_SMART: tbStyle |= AUI_TB_CLOCKWISE elif posMask in [AUI_MINIMIZE_POS_TOP, AUI_MINIMIZE_POS_BOTTOM]: tbStyle |= AUI_TB_HORZ_LAYOUT if posMask == AUI_MINIMIZE_POS_TOP: dockDirection = AUI_DOCK_TOP else: dockDirection = AUI_DOCK_BOTTOM else: tbStyle |= AUI_TB_VERTICAL if captMask == AUI_MINIMIZE_CAPT_SMART: tbStyle |= AUI_TB_CLOCKWISE if posMask == AUI_MINIMIZE_POS_LEFT: dockDirection = AUI_DOCK_LEFT elif posMask == AUI_MINIMIZE_POS_RIGHT: dockDirection = AUI_DOCK_RIGHT elif posMask == AUI_MINIMIZE_POS_BOTTOM: dockDirection = AUI_DOCK_BOTTOM # Create a new toolbar # give it the same name as the minimized pane with _min appended win_rect = paneInfo.window.GetScreenRect() if posMask != AUI_MINIMIZE_POS_TOOLBAR: minimize_toolbar = auibar.AuiToolBar(self.GetManagedWindow(), agwStyle=tbStyle) minimize_toolbar.Hide() minimize_toolbar.SetToolBitmapSize(wx.Size(16, 16)) if paneInfo.icon and paneInfo.icon.IsOk(): restore_bitmap = paneInfo.icon else: restore_bitmap = self._art._restore_bitmap if posMask == AUI_MINIMIZE_POS_TOOLBAR: xsize, ysize = minimize_toolbar.GetToolBitmapSize() if xsize != restore_bitmap.GetWidth(): img = restore_bitmap.ConvertToImage() img.Rescale(xsize, ysize, wx.IMAGE_QUALITY_HIGH) restore_bitmap = img.ConvertToBitmap() target = None if posMask == AUI_MINIMIZE_POS_TOOLBAR: target = paneInfo.name minimize_toolbar.AddSimpleTool(ID_RESTORE_FRAME, paneInfo.caption, restore_bitmap, _(six.u("Restore %s"))%paneInfo.caption, target=target) minimize_toolbar.SetAuiManager(self) minimize_toolbar.Realize() toolpanelname = paneInfo.name + "_min" if paneInfo.IsMaximized(): paneInfo.SetFlag(paneInfo.wasMaximized, True) if posMask != AUI_MINIMIZE_POS_TOOLBAR: if dockDirection == AUI_DOCK_TOP: self.AddPane(minimize_toolbar, AuiPaneInfo(). \ Name(toolpanelname).Caption(paneInfo.caption). \ ToolbarPane().Top().BottomDockable(False). \ LeftDockable(False).RightDockable(False).DestroyOnClose()) elif dockDirection == AUI_DOCK_BOTTOM: self.AddPane(minimize_toolbar, AuiPaneInfo(). \ Name(toolpanelname).Caption(paneInfo.caption). \ ToolbarPane().Bottom().TopDockable(False). \ LeftDockable(False).RightDockable(False).DestroyOnClose()) elif dockDirection == AUI_DOCK_LEFT: self.AddPane(minimize_toolbar, AuiPaneInfo(). \ Name(toolpanelname).Caption(paneInfo.caption). \ ToolbarPane().Left().TopDockable(False). \ BottomDockable(False).RightDockable(False).DestroyOnClose()) elif dockDirection in [AUI_DOCK_RIGHT, AUI_DOCK_CENTER]: self.AddPane(minimize_toolbar, AuiPaneInfo(). \ Name(toolpanelname).Caption(paneInfo.caption). \ ToolbarPane().Right().TopDockable(False). \ LeftDockable(False).BottomDockable(False).DestroyOnClose()) arr = FindDocks(self._docks, paneInfo.dock_direction, paneInfo.dock_layer, paneInfo.dock_row) if arr: dock = arr[0] paneInfo.previousDockSize = dock.size paneInfo.previousDockPos = paneInfo.dock_pos # mark ourselves minimized paneInfo.Minimize() paneInfo.Show(False) self._has_minimized = True # last, hide the window if paneInfo.window and paneInfo.window.IsShown(): paneInfo.window.Show(False) minimize_toolbar.Show() if mgrUpdate: self.Update() if self._agwFlags & AUI_MGR_ANIMATE_FRAMES: self.AnimateDocking(win_rect, minimize_toolbar.GetScreenRect()) def OnRestoreMinimizedPane(self, event): """ Handles the ``EVT_AUI_PANE_MIN_RESTORE`` event for :class:`AuiManager`. :param `event`: an instance of :class:`AuiManagerEvent` to be processed. """ self.RestoreMinimizedPane(event.pane) def OnPaneDocked(self, event): """ Handles the ``EVT_AUI_PANE_DOCKED`` event for :class:`AuiManager`. :param `event`: an instance of :class:`AuiManagerEvent` to be processed. """ event.Skip() self.RemoveAutoNBCaption(event.GetPane()) def CreateNotebookBase(self, panes, paneInfo): """ Creates an auto-notebook base from a pane, and then add that pane as a page. :param list `panes`: set of panes to append new notebook base pane to :param `paneInfo`: the pane to be converted to a new notebook, an instance of :class:`AuiPaneInfo`. """ # Create base notebook pane ... nbid = len(self._notebooks) baseInfo = AuiPaneInfo() baseInfo.SetDockPos(paneInfo).NotebookControl(nbid). \ CloseButton(False).SetNameFromNotebookId(). \ NotebookDockable(False).Floatable(paneInfo.IsFloatable()) baseInfo.best_size = paneInfo.best_size panes.append(baseInfo) # add original pane as tab ... paneInfo.NotebookPage(nbid) def RemoveAutoNBCaption(self, pane): """ Removes the caption on newly created automatic notebooks. :param `pane`: an instance of :class:`AuiPaneInfo` (the target notebook). """ if self._agwFlags & AUI_MGR_AUTONB_NO_CAPTION == 0: return False def RemoveCaption(): """ Sub-function used to remove the pane caption on automatic notebooks. """ if pane.HasNotebook(): notebook = self._notebooks[pane.notebook_id] self.GetPane(notebook).CaptionVisible(False).PaneBorder(False) self.Update() # it seems the notebook isnt created by this stage, so remove # the caption a moment later wx.CallAfter(RemoveCaption) return True def RestoreMinimizedPane(self, paneInfo): """ Restores a previously minimized pane. :param `paneInfo`: a :class:`AuiPaneInfo` instance for the pane to be restored. """ panename = paneInfo.name if paneInfo.minimize_mode & AUI_MINIMIZE_POS_TOOLBAR: pane = self.GetPane(panename) hasTarget = True else: panename = panename[0:-4] hasTarget = False pane = self.GetPane(panename) pane.SetFlag(pane.needsRestore, True) if not pane.IsOk(): panename = paneInfo.name pane = self.GetPane(panename) paneInfo = self.GetPane(panename + "_min") if not paneInfo.IsOk(): # Already minimized return if pane.IsOk(): if not pane.IsMinimized(): return if pane.HasFlag(pane.wasMaximized): self.SavePreviousDockSizes(pane) self.ShowPane(pane.window, True) pane.Show(True) self._has_minimized = False pane.SetFlag(pane.optionMinimized, False) if hasTarget: targetName = pane.minimize_target toolbarPane = self.GetPane(targetName) toolbar = toolbarPane.window item = toolbar.FindToolByLabel(pane.caption) toolbar.DeleteTool(item.id) else: paneInfo.window.Show(False) self.DetachPane(paneInfo.window) paneInfo.Show(False) paneInfo.Hide() self.Update() def AnimateDocking(self, win_rect, pane_rect): """ Animates the minimization/docking of a pane a la Eclipse, using a :class:`ScreenDC` to draw a "moving docking rectangle" on the screen. :param Rect `win_rect`: the original pane screen rectangle; :param Rect `pane_rect`: the newly created toolbar/pane screen rectangle. :note: This functionality is not available on wxMAC as this platform doesn't have the ability to use :class:`ScreenDC` to draw on-screen and on Windows > Vista. """ if wx.Platform == "__WXMAC__": # No wx.ScreenDC on the Mac... return if wx.Platform == "__WXMSW__" and wx.GetOsVersion()[1] > 5: # No easy way to handle this on Vista... return xstart, ystart = win_rect.x, win_rect.y xend, yend = pane_rect.x, pane_rect.y step = self.GetAnimationStep() wstep = int(abs(win_rect.width - pane_rect.width)/step) hstep = int(abs(win_rect.height - pane_rect.height)/step) xstep = int(win_rect.x - pane_rect.x)/step ystep = int(win_rect.y - pane_rect.y)/step dc = wx.ScreenDC() dc.SetLogicalFunction(wx.INVERT) dc.SetBrush(wx.TRANSPARENT_BRUSH) dc.SetPen(wx.LIGHT_GREY_PEN) for i in range(int(step)): width, height = win_rect.width - i*wstep, win_rect.height - i*hstep x, y = xstart - i*xstep, ystart - i*ystep new_rect = wx.Rect(x, y, width, height) dc.DrawRoundedRectangle(new_rect, 3) wx.SafeYield() wx.MilliSleep(10) dc.DrawRoundedRectangle(new_rect, 3) def SmoothDock(self, paneInfo): """ This method implements a smooth docking effect for floating panes, similar to what the PyQT library does with its floating windows. :param `paneInfo`: an instance of :class:`AuiPaneInfo`. :note: The smooth docking effect can only be used if you set the ``AUI_MGR_SMOOTH_DOCKING`` style to :class:`AuiManager`. """ if paneInfo.IsToolbar(): return if not paneInfo.frame or self._hint_rect.IsEmpty(): return hint_rect = self._hint_rect win_rect = paneInfo.frame.GetScreenRect() xstart, ystart = win_rect.x, win_rect.y xend, yend = hint_rect.x, hint_rect.y step = self.GetAnimationStep()/3 wstep = int((win_rect.width - hint_rect.width)/step) hstep = int((win_rect.height - hint_rect.height)/step) xstep = int((win_rect.x - hint_rect.x))/step ystep = int((win_rect.y - hint_rect.y))/step for i in range(int(step)): width, height = win_rect.width - i*wstep, win_rect.height - i*hstep x, y = xstart - i*xstep, ystart - i*ystep new_rect = wx.Rect(x, y, width, height) paneInfo.frame.SetRect(new_rect) wx.MilliSleep(10) def SetSnapLimits(self, x, y): """ Modifies the snap limits used when snapping the `managed_window` to the screen (using :meth:`SnapToScreen`) or when snapping the floating panes to one side of the `managed_window` (using :meth:`SnapPane`). To change the limit after which the `managed_window` or the floating panes are automatically stickled to the screen border (or to the `managed_window` side), set these two variables. Default values are 15 pixels. :param integer `x`: the minimum horizontal distance below which the snap occurs; :param integer `y`: the minimum vertical distance below which the snap occurs. """ self._snap_limits = (x, y) self.Snap() def Snap(self): """ Snaps the main frame to specified position on the screen. :see: :meth:`SnapToScreen` """ snap, hAlign, vAlign, monitor = self._is_docked if not snap: return managed_window = self.GetManagedWindow() snap_pos = self.GetSnapPosition() wnd_pos = managed_window.GetPosition() snapX, snapY = self._snap_limits if abs(snap_pos.x - wnd_pos.x) < snapX and abs(snap_pos.y - wnd_pos.y) < snapY: managed_window.SetPosition(snap_pos) def SnapToScreen(self, snap=True, monitor=0, hAlign=wx.RIGHT, vAlign=wx.TOP): """ Snaps the main frame to specified position on the screen. :param bool `snap`: whether to snap the main frame or not; :param integer `monitor`: the monitor display in which snapping the window; :param integer `hAlign`: the horizontal alignment of the snapping position; :param integer `vAlign`: the vertical alignment of the snapping position. """ if not snap: self._is_docked = (False, wx.RIGHT, wx.TOP, 0) return displayCount = wx.Display.GetCount() if monitor > displayCount: raise Exception("Invalid monitor selected: you only have %d monitors"%displayCount) self._is_docked = (True, hAlign, vAlign, monitor) self.GetManagedWindow().SetPosition(self.GetSnapPosition()) def GetSnapPosition(self): """ Returns the main frame snapping position. """ snap, hAlign, vAlign, monitor = self._is_docked display = wx.Display(monitor) area = display.GetClientArea() size = self.GetManagedWindow().GetSize() pos = wx.Point() if hAlign == wx.LEFT: pos.x = area.x elif hAlign == wx.CENTER: pos.x = area.x + (area.width - size.x)/2 else: pos.x = area.x + area.width - size.x if vAlign == wx.TOP: pos.y = area.y elif vAlign == wx.CENTER: pos.y = area.y + (area.height - size.y)/2 else: pos.y = area.y + area.height - size.y return pos def GetAnimationStep(self): """ Returns the animation step speed (a float) to use in :meth:`AnimateDocking`. """ return self._animation_step def SetAnimationStep(self, step): """ Sets the animation step speed (a float) to use in :meth:`AnimateDocking`. :param float `step`: the animation speed. """ self._animation_step = float(step) def RequestUserAttention(self, pane_window): """ Requests the user attention by intermittently highlighting the pane caption. :param Window `pane_window`: the window managed by the pane; """ # try to find the pane paneInfo = self.GetPane(pane_window) if not paneInfo.IsOk(): raise Exception("Pane window not found") dc = wx.ClientDC(self._frame) # if the frame is about to be deleted, don't bother if not self._frame or self._frame.IsBeingDeleted(): return if not self._frame.GetSizer(): return for part in self._uiparts: if part.pane == paneInfo: self._art.RequestUserAttention(dc, self._frame, part.pane.caption, part.rect, part.pane) self._frame.RefreshRect(part.rect, True) break def StartPreviewTimer(self, toolbar): """ Starts a timer for sliding in and out a minimized pane. :param `toolbar`: the :class:`~lib.agw.aui.auibar.AuiToolBar` containing the minimized pane tool. """ toolbar_pane = self.GetPane(toolbar) toolbar_name = toolbar_pane.name pane_name = toolbar_name[0:-4] self._sliding_pane = self.GetPane(pane_name) self._sliding_rect = toolbar.GetScreenRect() self._sliding_direction = toolbar_pane.dock_direction self._sliding_frame = None self._preview_timer.Start(1000, wx.TIMER_ONE_SHOT) def StopPreviewTimer(self): """ Stops a timer for sliding in and out a minimized pane. """ if self._preview_timer.IsRunning(): self._preview_timer.Stop() self.SlideOut() self._sliding_pane = None def SlideIn(self, event): """ Handles the ``wx.EVT_TIMER`` event for :class:`AuiManager`. :param `event`: a :class:`TimerEvent` to be processed. :note: This is used solely for sliding in and out minimized panes. """ window = self._sliding_pane.window self._sliding_frame = wx.MiniFrame(None, -1, title=_("Pane Preview"), style=wx.FRAME_TOOL_WINDOW | wx.STAY_ON_TOP | wx.FRAME_NO_TASKBAR | wx.CAPTION) window.Reparent(self._sliding_frame) self._sliding_frame.SetSize((0, 0)) window.Show() self._sliding_frame.Show() size = window.GetBestSize() startX, startY, stopX, stopY = GetSlidingPoints(self._sliding_rect, size, self._sliding_direction) step = stopX/10 window_size = 0 for i in range(0, stopX, step): window_size = i self._sliding_frame.SetSize(startX, startY, window_size, stopY) self._sliding_frame.Refresh() self._sliding_frame.Update() wx.MilliSleep(10) self._sliding_frame.SetSize(startX, startY, stopX, stopY) self._sliding_frame.Refresh() self._sliding_frame.Update() def SlideOut(self): """ Slides out a preview of a minimized pane. :note: This is used solely for sliding in and out minimized panes. """ if not self._sliding_frame: return window = self._sliding_frame.GetChildren()[0] size = window.GetBestSize() startX, startY, stopX, stopY = GetSlidingPoints(self._sliding_rect, size, self._sliding_direction) step = stopX/10 window_size = 0 for i in range(stopX, 0, -step): window_size = i self._sliding_frame.SetSize(startX, startY, window_size, stopY) self._sliding_frame.Refresh() self._sliding_frame.Update() self._frame.RefreshRect(wx.Rect(startX+window_size, startY, step, stopY)) self._frame.Update() wx.MilliSleep(10) self._sliding_frame.SetSize(startX, startY, 0, stopY) window.Hide() window.Reparent(self._frame) self._sliding_frame.Hide() self._sliding_frame.Destroy() self._sliding_frame = None self._sliding_pane = None class AuiManager_DCP(AuiManager): """ A class similar to :class:`AuiManager` but with a Dummy Center Pane (**DCP**). The code for this class is still flickery due to the call to :func:`CallAfter` and the double-update call. """ def __init__(self, *args, **keys): """ See :meth:`AuiManager.__init__` for the class construction. """ AuiManager.__init__(self, *args, **keys) self.hasDummyPane = False def _createDummyPane(self): """ Creates a Dummy Center Pane (**DCP**). """ if self.hasDummyPane: return self.hasDummyPane = True dummy = wx.Panel(self.GetManagedWindow()) info = AuiPaneInfo().CenterPane().NotebookDockable(True).Name('dummyCenterPane').DestroyOnClose(True) self.AddPane(dummy, info) def _destroyDummyPane(self): """ Destroys the Dummy Center Pane (**DCP**). """ if not self.hasDummyPane: return self.hasDummyPane = False self.ClosePane(self.GetPane('dummyCenterPane')) def Update(self): """ This method is called after any number of changes are made to any of the managed panes. :meth:`Update` must be invoked after :meth:`AuiManager.AddPane` or :meth:`AuiManager.InsertPane` are called in order to "realize" or "commit" the changes. In addition, any number of changes may be made to :class:`AuiManager` structures (retrieved with :meth:`AuiManager.GetPane`), but to realize the changes, :meth:`Update` must be called. This construction allows pane flicker to be avoided by updating the whole layout at one time. """ AuiManager.Update(self) # check if there's already a center pane (except our dummy pane) dummyCenterPane = self.GetPane('dummyCenterPane') haveCenterPane = any((pane != dummyCenterPane) and (pane.dock_direction == AUI_DOCK_CENTER) and not pane.IsFloating() and pane.IsShown() for pane in self.GetAllPanes()) if haveCenterPane: if self.hasDummyPane: # there's our dummy pane and also another center pane, therefor let's remove our dummy def do(): self._destroyDummyPane() self.Update() wx.CallAfter(do) else: # if we get here, there's no center pane, create our dummy if not self.hasDummyPane: self._createDummyPane()