1 # --------------------------------------------------------------------------- #
2 # FOLDPANELBAR wxPython IMPLEMENTATION
3 # Ported From Jorgen Bodde & Julian Smart (Extended Demo) C++ Code By:
5 # Andrea Gavana, @ 23 Mar 2005
6 # Latest Revision: 28 Mar 2005, 22.30 CET
11 # All The C++ TODOs Are Still Alive. I Am Not Able to Read Jorges's Mind
12 # So I Don't Really Know What Will Be The New Features/Additions He Will
13 # Make On His Code. At The Moment They Are:
15 # 1. OnPaint Function In CaptionBar Class:
16 # TODO: Maybe First A Memory Dc Should Draw All, And Then Paint It On The
17 # Caption. This Way A Flickering Arrow During Resize Is Not Visible.
19 # 2. OnChar Function In CaptionBar Class:
20 # TODO: This Is Easy To Do But I Don't Have Any Useful Idea On Which Kind
21 # Of Features To Add. Does Anyone Have An Intelligent Idea?
23 # 3. AddFoldPanelWindow Function In FoldPanelBar Class:
24 # TODO: Take Old And New Heights, And If Difference, Reposition All The
25 # Lower Panels. This Is Because The User Can Add New wxWindow Controls
26 # Somewhere In Between When Other Panels Are Already Present.
27 # Don't Know What It Means. Probably Is My Poor English...
29 # 4. OnSizePanel Function In FoldPanelBar Class:
30 # TODO: A Smart Way To Check Wether The Old - New Width Of The
31 # Panel Changed, If So No Need To Resize The Fold Panel Items
33 # 5. Implementing Styles Like FPB_SINGLE_FOLD and FPB_EXCLUSIVE_FOLD
34 # TODO: Jorgen Has Left Undone These Jobs. I Don't Really Get What They
35 # Should Supposed To Do, So If Someone Could Enlight Me, Please Let Me Know.
38 # For The Original TODO List From Jorgen, Please Refer To:
39 # http://www.solidsteel.nl/jorg/components/foldpanel/wxFoldPanelBar.php#todo_list
43 # For All Kind Of Problems, Requests Of Enhancements And Bug Reports, Please
46 # andrea.gavana@agip.it
49 # Or, Obviously, To The wxPython Mailing List!!!
53 # --------------------------------------------------------------------------- #
57 The `FoldPanelBar` is a control that contains multiple panels (of type
58 `FoldPanelItem`) that can be expanded or collapsed. The captionbar of
59 the FoldPanel can be customized by setting it to a horizontal gradient
60 style, vertical gradient style, a single color, a rectangle or filled
61 rectangle. The FoldPanel items can be collapsed in place or to the
62 bottom of the control. `wx.Window` derived controls can be added
63 dynamically, and separated by separator lines. FoldPanelBar is
64 freeware and distributed under the wxPython license.
70 The internals of the FoldPanelBar is a list of FoldPanelItem objects. Through
71 the reference of FoldPanel these panels can be controlled by adding new controls
72 to a FoldPanel or adding new FoldPanels to the FoldPanelBar.
73 The CaptionBar fires events to the parent (container of all panel items) when a
74 sub-panel needs resizing (either folding or expanding). The fold or expand process
75 is simply a resize of the panel so it looks like all controls on it are gone. All
76 controls are still child of the FoldPanel they are located on. If they don't
77 handle the event (and they won't) then the owner of the FoldPanelBar gets the
78 events. This is what you need to handle the controls. There isn't much to it just
79 a lot of calculations to see what panel belongs where. There are no sizers
80 involved in the panels, everything is purely x-y positioning.
83 What can it do and what not?
84 ----------------------------
87 * Run-time addition of panels (no deletion just yet)
88 * Run time addition of controls to the panel (it will be resized accordingly)
89 * Creating panels in collapsed mode or expanded mode
90 * Various modes of caption behaviour and filling to make it more appealing
91 * Panels can be folded and collapsed (or all of them) to allow more space
95 * Selection of a panel like in a list ctrl
96 * Dragging and dropping the panels
97 * Re-ordering the panels (not yet)
103 FoldPanelBar is supported on the following platforms:
104 * Windows (Verified on Windows XP, 2000)
105 * Linux/Unix (GTK2) (Thanks To Toni Brkic And Robin Dunn)
106 * Mac OSX (Thanks To Robin Dunn For The CaptionBar Size Patch)
109 Latest Revision: Andrea Gavana @ 30 Mar 2005, 22.30 CET
115 #----------------------------------------------------------------------
116 # Collapsed And Expanded Bitmap Images
117 # Created With img2py.py
118 #----------------------------------------------------------------------
120 def GetCollapsedIconData():
122 '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\
123 \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\
124 \x00\x007IDAT8\x8dcddbf\xa0\x040Q\xa4{\xf0\x1b\xf0\xff\xdf\xdf\xff\x03\xe7\
125 \x02\x98\xed\x84\\A\x1b\x17\xa0\xdb\x8a\xcf\x15\xd4w\x01.\xdbp\x89S\xec\x02\
126 \xc6\xd1\xbc\xc0\x00\x00\x9a\xf5\x1b\xfa\xf9m$?\x00\x00\x00\x00IEND\xaeB`\
129 def GetCollapsedIconBitmap():
130 return wx
.BitmapFromImage(GetCollapsedIconImage())
132 def GetCollapsedIconImage():
134 stream
= cStringIO
.StringIO(GetCollapsedIconData())
135 return wx
.ImageFromStream(stream
)
137 #----------------------------------------------------------------------
138 def GetExpandedIconData():
140 '\x89PNG\r\n\x1a\n\x00\x00\x00\rIHDR\x00\x00\x00\x10\x00\x00\x00\x10\x08\x06\
141 \x00\x00\x00\x1f\xf3\xffa\x00\x00\x00\x04sBIT\x08\x08\x08\x08|\x08d\x88\x00\
142 \x00\x00BIDAT8\x8dcddbf\xa0\x040Q\xa4{P\x18\xc0\x82.\xf0\xff\xdf\xdf\xff\xb8\
143 \x143213R\xdd\x05\x18\x06`\xb3\x05\x9f8m\x02\x11\xdd6\\\xb6\xd3\xce\x05\xc8\
144 \xb6\xe2\xb3\x9d*.`\x1c\xcd\x0b\x0c\x00\x9e\xbc\x04W\x19\xcfa\xb5\x00\x00\
145 \x00\x00IEND\xaeB`\x82'
147 def GetExpandedIconBitmap():
148 return wx
.BitmapFromImage(GetExpandedIconImage())
150 def GetExpandedIconImage():
152 stream
= cStringIO
.StringIO(GetExpandedIconData())
153 return wx
.ImageFromStream(stream
)
155 #----------------------------------------------------------------------
157 #----------------------------------------------------------------------
158 # FOLDPANELBAR Starts Here
159 #----------------------------------------------------------------------
163 #- CAPTIONBAR_GRADIENT_V: Draws a vertical gradient from top to bottom
164 #- CAPTIONBAR_GRADIENT_H: Draws a horizontal gradient from left to right
165 #- CAPTIONBAR_SINGLE: Draws a single filled rectangle to draw the caption
166 #- CAPTIONBAR_RECTANGLE: Draws a single colour with a rectangle around the caption
167 #- CAPTIONBAR_FILLED_RECTANGLE: Draws a filled rectangle and a border around it
169 CAPTIONBAR_NOSTYLE
= 0
170 CAPTIONBAR_GRADIENT_V
= 1
171 CAPTIONBAR_GRADIENT_H
= 2
172 CAPTIONBAR_SINGLE
= 3
173 CAPTIONBAR_RECTANGLE
= 4
174 CAPTIONBAR_FILLED_RECTANGLE
= 5
179 # pixels of the bmp to be aligned from the right filled with space
180 FPB_BMP_RIGHTSPACE
= 2
182 # Not yet supported but added for future reference. Single fold forces
183 # other panels to close when they are open, and only opens the current panel.
184 # This will allow the open panel to gain the full size left in the client area
185 FPB_SINGLE_FOLD
= 0x0001
187 # All panels are stacked to the bottom. When they are expanded again they
189 FPB_COLLAPSE_TO_BOTTOM
= 0x0002
191 # Not yet supported, but added for future reference. Single fold plus panels
192 # will be stacked at the bottom
193 FPB_EXCLUSIVE_FOLD
= FPB_SINGLE_FOLD | FPB_COLLAPSE_TO_BOTTOM
196 FPB_HORIZONTAL
= wx
.HORIZONTAL
197 FPB_VERTICAL
= wx
.VERTICAL
199 # Default Extrastyle of the FoldPanelBar
200 FPB_DEFAULT_EXTRASTYLE
= 0
201 # Default style of the FoldPanelBar
202 FPB_DEFAULT_STYLE
= wx
.TAB_TRAVERSAL | wx
.NO_BORDER
204 # FoldPanelItem default settings
208 FPB_DEFAULT_LEFTSPACING
= 5
209 FPB_DEFAULT_RIGHTSPACING
= 10
210 FPB_DEFAULT_SPACING
= 8
212 FPB_DEFAULT_LEFTLINESPACING
= 2
213 FPB_DEFAULT_RIGHTLINESPACING
= 2
216 # ------------------------------------------------------------------------------ #
217 # class CaptionBarStyle
218 # ------------------------------------------------------------------------------ #
220 class CaptionBarStyle
:
222 This class encapsulates the styles you wish to set for the
223 `CaptionBar` (this is the part of the FoldPanel where the caption
224 is displayed). It can either be applied at creation time be
225 reapplied when styles need to be changed.
227 At construction time, all styles are set to their default
228 transparency. This means none of the styles will be applied to
229 the `CaptionBar` in question, meaning it will be created using the
230 default internals. When setting i.e the color, font or panel
231 style, these styles become active to be used.
236 """ Default constructor for this class."""
241 def ResetDefaults(self
):
242 """ Resets default CaptionBarStyle."""
243 self
._firstColourUsed
= False
244 self
._secondColourUsed
= False
245 self
._textColourUsed
= False
246 self
._captionFontUsed
= False
247 self
._captionStyleUsed
= False
248 self
._captionStyle
= CAPTIONBAR_GRADIENT_V
251 # ------- CaptionBar Font -------
253 def SetCaptionFont(self
, font
):
255 Sets font for the caption bar.
257 If this is not set, the font property is undefined and will
258 not be used. Use `CaptionFontUsed` to check if this style is
261 self
._captionFont
= font
262 self
._captionFontUsed
= True
265 def CaptionFontUsed(self
):
266 """ Checks if the caption bar font is set. """
267 return self
._captionFontUsed
270 def GetCaptionFont(self
):
272 Returns the font for the caption bar.
274 Please be warned this will result in an assertion failure when
275 this property is not previously set.
277 :see: `SetCaptionFont`, `CaptionFontUsed`
279 return self
._captionFont
282 # ------- First Colour -------
284 def SetFirstColour(self
, colour
):
286 Sets first colour for the caption bar.
288 If this is not set, the colour property is undefined and will
289 not be used. Use `FirstColourUsed` to check if this style is
292 self
._firstColour
= colour
293 self
._firstColourUsed
= True
296 def FirstColourUsed(self
):
297 """ Checks if the first colour of the caption bar is set."""
298 return self
._firstColourUsed
301 def GetFirstColour(self
):
303 Returns the first colour for the caption bar.
305 Please be warned this will result in an assertion failure when
306 this property is not previously set.
308 :see: `SetFirstColour`, `FirstColourUsed`
310 return self
._firstColour
313 # ------- Second Colour -------
315 def SetSecondColour(self
, colour
):
317 Sets second colour for the caption bar.
319 If this is not set, the colour property is undefined and will
320 not be used. Use `SecondColourUsed` to check if this style is
323 self
._secondColour
= colour
324 self
._secondColourUsed
= True
327 def SecondColourUsed(self
):
328 """ Checks if the second colour of the caption bar is set."""
329 return self
._secondColourUsed
332 def GetSecondColour(self
):
334 Returns the second colour for the caption bar.
336 Please be warned this will result in an assertion failure when
337 this property is not previously set.
339 :see: `SetSecondColour`, `SecondColourUsed`
341 return self
._secondColour
344 # ------- Caption Text Colour -------
346 def SetCaptionColour(self
, colour
):
348 Sets caption colour for the caption bar.
350 If this is not set, the colour property is undefined and will
351 not be used. Use `CaptionColourUsed` to check if this style is
354 self
._textColour
= colour
355 self
._textColourUsed
= True
358 def CaptionColourUsed(self
):
359 """ Checks if the caption colour of the caption bar is set."""
360 return self
._textColourUsed
363 def GetCaptionColour(self
):
365 Returns the caption colour for the caption bar.
367 Please be warned this will result in an assertion failure
368 when this property is not previously set.
369 See also SetCaptionColour(), CaptionColourUsed()
371 return self
._textColour
374 # ------- CaptionStyle -------
376 def SetCaptionStyle(self
, style
):
378 Sets caption style for the caption bar.
380 If this is not set, the property is undefined and will not be
381 used. Use CaptionStyleUsed() to check if this style is used.
382 The following styles can be applied:
384 * CAPTIONBAR_GRADIENT_V: Draws a vertical gradient from top to bottom
386 * CAPTIONBAR_GRADIENT_H: Draws a horizontal gradient from
389 * CAPTIONBAR_SINGLE: Draws a single filled rectangle to
392 * CAPTIONBAR_RECTANGLE: Draws a single colour with a
393 rectangle around the caption
395 * CAPTIONBAR_FILLED_RECTANGLE: Draws a filled rectangle
396 and a border around it
399 self
._captionStyle
= style
400 self
._captionStyleUsed
= True
403 def CaptionStyleUsed(self
):
404 """ Checks if the caption style of the caption bar is set."""
405 return self
._captionStyleUsed
408 def GetCaptionStyle(self
):
410 Returns the caption style for the caption bar.
412 Please be warned this will result in an assertion failure
413 when this property is not previously set.
415 :see: `SetCaptionStyle`, `CaptionStyleUsed`
417 return self
._captionStyle
420 #-----------------------------------#
422 #-----------------------------------#
423 wxEVT_CAPTIONBAR
= wx
.NewEventType()
424 EVT_CAPTIONBAR
= wx
.PyEventBinder(wxEVT_CAPTIONBAR
, 0)
427 # ---------------------------------------------------------------------------- #
428 # class CaptionBarEvent
429 # ---------------------------------------------------------------------------- #
431 class CaptionBarEvent(wx
.PyCommandEvent
):
433 This event will be sent when a EVT_CAPTIONBAR is mapped in the parent.
434 It is to notify the parent that the bar is now in collapsed or expanded
435 state. The parent should re-arrange the associated windows accordingly
437 def __init__(self
, evtType
):
438 """ Default Constructor For This Class."""
439 wx
.PyCommandEvent
.__init
__(self
, evtType
)
442 def GetFoldStatus(self
):
444 Returns whether the bar is expanded or collapsed. True means
447 return not self
._bar
.IsCollapsed()
451 """ Returns The CaptionBar Selected."""
455 def SetTag(self
, tag
):
456 """ Assign A Tag To The Selected CaptionBar."""
461 """ Returns The Tag Assigned To The Selected CaptionBar."""
465 def SetBar(self
, bar
):
467 Sets the bar associated with this event.
469 Should not used by any other then the originator of the event.
474 # -------------------------------------------------------------------------------- #
476 # -------------------------------------------------------------------------------- #
478 class CaptionBar(wx
.Window
):
480 This class is a graphical caption component that consists of a
481 caption and a clickable arrow.
483 The CaptionBar fires an event EVT_CAPTIONBAR which is a
484 `CaptionBarEvent`. This event can be caught and the parent window
485 can act upon the collapsed or expanded state of the bar (which is
486 actually just the icon which changed). The parent panel can
487 reduce size or expand again.
490 # Define Empty CaptionBar Style
491 EmptyCaptionBarStyle
= CaptionBarStyle()
493 def __init__(self
, parent
, id, pos
, size
, caption
="",
494 foldIcons
=None, cbstyle
=EmptyCaptionBarStyle
,
495 rightIndent
=FPB_BMP_RIGHTSPACE
,
496 iconWidth
=16, iconHeight
=16, collapsed
=False):
497 """ Default Class Constructor."""
499 wx
.Window
.__init
__(self
, parent
, wx
.ID_ANY
, pos
=wx
.DefaultPosition
,
500 size
=(20,20), style
=wx
.NO_BORDER
)
502 self
._controlCreated
= False
503 self
._collapsed
= collapsed
504 self
.ApplyCaptionStyle(cbstyle
, True)
506 if foldIcons
is None:
507 foldIcons
= wx
.ImageList(16, 16)
509 bmp
= GetExpandedIconBitmap()
511 bmp
= GetCollapsedIconBitmap()
516 assert foldIcons
.GetImageCount() > 1
517 iconWidth
, iconHeight
= foldIcons
.GetSize(0)
519 self
._caption
= caption
520 self
._foldIcons
= foldIcons
521 self
._style
= cbstyle
522 self
._rightIndent
= rightIndent
523 self
._iconWidth
= iconWidth
524 self
._iconHeight
= iconHeight
525 self
._oldSize
= wx
.Size(20,20)
527 self
._controlCreated
= True
529 self
.Bind(wx
.EVT_PAINT
, self
.OnPaint
)
530 self
.Bind(wx
.EVT_SIZE
, self
.OnSize
)
531 self
.Bind(wx
.EVT_MOUSE_EVENTS
, self
.OnMouseEvent
)
532 self
.Bind(wx
.EVT_CHAR
, self
.OnChar
)
535 def ApplyCaptionStyle(self
, cbstyle
=EmptyCaptionBarStyle
, applyDefault
=True):
536 """ Applies the style defined in cbstyle to the CaptionBar."""
542 # get first colour from style or make it default
543 if not newstyle
.FirstColourUsed():
544 newstyle
.SetFirstColour(wx
.WHITE
)
546 # get second colour from style or make it default
547 if not newstyle
.SecondColourUsed():
548 # make the second colour slightly darker then the background
549 color
= self
.GetParent().GetBackgroundColour()
550 r
, g
, b
= int(color
.Red()), int(color
.Green()), int(color
.Blue())
551 color
= ((r
>> 1) + 20, (g
>> 1) + 20, (b
>> 1) + 20)
552 newstyle
.SetSecondColour(wx
.Colour(color
[0], color
[1], color
[2]))
555 if not newstyle
.CaptionColourUsed():
556 newstyle
.SetCaptionColour(wx
.BLACK
)
559 if not newstyle
.CaptionFontUsed():
560 newstyle
.SetCaptionFont(self
.GetParent().GetFont())
562 # apply caption style
563 if not newstyle
.CaptionStyleUsed():
564 newstyle
.SetCaptionStyle(CAPTIONBAR_GRADIENT_V
)
566 self
._style
= newstyle
569 def SetCaptionStyle(self
, cbstyle
=EmptyCaptionBarStyle
, applyDefault
=True):
571 Sets CaptionBar styles with CapionBarStyle class.
573 All styles that are actually set, are applied. If you set
574 applyDefault to True, all other (not defined) styles will be
575 set to default. If it is False, the styles which are not set
576 in the CaptionBarStyle will be ignored.
578 self
.ApplyCaptionStyle(cbstyle
, applyDefault
)
582 def GetCaptionStyle(self
):
584 Returns the current style of the captionbar in a
585 `CaptionBarStyle` class.
587 This can be used to change and set back the changes.
592 def IsCollapsed(self
):
594 Returns wether the status of the bar is expanded or collapsed.
596 return self
._collapsed
599 def SetRightIndent(self
, pixels
):
601 Sets the amount of pixels on the right from which the bitmap
604 If this is 0, it will be drawn all the way to the right,
605 default is equal to FPB_BMP_RIGHTSPACE. Assign this before
606 assigning an image list to prevent a redraw.
609 self
._rightIndent
= pixels
616 This sets the internal state / representation to collapsed.
618 This does not trigger a `CaptionBarEvent` to be sent to the
621 self
._collapsed
= True
622 self
.RedrawIconBitmap()
627 This sets the internal state / representation to expanded.
629 This does not trigger a `CaptionBarEvent` to be sent to the
632 self
._collapsed
= False
633 self
.RedrawIconBitmap()
636 def SetBoldFont(self
):
637 """ Sets the CaptionBarFont weight to BOLD."""
639 self
.GetFont().SetWeight(wx
.BOLD
)
642 def SetNormalFont(self
):
643 """ Sets the CaptionBarFont weight to NORMAL."""
645 self
.GetFont().SetWeight(wx
.NORMAL
)
648 def IsVertical(self
):
650 Returns wether the CaptionBar Has Default Orientation Or Not.
655 fld
= self
.GetParent().GetGrandParent()
656 if isinstance(fld
, FoldPanelBar
):
657 return self
.GetParent().GetGrandParent().IsVertical()
659 raise "ERROR: Wrong Parent " + repr(fld
)
662 def OnPaint(self
, event
):
663 """ The paint event for flat or gradient fill. """
665 if not self
._controlCreated
:
669 dc
= wx
.PaintDC(self
)
670 wndRect
= self
.GetRect()
671 vertical
= self
.IsVertical()
673 # TODO: Maybe first a memory DC should draw all, and then paint it on
674 # the caption. This way a flickering arrow during resize is not visible
676 self
.FillCaptionBackground(dc
)
677 dc
.SetFont(self
._style
.GetCaptionFont())
680 dc
.DrawText(self
._caption
, 4, FPB_EXTRA_Y
/2)
682 dc
.DrawRotatedText(self
._caption
, FPB_EXTRA_Y
/2,
683 wndRect
.GetBottom() - 4, 90)
685 # draw small icon, either collapsed or expanded
686 # based on the state of the bar. If we have any bmp's
690 index
= self
._collapsed
693 drw
= wndRect
.GetRight() - self
._iconWidth
- self
._rightIndent
694 self
._foldIcons
.Draw(index
, dc
, drw
,
695 (wndRect
.GetHeight() - self
._iconHeight
)/2,
696 wx
.IMAGELIST_DRAW_TRANSPARENT
)
698 self
._foldIcons
.Draw(index
, dc
,
699 (wndRect
.GetWidth() - self
._iconWidth
)/2,
700 self
._rightIndent
, wx
.IMAGELIST_DRAW_TRANSPARENT
)
705 def FillCaptionBackground(self
, dc
):
707 Fills the background of the caption with either a gradient or
711 style
= self
._style
.GetCaptionStyle()
713 if style
== CAPTIONBAR_GRADIENT_V
:
714 if self
.IsVertical():
715 self
.DrawVerticalGradient(dc
, self
.GetRect())
717 self
.DrawHorizontalGradient(dc
, self
.GetRect())
719 elif style
== CAPTIONBAR_GRADIENT_H
:
720 if self
.IsVertical():
721 self
.DrawHorizontalGradient(dc
, self
.GetRect())
723 self
.DrawVerticalGradient(dc
, self
.GetRect())
725 elif style
== CAPTIONBAR_SINGLE
:
726 self
.DrawSingleColour(dc
, self
.GetRect())
727 elif style
== CAPTIONBAR_RECTANGLE
or style
== CAPTIONBAR_FILLED_RECTANGLE
:
728 self
.DrawSingleRectangle(dc
, self
.GetRect())
730 raise "STYLE Error: Undefined Style Selected: " + repr(style
)
733 def OnMouseEvent(self
, event
):
735 Catches the mouse click-double click.
737 If clicked on the arrow (single) or double on the caption we
738 change state and an event must be fired to let this panel
744 if event
.LeftDown() and self
._foldIcons
:
746 pt
= event
.GetPosition()
747 rect
= self
.GetRect()
748 vertical
= self
.IsVertical()
750 drw
= (rect
.GetWidth() - self
._iconWidth
- self
._rightIndent
)
751 if vertical
and pt
.x
> drw
or not vertical
and \
752 pt
.y
< (self
._iconHeight
+ self
._rightIndent
):
755 elif event
.LeftDClick():
758 # send the collapse, expand event to the parent
761 event
= CaptionBarEvent(wxEVT_CAPTIONBAR
)
763 self
.GetEventHandler().ProcessEvent(event
)
767 def OnChar(self
, event
):
768 """ Unused Methods. Any Ideas?!?"""
769 # TODO: Anything here?
773 def DoGetBestSize(self
):
775 Returns the best size for this panel, based upon the font
776 assigned to this window, and the caption string
779 if self
.IsVertical():
780 x
, y
= self
.GetTextExtent(self
._caption
)
782 y
, x
= self
.GetTextExtent(self
._caption
)
784 if x
< self
._iconWidth
:
787 if y
< self
._iconHeight
:
790 # TODO: The extra FPB_EXTRA_X constants should be adjustable as well
792 return wx
.Size(x
+ FPB_EXTRA_X
, y
+ FPB_EXTRA_Y
)
795 def DrawVerticalGradient(self
, dc
, rect
):
796 """ Gradient fill from colour 1 to colour 2 with top to bottom. """
798 if rect
.height
< 1 or rect
.width
< 1:
801 dc
.SetPen(wx
.TRANSPARENT_PEN
)
803 # calculate gradient coefficients
804 col2
= self
._style
.GetSecondColour()
805 col1
= self
._style
.GetFirstColour()
807 r1
, g1
, b1
= int(col1
.Red()), int(col1
.Green()), int(col1
.Blue())
808 r2
, g2
, b2
= int(col2
.Red()), int(col2
.Green()), int(col2
.Blue())
810 flrect
= float(rect
.height
)
812 rstep
= float((r2
- r1
)) / flrect
813 gstep
= float((g2
- g1
)) / flrect
814 bstep
= float((b2
- b1
)) / flrect
818 for y
in range(rect
.y
, rect
.y
+ rect
.height
):
819 currCol
= (r1
+ rf
, g1
+ gf
, b1
+ bf
)
821 dc
.SetBrush(wx
.Brush(currCol
, wx
.SOLID
))
822 dc
.DrawRectangle(rect
.x
, rect
.y
+ (y
- rect
.y
), rect
.width
, rect
.height
)
828 def DrawHorizontalGradient(self
, dc
, rect
):
829 """ Gradient fill from colour 1 to colour 2 with left to right. """
831 if rect
.height
< 1 or rect
.width
< 1:
834 dc
.SetPen(wx
.TRANSPARENT_PEN
)
836 # calculate gradient coefficients
837 col2
= self
._style
.GetSecondColour()
838 col1
= self
._style
.GetFirstColour()
840 r1
, g1
, b1
= int(col1
.Red()), int(col1
.Green()), int(col1
.Blue())
841 r2
, g2
, b2
= int(col2
.Red()), int(col2
.Green()), int(col2
.Blue())
843 flrect
= float(rect
.width
)
845 rstep
= float((r2
- r1
)) / flrect
846 gstep
= float((g2
- g1
)) / flrect
847 bstep
= float((b2
- b1
)) / flrect
851 for x
in range(rect
.x
, rect
.x
+ rect
.width
):
852 currCol
= (r1
+ rf
, g1
+ gf
, b1
+ bf
)
854 dc
.SetBrush(wx
.Brush(currCol
, wx
.SOLID
))
855 dc
.DrawRectangle(rect
.x
+ (x
- rect
.x
), rect
.y
, 1, rect
.height
)
861 def DrawSingleColour(self
, dc
, rect
):
862 """ Single colour fill. This is the most easy one to find. """
864 if rect
.height
< 1 or rect
.width
< 1:
867 dc
.SetPen(wx
.TRANSPARENT_PEN
)
869 # draw simple rectangle
870 dc
.SetBrush(wx
.Brush(self
._style
.GetFirstColour(), wx
.SOLID
))
871 dc
.DrawRectangle(rect
.x
, rect
.y
, rect
.width
, rect
.height
)
874 def DrawSingleRectangle(self
, dc
, rect
):
875 """ Single rectangle. This is the most easy one to find. """
877 if rect
.height
< 2 or rect
.width
< 1:
880 # single frame, set up internal fill colour
882 if self
._style
.GetCaptionStyle() == CAPTIONBAR_RECTANGLE
:
883 color
= self
.GetParent().GetBackgroundColour()
884 br
= wx
.Brush(color
, wx
.SOLID
)
886 color
= self
._style
.GetFirstColour()
887 br
= wx
.Brush(color
, wx
.SOLID
)
889 # setup the pen frame
891 pen
= wx
.Pen(self
._style
.GetSecondColour())
894 dc
.DrawRectangle(rect
.x
, rect
.y
, rect
.width
, rect
.height
- 1)
896 bgpen
= wx
.Pen(self
.GetParent().GetBackgroundColour())
898 dc
.DrawLine(rect
.x
, rect
.y
+ rect
.height
- 1, rect
.x
+ rect
.width
,
899 rect
.y
+ rect
.height
- 1)
902 def OnSize(self
, event
):
903 """ Handles the size events for the CaptionBar."""
905 if not self
._controlCreated
:
909 size
= event
.GetSize()
913 # What I am doing here is simply invalidating the part of the window
914 # exposed. So when I make a rect with as width the newly exposed part,
915 # and the x,y of the old window size origin, I don't need a bitmap
916 # calculation in it, or do I ? The bitmap needs redrawing anyway.
917 # Leave it like this until I figured it out.
919 # set rect to redraw as old bitmap area which is entitled to redraw
921 rect
= wx
.Rect(size
.GetWidth() - self
._iconWidth
- self
._rightIndent
, 0,
922 self
._iconWidth
+ self
._rightIndent
,
923 self
._iconWidth
+ self
._rightIndent
)
925 # adjust rectangle when more is slided so we need to redraw all
926 # the old stuff but not all (ugly flickering)
928 diffX
= size
.GetWidth() - self
._oldSize
.GetWidth()
932 # adjust the rect with all the crap to redraw
934 rect
.SetWidth(rect
.GetWidth() + diffX
+ 10)
935 rect
.SetX(rect
.GetX() - diffX
- 10)
937 self
.RefreshRect(rect
)
941 rect
= self
.GetRect()
942 self
.RefreshRect(rect
)
947 def RedrawIconBitmap(self
):
948 """ Redraws the icons (if they exists). """
952 # invalidate the bitmap area and force a redraw
954 rect
= self
.GetRect()
956 rect
.SetX(rect
.GetWidth() - self
._iconWidth
- self
._rightIndent
)
957 rect
.SetWidth(self
._iconWidth
+ self
._rightIndent
)
958 self
.RefreshRect(rect
)
961 # ---------------------------------------------------------------------------------- #
963 # ---------------------------------------------------------------------------------- #
965 class FoldPanelBar(wx
.Panel
):
967 The FoldPanelBar is a class which can maintain a list of
968 collapsable panels. Once a panel is collapsed, only it's caption
969 bar is visible to the user. This will provide more space for the
970 other panels, or allow the user to close panels which are not used
971 often to get the most out of the work area.
973 This control is easy to use. Simply create it as a child for a
974 panel or sash window, and populate panels with
975 `AddFoldPanel`. Then use the AdddFoldPanelWindow` to add
976 `wx.Window` derived controls to the current fold panel. Use
977 `AddFoldPanelSeparator` to put separators between the groups of
978 controls that need a visual separator to group them
979 together. After all is constructed, the user can fold the panels
980 by doubleclicking on the bar or single click on the arrow, which
981 will indicate the collapsed or expanded state.
983 # Define Empty CaptionBar Style
984 EmptyCaptionBarStyle
= CaptionBarStyle()
986 def __init__(self
, parent
, id=-1, pos
=wx
.DefaultPosition
, size
=wx
.DefaultSize
,
987 style
=FPB_DEFAULT_STYLE
, extraStyle
=FPB_DEFAULT_EXTRASTYLE
):
988 """ Default Class Constructor. """
990 self
._controlCreated
= False
991 self
._extraStyle
= extraStyle
993 # make sure there is any orientation
994 if style
& FPB_HORIZONTAL
!= FPB_HORIZONTAL
:
995 style
= style | FPB_VERTICAL
997 if style
& FPB_HORIZONTAL
== 4:
998 self
._isVertical
= False
1000 self
._isVertical
= True
1003 # create the panel (duh!). This causes a size event, which we are going
1004 # to skip when we are not initialised
1006 wx
.Panel
.__init
__(self
, parent
, id, pos
, size
, style
)
1008 # the fold panel area
1010 self
._foldPanel
= wx
.Panel(self
, wx
.ID_ANY
, pos
, size
,
1011 wx
.NO_BORDER | wx
.TAB_TRAVERSAL
)
1013 self
._controlCreated
= True
1016 self
.Bind(EVT_CAPTIONBAR
, self
.OnPressCaption
)
1017 self
.Bind(wx
.EVT_SIZE
, self
.OnSizePanel
)
1020 def AddFoldPanel(self
, caption
="", collapsed
=False, foldIcons
=None,
1021 cbstyle
=EmptyCaptionBarStyle
):
1023 Adds a fold panel to the list of panels.
1025 If the flag collapsed is set to True, the panel is collapsed
1026 initially. The FoldPanel item which is returned, can be used
1027 as a reference to perform actions upon the fold panel like
1028 collapsing it, expanding it, or deleting it from the list.
1030 Use this foldpanel to add windows to it. Please consult
1031 `AddFoldPanelWindow` and `AddFoldPanelSeparator` to know how
1032 to add items derived from `wx.Window` to the panels.
1035 # create a fold panel item, which is first only the caption.
1036 # the user can now add a panel area which will be folded in
1039 if foldIcons
is None:
1040 foldIcons
= wx
.ImageList(16, 16)
1042 bmp
= GetExpandedIconBitmap()
1044 bmp
= GetCollapsedIconBitmap()
1047 item
= FoldPanelItem(self
._foldPanel
, -1, caption
=caption
,
1048 foldIcons
=foldIcons
,
1049 collapsed
=collapsed
, cbstyle
=cbstyle
)
1052 if len(self
._panels
) > 0:
1053 pos
= self
._panels
[-1].GetItemPos() + self
._panels
[-1].GetPanelLength()
1055 item
.Reposition(pos
)
1056 self
._panels
.append(item
)
1061 def AddFoldPanelWindow(self
, panel
, window
, flags
=FPB_ALIGN_WIDTH
,
1062 Spacing
=FPB_DEFAULT_SPACING
,
1063 leftSpacing
=FPB_DEFAULT_LEFTLINESPACING
,
1064 rightSpacing
=FPB_DEFAULT_RIGHTLINESPACING
):
1066 Adds a `wx.Window` derived instance to the referenced
1069 IMPORTANT: Make the window be a child of the FoldPanel. See
1070 example that follows. The flags to be used are:
1072 * FPB_ALIGN_WIDTH: Which means the wxWindow to be added
1073 will be aligned to fit the width of the FoldPanel when
1074 it is resized. Very handy for sizer items, buttons and
1077 * FPB_ALIGN_LEFT: Alligns left instead of fitting the
1078 width of the child window to be added. Use either this
1079 one or FPB_ALIGN_WIDTH.
1081 The wx.Window to be added can be slightly indented from left
1082 and right so it is more visibly placed in the FoldPanel. Use
1083 Spacing > 0 to give the control an y offset from the previous
1084 wx.Window added, use leftSpacing to give it a slight indent
1085 from the left, and rightSpacing also reserves a little space
1086 on the right so the wxWindow can be properly placed in the
1089 The following example adds a FoldPanel to the FoldPanelBar and
1090 adds two wx.Window derived controls to the FoldPanel::
1092 # create the FoldPanelBar
1093 >>> m_pnl = FoldPanelBar(self, wx.ID_ANY, wx.DefaultPosition,
1094 wx.DefaultSize, FPB_DEFAULT_STYLE,
1095 FPB_COLLAPSE_TO_BOTTOM)
1097 # add a foldpanel to the control. "Test me" is the caption and it is
1098 # initially not collapsed.
1099 >>> item = m_pnl.AddFoldPanel("Test me", False)
1101 # now add a button to the fold panel. Mind that the button should be
1102 # made child of the FoldPanel and not of the main form.
1103 >>> m_pnl.AddFoldPanelWindow(item, wx.Button(item, ID_COLLAPSEME,
1106 # add a separator between the two controls. This is purely a visual
1107 # line that can have a certain color and also the indents and width
1108 # aligning like a control.
1109 >>> m_pnl.AddFoldPanelSeparator(item)
1111 # now add a text ctrl. Also very easy. Align this on width so that
1112 # when the control gets wider the text control also sizes along.
1113 >>> m_pnl.AddFoldPanelWindow(item, wx.TextCtrl(item, wx.ID_ANY, "Comment"),
1114 FPB_ALIGN_WIDTH, FPB_DEFAULT_SPACING, 20)
1119 item
= self
._panels
.index(panel
)
1121 raise "ERROR: Invalid Panel Passed To AddFoldPanelWindow: " + repr(panel
)
1123 panel
.AddWindow(window
, flags
, Spacing
, leftSpacing
, rightSpacing
)
1125 # TODO: Take old and new height, and if difference, reposition all the lower
1126 # panels this is because the user can add new wxWindow controls somewhere in
1127 # between when other panels are already present.
1132 def AddFoldPanelSeparator(self
, panel
, colour
=wx
.BLACK
,
1133 Spacing
=FPB_DEFAULT_SPACING
,
1134 leftSpacing
=FPB_DEFAULT_LEFTLINESPACING
,
1135 rightSpacing
=FPB_DEFAULT_RIGHTLINESPACING
):
1137 Adds a separator line to the current FoldPanel.
1139 The seperator is a simple line which is drawn and is no real
1140 component. It can be used to separate groups of controls
1141 which belong to each other. The colour is adjustable, and it
1142 takes the same Spacing, leftSpacing and rightSpacing as
1143 `AddFoldPanelWindow`.
1147 item
= self
._panels
.index(panel
)
1149 raise "ERROR: Invalid Panel Passed To AddFoldPanelSeparator: " + repr(panel
)
1151 panel
.AddSeparator(colour
, Spacing
, leftSpacing
, rightSpacing
)
1155 def OnSizePanel(self
, event
):
1156 """ Handles the EVT_SIZE event for the FoldPanelBar. """
1158 # skip all stuff when we are not initialised yet
1160 if not self
._controlCreated
:
1164 foldrect
= self
.GetRect()
1166 # fold panel itself. If too little space,
1172 self
._foldPanel
.SetSize(foldrect
[2:])
1174 if self
._extraStyle
& FPB_COLLAPSE_TO_BOTTOM
:
1175 rect
= self
.RepositionCollapsedToBottom()
1176 vertical
= self
.IsVertical()
1177 if vertical
and rect
.GetHeight() > 0 or not vertical
and rect
.GetWidth() > 0:
1178 self
.RefreshRect(rect
)
1180 # TODO: A smart way to check wether the old - new width of the
1181 # panel changed, if so no need to resize the fold panel items
1183 self
.RedisplayFoldPanelItems()
1186 def OnPressCaption(self
, event
):
1187 """ Handles the EVT_CAPTIONBAR event in the FoldPanelBar. """
1189 # act upon the folding or expanding status of the bar
1190 # to expand or collapse the panel(s)
1192 if event
.GetFoldStatus():
1193 self
.Collapse(event
.GetTag())
1195 self
.Expand(event
.GetTag())
1200 def RefreshPanelsFrom(self
, item
):
1201 """ Refreshes all the panels from given index down to last one. """
1204 i
= self
._panels
.index(item
)
1206 raise "ERROR: Invalid Panel Passed To RefreshPanelsFrom: " + repr(item
)
1210 # if collapse to bottom is on, the panels that are not expanded
1211 # should be drawn at the bottom. All panels that are expanded
1212 # are drawn on top. The last expanded panel gets all the extra space
1214 if self
._extraStyle
& FPB_COLLAPSE_TO_BOTTOM
:
1218 for panels
in self
._panels
:
1220 if panels
.IsExpanded():
1221 offset
= offset
+ panels
.Reposition(offset
)
1223 # put all non collapsed panels at the bottom where there is space,
1224 # else put them right behind the expanded ones
1226 self
.RepositionCollapsedToBottom()
1230 pos
= self
._panels
[i
].GetItemPos() + self
._panels
[i
].GetPanelLength()
1231 for j
in range(i
+1, len(self
._panels
)):
1232 pos
= pos
+ self
._panels
[j
].Reposition(pos
)
1237 def RedisplayFoldPanelItems(self
):
1238 """ Resizes the fold panels so they match the width. """
1239 # resize them all. No need to reposition
1240 for panels
in self
._panels
:
1241 panels
.ResizePanel()
1245 def RepositionCollapsedToBottom(self
):
1247 Repositions all the collapsed panels to the bottom.
1249 When it is not possible to align them to the bottom, stick
1250 them behind the visible panels. The Rect holds the slack area
1251 left between last repositioned panel and the bottom
1252 panels. This needs to get a refresh.
1255 value
= wx
.Rect(0,0,0,0)
1256 vertical
= self
.IsVertical()
1258 # determine wether the number of panels left
1259 # times the size of their captions is enough
1260 # to be placed in the left over space
1264 collapsed
, expanded
, values
= self
.GetPanelsLength(collapsed
, expanded
)
1266 # if no room stick them behind the normal ones, else
1269 if (vertical
and [self
.GetSize().GetHeight()] or \
1270 [self
.GetSize().GetWidth()])[0] - expanded
- collapsed
< 0:
1274 # value is the region which is left unpainted
1275 # I will send it back as 'slack' so it does not need to
1278 value
.SetHeight(self
.GetSize().GetHeight())
1279 value
.SetWidth(self
.GetSize().GetWidth())
1282 value
.SetY(expanded
)
1283 value
.SetHeight(value
.GetHeight() - expanded
)
1285 value
.SetX(expanded
)
1286 value
.SetWidth(value
.GetWidth() - expanded
)
1288 offset
= (vertical
and [self
.GetSize().GetHeight()] or \
1289 [self
.GetSize().GetWidth()])[0] - collapsed
1294 for panels
in self
._panels
:
1295 if not panels
.IsExpanded():
1296 offset
= offset
+ panels
.Reposition(offset
)
1301 def GetPanelsLength(self
, collapsed
, expanded
):
1303 Returns the length of the panels that are expanded and
1306 This is useful to determine quickly what size is used to
1307 display, and what is left at the bottom (right) to align the
1313 # assumed here that all the panels that are expanded
1314 # are positioned after each other from 0,0 to end.
1316 for j
in range(0, len(self
._panels
)):
1317 offset
= self
._panels
[j
].GetPanelLength()
1318 value
= value
+ offset
1319 if self
._panels
[j
].IsExpanded():
1320 expanded
= expanded
+ offset
1322 collapsed
= collapsed
+ offset
1324 return collapsed
, expanded
, value
1327 def Collapse(self
, foldpanel
):
1329 Collapses the given FoldPanel reference, and updates the
1332 In the FPB_COLLAPSE_TO_BOTTOM style, all collapsed captions
1333 are put at the bottom of the control. In the normal mode, they
1334 stay where they are.
1338 item
= self
._panels
.index(foldpanel
)
1340 raise "ERROR: Invalid Panel Passed To Collapse: " + repr(foldpanel
)
1342 foldpanel
.Collapse()
1343 self
.RefreshPanelsFrom(foldpanel
)
1346 def Expand(self
, foldpanel
):
1348 Expands the given FoldPanel reference, and updates the
1351 In the FPB_COLLAPSE_TO_BOTTOM style, they will be removed from
1352 the bottom and the order where the panel originally was placed
1357 self
.RefreshPanelsFrom(foldpanel
)
1360 def ApplyCaptionStyle(self
, foldpanel
, cbstyle
):
1362 Sets the style of the caption bar (`CaptionBar`) of the
1365 The changes are applied immediately. All styles not set in the
1366 CaptionBarStyle class are not applied. Use the CaptionBar
1367 reference to indicate what captionbar you want to apply the
1368 style to. To apply one style to all CaptionBar items, use
1369 `ApplyCaptionStyleAll`
1371 foldpanel
.ApplyCaptionStyle(cbstyle
)
1374 def ApplyCaptionStyleAll(self
, cbstyle
):
1376 Sets the style of all the caption bars of the FoldPanel.
1378 The changes are applied immediately.
1380 for panels
in self
._panels
:
1381 self
.ApplyCaptionStyle(panels
, cbstyle
)
1384 def GetCaptionStyle(self
, foldpanel
):
1386 Returns the currently used caption style for the FoldPanel.
1388 It is returned as a CaptionBarStyle class. After modifying it,
1389 it can be set again.
1391 return foldpanel
.GetCaptionStyle()
1394 def IsVertical(self
):
1396 Returns whether the CaptionBar has default orientation or not.
1398 Default is vertical.
1400 return self
._isVertical
1403 def GetFoldPanel(self
, item
):
1405 Returns the panel associated with the index "item".
1407 See the example at the bottom of the module, especially the events
1408 for the "Collapse Me" and "Expand Me" buttons.
1411 ind
= self
._panels
[item
]
1412 return self
._panels
[item
]
1414 raise "ERROR: List Index Out Of Range Or Bad Item Passed: " + repr(item
) + \
1415 ". Item Should Be An Integer Between " + repr(0) + " And " + \
1416 repr(len(self
._panels
))
1420 """ Returns the number of panels in the FoldPanelBar. """
1423 return len(self
._panels
)
1425 raise "ERROR: No Panels Have Been Added To FoldPanelBar"
1429 # --------------------------------------------------------------------------------- #
1430 # class FoldPanelItem
1431 # --------------------------------------------------------------------------------- #
1433 class FoldPanelItem(wx
.Panel
):
1435 This class is a child sibling of the `FoldPanelBar` class. It will
1436 contain a `CaptionBar` class for receiving of events, and a the
1437 rest of the area can be populated by a `wx.Panel` derived class.
1439 # Define Empty CaptionBar Style
1440 EmptyCaptionBarStyle
= CaptionBarStyle()
1442 def __init__(self
, parent
, id=wx
.ID_ANY
, caption
="", foldIcons
=None,
1443 collapsed
=False, cbstyle
=EmptyCaptionBarStyle
):
1444 """ Default Class Constructor. """
1446 wx
.Panel
.__init
__(self
, parent
, id, style
=wx
.CLIP_CHILDREN
)
1447 self
._controlCreated
= False
1450 self
._LastInsertPos
= 0
1452 self
._userSized
= False
1454 if foldIcons
is None:
1455 foldIcons
= wx
.ImageList(16, 16)
1457 bmp
= GetExpandedIconBitmap()
1459 bmp
= GetCollapsedIconBitmap()
1462 self
._foldIcons
= foldIcons
1464 # create the caption bar, in collapsed or expanded state
1466 self
._captionBar
= CaptionBar(self
, wx
.ID_ANY
, wx
.Point(0,0),
1467 size
=wx
.DefaultSize
, caption
=caption
,
1468 foldIcons
=foldIcons
, cbstyle
=cbstyle
)
1471 self
._captionBar
.Collapse()
1473 self
._controlCreated
= True
1475 # make initial size for component, if collapsed, the
1476 # size is determined on the panel height and won't change
1478 size
= self
._captionBar
.GetSize()
1480 self
._PanelSize
= (self
.IsVertical() and [size
.GetHeight()] or \
1481 [size
.GetWidth()])[0]
1483 self
._LastInsertPos
= self
._PanelSize
1486 self
.Bind(EVT_CAPTIONBAR
, self
.OnPressCaption
)
1487 self
.Bind(wx
.EVT_PAINT
, self
.OnPaint
)
1490 def AddWindow(self
, window
, flags
=FPB_ALIGN_WIDTH
, Spacing
=FPB_DEFAULT_SPACING
,
1491 leftSpacing
=FPB_DEFAULT_LEFTLINESPACING
,
1492 rightSpacing
=FPB_DEFAULT_RIGHTLINESPACING
):
1494 Adds a window item to the list of items on this panel.
1496 The flags are FPB_ALIGN_LEFT for a non sizing window element,
1497 and FPB_ALIGN_WIDTH for a width aligned item. The Spacing
1498 parameter reserves a number of pixels before the window
1499 element, and leftSpacing is an indent. rightSpacing is only
1500 relevant when the style FPB_ALIGN_WIDTH is chosen.
1503 wi
= FoldWindowItem(self
, window
, Type
="WINDOW", flags
=flags
, Spacing
=Spacing
,
1504 leftSpacing
=leftSpacing
, rightSpacing
=rightSpacing
)
1506 self
._items
.append(wi
)
1508 vertical
= self
.IsVertical()
1510 self
._Spacing
= Spacing
1511 self
._leftSpacing
= leftSpacing
1512 self
._rightSpacing
= rightSpacing
1514 xpos
= (vertical
and [leftSpacing
] or [self
._LastInsertPos
+ Spacing
])[0]
1515 ypos
= (vertical
and [self
._LastInsertPos
+ Spacing
] or [leftSpacing
])[0]
1517 window
.SetDimensions(xpos
, ypos
, -1, -1, wx
.SIZE_USE_EXISTING
)
1519 self
._LastInsertPos
= self
._LastInsertPos
+ wi
.GetWindowLength(vertical
)
1523 def AddSeparator(self
, colour
=wx
.BLACK
, Spacing
=FPB_DEFAULT_SPACING
,
1524 leftSpacing
=FPB_DEFAULT_LEFTSPACING
,
1525 rightSpacing
=FPB_DEFAULT_RIGHTSPACING
):
1527 Adds a separator item to the list of items on this panel. """
1529 wi
= FoldWindowItem(self
, window
=None, Type
="SEPARATOR",
1530 flags
=FPB_ALIGN_WIDTH
, y
=self
._LastInsertPos
,
1531 colour
=colour
, Spacing
=Spacing
, leftSpacing
=leftSpacing
,
1532 rightSpacing
=rightSpacing
)
1534 self
._items
.append(wi
)
1535 self
._LastInsertPos
= self
._LastInsertPos
+ \
1536 wi
.GetWindowLength(self
.IsVertical())
1541 def Reposition(self
, pos
):
1543 Repositions this FoldPanelBar and reports the length occupied
1544 for the next FoldPanelBar in the list.
1546 # NOTE: Call Resize before Reposition when an item is added, because the new
1547 # size needed will be calculated by Resize. Of course the relative position
1548 # of the controls have to be correct in respect to the caption bar
1552 vertical
= self
.IsVertical()
1553 xpos
= (vertical
and [-1] or [pos
])[0]
1554 ypos
= (vertical
and [pos
] or [-1])[0]
1556 self
.SetDimensions(xpos
, ypos
, -1, -1, wx
.SIZE_USE_EXISTING
)
1561 return self
.GetPanelLength()
1564 def OnPressCaption(self
, event
):
1565 """ Handles the EVT_CAPTIONBAR event in the FoldPanelItem. """
1567 # tell the upper container we are responsible
1568 # for this event, so it can fold the panel item
1575 def ResizePanel(self
):
1576 """ Resizes the panel. """
1578 # prevent unnecessary updates by blocking repaints for a sec
1582 vertical
= self
.IsVertical()
1583 # force this panel to take the width of the parent panel and the y of the
1584 # user or calculated width (which will be recalculated by the contents here)
1587 if self
._captionBar
.IsCollapsed():
1588 size
= self
._captionBar
.GetSize()
1589 self
._PanelSize
= (vertical
and [size
.GetHeight()] or [size
.GetWidth()])[0]
1591 size
= self
.GetBestSize()
1592 self
._PanelSize
= (vertical
and [size
.GetHeight()] or [size
.GetWidth()])[0]
1596 size
.SetHeight(self
._UserSize
)
1598 size
.SetWidth(self
._UserSize
)
1600 pnlsize
= self
.GetParent().GetSize()
1603 size
.SetWidth(pnlsize
.GetWidth())
1605 size
.SetHeight(pnlsize
.GetHeight())
1607 # resize caption bar
1608 xsize
= (vertical
and [size
.GetWidth()] or [-1])[0]
1609 ysize
= (vertical
and [-1] or [size
.GetHeight()])[0]
1611 self
._captionBar
.SetSize((xsize
, ysize
))
1616 # go by all the controls and call Layout
1618 for items
in self
._items
:
1619 items
.ResizeItem((vertical
and [size
.GetWidth()] or \
1620 [size
.GetHeight()])[0], vertical
)
1625 def OnPaint(self
, event
):
1626 """ Handles the EVT_PAINT event in the FoldPanelItem. """
1628 # draw all the items that are lines
1630 dc
= wx
.PaintDC(self
)
1631 vertical
= self
.IsVertical()
1633 for item
in self
._items
:
1635 if item
.GetType() == "SEPARATOR":
1636 pen
= wx
.Pen(item
.GetLineColour(), 1, wx
.SOLID
)
1638 a
= item
.GetLeftSpacing()
1639 b
= item
.GetLineY() + item
.GetSpacing()
1640 c
= item
.GetLineLength()
1644 dc
.DrawLine(a
, b
, d
, b
)
1646 dc
.DrawLine(b
, a
, b
, d
)
1651 def IsVertical(self
):
1653 Returns wether the CaptionBar Has Default Orientation Or Not.
1655 Default is vertical.
1658 # grandparent of FoldPanelItem is FoldPanelBar
1659 # default is vertical
1661 if isinstance(self
.GetGrandParent(), FoldPanelBar
):
1662 return self
.GetGrandParent().IsVertical()
1664 raise "ERROR: Wrong Parent " + repr(self
.GetGrandParent())
1667 def IsExpanded(self
):
1669 Returns expanded or collapsed status. If the panel is
1670 expanded, True is returned.
1673 return not self
._captionBar
.IsCollapsed()
1676 def GetItemPos(self
):
1677 """ Returns item's position. """
1679 return self
._itemPos
1683 # this should not be called by the user, because it doesn't trigger the
1684 # parent to tell it that we are collapsed or expanded, it only changes
1687 self
._captionBar
.Collapse()
1692 # this should not be called by the user, because it doesn't trigger the
1693 # parent to tell it that we are collapsed or expanded, it only changes
1696 self
._captionBar
.Expand()
1700 def GetPanelLength(self
):
1701 """ Returns size of panel. """
1703 if self
._captionBar
.IsCollapsed():
1704 return self
.GetCaptionLength()
1705 elif self
._userSized
:
1706 return self
._UserSize
1708 return self
._PanelSize
1711 def GetCaptionLength(self
):
1713 Returns height of caption only. This is for folding
1714 calculation purposes.
1717 size
= self
._captionBar
.GetSize()
1718 return (self
.IsVertical() and [size
.GetHeight()] or [size
.GetWidth()])[0]
1721 def ApplyCaptionStyle(self
, cbstyle
):
1722 """ Applies the style defined in cbstyle to the CaptionBar."""
1724 self
._captionBar
.SetCaptionStyle(cbstyle
)
1727 def GetCaptionStyle(self
):
1729 Returns the current style of the captionbar in a
1730 CaptionBarStyle class.
1732 This can be used to change and set back the changes.
1735 return self
._captionBar
.GetCaptionStyle()
1738 # ----------------------------------------------------------------------------------- #
1739 # class FoldWindowItem
1740 # ----------------------------------------------------------------------------------- #
1742 class FoldWindowItem
:
1744 This class is a child sibling of the `FoldPanelItem` class. It
1745 will contain wx.Window that can be either a separator (a colored
1746 line simulated by a wx.Window) or a wxPython controls (such as a
1747 wx.Button, a wx.ListCtrl etc...).
1749 def __init__(self
, parent
, window
=None, **kw
):
1751 Default Class Constructor
1755 Type = "WINDOW", flags = FPB_ALIGN_WIDTH,
1756 Spacing = FPB_DEFAULT_SPACING,
1757 leftSpacing = FPB_DEFAULT_LEFTSPACING,
1758 rightSpacing = FPB_DEFAULT_RIGHTSPACING
1763 y, lineColor = wx.BLACK,
1764 flags = FPB_ALIGN_WIDTH,
1765 Spacing = FPB_DEFAULT_SPACING,
1766 leftSpacing = FPB_DEFAULT_LEFTLINESPACING,
1767 rightSpacing = FPB_DEFAULT_RIGHTLINESPACING
1771 if not kw
.has_key("Type"):
1772 raise 'ERROR: Missing Window Type Information. This Should Be "WINDOW" Or "SEPARATOR"'
1774 if kw
.get("Type") == "WINDOW":
1775 # Window constructor. This initialises the class as a wx.Window Type
1777 if kw
.has_key("flags"):
1778 self
._flags
= kw
.get("flags")
1780 self
._flags
= FPB_ALIGN_WIDTH
1781 if kw
.has_key("Spacing"):
1782 self
._Spacing
= kw
.get("Spacing")
1784 self
._Spacing
= FPB_DEFAULT_SPACING
1785 if kw
.has_key("leftSpacing"):
1786 self
._leftSpacing
= kw
.get("leftSpacing")
1788 self
._leftSpacing
= FPB_DEFAULT_LEFTSPACING
1789 if kw
.has_key("rightSpacing"):
1790 self
._rightSpacing
= kw
.get("rightSpacing")
1792 self
._rightSpacing
= FPB_DEFAULT_RIGHTSPACING
1795 self
._sepLineColour
= None
1799 elif kw
.get("Type") == "SEPARATOR":
1800 # separator constructor. This initialises the class as a separator type
1803 self
._lineY
= kw
.get("y")
1805 raise "ERROR: Undefined Y Position For The Separator"
1806 if kw
.has_key("lineColour"):
1807 self
._sepLineColour
= kw
.get("lineColour")
1809 self
._sepLineColour
= wx
.BLACK
1810 if kw
.has_key("flags"):
1811 self
._flags
= kw
.get("flags")
1813 self
._flags
= FPB_ALIGN_WIDTH
1814 if kw
.has_key("Spacing"):
1815 self
._Spacing
= kw
.get("Spacing")
1817 self
._Spacing
= FPB_DEFAULT_SPACING
1818 if kw
.has_key("leftSpacing"):
1819 self
._leftSpacing
= kw
.get("leftSpacing")
1821 self
._leftSpacing
= FPB_DEFAULT_LEFTSPACING
1822 if kw
.has_key("rightSpacing"):
1823 self
._rightSpacing
= kw
.get("rightSpacing")
1825 self
._rightSpacing
= FPB_DEFAULT_RIGHTSPACING
1830 raise "ERROR: Undefined Window Type Selected: " + repr(kw
.get("Type"))
1832 self
._type
= kw
.get("Type")
1833 self
._lineLength
= 0
1842 def GetLineLength(self
):
1843 return self
._lineLength
1845 def GetLineColour(self
):
1846 return self
._sepLineColour
1848 def GetLeftSpacing(self
):
1849 return self
._leftSpacing
1851 def GetRightSpacing(self
):
1852 return self
._rightSpacing
1854 def GetSpacing(self
):
1855 return self
._Spacing
1858 def GetWindowLength(self
, vertical
=True):
1860 Returns space needed by the window if type is FoldWindowItem
1861 "WINDOW" and returns the total size plus the extra spacing.
1865 if self
._type
== "WINDOW":
1866 size
= self
._wnd
.GetSize()
1867 value
= (vertical
and [size
.GetHeight()] or [size
.GetWidth()])[0] + \
1870 elif self
._type
== "SEPARATOR":
1871 value
= 1 + self
._Spacing
1876 def ResizeItem(self
, size
, vertical
=True):
1878 Resizes the element, whatever it is.
1880 A separator or line will be always aligned by width or height
1881 depending on orientation of the whole panel.
1884 if self
._flags
& FPB_ALIGN_WIDTH
:
1885 # align by taking full width
1886 mySize
= size
- self
._leftSpacing
- self
._rightSpacing
1889 mySize
= 10 # can't have negative width
1891 if self
._type
== "SEPARATOR":
1892 self
._lineLength
= mySize
1894 xsize
= (vertical
and [mySize
] or [-1])[0]
1895 ysize
= (vertical
and [-1] or [mySize
])[0]
1897 self
._wnd
.SetSize((xsize
, ysize
))