Poodletooth-iLand/panda/python/Lib/site-packages/wx/lib/agw/aui/tabart.py

2803 lines
107 KiB
Python
Raw Normal View History

2015-03-06 12:11:40 +00:00
# -*- coding: utf-8 -*-
#----------------------------------------------------------------------------
# Name: tabart.py
# Purpose:
#
# Author: Andrea Gavana <andrea.gavana@gmail.com>
#
# Created:
# Version:
# Date: 31 March 2009
# Licence: wxWindows license
# Tags: phoenix-port, unittest, documented, py3-port
#----------------------------------------------------------------------------
"""
Tab art provider code - a tab provider provides all drawing functionality to
the :class:`~lib.agw.aui.auibook.AuiNotebook`. This allows the
:class:`~lib.agw.aui.auibook.AuiNotebook` to have a plugable look-and-feel.
By default, a :class:`~lib.agw.aui.auibook.AuiNotebook` uses an instance of this class
called :class:`AuiDefaultTabArt` which provides bitmap art and a colour scheme that is
adapted to the major platforms' look. You can either derive from that class to alter its
behaviour or write a completely new tab art class.
Call :meth:`AuiNotebook.SetArtProvider() <lib.agw.aui.auibook.AuiNotebook.SetArtProvider>`
to make use this new tab art.
"""
__author__ = "Andrea Gavana <andrea.gavana@gmail.com>"
__date__ = "31 March 2009"
import wx
if wx.Platform == '__WXMAC__':
try:
import Carbon.Appearance
except ImportError:
CARBON = False
else:
CARBON = True
from .aui_utilities import BitmapFromBits, StepColour, IndentPressedBitmap, ChopText
from .aui_utilities import GetBaseColour, DrawMACCloseButton, LightColour, TakeScreenShot
from .aui_utilities import CopyAttributes
from .aui_constants import *
# -- GUI helper classes and functions --
class AuiCommandCapture(wx.EvtHandler):
""" A class to handle the dropdown window menu. """
def __init__(self):
""" Default class constructor. """
wx.EvtHandler.__init__(self)
self._last_id = 0
def GetCommandId(self):
""" Returns the event command identifier. """
return self._last_id
def ProcessEvent(self, event):
"""
Processes an event, searching event tables and calling zero or more suitable
event handler function(s).
:param `event`: the event to process.
:note: Normally, your application would not call this function: it is called
in the wxPython implementation to dispatch incoming user interface events
to the framework (and application).
However, you might need to call it if implementing new functionality (such as
a new control) where you define new event types, as opposed to allowing the
user to override functions.
An instance where you might actually override the :meth:`ProcessEvent` function is where
you want to direct event processing to event handlers not normally noticed by
wxPython. For example, in the document/view architecture, documents and views
are potential event handlers. When an event reaches a frame, :meth:`ProcessEvent` will
need to be called on the associated document and view in case event handler
functions are associated with these objects.
The normal order of event table searching is as follows:
1. If the object is disabled (via a call to :meth:`EvtHandler.SetEvtHandlerEnabled`) the function
skips to step (6).
2. If the object is a :class:`Window`, :meth:`ProcessEvent` is recursively called on the window's
:class:`Validator`. If this returns ``True``, the function exits.
3. wxWidgets `SearchEventTable` is called for this event handler. If this fails, the
base class table is tried, and so on until no more tables exist or an appropriate
function was found, in which case the function exits.
4. The search is applied down the entire chain of event handlers (usually the chain
has a length of one). If this succeeds, the function exits.
5. If the object is a :class:`Window` and the event is a :class:`CommandEvent`, :meth:`ProcessEvent` is
recursively applied to the parent window's event handler. If this returns ``True``,
the function exits.
6. Finally, :meth:`ProcessEvent` is called on the :class:`App` object.
"""
if event.GetEventType() == wx.wxEVT_COMMAND_MENU_SELECTED:
self._last_id = event.GetId()
return True
if self.GetNextHandler():
return self.GetNextHandler().ProcessEvent(event)
return False
class AuiDefaultTabArt(object):
"""
Tab art provider code - a tab provider provides all drawing functionality to the :class:`~lib.agw.aui.auibook.AuiNotebook`.
This allows the :class:`~lib.agw.aui.auibook.AuiNotebook` to have a plugable look-and-feel.
By default, a :class:`~lib.agw.aui.auibook.AuiNotebook` uses an instance of this class called
:class:`AuiDefaultTabArt` which provides bitmap art and a colour scheme that is adapted to the major platforms'
look. You can either derive from that class to alter its behaviour or write a
completely new tab art class. Call :meth:`AuiNotebook.SetArtProvider() <lib.agw.aui.auibook.AuiNotebook.SetArtProvider>` to make use this
new tab art.
"""
def __init__(self):
""" Default class constructor. """
self._normal_font = wx.SystemSettings.GetFont(wx.SYS_DEFAULT_GUI_FONT)
self._selected_font = wx.SystemSettings.GetFont(wx.SYS_DEFAULT_GUI_FONT)
self._selected_font.SetWeight(wx.FONTWEIGHT_BOLD)
self._measuring_font = self._selected_font
self._fixed_tab_width = 100
self._tab_ctrl_height = 0
self._buttonRect = wx.Rect()
self.SetDefaultColours()
if wx.Platform == "__WXMAC__":
bmp_colour = wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DDKSHADOW)
self._active_close_bmp = DrawMACCloseButton(bmp_colour)
self._disabled_close_bmp = DrawMACCloseButton(wx.Colour(128, 128, 128))
else:
self._active_close_bmp = BitmapFromBits(nb_close_bits, 16, 16, wx.BLACK)
self._disabled_close_bmp = BitmapFromBits(nb_close_bits, 16, 16, wx.Colour(128, 128, 128))
self._hover_close_bmp = self._active_close_bmp
self._pressed_close_bmp = self._active_close_bmp
self._active_left_bmp = BitmapFromBits(nb_left_bits, 16, 16, wx.BLACK)
self._disabled_left_bmp = BitmapFromBits(nb_left_bits, 16, 16, wx.Colour(128, 128, 128))
self._active_right_bmp = BitmapFromBits(nb_right_bits, 16, 16, wx.BLACK)
self._disabled_right_bmp = BitmapFromBits(nb_right_bits, 16, 16, wx.Colour(128, 128, 128))
self._active_windowlist_bmp = BitmapFromBits(nb_list_bits, 16, 16, wx.BLACK)
self._disabled_windowlist_bmp = BitmapFromBits(nb_list_bits, 16, 16, wx.Colour(128, 128, 128))
if wx.Platform == "__WXMAC__":
k = Carbon.Appearance.kThemeBrushFocusHighlight if CARBON else 19
# Get proper highlight colour for focus rectangle from the
# current Mac theme. kThemeBrushFocusHighlight is
# available on Mac OS 8.5 and higher
if hasattr(wx, 'MacThemeColour'):
c = wx.MacThemeColour(k)
else:
brush = wx.Brush(wx.BLACK)
brush.MacSetTheme(k)
c = brush.GetColour()
self._focusPen = wx.Pen(c, 2, wx.SOLID)
else:
self._focusPen = wx.Pen(wx.BLACK, 1, wx.USER_DASH)
self._focusPen.SetDashes([1, 1])
self._focusPen.SetCap(wx.CAP_BUTT)
def SetBaseColour(self, base_colour):
"""
Sets a new base colour.
:param `base_colour`: an instance of :class:`Colour`.
"""
self._base_colour = base_colour
self._base_colour_pen = wx.Pen(self._base_colour)
self._base_colour_brush = wx.Brush(self._base_colour)
def SetDefaultColours(self, base_colour=None):
"""
Sets the default colours, which are calculated from the given base colour.
:param `base_colour`: an instance of :class:`Colour`. If defaulted to ``None``, a colour
is generated accordingly to the platform and theme.
"""
if base_colour is None:
base_colour = GetBaseColour()
self.SetBaseColour( base_colour )
self._border_colour = StepColour(base_colour, 75)
self._border_pen = wx.Pen(self._border_colour)
self._background_top_colour = StepColour(self._base_colour, 90)
self._background_bottom_colour = StepColour(self._base_colour, 170)
self._tab_top_colour = self._base_colour
self._tab_bottom_colour = wx.WHITE
self._tab_gradient_highlight_colour = wx.WHITE
self._tab_inactive_top_colour = self._base_colour
self._tab_inactive_bottom_colour = StepColour(self._tab_inactive_top_colour, 160)
self._tab_text_colour = lambda page: page.text_colour
self._tab_disabled_text_colour = wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT)
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def SetAGWFlags(self, agwFlags):
"""
Sets the tab art flags.
:param integer `agwFlags`: a combination of the following values:
==================================== ==================================
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:`~lib.agw.aui.framemanager.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
==================================== ==================================
"""
self._agwFlags = agwFlags
def GetAGWFlags(self):
"""
Returns the tab art flags.
:see: :meth:`~AuiDefaultTabArt.SetAGWFlags` for a list of possible return values.
"""
return self._agwFlags
def SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth):
"""
Sets the tab sizing information.
:param Size `tab_ctrl_size`: the size of the tab control area;
:param integer `tab_count`: the number of tabs;
:param tuple `minMaxTabWidth`: a tuple containing the minimum and maximum tab widths
to be used when the ``AUI_NB_TAB_FIXED_WIDTH`` style is active.
"""
self._fixed_tab_width = 100
minTabWidth, maxTabWidth = minMaxTabWidth
tot_width = tab_ctrl_size.x - self.GetIndentSize() - 4
agwFlags = self.GetAGWFlags()
if agwFlags & AUI_NB_CLOSE_BUTTON:
tot_width -= self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_WINDOWLIST_BUTTON:
tot_width -= self._active_windowlist_bmp.GetWidth()
if tab_count > 0:
self._fixed_tab_width = tot_width/tab_count
if self._fixed_tab_width < 100:
self._fixed_tab_width = 100
if self._fixed_tab_width > tot_width/2:
self._fixed_tab_width = tot_width/2
if self._fixed_tab_width > 220:
self._fixed_tab_width = 220
if minTabWidth > -1:
self._fixed_tab_width = max(self._fixed_tab_width, minTabWidth)
if maxTabWidth > -1:
self._fixed_tab_width = min(self._fixed_tab_width, maxTabWidth)
self._tab_ctrl_height = tab_ctrl_size.y
def DrawBackground(self, dc, wnd, rect):
"""
Draws the tab area background.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param Rect `rect`: the tab control rectangle.
"""
self._buttonRect = wx.Rect()
# draw background
agwFlags = self.GetAGWFlags()
if agwFlags & AUI_NB_BOTTOM:
r = wx.Rect(rect.x, rect.y, rect.width+2, rect.height)
# TODO: else if (agwFlags & AUI_NB_LEFT)
# TODO: else if (agwFlags & AUI_NB_RIGHT)
else: #for AUI_NB_TOP
r = wx.Rect(rect.x, rect.y, rect.width+2, rect.height-3)
dc.GradientFillLinear(r, self._background_top_colour, self._background_bottom_colour, wx.SOUTH)
# draw base lines
dc.SetPen(self._border_pen)
y = rect.GetHeight()
w = rect.GetWidth()
if agwFlags & AUI_NB_BOTTOM:
dc.SetBrush(wx.Brush(self._background_bottom_colour))
dc.DrawRectangle(-1, 0, w+2, 4)
# TODO: else if (agwFlags & AUI_NB_LEFT)
# TODO: else if (agwFlags & AUI_NB_RIGHT)
else: # for AUI_NB_TOP
dc.SetBrush(self._base_colour_brush)
dc.DrawRectangle(-1, y-4, w+2, 4)
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# if the caption is empty, measure some temporary text
caption = page.caption
if not caption:
caption = "Xj"
dc.SetFont(self._selected_font)
selected_textx, selected_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
dc.SetFont(self._normal_font)
normal_textx, normal_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
control = page.control
# figure out the size of the tab
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap,
page.active, close_button_state, control)
tab_height = self._tab_ctrl_height - 3
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
caption = page.caption
# select pen, brush and font for the tab to be drawn
if page.active:
dc.SetFont(self._selected_font)
textx, texty = selected_textx, selected_texty
else:
dc.SetFont(self._normal_font)
textx, texty = normal_textx, normal_texty
if not page.enabled:
dc.SetTextForeground(self._tab_disabled_text_colour)
pagebitmap = page.dis_bitmap
else:
dc.SetTextForeground(self._tab_text_colour(page))
pagebitmap = page.bitmap
# create points that will make the tab outline
clip_width = tab_width
if tab_x + clip_width > in_rect.x + in_rect.width:
clip_width = in_rect.x + in_rect.width - tab_x
# since the above code above doesn't play well with WXDFB or WXCOCOA,
# we'll just use a rectangle for the clipping region for now --
dc.SetClippingRegion(tab_x, tab_y, clip_width+1, tab_height-3)
border_points = [wx.Point() for i in range(6)]
agwFlags = self.GetAGWFlags()
if agwFlags & AUI_NB_BOTTOM:
border_points[0] = wx.Point(tab_x, tab_y)
border_points[1] = wx.Point(tab_x, tab_y+tab_height-6)
border_points[2] = wx.Point(tab_x+2, tab_y+tab_height-4)
border_points[3] = wx.Point(tab_x+tab_width-2, tab_y+tab_height-4)
border_points[4] = wx.Point(tab_x+tab_width, tab_y+tab_height-6)
border_points[5] = wx.Point(tab_x+tab_width, tab_y)
else: #if (agwFlags & AUI_NB_TOP)
border_points[0] = wx.Point(tab_x, tab_y+tab_height-4)
border_points[1] = wx.Point(tab_x, tab_y+2)
border_points[2] = wx.Point(tab_x+2, tab_y)
border_points[3] = wx.Point(tab_x+tab_width-2, tab_y)
border_points[4] = wx.Point(tab_x+tab_width, tab_y+2)
border_points[5] = wx.Point(tab_x+tab_width, tab_y+tab_height-4)
# TODO: else if (agwFlags & AUI_NB_LEFT)
# TODO: else if (agwFlags & AUI_NB_RIGHT)
drawn_tab_yoff = border_points[1].y
drawn_tab_height = border_points[0].y - border_points[1].y
if page.active:
# draw active tab
# draw base background colour
r = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.SetPen(self._base_colour_pen)
dc.SetBrush(self._base_colour_brush)
dc.DrawRectangle(r.x+1, r.y+1, r.width-1, r.height-4)
# this white helps fill out the gradient at the top of the tab
dc.SetPen( wx.Pen(self._tab_gradient_highlight_colour) )
dc.SetBrush( wx.Brush(self._tab_gradient_highlight_colour) )
dc.DrawRectangle(r.x+2, r.y+1, r.width-3, r.height-4)
# these two points help the rounded corners appear more antialiased
dc.SetPen(self._base_colour_pen)
dc.DrawPoint(r.x+2, r.y+1)
dc.DrawPoint(r.x+r.width-2, r.y+1)
# set rectangle down a bit for gradient drawing
r.SetHeight(r.GetHeight()/2)
r.x += 2
r.width -= 2
r.y += r.height
r.y -= 2
# draw gradient background
top_colour = self._tab_bottom_colour
bottom_colour = self._tab_top_colour
dc.GradientFillLinear(r, bottom_colour, top_colour, wx.NORTH)
else:
# draw inactive tab
r = wx.Rect(tab_x, tab_y+1, tab_width, tab_height-3)
# start the gradent up a bit and leave the inside border inset
# by a pixel for a 3D look. Only the top half of the inactive
# tab will have a slight gradient
r.x += 3
r.y += 1
r.width -= 4
r.height /= 2
r.height -= 1
# -- draw top gradient fill for glossy look
top_colour = self._tab_inactive_top_colour
bottom_colour = self._tab_inactive_bottom_colour
dc.GradientFillLinear(r, bottom_colour, top_colour, wx.NORTH)
r.y += r.height
r.y -= 1
# -- draw bottom fill for glossy look
top_colour = self._tab_inactive_bottom_colour
bottom_colour = self._tab_inactive_bottom_colour
dc.GradientFillLinear(r, top_colour, bottom_colour, wx.SOUTH)
# draw tab outline
dc.SetPen(self._border_pen)
dc.SetBrush(wx.TRANSPARENT_BRUSH)
dc.DrawPolygon(border_points)
# there are two horizontal grey lines at the bottom of the tab control,
# this gets rid of the top one of those lines in the tab control
if page.active:
if agwFlags & AUI_NB_BOTTOM:
dc.SetPen(wx.Pen(self._background_bottom_colour))
# TODO: else if (agwFlags & AUI_NB_LEFT)
# TODO: else if (agwFlags & AUI_NB_RIGHT)
else: # for AUI_NB_TOP
dc.SetPen(self._base_colour_pen)
dc.DrawLine(border_points[0].x+1,
border_points[0].y,
border_points[5].x,
border_points[5].y)
text_offset = tab_x + 8
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset += close_button_width - 5
bitmap_offset = 0
if pagebitmap.IsOk():
bitmap_offset = tab_x + 8
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT and close_button_width:
bitmap_offset += close_button_width - 5
# draw bitmap
dc.DrawBitmap(pagebitmap,
bitmap_offset,
drawn_tab_yoff + (drawn_tab_height/2) - (pagebitmap.GetHeight()/2),
True)
text_offset = bitmap_offset + pagebitmap.GetWidth()
text_offset += 3 # bitmap padding
else:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT == 0 or not close_button_width:
text_offset = tab_x + 8
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width)
ypos = drawn_tab_yoff + (drawn_tab_height)/2 - (texty/2) - 1
offset_focus = text_offset
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
textx += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
# draw focus rectangle
if (agwFlags & AUI_NB_NO_TAB_FOCUS) == 0:
self.DrawFocusRectangle(dc, page, wnd, draw_text, offset_focus, bitmap_offset, drawn_tab_yoff, drawn_tab_height, rectx, recty)
out_button_rect = wx.Rect()
# draw close button if necessary
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
bmp = self._disabled_close_bmp
if close_button_state == AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif close_button_state == AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
shift = (agwFlags & AUI_NB_BOTTOM and [1] or [0])[0]
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + 4, tab_y + (tab_height - bmp.GetHeight())/2 - shift,
close_button_width, tab_height)
else:
rect = wx.Rect(tab_x + tab_width - close_button_width - 1,
tab_y + (tab_height - bmp.GetHeight())/2 - shift,
close_button_width, tab_height)
rect = IndentPressedBitmap(rect, close_button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_button_rect = rect
out_tab_rect = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent
def SetCustomButton(self, bitmap_id, button_state, bmp):
"""
Sets a custom bitmap for the close, left, right and window list buttons.
:param integer `bitmap_id`: the button identifier;
:param integer `button_state`: the button state;
:param Bitmap `bmp`: the custom bitmap to use for the button.
"""
if bitmap_id == AUI_BUTTON_CLOSE:
if button_state == AUI_BUTTON_STATE_NORMAL:
self._active_close_bmp = bmp
self._hover_close_bmp = self._active_close_bmp
self._pressed_close_bmp = self._active_close_bmp
self._disabled_close_bmp = self._active_close_bmp
elif button_state == AUI_BUTTON_STATE_HOVER:
self._hover_close_bmp = bmp
elif button_state == AUI_BUTTON_STATE_PRESSED:
self._pressed_close_bmp = bmp
else:
self._disabled_close_bmp = bmp
elif bitmap_id == AUI_BUTTON_LEFT:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_left_bmp = bmp
else:
self._active_left_bmp = bmp
elif bitmap_id == AUI_BUTTON_RIGHT:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_right_bmp = bmp
else:
self._active_right_bmp = bmp
elif bitmap_id == AUI_BUTTON_WINDOWLIST:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_windowlist_bmp = bmp
else:
self._active_windowlist_bmp = bmp
def GetIndentSize(self):
""" Returns the tabs indent size. """
return 5
def GetTabSize(self, dc, wnd, caption, bitmap, active, close_button_state, control=None):
"""
Returns the tab size for the given caption, bitmap and button state.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param string `caption`: the tab text caption;
:param Bitmap `bitmap`: the bitmap displayed on the tab;
:param bool `active`: whether the tab is selected or not;
:param integer `close_button_state`: the state of the close button on the tab;
:param Window `control`: a :class:`Window` instance inside a tab (or ``None``).
"""
dc.SetFont(self._measuring_font)
measured_textx, measured_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
# add padding around the text
tab_width = measured_textx
tab_height = measured_texty
# if the close button is showing, add space for it
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
tab_width += self._active_close_bmp.GetWidth() + 3
# if there's a bitmap, add space for it
if bitmap.IsOk():
tab_width += bitmap.GetWidth()
tab_width += 3 # right side bitmap padding
tab_height = max(tab_height, bitmap.GetHeight())
# add padding
tab_width += 16
tab_height += 10
agwFlags = self.GetAGWFlags()
if agwFlags & AUI_NB_TAB_FIXED_WIDTH:
tab_width = self._fixed_tab_width
if control:
tab_width += control.GetSize().GetWidth() + 4
x_extent = tab_width
return (tab_width, tab_height), x_extent
def DrawButton(self, dc, wnd, in_rect, button, orientation):
"""
Draws a button on the tab or on the tab area, depending on the button identifier.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param `button`: an instance of the button class;
:param integer `orientation`: the tab orientation.
"""
bitmap_id, button_state = button.id, button.cur_state
if bitmap_id == AUI_BUTTON_CLOSE:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_close_bmp
elif button_state & AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif button_state & AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
else:
bmp = self._active_close_bmp
elif bitmap_id == AUI_BUTTON_LEFT:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_left_bmp
else:
bmp = self._active_left_bmp
elif bitmap_id == AUI_BUTTON_RIGHT:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_right_bmp
else:
bmp = self._active_right_bmp
elif bitmap_id == AUI_BUTTON_WINDOWLIST:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_windowlist_bmp
else:
bmp = self._active_windowlist_bmp
else:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = button.dis_bitmap
else:
bmp = button.bitmap
if not bmp.IsOk():
return
rect = wx.Rect(*in_rect)
if orientation == wx.LEFT:
rect.SetX(in_rect.x)
rect.SetY(((in_rect.y + in_rect.height)/2) - (bmp.GetHeight()/2))
rect.SetWidth(bmp.GetWidth())
rect.SetHeight(bmp.GetHeight())
else:
rect = wx.Rect(in_rect.x + in_rect.width - bmp.GetWidth(),
((in_rect.y + in_rect.height)/2) - (bmp.GetHeight()/2),
bmp.GetWidth(), bmp.GetHeight())
rect = IndentPressedBitmap(rect, button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_rect = rect
if bitmap_id == AUI_BUTTON_RIGHT:
self._buttonRect = wx.Rect(rect.x, rect.y, 30, rect.height)
return out_rect
def DrawFocusRectangle(self, dc, page, wnd, draw_text, text_offset, bitmap_offset, drawn_tab_yoff, drawn_tab_height, textx, texty):
"""
Draws the focus rectangle on a tab.
:param `dc`: a :class:`DC` device context;
:param `page`: the page associated with the tab;
:param `wnd`: a :class:`Window` instance object;
:param string `draw_text`: the text that has been drawn on the tab;
:param integer `text_offset`: the text offset on the tab;
:param integer `bitmap_offset`: the bitmap offset on the tab;
:param integer `drawn_tab_yoff`: the y offset of the tab text;
:param integer `drawn_tab_height`: the height of the tab;
:param integer `textx`: the x text extent;
:param integer `texty`: the y text extent.
"""
if self.GetAGWFlags() & AUI_NB_NO_TAB_FOCUS:
return
if page.active and wx.Window.FindFocus() == wnd:
focusRectText = wx.Rect(text_offset, (drawn_tab_yoff + (drawn_tab_height)/2 - (texty/2)),
textx, texty)
if page.bitmap.IsOk():
focusRectBitmap = wx.Rect(bitmap_offset, drawn_tab_yoff + (drawn_tab_height/2) - (page.bitmap.GetHeight()/2),
page.bitmap.GetWidth(), page.bitmap.GetHeight())
if page.bitmap.IsOk() and draw_text == "":
focusRect = wx.Rect(*focusRectBitmap)
elif not page.bitmap.IsOk() and draw_text != "":
focusRect = wx.Rect(*focusRectText)
elif page.bitmap.IsOk() and draw_text != "":
focusRect = focusRectText.Union(focusRectBitmap)
focusRect.Inflate(2, 2)
dc.SetBrush(wx.TRANSPARENT_BRUSH)
dc.SetPen(self._focusPen)
dc.DrawRoundedRectangle(focusRect, 2)
def GetBestTabCtrlSize(self, wnd, pages, required_bmp_size):
"""
Returns the best tab control size.
:param `wnd`: a :class:`Window` instance object;
:param list `pages`: the pages associated with the tabs;
:param Size `required_bmp_size`: the size of the bitmap on the tabs.
"""
dc = wx.ClientDC(wnd)
dc.SetFont(self._measuring_font)
# sometimes a standard bitmap size needs to be enforced, especially
# if some tabs have bitmaps and others don't. This is important because
# it prevents the tab control from resizing when tabs are added.
measure_bmp = wx.NullBitmap
if required_bmp_size.IsFullySpecified():
measure_bmp = wx.Bitmap(required_bmp_size.x, required_bmp_size.y)
max_y = 0
for page in pages:
if measure_bmp.IsOk():
bmp = measure_bmp
else:
bmp = page.bitmap
# we don't use the caption text because we don't
# want tab heights to be different in the case
# of a very short piece of text on one tab and a very
# tall piece of text on another tab
s, x_ext = self.GetTabSize(dc, wnd, page.caption, bmp, True, AUI_BUTTON_STATE_HIDDEN, None)
max_y = max(max_y, s[1])
if page.control:
controlW, controlH = page.control.GetSize()
max_y = max(max_y, controlH+4)
return max_y + 2
def SetNormalFont(self, font):
"""
Sets the normal font for drawing tab labels.
:param Font `font`: the new font to use to draw tab labels in their normal, un-selected state.
"""
self._normal_font = font
def SetSelectedFont(self, font):
"""
Sets the selected tab font for drawing tab labels.
:param Font `font`: the new font to use to draw tab labels in their selected state.
"""
self._selected_font = font
def SetMeasuringFont(self, font):
"""
Sets the font for calculating text measurements.
:param Font `font`: the new font to use to measure tab labels text extents.
"""
self._measuring_font = font
def GetNormalFont(self):
""" Returns the normal font for drawing tab labels. """
return self._normal_font
def GetSelectedFont(self):
""" Returns the selected tab font for drawing tab labels. """
return self._selected_font
def GetMeasuringFont(self):
""" Returns the font for calculating text measurements. """
return self._measuring_font
def ShowDropDown(self, wnd, pages, active_idx):
"""
Shows the drop-down window menu on the tab area.
:param `wnd`: a :class:`Window` derived window instance;
:param list `pages`: the pages associated with the tabs;
:param integer `active_idx`: the active tab index.
"""
useImages = self.GetAGWFlags() & AUI_NB_USE_IMAGES_DROPDOWN
menuPopup = wx.Menu()
longest = 0
for i, page in enumerate(pages):
caption = page.caption
# if there is no caption, make it a space. This will prevent
# an assert in the menu code.
if caption == "":
caption = " "
# Save longest caption width for calculating menu width with
width = wnd.GetTextExtent(caption)[0]
if width > longest:
longest = width
if useImages:
menuItem = wx.MenuItem(menuPopup, 1000+i, caption)
if page.bitmap:
menuItem.SetBitmap(page.bitmap)
menuPopup.Append(menuItem)
else:
menuPopup.AppendCheckItem(1000+i, caption)
menuPopup.Enable(1000+i, page.enabled)
if active_idx != -1 and not useImages:
menuPopup.Check(1000+active_idx, True)
# find out the screen coordinate at the bottom of the tab ctrl
cli_rect = wnd.GetClientRect()
# Calculate the approximate size of the popupmenu for setting the
# position of the menu when its shown.
# Account for extra padding on left/right of text on mac menus
if wx.Platform in ['__WXMAC__', '__WXMSW__']:
longest += 32
# Bitmap/Checkmark width + padding
longest += 20
if self.GetAGWFlags() & AUI_NB_CLOSE_BUTTON:
longest += 16
pt = wx.Point(cli_rect.x + cli_rect.GetWidth() - longest,
cli_rect.y + cli_rect.height)
cc = AuiCommandCapture()
wnd.PushEventHandler(cc)
wnd.PopupMenu(menuPopup, pt)
command = cc.GetCommandId()
wnd.PopEventHandler(True)
if command >= 1000:
return command - 1000
return -1
class AuiSimpleTabArt(object):
""" A simple-looking implementation of a tab art. """
def __init__(self):
""" Default class constructor. """
self._normal_font = wx.SystemSettings.GetFont(wx.SYS_DEFAULT_GUI_FONT)
self._selected_font = wx.SystemSettings.GetFont(wx.SYS_DEFAULT_GUI_FONT)
self._selected_font.SetWeight(wx.FONTWEIGHT_BOLD)
self._measuring_font = self._selected_font
self._agwFlags = 0
self._fixed_tab_width = 100
base_colour = wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DFACE)
background_colour = base_colour
normaltab_colour = base_colour
selectedtab_colour = wx.WHITE
self._bkbrush = wx.Brush(background_colour)
self._normal_bkbrush = wx.Brush(normaltab_colour)
self._normal_bkpen = wx.Pen(normaltab_colour)
self._selected_bkbrush = wx.Brush(selectedtab_colour)
self._selected_bkpen = wx.Pen(selectedtab_colour)
self._active_close_bmp = BitmapFromBits(nb_close_bits, 16, 16, wx.BLACK)
self._disabled_close_bmp = BitmapFromBits(nb_close_bits, 16, 16, wx.Colour(128, 128, 128))
self._active_left_bmp = BitmapFromBits(nb_left_bits, 16, 16, wx.BLACK)
self._disabled_left_bmp = BitmapFromBits(nb_left_bits, 16, 16, wx.Colour(128, 128, 128))
self._active_right_bmp = BitmapFromBits(nb_right_bits, 16, 16, wx.BLACK)
self._disabled_right_bmp = BitmapFromBits(nb_right_bits, 16, 16, wx.Colour(128, 128, 128))
self._active_windowlist_bmp = BitmapFromBits(nb_list_bits, 16, 16, wx.BLACK)
self._disabled_windowlist_bmp = BitmapFromBits(nb_list_bits, 16, 16, wx.Colour(128, 128, 128))
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def SetAGWFlags(self, agwFlags):
"""
Sets the tab art flags.
:param integer `agwFlags`: a combination of the following values:
==================================== ==================================
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:`~lib.agw.aui.framemanager.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
==================================== ==================================
"""
self._agwFlags = agwFlags
def GetAGWFlags(self):
"""
Returns the tab art flags.
:see: :meth:`~AuiSimpleTabArt.SetAGWFlags` for a list of possible return values.
"""
return self._agwFlags
def SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth):
"""
Sets the tab sizing information.
:param Size `tab_ctrl_size`: the size of the tab control area;
:param integer `tab_count`: the number of tabs;
:param tuple `minMaxTabWidth`: a tuple containing the minimum and maximum tab widths
to be used when the ``AUI_NB_TAB_FIXED_WIDTH`` style is active.
"""
self._fixed_tab_width = 100
minTabWidth, maxTabWidth = minMaxTabWidth
tot_width = tab_ctrl_size.x - self.GetIndentSize() - 4
if self._agwFlags & AUI_NB_CLOSE_BUTTON:
tot_width -= self._active_close_bmp.GetWidth()
if self._agwFlags & AUI_NB_WINDOWLIST_BUTTON:
tot_width -= self._active_windowlist_bmp.GetWidth()
if tab_count > 0:
self._fixed_tab_width = tot_width/tab_count
if self._fixed_tab_width < 100:
self._fixed_tab_width = 100
if self._fixed_tab_width > tot_width/2:
self._fixed_tab_width = tot_width/2
if self._fixed_tab_width > 220:
self._fixed_tab_width = 220
if minTabWidth > -1:
self._fixed_tab_width = max(self._fixed_tab_width, minTabWidth)
if maxTabWidth > -1:
self._fixed_tab_width = min(self._fixed_tab_width, maxTabWidth)
self._tab_ctrl_height = tab_ctrl_size.y
def DrawBackground(self, dc, wnd, rect):
"""
Draws the tab area background.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param Rect `rect`: the tab control rectangle.
"""
# draw background
dc.SetBrush(self._bkbrush)
dc.SetPen(wx.TRANSPARENT_PEN)
dc.DrawRectangle(-1, -1, rect.GetWidth()+2, rect.GetHeight()+2)
# draw base line
dc.SetPen(wx.GREY_PEN)
dc.DrawLine(0, rect.GetHeight()-1, rect.GetWidth(), rect.GetHeight()-1)
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# if the caption is empty, measure some temporary text
caption = page.caption
if caption == "":
caption = "Xj"
agwFlags = self.GetAGWFlags()
dc.SetFont(self._selected_font)
selected_textx, selected_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
dc.SetFont(self._normal_font)
normal_textx, normal_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
control = page.control
# figure out the size of the tab
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap,
page.active, close_button_state, control)
tab_height = tab_size[1]
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
caption = page.caption
# select pen, brush and font for the tab to be drawn
if page.active:
dc.SetPen(self._selected_bkpen)
dc.SetBrush(self._selected_bkbrush)
dc.SetFont(self._selected_font)
textx = selected_textx
texty = selected_texty
else:
dc.SetPen(self._normal_bkpen)
dc.SetBrush(self._normal_bkbrush)
dc.SetFont(self._normal_font)
textx = normal_textx
texty = normal_texty
if not page.enabled:
dc.SetTextForeground(wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT))
else:
dc.SetTextForeground(page.text_colour)
# -- draw line --
points = [wx.Point() for i in range(7)]
points[0].x = tab_x
points[0].y = tab_y + tab_height - 1
points[1].x = tab_x + tab_height - 3
points[1].y = tab_y + 2
points[2].x = tab_x + tab_height + 3
points[2].y = tab_y
points[3].x = tab_x + tab_width - 2
points[3].y = tab_y
points[4].x = tab_x + tab_width
points[4].y = tab_y + 2
points[5].x = tab_x + tab_width
points[5].y = tab_y + tab_height - 1
points[6] = points[0]
dc.SetClippingRegion(in_rect)
dc.DrawPolygon(points)
dc.SetPen(wx.GREY_PEN)
dc.DrawLines(points)
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
if control:
text_offset = tab_x + (tab_height/2) + close_button_width - (textx/2) - 2
else:
text_offset = tab_x + (tab_height/2) + ((tab_width+close_button_width)/2) - (textx/2) - 2
else:
if control:
text_offset = tab_x + (tab_height/2) + close_button_width - (textx/2)
else:
text_offset = tab_x + (tab_height/2) + ((tab_width-close_button_width)/2) - (textx/2)
else:
text_offset = tab_x + (tab_height/3) + (tab_width/2) - (textx/2)
if control:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset = tab_x + (tab_height/3) - (textx/2) + close_button_width + 2
else:
text_offset = tab_x + (tab_height/3) - (textx/2)
# set minimum text offset
if text_offset < tab_x + tab_height:
text_offset = tab_x + tab_height
# chop text if necessary
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x))
else:
draw_text = ChopText(dc, caption,
tab_width - (text_offset-tab_x) - close_button_width)
ypos = (tab_y + tab_height)/2 - (texty/2) + 1
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
# draw focus rectangle
if page.active and wx.Window.FindFocus() == wnd and (agwFlags & AUI_NB_NO_TAB_FOCUS) == 0:
focusRect = wx.Rect(text_offset, ((tab_y + tab_height)/2 - (texty/2) + 1),
selected_textx, selected_texty)
focusRect.Inflate(2, 2)
# TODO:
# This should be uncommented when DrawFocusRect will become
# available in wxPython
# wx.RendererNative.Get().DrawFocusRect(wnd, dc, focusRect, 0)
out_button_rect = wx.Rect()
# draw close button if necessary
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
if page.active:
bmp = self._active_close_bmp
else:
bmp = self._disabled_close_bmp
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + tab_height - 2,
tab_y + (tab_height/2) - (bmp.GetHeight()/2) + 1,
close_button_width, tab_height - 1)
else:
rect = wx.Rect(tab_x + tab_width - close_button_width - 1,
tab_y + (tab_height/2) - (bmp.GetHeight()/2) + 1,
close_button_width, tab_height - 1)
self.DrawButtons(dc, rect, bmp, wx.WHITE, close_button_state)
out_button_rect = wx.Rect(*rect)
out_tab_rect = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent
def DrawButtons(self, dc, _rect, bmp, bkcolour, button_state):
"""
Convenience method to draw tab buttons.
:param `dc`: a :class:`DC` device context;
:param Rect `_rect`: the tab rectangle;
:param Bitmap `bmp`: the tab bitmap;
:param Colour `bkcolour`: the tab background colour;
:param integer `button_state`: the state of the tab button.
"""
rect = wx.Rect(*_rect)
if button_state == AUI_BUTTON_STATE_PRESSED:
rect.x += 1
rect.y += 1
if button_state in [AUI_BUTTON_STATE_HOVER, AUI_BUTTON_STATE_PRESSED]:
dc.SetBrush(wx.Brush(StepColour(bkcolour, 120)))
dc.SetPen(wx.Pen(StepColour(bkcolour, 75)))
# draw the background behind the button
dc.DrawRectangle(rect.x, rect.y, 15, 15)
# draw the button itself
dc.DrawBitmap(bmp, rect.x, rect.y, True)
def GetIndentSize(self):
""" Returns the tabs indent size. """
return 0
def GetTabSize(self, dc, wnd, caption, bitmap, active, close_button_state, control=None):
"""
Returns the tab size for the given caption, bitmap and button state.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param string `caption`: the tab text caption;
:param Bitmap `bitmap`: the bitmap displayed on the tab;
:param bool `active`: whether the tab is selected or not;
:param integer `close_button_state`: the state of the close button on the tab;
:param Window `control`: a :class:`Window` instance inside a tab (or ``None``).
"""
dc.SetFont(self._measuring_font)
measured_textx, measured_texty, dummy = dc.GetFullMultiLineTextExtent(caption)
tab_height = measured_texty + 4
tab_width = measured_textx + tab_height + 5
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
tab_width += self._active_close_bmp.GetWidth()
if self._agwFlags & AUI_NB_TAB_FIXED_WIDTH:
tab_width = self._fixed_tab_width
if control:
controlW, controlH = control.GetSize()
tab_width += controlW + 4
x_extent = tab_width - (tab_height/2) - 1
return (tab_width, tab_height), x_extent
def DrawButton(self, dc, wnd, in_rect, button, orientation):
"""
Draws a button on the tab or on the tab area, depending on the button identifier.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param `button`: an instance of the button class;
:param integer `orientation`: the tab orientation.
"""
bitmap_id, button_state = button.id, button.cur_state
if bitmap_id == AUI_BUTTON_CLOSE:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_close_bmp
else:
bmp = self._active_close_bmp
elif bitmap_id == AUI_BUTTON_LEFT:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_left_bmp
else:
bmp = self._active_left_bmp
elif bitmap_id == AUI_BUTTON_RIGHT:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_right_bmp
else:
bmp = self._active_right_bmp
elif bitmap_id == AUI_BUTTON_WINDOWLIST:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = self._disabled_windowlist_bmp
else:
bmp = self._active_windowlist_bmp
else:
if button_state & AUI_BUTTON_STATE_DISABLED:
bmp = button.dis_bitmap
else:
bmp = button.bitmap
if not bmp.IsOk():
return
rect = wx.Rect(*in_rect)
if orientation == wx.LEFT:
rect.SetX(in_rect.x)
rect.SetY(((in_rect.y + in_rect.height)/2) - (bmp.GetHeight()/2))
rect.SetWidth(bmp.GetWidth())
rect.SetHeight(bmp.GetHeight())
else:
rect = wx.Rect(in_rect.x + in_rect.width - bmp.GetWidth(),
((in_rect.y + in_rect.height)/2) - (bmp.GetHeight()/2),
bmp.GetWidth(), bmp.GetHeight())
self.DrawButtons(dc, rect, bmp, wx.WHITE, button_state)
out_rect = wx.Rect(*rect)
return out_rect
def ShowDropDown(self, wnd, pages, active_idx):
"""
Shows the drop-down window menu on the tab area.
:param `wnd`: a :class:`Window` derived window instance;
:param list `pages`: the pages associated with the tabs;
:param integer `active_idx`: the active tab index.
"""
menuPopup = wx.Menu()
useImages = self.GetAGWFlags() & AUI_NB_USE_IMAGES_DROPDOWN
for i, page in enumerate(pages):
if useImages:
menuItem = wx.MenuItem(menuPopup, 1000+i, page.caption)
if page.bitmap:
menuItem.SetBitmap(page.bitmap)
menuPopup.Append(menuItem)
else:
menuPopup.AppendCheckItem(1000+i, page.caption)
menuPopup.Enable(1000+i, page.enabled)
if active_idx != -1 and not useImages:
menuPopup.Check(1000+active_idx, True)
# find out where to put the popup menu of window
# items. Subtract 100 for now to center the menu
# a bit, until a better mechanism can be implemented
pt = wx.GetMousePosition()
pt = wnd.ScreenToClient(pt)
if pt.x < 100:
pt.x = 0
else:
pt.x -= 100
# find out the screen coordinate at the bottom of the tab ctrl
cli_rect = wnd.GetClientRect()
pt.y = cli_rect.y + cli_rect.height
cc = AuiCommandCapture()
wnd.PushEventHandler(cc)
wnd.PopupMenu(menuPopup, pt)
command = cc.GetCommandId()
wnd.PopEventHandler(True)
if command >= 1000:
return command-1000
return -1
def GetBestTabCtrlSize(self, wnd, pages, required_bmp_size):
"""
Returns the best tab control size.
:param `wnd`: a :class:`Window` instance object;
:param list `pages`: the pages associated with the tabs;
:param Size `required_bmp_size`: the size of the bitmap on the tabs.
"""
dc = wx.ClientDC(wnd)
dc.SetFont(self._measuring_font)
s, x_extent = self.GetTabSize(dc, wnd, "ABCDEFGHIj", wx.NullBitmap, True,
AUI_BUTTON_STATE_HIDDEN, None)
max_y = s[1]
for page in pages:
if page.control:
controlW, controlH = page.control.GetSize()
max_y = max(max_y, controlH+4)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(page.caption)
max_y = max(max_y, texty)
return max_y + 3
def SetNormalFont(self, font):
"""
Sets the normal font for drawing tab labels.
:param Font `font`: the new font to use to draw tab labels in their normal, un-selected state.
"""
self._normal_font = font
def SetSelectedFont(self, font):
"""
Sets the selected tab font for drawing tab labels.
:param Font `font`: the new font to use to draw tab labels in their selected state.
"""
self._selected_font = font
def SetMeasuringFont(self, font):
"""
Sets the font for calculating text measurements.
:param Font `font`: the new font to use to measure tab labels text extents.
"""
self._measuring_font = font
def GetNormalFont(self):
""" Returns the normal font for drawing tab labels. """
return self._normal_font
def GetSelectedFont(self):
""" Returns the selected tab font for drawing tab labels. """
return self._selected_font
def GetMeasuringFont(self):
""" Returns the font for calculating text measurements. """
return self._measuring_font
def SetCustomButton(self, bitmap_id, button_state, bmp):
"""
Sets a custom bitmap for the close, left, right and window list buttons.
:param integer `bitmap_id`: the button identifier;
:param integer `button_state`: the button state;
:param Bitmap `bmp`: the custom bitmap to use for the button.
"""
if bitmap_id == AUI_BUTTON_CLOSE:
if button_state == AUI_BUTTON_STATE_NORMAL:
self._active_close_bmp = bmp
self._hover_close_bmp = self._active_close_bmp
self._pressed_close_bmp = self._active_close_bmp
self._disabled_close_bmp = self._active_close_bmp
elif button_state == AUI_BUTTON_STATE_HOVER:
self._hover_close_bmp = bmp
elif button_state == AUI_BUTTON_STATE_PRESSED:
self._pressed_close_bmp = bmp
else:
self._disabled_close_bmp = bmp
elif bitmap_id == AUI_BUTTON_LEFT:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_left_bmp = bmp
else:
self._active_left_bmp = bmp
elif bitmap_id == AUI_BUTTON_RIGHT:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_right_bmp = bmp
else:
self._active_right_bmp = bmp
elif bitmap_id == AUI_BUTTON_WINDOWLIST:
if button_state & AUI_BUTTON_STATE_DISABLED:
self._disabled_windowlist_bmp = bmp
else:
self._active_windowlist_bmp = bmp
class VC71TabArt(AuiDefaultTabArt):
""" A class to draw tabs using the Visual Studio 2003 (VC71) style. """
def __init__(self):
""" Default class constructor. """
AuiDefaultTabArt.__init__(self)
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# Visual studio 7.1 style
# This code is based on the renderer included in FlatNotebook
# figure out the size of the tab
control = page.control
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap, page.active,
close_button_state, control)
tab_height = self._tab_ctrl_height - 3
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
clip_width = tab_width
if tab_x + clip_width > in_rect.x + in_rect.width - 4:
clip_width = (in_rect.x + in_rect.width) - tab_x - 4
dc.SetClippingRegion(tab_x, tab_y, clip_width + 1, tab_height - 3)
agwFlags = self.GetAGWFlags()
if agwFlags & AUI_NB_BOTTOM:
tab_y -= 1
dc.SetPen((page.active and [wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DHIGHLIGHT))] or \
[wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DSHADOW))])[0])
dc.SetBrush((page.active and [wx.Brush(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DFACE))] or \
[wx.TRANSPARENT_BRUSH])[0])
if page.active:
tabH = tab_height - 2
dc.DrawRectangle(tab_x, tab_y, tab_width, tabH)
rightLineY1 = (agwFlags & AUI_NB_BOTTOM and [vertical_border_padding - 2] or \
[vertical_border_padding - 1])[0]
rightLineY2 = tabH + 3
dc.SetPen(wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DSHADOW)))
dc.DrawLine(tab_x + tab_width - 1, rightLineY1 + 1, tab_x + tab_width - 1, rightLineY2)
if agwFlags & AUI_NB_BOTTOM:
dc.DrawLine(tab_x + 1, rightLineY2 - 3 , tab_x + tab_width - 1, rightLineY2 - 3)
dc.SetPen(wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DDKSHADOW)))
dc.DrawLine(tab_x + tab_width, rightLineY1, tab_x + tab_width, rightLineY2)
if agwFlags & AUI_NB_BOTTOM:
dc.DrawLine(tab_x, rightLineY2 - 2, tab_x + tab_width, rightLineY2 - 2)
else:
# We dont draw a rectangle for non selected tabs, but only
# vertical line on the right
blackLineY1 = (agwFlags & AUI_NB_BOTTOM and [vertical_border_padding + 2] or \
[vertical_border_padding + 1])[0]
blackLineY2 = tab_height - 5
dc.DrawLine(tab_x + tab_width, blackLineY1, tab_x + tab_width, blackLineY2)
border_points = [0, 0]
if agwFlags & AUI_NB_BOTTOM:
border_points[0] = wx.Point(tab_x, tab_y)
border_points[1] = wx.Point(tab_x, tab_y + tab_height - 6)
else: # if (agwFlags & AUI_NB_TOP)
border_points[0] = wx.Point(tab_x, tab_y + tab_height - 4)
border_points[1] = wx.Point(tab_x, tab_y + 2)
drawn_tab_yoff = border_points[1].y
drawn_tab_height = border_points[0].y - border_points[1].y
text_offset = tab_x + 8
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset += close_button_width - 5
if not page.enabled:
dc.SetTextForeground(wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT))
pagebitmap = page.dis_bitmap
else:
dc.SetTextForeground(page.text_colour)
pagebitmap = page.bitmap
shift = 0
if agwFlags & AUI_NB_BOTTOM:
shift = (page.active and [1] or [2])[0]
bitmap_offset = 0
if pagebitmap.IsOk():
bitmap_offset = tab_x + 8
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT and close_button_width:
bitmap_offset += close_button_width - 5
# draw bitmap
dc.DrawBitmap(pagebitmap, bitmap_offset,
drawn_tab_yoff + (drawn_tab_height/2) - (pagebitmap.GetHeight()/2) + shift,
True)
text_offset = bitmap_offset + pagebitmap.GetWidth()
text_offset += 3 # bitmap padding
else:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT == 0 or not close_button_width:
text_offset = tab_x + 8
# if the caption is empty, measure some temporary text
caption = page.caption
if caption == "":
caption = "Xj"
if page.active:
dc.SetFont(self._selected_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
else:
dc.SetFont(self._normal_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width)
ypos = drawn_tab_yoff + (drawn_tab_height)/2 - (texty/2) - 1 + shift
offset_focus = text_offset
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
textx += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
out_button_rect = wx.Rect()
# draw focus rectangle
if (agwFlags & AUI_NB_NO_TAB_FOCUS) == 0:
self.DrawFocusRectangle(dc, page, wnd, draw_text, offset_focus, bitmap_offset, drawn_tab_yoff+shift,
drawn_tab_height+shift, rectx, recty)
# draw 'x' on tab (if enabled)
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
bmp = self._disabled_close_bmp
if close_button_state == AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif close_button_state == AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + 4,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
else:
rect = wx.Rect(tab_x + tab_width - close_button_width - 3,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
# Indent the button if it is pressed down:
rect = IndentPressedBitmap(rect, close_button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_button_rect = rect
out_tab_rect = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent
class FF2TabArt(AuiDefaultTabArt):
""" A class to draw tabs using the Firefox 2 (FF2) style. """
def __init__(self):
""" Default class constructor. """
AuiDefaultTabArt.__init__(self)
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def GetTabSize(self, dc, wnd, caption, bitmap, active, close_button_state, control):
"""
Returns the tab size for the given caption, bitmap and button state.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param string `caption`: the tab text caption;
:param Bitmap `bitmap`: the bitmap displayed on the tab;
:param bool `active`: whether the tab is selected or not;
:param integer `close_button_state`: the state of the close button on the tab;
:param Window `control`: a :class:`Window` instance inside a tab (or ``None``).
"""
tab_size, x_extent = AuiDefaultTabArt.GetTabSize(self, dc, wnd, caption, bitmap,
active, close_button_state, control)
tab_width, tab_height = tab_size
# add some vertical padding
tab_height += 2
return (tab_width, tab_height), x_extent
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# Firefox 2 style
control = page.control
# figure out the size of the tab
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap,
page.active, close_button_state, control)
tab_height = self._tab_ctrl_height - 2
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
clip_width = tab_width
if tab_x + clip_width > in_rect.x + in_rect.width - 4:
clip_width = (in_rect.x + in_rect.width) - tab_x - 4
dc.SetClippingRegion(tab_x, tab_y, clip_width + 1, tab_height - 3)
tabPoints = [wx.Point() for i in range(7)]
adjust = 0
if not page.active:
adjust = 1
agwFlags = self.GetAGWFlags()
tabPoints[0].x = tab_x + 3
tabPoints[0].y = (agwFlags & AUI_NB_BOTTOM and [3] or [tab_height - 2])[0]
tabPoints[1].x = tabPoints[0].x
tabPoints[1].y = (agwFlags & AUI_NB_BOTTOM and [tab_height - (vertical_border_padding + 2) - adjust] or \
[(vertical_border_padding + 2) + adjust])[0]
tabPoints[2].x = tabPoints[1].x+2
tabPoints[2].y = (agwFlags & AUI_NB_BOTTOM and [tab_height - vertical_border_padding - adjust] or \
[vertical_border_padding + adjust])[0]
tabPoints[3].x = tab_x + tab_width - 2
tabPoints[3].y = tabPoints[2].y
tabPoints[4].x = tabPoints[3].x + 2
tabPoints[4].y = tabPoints[1].y
tabPoints[5].x = tabPoints[4].x
tabPoints[5].y = tabPoints[0].y
tabPoints[6].x = tabPoints[0].x
tabPoints[6].y = tabPoints[0].y
rr = wx.Rect(tabPoints[2], tabPoints[5])
self.DrawTabBackground(dc, rr, page.active, (agwFlags & AUI_NB_BOTTOM) == 0)
dc.SetBrush(wx.TRANSPARENT_BRUSH)
dc.SetPen(wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_BTNSHADOW)))
# Draw the tab as rounded rectangle
dc.DrawPolygon(tabPoints)
if page.active:
dc.DrawLine(tabPoints[0].x + 1, tabPoints[0].y, tabPoints[5].x , tabPoints[0].y)
drawn_tab_yoff = tabPoints[1].y
drawn_tab_height = tabPoints[0].y - tabPoints[2].y
text_offset = tab_x + 8
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset += close_button_width - 4
if not page.enabled:
dc.SetTextForeground(wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT))
pagebitmap = page.dis_bitmap
else:
dc.SetTextForeground(page.text_colour)
pagebitmap = page.bitmap
shift = -1
if agwFlags & AUI_NB_BOTTOM:
shift = 2
bitmap_offset = 0
if pagebitmap.IsOk():
bitmap_offset = tab_x + 8
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT and close_button_width:
bitmap_offset += close_button_width - 4
# draw bitmap
dc.DrawBitmap(pagebitmap, bitmap_offset,
drawn_tab_yoff + (drawn_tab_height/2) - (pagebitmap.GetHeight()/2) + shift,
True)
text_offset = bitmap_offset + pagebitmap.GetWidth()
text_offset += 3 # bitmap padding
else:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT == 0 or not close_button_width:
text_offset = tab_x + 8
# if the caption is empty, measure some temporary text
caption = page.caption
if caption == "":
caption = "Xj"
if page.active:
dc.SetFont(self._selected_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
else:
dc.SetFont(self._normal_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width + 1)
else:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width)
ypos = drawn_tab_yoff + drawn_tab_height/2 - texty/2 - 1 + shift
offset_focus = text_offset
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
textx += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
# draw focus rectangle
if (agwFlags & AUI_NB_NO_TAB_FOCUS) == 0:
self.DrawFocusRectangle(dc, page, wnd, draw_text, offset_focus, bitmap_offset, drawn_tab_yoff+shift,
drawn_tab_height, rectx, recty)
out_button_rect = wx.Rect()
# draw 'x' on tab (if enabled)
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
bmp = self._disabled_close_bmp
if close_button_state == AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif close_button_state == AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + 5,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
else:
rect = wx.Rect(tab_x + tab_width - close_button_width - 3,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
# Indent the button if it is pressed down:
rect = IndentPressedBitmap(rect, close_button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_button_rect = rect
out_tab_rect = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent
def DrawTabBackground(self, dc, rect, focus, upperTabs):
"""
Draws the tab background for the Firefox 2 style.
This is more consistent with :class:`~lib.agw.flatnotebook.FlatNotebook` than before.
:param `dc`: a :class:`DC` device context;
:param Rect `rect`: rectangle the tab should be confined to;
:param bool `focus`: whether the tab has focus or not;
:param bool `upperTabs`: whether the style is ``AUI_NB_TOP`` or ``AUI_NB_BOTTOM``.
"""
# Define the rounded rectangle base on the given rect
# we need an array of 9 points for it
regPts = [wx.Point() for indx in range(9)]
if focus:
if upperTabs:
leftPt = wx.Point(rect.x, rect.y + (rect.height / 10)*8)
rightPt = wx.Point(rect.x + rect.width - 2, rect.y + (rect.height / 10)*8)
else:
leftPt = wx.Point(rect.x, rect.y + (rect.height / 10)*5)
rightPt = wx.Point(rect.x + rect.width - 2, rect.y + (rect.height / 10)*5)
else:
leftPt = wx.Point(rect.x, rect.y + (rect.height / 2))
rightPt = wx.Point(rect.x + rect.width - 2, rect.y + (rect.height / 2))
# Define the top region
top = wx.Rect(rect.GetTopLeft(), rightPt)
bottom = wx.Rect(leftPt, rect.GetBottomRight())
topStartColour = wx.WHITE
if not focus:
topStartColour = LightColour(wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DFACE), 50)
topEndColour = wx.SystemSettings.GetColour(wx.SYS_COLOUR_3DFACE)
bottomStartColour = topEndColour
bottomEndColour = topEndColour
# Incase we use bottom tabs, switch the colours
if upperTabs:
if focus:
dc.GradientFillLinear(top, topStartColour, topEndColour, wx.SOUTH)
dc.GradientFillLinear(bottom, bottomStartColour, bottomEndColour, wx.SOUTH)
else:
dc.GradientFillLinear(top, topEndColour , topStartColour, wx.SOUTH)
dc.GradientFillLinear(bottom, bottomStartColour, bottomEndColour, wx.SOUTH)
else:
if focus:
dc.GradientFillLinear(bottom, topEndColour, bottomEndColour, wx.SOUTH)
dc.GradientFillLinear(top, topStartColour, topStartColour, wx.SOUTH)
else:
dc.GradientFillLinear(bottom, bottomStartColour, bottomEndColour, wx.SOUTH)
dc.GradientFillLinear(top, topEndColour, topStartColour, wx.SOUTH)
dc.SetBrush(wx.TRANSPARENT_BRUSH)
class VC8TabArt(AuiDefaultTabArt):
""" A class to draw tabs using the Visual Studio 2005 (VC8) style. """
def __init__(self):
""" Default class constructor. """
AuiDefaultTabArt.__init__(self)
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth):
"""
Sets the tab sizing information.
:param Size `tab_ctrl_size`: the size of the tab control area;
:param integer `tab_count`: the number of tabs;
:param tuple `minMaxTabWidth`: a tuple containing the minimum and maximum tab widths
to be used when the ``AUI_NB_TAB_FIXED_WIDTH`` style is active.
"""
AuiDefaultTabArt.SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth)
minTabWidth, maxTabWidth = minMaxTabWidth
if minTabWidth > -1:
self._fixed_tab_width = max(self._fixed_tab_width, minTabWidth)
if maxTabWidth > -1:
self._fixed_tab_width = min(self._fixed_tab_width, maxTabWidth)
self._fixed_tab_width -= 5
def GetTabSize(self, dc, wnd, caption, bitmap, active, close_button_state, control=None):
"""
Returns the tab size for the given caption, bitmap and button state.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param string `caption`: the tab text caption;
:param Bitmap `bitmap`: the bitmap displayed on the tab;
:param bool `active`: whether the tab is selected or not;
:param integer `close_button_state`: the state of the close button on the tab;
:param Window `control`: a :class:`Window` instance inside a tab (or ``None``).
"""
tab_size, x_extent = AuiDefaultTabArt.GetTabSize(self, dc, wnd, caption, bitmap,
active, close_button_state, control)
tab_width, tab_height = tab_size
# add some padding
tab_width += 10
if not bitmap.IsOk():
tab_width += 5
tab_height += 2
return (tab_width, tab_height), x_extent
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# Visual Studio 8 style
control = page.control
# figure out the size of the tab
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap,
page.active, close_button_state, control)
tab_height = self._tab_ctrl_height - 1
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
clip_width = tab_width + 3
if tab_x + clip_width > in_rect.x + in_rect.width - 4:
clip_width = (in_rect.x + in_rect.width) - tab_x - 4
tabPoints = [wx.Point() for i in range(8)]
# If we draw the first tab or the active tab,
# we draw a full tab, else we draw a truncated tab
#
# X(2) X(3)
# X(1) X(4)
#
# X(5)
#
# X(0),(7) X(6)
#
#
adjust = 0
if not page.active:
adjust = 1
agwFlags = self.GetAGWFlags()
tabPoints[0].x = (agwFlags & AUI_NB_BOTTOM and [tab_x] or [tab_x + adjust])[0]
tabPoints[0].y = (agwFlags & AUI_NB_BOTTOM and [2] or [tab_height - 3])[0]
tabPoints[1].x = tabPoints[0].x + tab_height - vertical_border_padding - 3 - adjust
tabPoints[1].y = (agwFlags & AUI_NB_BOTTOM and [tab_height - (vertical_border_padding+2)] or \
[(vertical_border_padding+2)])[0]
tabPoints[2].x = tabPoints[1].x + 4
tabPoints[2].y = (agwFlags & AUI_NB_BOTTOM and [tab_height - vertical_border_padding] or \
[vertical_border_padding])[0]
tabPoints[3].x = tabPoints[2].x + tab_width - tab_height + vertical_border_padding
tabPoints[3].y = (agwFlags & AUI_NB_BOTTOM and [tab_height - vertical_border_padding] or \
[vertical_border_padding])[0]
tabPoints[4].x = tabPoints[3].x + 1
tabPoints[4].y = (agwFlags & AUI_NB_BOTTOM and [tabPoints[3].y - 1] or [tabPoints[3].y + 1])[0]
tabPoints[5].x = tabPoints[4].x + 1
tabPoints[5].y = (agwFlags & AUI_NB_BOTTOM and [(tabPoints[4].y - 1)] or [tabPoints[4].y + 1])[0]
tabPoints[6].x = tabPoints[2].x + tab_width - tab_height + 2 + vertical_border_padding
tabPoints[6].y = tabPoints[0].y
tabPoints[7].x = tabPoints[0].x
tabPoints[7].y = tabPoints[0].y
self.FillVC8GradientColour(dc, tabPoints, page.active)
dc.SetBrush(wx.TRANSPARENT_BRUSH)
dc.SetPen(wx.Pen(wx.SystemSettings.GetColour(wx.SYS_COLOUR_BTNSHADOW)))
dc.DrawPolygon(tabPoints)
if page.active:
# Delete the bottom line (or the upper one, incase we use wxBOTTOM)
dc.SetPen(wx.WHITE_PEN)
dc.DrawLine(tabPoints[0].x, tabPoints[0].y, tabPoints[6].x, tabPoints[6].y)
dc.SetClippingRegion(tab_x, tab_y, clip_width + 2, tab_height - 3)
drawn_tab_yoff = tabPoints[1].y
drawn_tab_height = tabPoints[0].y - tabPoints[2].y
text_offset = tab_x + 20
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset += close_button_width
if not page.enabled:
dc.SetTextForeground(wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT))
pagebitmap = page.dis_bitmap
else:
dc.SetTextForeground(page.text_colour)
pagebitmap = page.bitmap
shift = 0
if agwFlags & AUI_NB_BOTTOM:
shift = (page.active and [1] or [2])[0]
bitmap_offset = 0
if pagebitmap.IsOk():
bitmap_offset = tab_x + 20
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT and close_button_width:
bitmap_offset += close_button_width
# draw bitmap
dc.DrawBitmap(pagebitmap, bitmap_offset,
drawn_tab_yoff + (drawn_tab_height/2) - (pagebitmap.GetHeight()/2) + shift,
True)
text_offset = bitmap_offset + pagebitmap.GetWidth()
text_offset += 3 # bitmap padding
else:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT == 0 or not close_button_width:
text_offset = tab_x + tab_height
# if the caption is empty, measure some temporary text
caption = page.caption
if caption == "":
caption = "Xj"
if page.active:
dc.SetFont(self._selected_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
else:
dc.SetFont(self._normal_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x))
else:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width)
ypos = drawn_tab_yoff + drawn_tab_height/2 - texty/2 - 1 + shift
offset_focus = text_offset
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
textx += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
# draw focus rectangle
if (agwFlags & AUI_NB_NO_TAB_FOCUS) == 0:
self.DrawFocusRectangle(dc, page, wnd, draw_text, offset_focus, bitmap_offset, drawn_tab_yoff+shift,
drawn_tab_height+shift, rectx, recty)
out_button_rect = wx.Rect()
# draw 'x' on tab (if enabled)
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
bmp = self._disabled_close_bmp
if close_button_state == AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif close_button_state == AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
if page.active:
xpos = tab_x + tab_width - close_button_width + 3
else:
xpos = tab_x + tab_width - close_button_width - 5
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + 20,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
else:
rect = wx.Rect(xpos,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + shift,
close_button_width, tab_height)
# Indent the button if it is pressed down:
rect = IndentPressedBitmap(rect, close_button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_button_rect = rect
out_tab_rect = wx.Rect(tab_x, tab_y, x_extent, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent
def FillVC8GradientColour(self, dc, tabPoints, active):
"""
Fills the tab with the Visual Studio 2005 gradient background.
:param `dc`: a :class:`DC` device context;
:param list `tabPoints`: a list of :class:`Point` objects describing the tab shape;
:param bool `active`: whether the tab is selected or not.
"""
xList = [pt.x for pt in tabPoints]
yList = [pt.y for pt in tabPoints]
minx, maxx = min(xList), max(xList)
miny, maxy = min(yList), max(yList)
rect = wx.Rect(minx, maxy, maxx-minx, miny-maxy+1)
region = wx.Region(tabPoints)
if self._buttonRect.width > 0:
buttonRegion = wx.Region(*self._buttonRect)
region.Xor(buttonRegion)
dc.SetDeviceClippingRegion(region)
if active:
bottom_colour = top_colour = wx.WHITE
else:
bottom_colour = StepColour(self._base_colour, 90)
top_colour = StepColour(self._base_colour, 170)
dc.GradientFillLinear(rect, top_colour, bottom_colour, wx.SOUTH)
dc.DestroyClippingRegion()
class ChromeTabArt(AuiDefaultTabArt):
"""
A class to draw tabs using the Google Chrome browser style.
It uses custom bitmap to render the tabs, so that the look and feel is as close
as possible to the Chrome style.
"""
def __init__(self):
""" Default class constructor. """
AuiDefaultTabArt.__init__(self)
self.SetBitmaps(mirror=False)
closeBmp = tab_close.GetBitmap()
closeHBmp = tab_close_h.GetBitmap()
closePBmp = tab_close_p.GetBitmap()
self.SetCustomButton(AUI_BUTTON_CLOSE, AUI_BUTTON_STATE_NORMAL, closeBmp)
self.SetCustomButton(AUI_BUTTON_CLOSE, AUI_BUTTON_STATE_HOVER, closeHBmp)
self.SetCustomButton(AUI_BUTTON_CLOSE, AUI_BUTTON_STATE_PRESSED, closePBmp)
def SetAGWFlags(self, agwFlags):
"""
Sets the tab art flags.
:param integer `agwFlags`: a combination of the following values:
==================================== ==================================
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:`~lib.agw.aui.framemanager.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
==================================== ==================================
:note: Overridden from :class:`AuiDefaultTabArt`.
"""
if agwFlags & AUI_NB_TOP:
self.SetBitmaps(mirror=False)
elif agwFlags & AUI_NB_BOTTOM:
self.SetBitmaps(mirror=True)
AuiDefaultTabArt.SetAGWFlags(self, agwFlags)
def SetBitmaps(self, mirror):
"""
Assigns the tab custom bitmaps
:param bool `mirror`: whether to vertically mirror the bitmap or not.
"""
bmps = [tab_active_left.GetBitmap(), tab_active_center.GetBitmap(),
tab_active_right.GetBitmap(), tab_inactive_left.GetBitmap(),
tab_inactive_center.GetBitmap(), tab_inactive_right.GetBitmap()]
if mirror:
for indx, bmp in enumerate(bmps):
img = bmp.ConvertToImage()
img = img.Mirror(horizontally=False)
bmps[indx] = img.ConvertToBitmap()
self._leftActiveBmp = bmps[0]
self._centerActiveBmp = bmps[1]
self._rightActiveBmp = bmps[2]
self._leftInactiveBmp = bmps[3]
self._centerInactiveBmp = bmps[4]
self._rightInactiveBmp = bmps[5]
def Clone(self):
""" Clones the art object. """
art = type(self)()
art.SetNormalFont(self.GetNormalFont())
art.SetSelectedFont(self.GetSelectedFont())
art.SetMeasuringFont(self.GetMeasuringFont())
art = CopyAttributes(art, self)
return art
def SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth):
"""
Sets the tab sizing information.
:param Size `tab_ctrl_size`: the size of the tab control area;
:param integer `tab_count`: the number of tabs;
:param tuple `minMaxTabWidth`: a tuple containing the minimum and maximum tab widths
to be used when the ``AUI_NB_TAB_FIXED_WIDTH`` style is active.
"""
AuiDefaultTabArt.SetSizingInfo(self, tab_ctrl_size, tab_count, minMaxTabWidth)
minTabWidth, maxTabWidth = minMaxTabWidth
if minTabWidth > -1:
self._fixed_tab_width = max(self._fixed_tab_width, minTabWidth)
if maxTabWidth > -1:
self._fixed_tab_width = min(self._fixed_tab_width, maxTabWidth)
self._fixed_tab_width -= 5
def GetTabSize(self, dc, wnd, caption, bitmap, active, close_button_state, control=None):
"""
Returns the tab size for the given caption, bitmap and button state.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param string `caption`: the tab text caption;
:param Bitmap `bitmap`: the bitmap displayed on the tab;
:param bool `active`: whether the tab is selected or not;
:param integer `close_button_state`: the state of the close button on the tab;
:param Window `control`: a :class:`Window` instance inside a tab (or ``None``).
"""
tab_size, x_extent = AuiDefaultTabArt.GetTabSize(self, dc, wnd, caption, bitmap,
active, close_button_state, control)
tab_width, tab_height = tab_size
# add some padding
tab_width += self._leftActiveBmp.GetWidth()
tab_height += 2
tab_height = max(tab_height, self._centerActiveBmp.GetHeight())
return (tab_width, tab_height), x_extent
def DrawTab(self, dc, wnd, page, in_rect, close_button_state, paint_control=False):
"""
Draws a single tab.
:param `dc`: a :class:`DC` device context;
:param `wnd`: a :class:`Window` instance object;
:param `page`: the tab control page associated with the tab;
:param Rect `in_rect`: rectangle the tab should be confined to;
:param integer `close_button_state`: the state of the close button on the tab;
:param bool `paint_control`: whether to draw the control inside a tab (if any) on a :class:`MemoryDC`.
"""
# Chrome tab style
control = page.control
# figure out the size of the tab
tab_size, x_extent = self.GetTabSize(dc, wnd, page.caption, page.bitmap, page.active,
close_button_state, control)
agwFlags = self.GetAGWFlags()
tab_height = self._tab_ctrl_height - 1
tab_width = tab_size[0]
tab_x = in_rect.x
tab_y = in_rect.y + in_rect.height - tab_height
clip_width = tab_width
if tab_x + clip_width > in_rect.x + in_rect.width - 4:
clip_width = (in_rect.x + in_rect.width) - tab_x - 4
dc.SetClippingRegion(tab_x, tab_y, clip_width + 1, tab_height - 3)
drawn_tab_yoff = 1
if page.active:
left = self._leftActiveBmp
center = self._centerActiveBmp
right = self._rightActiveBmp
else:
left = self._leftInactiveBmp
center = self._centerInactiveBmp
right = self._rightInactiveBmp
dc.DrawBitmap(left, tab_x, tab_y)
leftw = left.GetWidth()
centerw = center.GetWidth()
rightw = right.GetWidth()
available = tab_x + tab_width - rightw
posx = tab_x + leftw
while 1:
if posx >= available:
break
dc.DrawBitmap(center, posx, tab_y)
posx += centerw
dc.DrawBitmap(right, posx, tab_y)
drawn_tab_height = center.GetHeight()
text_offset = tab_x + leftw
close_button_width = 0
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
text_offset += close_button_width
if not page.enabled:
dc.SetTextForeground(wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT))
pagebitmap = page.dis_bitmap
else:
dc.SetTextForeground(page.text_colour)
pagebitmap = page.bitmap
bitmap_offset = 0
if pagebitmap.IsOk():
bitmap_offset = tab_x + leftw
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT and close_button_width:
bitmap_offset += close_button_width
# draw bitmap
dc.DrawBitmap(pagebitmap, bitmap_offset,
drawn_tab_yoff + (drawn_tab_height/2) - (pagebitmap.GetHeight()/2),
True)
text_offset = bitmap_offset + pagebitmap.GetWidth()
text_offset += 3 # bitmap padding
else:
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT == 0 or not close_button_width:
text_offset = tab_x + leftw
# if the caption is empty, measure some temporary text
caption = page.caption
if caption == "":
caption = "Xj"
if page.active:
dc.SetFont(self._selected_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
else:
dc.SetFont(self._normal_font)
textx, texty, dummy = dc.GetFullMultiLineTextExtent(caption)
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - leftw)
else:
draw_text = ChopText(dc, caption, tab_width - (text_offset-tab_x) - close_button_width - leftw)
ypos = drawn_tab_yoff + drawn_tab_height/2 - texty/2 - 1
if control:
if control.GetPosition() != wx.Point(text_offset+1, ypos):
control.SetPosition(wx.Point(text_offset+1, ypos))
if not control.IsShown():
control.Show()
if paint_control:
bmp = TakeScreenShot(control.GetScreenRect())
dc.DrawBitmap(bmp, text_offset+1, ypos, True)
controlW, controlH = control.GetSize()
text_offset += controlW + 4
# draw tab text
rectx, recty, dummy = dc.GetFullMultiLineTextExtent(draw_text)
dc.DrawLabel(draw_text, wx.Rect(text_offset, ypos, rectx, recty))
out_button_rect = wx.Rect()
# draw 'x' on tab (if enabled)
if close_button_state != AUI_BUTTON_STATE_HIDDEN:
close_button_width = self._active_close_bmp.GetWidth()
bmp = self._disabled_close_bmp
if close_button_state == AUI_BUTTON_STATE_HOVER:
bmp = self._hover_close_bmp
elif close_button_state == AUI_BUTTON_STATE_PRESSED:
bmp = self._pressed_close_bmp
if agwFlags & AUI_NB_CLOSE_ON_TAB_LEFT:
rect = wx.Rect(tab_x + leftw - 2,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + 1,
close_button_width, tab_height)
else:
rect = wx.Rect(tab_x + tab_width - close_button_width - rightw + 2,
drawn_tab_yoff + (drawn_tab_height / 2) - (bmp.GetHeight() / 2) + 1,
close_button_width, tab_height)
if agwFlags & AUI_NB_BOTTOM:
rect.y -= 1
# Indent the button if it is pressed down:
rect = IndentPressedBitmap(rect, close_button_state)
dc.DrawBitmap(bmp, rect.x, rect.y, True)
out_button_rect = rect
out_tab_rect = wx.Rect(tab_x, tab_y, tab_width, tab_height)
dc.DestroyClippingRegion()
return out_tab_rect, out_button_rect, x_extent