]> git.saurik.com Git - wxWidgets.git/blob - wxPython/wx/lib/foldpanelbar.py
Added a wx.xrc.XmlResourceHandler for the Ticker class
[wxWidgets.git] / wxPython / wx / lib / foldpanelbar.py
1 # --------------------------------------------------------------------------- #
2 # FOLDPANELBAR wxPython IMPLEMENTATION
3 # Ported From Jorgen Bodde & Julian Smart (Extended Demo) C++ Code By:
4 #
5 # Andrea Gavana, @ 23 Mar 2005
6 # Latest Revision: 28 Mar 2005, 22.30 CET
7 #
8 #
9 # TODO List
10 #
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:
14 #
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.
18 #
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?
22 #
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...
28 #
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
32 #
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.
36 #
37 #
38 # For The Original TODO List From Jorgen, Please Refer To:
39 # http://www.solidsteel.nl/jorg/components/foldpanel/wxFoldPanelBar.php#todo_list
40 #
41 #
42 #
43 # For All Kind Of Problems, Requests Of Enhancements And Bug Reports, Please
44 # Write To Me At:
45 #
46 # andrea.gavana@agip.it
47 # andrea_gavan@tin.it
48 #
49 # Or, Obviously, To The wxPython Mailing List!!!
50 #
51 #
52 # End Of Comments
53 # --------------------------------------------------------------------------- #
54
55
56 """
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.
65
66
67 How does it work
68 ----------------
69
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.
81
82
83 What can it do and what not?
84 ----------------------------
85
86 a) What it can do:
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
92
93 b) What it cannot do:
94
95 * Selection of a panel like in a list ctrl
96 * Dragging and dropping the panels
97 * Re-ordering the panels (not yet)
98
99
100 Supported platforms
101 -------------------
102
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)
107
108
109 Latest Revision: Andrea Gavana @ 30 Mar 2005, 22.30 CET
110
111 """
112
113 import wx
114
115 #----------------------------------------------------------------------
116 # Collapsed And Expanded Bitmap Images
117 # Created With img2py.py
118 #----------------------------------------------------------------------
119
120 def GetCollapsedIconData():
121 return \
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`\
127 \x82'
128
129 def GetCollapsedIconBitmap():
130 return wx.BitmapFromImage(GetCollapsedIconImage())
131
132 def GetCollapsedIconImage():
133 import cStringIO
134 stream = cStringIO.StringIO(GetCollapsedIconData())
135 return wx.ImageFromStream(stream)
136
137 #----------------------------------------------------------------------
138 def GetExpandedIconData():
139 return \
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'
146
147 def GetExpandedIconBitmap():
148 return wx.BitmapFromImage(GetExpandedIconImage())
149
150 def GetExpandedIconImage():
151 import cStringIO
152 stream = cStringIO.StringIO(GetExpandedIconData())
153 return wx.ImageFromStream(stream)
154
155 #----------------------------------------------------------------------
156
157 #----------------------------------------------------------------------
158 # FOLDPANELBAR Starts Here
159 #----------------------------------------------------------------------
160
161 # CAPTIONBAR STYLES
162 #
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
168
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
175
176 FPB_EXTRA_X = 10
177 FPB_EXTRA_Y = 4
178
179 # pixels of the bmp to be aligned from the right filled with space
180 FPB_BMP_RIGHTSPACE = 2
181
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
186
187 # All panels are stacked to the bottom. When they are expanded again they
188 # show up at the top
189 FPB_COLLAPSE_TO_BOTTOM = 0x0002
190
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
194
195 # Orientation Flag
196 FPB_HORIZONTAL = wx.HORIZONTAL
197 FPB_VERTICAL = wx.VERTICAL
198
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
203
204 # FoldPanelItem default settings
205 FPB_ALIGN_LEFT = 0
206 FPB_ALIGN_WIDTH = 1
207
208 FPB_DEFAULT_LEFTSPACING = 5
209 FPB_DEFAULT_RIGHTSPACING = 10
210 FPB_DEFAULT_SPACING = 8
211
212 FPB_DEFAULT_LEFTLINESPACING = 2
213 FPB_DEFAULT_RIGHTLINESPACING = 2
214
215
216 # ------------------------------------------------------------------------------ #
217 # class CaptionBarStyle
218 # ------------------------------------------------------------------------------ #
219
220 class CaptionBarStyle:
221 """
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.
226
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.
232
233 """
234
235 def __init__(self):
236 """ Default constructor for this class."""
237
238 self.ResetDefaults()
239
240
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
249
250
251 # ------- CaptionBar Font -------
252
253 def SetCaptionFont(self, font):
254 """
255 Sets font for the caption bar.
256
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
259 used.
260 """
261 self._captionFont = font
262 self._captionFontUsed = True
263
264
265 def CaptionFontUsed(self):
266 """ Checks if the caption bar font is set. """
267 return self._captionFontUsed
268
269
270 def GetCaptionFont(self):
271 """
272 Returns the font for the caption bar.
273
274 Please be warned this will result in an assertion failure when
275 this property is not previously set.
276
277 :see: `SetCaptionFont`, `CaptionFontUsed`
278 """
279 return self._captionFont
280
281
282 # ------- First Colour -------
283
284 def SetFirstColour(self, colour):
285 """
286 Sets first colour for the caption bar.
287
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
290 used.
291 """
292 self._firstColour = colour
293 self._firstColourUsed = True
294
295
296 def FirstColourUsed(self):
297 """ Checks if the first colour of the caption bar is set."""
298 return self._firstColourUsed
299
300
301 def GetFirstColour(self):
302 """
303 Returns the first colour for the caption bar.
304
305 Please be warned this will result in an assertion failure when
306 this property is not previously set.
307
308 :see: `SetFirstColour`, `FirstColourUsed`
309 """
310 return self._firstColour
311
312
313 # ------- Second Colour -------
314
315 def SetSecondColour(self, colour):
316 """
317 Sets second colour for the caption bar.
318
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
321 used.
322 """
323 self._secondColour = colour
324 self._secondColourUsed = True
325
326
327 def SecondColourUsed(self):
328 """ Checks if the second colour of the caption bar is set."""
329 return self._secondColourUsed
330
331
332 def GetSecondColour(self):
333 """
334 Returns the second colour for the caption bar.
335
336 Please be warned this will result in an assertion failure when
337 this property is not previously set.
338
339 :see: `SetSecondColour`, `SecondColourUsed`
340 """
341 return self._secondColour
342
343
344 # ------- Caption Text Colour -------
345
346 def SetCaptionColour(self, colour):
347 """
348 Sets caption colour for the caption bar.
349
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
352 used.
353 """
354 self._textColour = colour
355 self._textColourUsed = True
356
357
358 def CaptionColourUsed(self):
359 """ Checks if the caption colour of the caption bar is set."""
360 return self._textColourUsed
361
362
363 def GetCaptionColour(self):
364 """
365 Returns the caption colour for the caption bar.
366
367 Please be warned this will result in an assertion failure
368 when this property is not previously set.
369 See also SetCaptionColour(), CaptionColourUsed()
370 """
371 return self._textColour
372
373
374 # ------- CaptionStyle -------
375
376 def SetCaptionStyle(self, style):
377 """
378 Sets caption style for the caption bar.
379
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:
383
384 * CAPTIONBAR_GRADIENT_V: Draws a vertical gradient from top to bottom
385
386 * CAPTIONBAR_GRADIENT_H: Draws a horizontal gradient from
387 left to right
388
389 * CAPTIONBAR_SINGLE: Draws a single filled rectangle to
390 draw the caption
391
392 * CAPTIONBAR_RECTANGLE: Draws a single colour with a
393 rectangle around the caption
394
395 * CAPTIONBAR_FILLED_RECTANGLE: Draws a filled rectangle
396 and a border around it
397
398 """
399 self._captionStyle = style
400 self._captionStyleUsed = True
401
402
403 def CaptionStyleUsed(self):
404 """ Checks if the caption style of the caption bar is set."""
405 return self._captionStyleUsed
406
407
408 def GetCaptionStyle(self):
409 """
410 Returns the caption style for the caption bar.
411
412 Please be warned this will result in an assertion failure
413 when this property is not previously set.
414
415 :see: `SetCaptionStyle`, `CaptionStyleUsed`
416 """
417 return self._captionStyle
418
419
420 #-----------------------------------#
421 # CaptionBarEvent
422 #-----------------------------------#
423 wxEVT_CAPTIONBAR = wx.NewEventType()
424 EVT_CAPTIONBAR = wx.PyEventBinder(wxEVT_CAPTIONBAR, 0)
425
426
427 # ---------------------------------------------------------------------------- #
428 # class CaptionBarEvent
429 # ---------------------------------------------------------------------------- #
430
431 class CaptionBarEvent(wx.PyCommandEvent):
432 """
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
436 """
437 def __init__(self, evtType):
438 """ Default Constructor For This Class."""
439 wx.PyCommandEvent.__init__(self, evtType)
440
441
442 def GetFoldStatus(self):
443 """
444 Returns whether the bar is expanded or collapsed. True means
445 expanded.
446 """
447 return not self._bar.IsCollapsed()
448
449
450 def GetBar(self):
451 """ Returns The CaptionBar Selected."""
452 return self._bar
453
454
455 def SetTag(self, tag):
456 """ Assign A Tag To The Selected CaptionBar."""
457 self._tag = tag
458
459
460 def GetTag(self):
461 """ Returns The Tag Assigned To The Selected CaptionBar."""
462 return self._tag
463
464
465 def SetBar(self, bar):
466 """
467 Sets the bar associated with this event.
468
469 Should not used by any other then the originator of the event.
470 """
471 self._bar = bar
472
473
474 # -------------------------------------------------------------------------------- #
475 # class CaptionBar
476 # -------------------------------------------------------------------------------- #
477
478 class CaptionBar(wx.Window):
479 """
480 This class is a graphical caption component that consists of a
481 caption and a clickable arrow.
482
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.
488 """
489
490 # Define Empty CaptionBar Style
491 EmptyCaptionBarStyle = CaptionBarStyle()
492
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."""
498
499 wx.Window.__init__(self, parent, wx.ID_ANY, pos=wx.DefaultPosition,
500 size=(20,20), style=wx.NO_BORDER)
501
502 self._controlCreated = False
503 self._collapsed = collapsed
504 self.ApplyCaptionStyle(cbstyle, True)
505
506 if foldIcons is None:
507 foldIcons = wx.ImageList(16, 16)
508
509 bmp = GetExpandedIconBitmap()
510 foldIcons.Add(bmp)
511 bmp = GetCollapsedIconBitmap()
512 foldIcons.Add(bmp)
513
514 # set initial size
515 if foldIcons:
516 assert foldIcons.GetImageCount() > 1
517 iconWidth, iconHeight = foldIcons.GetSize(0)
518
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)
526
527 self._controlCreated = True
528
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)
533
534
535 def ApplyCaptionStyle(self, cbstyle=EmptyCaptionBarStyle, applyDefault=True):
536 """ Applies the style defined in cbstyle to the CaptionBar."""
537
538 newstyle = cbstyle
539
540 if applyDefault:
541
542 # get first colour from style or make it default
543 if not newstyle.FirstColourUsed():
544 newstyle.SetFirstColour(wx.WHITE)
545
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]))
553
554 # get text colour
555 if not newstyle.CaptionColourUsed():
556 newstyle.SetCaptionColour(wx.BLACK)
557
558 # get font colour
559 if not newstyle.CaptionFontUsed():
560 newstyle.SetCaptionFont(self.GetParent().GetFont())
561
562 # apply caption style
563 if not newstyle.CaptionStyleUsed():
564 newstyle.SetCaptionStyle(CAPTIONBAR_GRADIENT_V)
565
566 self._style = newstyle
567
568
569 def SetCaptionStyle(self, cbstyle=EmptyCaptionBarStyle, applyDefault=True):
570 """
571 Sets CaptionBar styles with CapionBarStyle class.
572
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.
577 """
578 self.ApplyCaptionStyle(cbstyle, applyDefault)
579 self.Refresh()
580
581
582 def GetCaptionStyle(self):
583 """
584 Returns the current style of the captionbar in a
585 `CaptionBarStyle` class.
586
587 This can be used to change and set back the changes.
588 """
589 return self._style
590
591
592 def IsCollapsed(self):
593 """
594 Returns wether the status of the bar is expanded or collapsed.
595 """
596 return self._collapsed
597
598
599 def SetRightIndent(self, pixels):
600 """
601 Sets the amount of pixels on the right from which the bitmap
602 is trailing.
603
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.
607 """
608 assert pixels >= 0
609 self._rightIndent = pixels
610 if self._foldIcons:
611 self.Refresh()
612
613
614 def Collapse(self):
615 """
616 This sets the internal state / representation to collapsed.
617
618 This does not trigger a `CaptionBarEvent` to be sent to the
619 parent.
620 """
621 self._collapsed = True
622 self.RedrawIconBitmap()
623
624
625 def Expand(self):
626 """
627 This sets the internal state / representation to expanded.
628
629 This does not trigger a `CaptionBarEvent` to be sent to the
630 parent.
631 """
632 self._collapsed = False
633 self.RedrawIconBitmap()
634
635
636 def SetBoldFont(self):
637 """ Sets the CaptionBarFont weight to BOLD."""
638
639 self.GetFont().SetWeight(wx.BOLD)
640
641
642 def SetNormalFont(self):
643 """ Sets the CaptionBarFont weight to NORMAL."""
644
645 self.GetFont().SetWeight(wx.NORMAL)
646
647
648 def IsVertical(self):
649 """
650 Returns wether the CaptionBar Has Default Orientation Or Not.
651
652 Default is vertical.
653 """
654
655 fld = self.GetParent().GetGrandParent()
656 if isinstance(fld, FoldPanelBar):
657 return self.GetParent().GetGrandParent().IsVertical()
658 else:
659 raise "ERROR: Wrong Parent " + repr(fld)
660
661
662 def OnPaint(self, event):
663 """ The paint event for flat or gradient fill. """
664
665 if not self._controlCreated:
666 event.Skip()
667 return
668
669 dc = wx.PaintDC(self)
670 wndRect = self.GetRect()
671 vertical = self.IsVertical()
672
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
675
676 self.FillCaptionBackground(dc)
677 dc.SetFont(self._style.GetCaptionFont())
678
679 if vertical:
680 dc.DrawText(self._caption, 4, FPB_EXTRA_Y/2)
681 else:
682 dc.DrawRotatedText(self._caption, FPB_EXTRA_Y/2,
683 wndRect.GetBottom() - 4, 90)
684
685 # draw small icon, either collapsed or expanded
686 # based on the state of the bar. If we have any bmp's
687
688 if self._foldIcons:
689
690 index = self._collapsed
691
692 if vertical:
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)
697 else:
698 self._foldIcons.Draw(index, dc,
699 (wndRect.GetWidth() - self._iconWidth)/2,
700 self._rightIndent, wx.IMAGELIST_DRAW_TRANSPARENT)
701
702 ## event.Skip()
703
704
705 def FillCaptionBackground(self, dc):
706 """
707 Fills the background of the caption with either a gradient or
708 a solid color.
709 """
710
711 style = self._style.GetCaptionStyle()
712
713 if style == CAPTIONBAR_GRADIENT_V:
714 if self.IsVertical():
715 self.DrawVerticalGradient(dc, self.GetRect())
716 else:
717 self.DrawHorizontalGradient(dc, self.GetRect())
718
719 elif style == CAPTIONBAR_GRADIENT_H:
720 if self.IsVertical():
721 self.DrawHorizontalGradient(dc, self.GetRect())
722 else:
723 self.DrawVerticalGradient(dc, self.GetRect())
724
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())
729 else:
730 raise "STYLE Error: Undefined Style Selected: " + repr(style)
731
732
733 def OnMouseEvent(self, event):
734 """
735 Catches the mouse click-double click.
736
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
739 collapse or expand.
740 """
741
742 send_event = False
743
744 if event.LeftDown() and self._foldIcons:
745
746 pt = event.GetPosition()
747 rect = self.GetRect()
748 vertical = self.IsVertical()
749
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):
753 send_event = True
754
755 elif event.LeftDClick():
756 send_event = True
757
758 # send the collapse, expand event to the parent
759
760 if send_event:
761 event = CaptionBarEvent(wxEVT_CAPTIONBAR)
762 event.SetBar(self)
763 self.GetEventHandler().ProcessEvent(event)
764 event.Skip()
765
766
767 def OnChar(self, event):
768 """ Unused Methods. Any Ideas?!?"""
769 # TODO: Anything here?
770 event.Skip()
771
772
773 def DoGetBestSize(self):
774 """
775 Returns the best size for this panel, based upon the font
776 assigned to this window, and the caption string
777 """
778
779 if self.IsVertical():
780 x, y = self.GetTextExtent(self._caption)
781 else:
782 y, x = self.GetTextExtent(self._caption)
783
784 if x < self._iconWidth:
785 x = self._iconWidth
786
787 if y < self._iconHeight:
788 y = self._iconHeight
789
790 # TODO: The extra FPB_EXTRA_X constants should be adjustable as well
791
792 return wx.Size(x + FPB_EXTRA_X, y + FPB_EXTRA_Y)
793
794
795 def DrawVerticalGradient(self, dc, rect):
796 """ Gradient fill from colour 1 to colour 2 with top to bottom. """
797
798 if rect.height < 1 or rect.width < 1:
799 return
800
801 dc.SetPen(wx.TRANSPARENT_PEN)
802
803 # calculate gradient coefficients
804 col2 = self._style.GetSecondColour()
805 col1 = self._style.GetFirstColour()
806
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())
809
810 flrect = float(rect.height)
811
812 rstep = float((r2 - r1)) / flrect
813 gstep = float((g2 - g1)) / flrect
814 bstep = float((b2 - b1)) / flrect
815
816 rf, gf, bf = 0, 0, 0
817
818 for y in range(rect.y, rect.y + rect.height):
819 currCol = (r1 + rf, g1 + gf, b1 + bf)
820
821 dc.SetBrush(wx.Brush(currCol, wx.SOLID))
822 dc.DrawRectangle(rect.x, rect.y + (y - rect.y), rect.width, rect.height)
823 rf = rf + rstep
824 gf = gf + gstep
825 bf = bf + bstep
826
827
828 def DrawHorizontalGradient(self, dc, rect):
829 """ Gradient fill from colour 1 to colour 2 with left to right. """
830
831 if rect.height < 1 or rect.width < 1:
832 return
833
834 dc.SetPen(wx.TRANSPARENT_PEN)
835
836 # calculate gradient coefficients
837 col2 = self._style.GetSecondColour()
838 col1 = self._style.GetFirstColour()
839
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())
842
843 flrect = float(rect.width)
844
845 rstep = float((r2 - r1)) / flrect
846 gstep = float((g2 - g1)) / flrect
847 bstep = float((b2 - b1)) / flrect
848
849 rf, gf, bf = 0, 0, 0
850
851 for x in range(rect.x, rect.x + rect.width):
852 currCol = (r1 + rf, g1 + gf, b1 + bf)
853
854 dc.SetBrush(wx.Brush(currCol, wx.SOLID))
855 dc.DrawRectangle(rect.x + (x - rect.x), rect.y, 1, rect.height)
856 rf = rf + rstep
857 gf = gf + gstep
858 bf = bf + bstep
859
860
861 def DrawSingleColour(self, dc, rect):
862 """ Single colour fill. This is the most easy one to find. """
863
864 if rect.height < 1 or rect.width < 1:
865 return
866
867 dc.SetPen(wx.TRANSPARENT_PEN)
868
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)
872
873
874 def DrawSingleRectangle(self, dc, rect):
875 """ Single rectangle. This is the most easy one to find. """
876
877 if rect.height < 2 or rect.width < 1:
878 return
879
880 # single frame, set up internal fill colour
881
882 if self._style.GetCaptionStyle() == CAPTIONBAR_RECTANGLE:
883 color = self.GetParent().GetBackgroundColour()
884 br = wx.Brush(color, wx.SOLID)
885 else:
886 color = self._style.GetFirstColour()
887 br = wx.Brush(color, wx.SOLID)
888
889 # setup the pen frame
890
891 pen = wx.Pen(self._style.GetSecondColour())
892 dc.SetPen(pen)
893 dc.SetBrush(br)
894 dc.DrawRectangle(rect.x, rect.y, rect.width, rect.height - 1)
895
896 bgpen = wx.Pen(self.GetParent().GetBackgroundColour())
897 dc.SetPen(bgpen)
898 dc.DrawLine(rect.x, rect.y + rect.height - 1, rect.x + rect.width,
899 rect.y + rect.height - 1)
900
901
902 def OnSize(self, event):
903 """ Handles the size events for the CaptionBar."""
904
905 if not self._controlCreated:
906 event.Skip()
907 return
908
909 size = event.GetSize()
910
911 if self._foldIcons:
912
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.
918
919 # set rect to redraw as old bitmap area which is entitled to redraw
920
921 rect = wx.Rect(size.GetWidth() - self._iconWidth - self._rightIndent, 0,
922 self._iconWidth + self._rightIndent,
923 self._iconWidth + self._rightIndent)
924
925 # adjust rectangle when more is slided so we need to redraw all
926 # the old stuff but not all (ugly flickering)
927
928 diffX = size.GetWidth() - self._oldSize.GetWidth()
929
930 if diffX > 1:
931
932 # adjust the rect with all the crap to redraw
933
934 rect.SetWidth(rect.GetWidth() + diffX + 10)
935 rect.SetX(rect.GetX() - diffX - 10)
936
937 self.RefreshRect(rect)
938
939 else:
940
941 rect = self.GetRect()
942 self.RefreshRect(rect)
943
944 self._oldSize = size
945
946
947 def RedrawIconBitmap(self):
948 """ Redraws the icons (if they exists). """
949
950 if self._foldIcons:
951
952 # invalidate the bitmap area and force a redraw
953
954 rect = self.GetRect()
955
956 rect.SetX(rect.GetWidth() - self._iconWidth - self._rightIndent)
957 rect.SetWidth(self._iconWidth + self._rightIndent)
958 self.RefreshRect(rect)
959
960
961 # ---------------------------------------------------------------------------------- #
962 # class FoldPanelBar
963 # ---------------------------------------------------------------------------------- #
964
965 class FoldPanelBar(wx.Panel):
966 """
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.
972
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.
982 """
983 # Define Empty CaptionBar Style
984 EmptyCaptionBarStyle = CaptionBarStyle()
985
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. """
989
990 self._controlCreated = False
991 self._extraStyle = extraStyle
992
993 # make sure there is any orientation
994 if style & FPB_HORIZONTAL != FPB_HORIZONTAL:
995 style = style | FPB_VERTICAL
996
997 if style & FPB_HORIZONTAL == 4:
998 self._isVertical = False
999 else:
1000 self._isVertical = True
1001
1002
1003 # create the panel (duh!). This causes a size event, which we are going
1004 # to skip when we are not initialised
1005
1006 wx.Panel.__init__(self, parent, id, pos, size, style)
1007
1008 # the fold panel area
1009
1010 self._foldPanel = wx.Panel(self, wx.ID_ANY, pos, size,
1011 wx.NO_BORDER | wx.TAB_TRAVERSAL)
1012
1013 self._controlCreated = True
1014 self._panels = []
1015
1016 self.Bind(EVT_CAPTIONBAR, self.OnPressCaption)
1017 self.Bind(wx.EVT_SIZE, self.OnSizePanel)
1018
1019
1020 def AddFoldPanel(self, caption="", collapsed=False, foldIcons=None,
1021 cbstyle=EmptyCaptionBarStyle):
1022 """
1023 Adds a fold panel to the list of panels.
1024
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.
1029
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.
1033 """
1034
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
1037 # when pressed.
1038
1039 if foldIcons is None:
1040 foldIcons = wx.ImageList(16, 16)
1041
1042 bmp = GetExpandedIconBitmap()
1043 foldIcons.Add(bmp)
1044 bmp = GetCollapsedIconBitmap()
1045 foldIcons.Add(bmp)
1046
1047 item = FoldPanelItem(self._foldPanel, -1, caption=caption,
1048 foldIcons=foldIcons,
1049 collapsed=collapsed, cbstyle=cbstyle)
1050
1051 pos = 0
1052 if len(self._panels) > 0:
1053 pos = self._panels[-1].GetItemPos() + self._panels[-1].GetPanelLength()
1054
1055 item.Reposition(pos)
1056 self._panels.append(item)
1057
1058 return item
1059
1060
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):
1065 """
1066 Adds a `wx.Window` derived instance to the referenced
1067 FoldPanel.
1068
1069 IMPORTANT: Make the window be a child of the FoldPanel. See
1070 example that follows. The flags to be used are:
1071
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
1075 text boxes.
1076
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.
1080
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
1087 FoldPanel.
1088
1089 The following example adds a FoldPanel to the FoldPanelBar and
1090 adds two wx.Window derived controls to the FoldPanel::
1091
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)
1096
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)
1100
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,
1104 "Collapse Me"))
1105
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)
1110
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)
1115
1116 """
1117
1118 try:
1119 item = self._panels.index(panel)
1120 except:
1121 raise "ERROR: Invalid Panel Passed To AddFoldPanelWindow: " + repr(panel)
1122
1123 panel.AddWindow(window, flags, Spacing, leftSpacing, rightSpacing)
1124
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.
1128
1129 return 0
1130
1131
1132 def AddFoldPanelSeparator(self, panel, colour=wx.BLACK,
1133 Spacing=FPB_DEFAULT_SPACING,
1134 leftSpacing=FPB_DEFAULT_LEFTLINESPACING,
1135 rightSpacing=FPB_DEFAULT_RIGHTLINESPACING):
1136 """
1137 Adds a separator line to the current FoldPanel.
1138
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`.
1144 """
1145
1146 try:
1147 item = self._panels.index(panel)
1148 except:
1149 raise "ERROR: Invalid Panel Passed To AddFoldPanelSeparator: " + repr(panel)
1150
1151 panel.AddSeparator(colour, Spacing, leftSpacing, rightSpacing)
1152 return 0
1153
1154
1155 def OnSizePanel(self, event):
1156 """ Handles the EVT_SIZE event for the FoldPanelBar. """
1157
1158 # skip all stuff when we are not initialised yet
1159
1160 if not self._controlCreated:
1161 event.Skip()
1162 return
1163
1164 foldrect = self.GetRect()
1165
1166 # fold panel itself. If too little space,
1167 # don't show it
1168
1169 foldrect.SetX(0)
1170 foldrect.SetY(0)
1171
1172 self._foldPanel.SetSize(foldrect[2:])
1173
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)
1179
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
1182
1183 self.RedisplayFoldPanelItems()
1184
1185
1186 def OnPressCaption(self, event):
1187 """ Handles the EVT_CAPTIONBAR event in the FoldPanelBar. """
1188
1189 # act upon the folding or expanding status of the bar
1190 # to expand or collapse the panel(s)
1191
1192 if event.GetFoldStatus():
1193 self.Collapse(event.GetTag())
1194 else:
1195 self.Expand(event.GetTag())
1196
1197 event.Skip()
1198
1199
1200 def RefreshPanelsFrom(self, item):
1201 """ Refreshes all the panels from given index down to last one. """
1202
1203 try:
1204 i = self._panels.index(item)
1205 except:
1206 raise "ERROR: Invalid Panel Passed To RefreshPanelsFrom: " + repr(item)
1207
1208 self.Freeze()
1209
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
1213
1214 if self._extraStyle & FPB_COLLAPSE_TO_BOTTOM:
1215
1216 offset = 0
1217
1218 for panels in self._panels:
1219
1220 if panels.IsExpanded():
1221 offset = offset + panels.Reposition(offset)
1222
1223 # put all non collapsed panels at the bottom where there is space,
1224 # else put them right behind the expanded ones
1225
1226 self.RepositionCollapsedToBottom()
1227
1228 else:
1229
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)
1233
1234 self.Thaw()
1235
1236
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()
1242 panels.Refresh()
1243
1244
1245 def RepositionCollapsedToBottom(self):
1246 """
1247 Repositions all the collapsed panels to the bottom.
1248
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.
1253 """
1254
1255 value = wx.Rect(0,0,0,0)
1256 vertical = self.IsVertical()
1257
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
1261
1262 expanded = 0
1263 collapsed = 0
1264 collapsed, expanded, values = self.GetPanelsLength(collapsed, expanded)
1265
1266 # if no room stick them behind the normal ones, else
1267 # at the bottom
1268
1269 if (vertical and [self.GetSize().GetHeight()] or \
1270 [self.GetSize().GetWidth()])[0] - expanded - collapsed < 0:
1271 offset = expanded
1272 else:
1273
1274 # value is the region which is left unpainted
1275 # I will send it back as 'slack' so it does not need to
1276 # be recalculated.
1277
1278 value.SetHeight(self.GetSize().GetHeight())
1279 value.SetWidth(self.GetSize().GetWidth())
1280
1281 if vertical:
1282 value.SetY(expanded)
1283 value.SetHeight(value.GetHeight() - expanded)
1284 else:
1285 value.SetX(expanded)
1286 value.SetWidth(value.GetWidth() - expanded)
1287
1288 offset = (vertical and [self.GetSize().GetHeight()] or \
1289 [self.GetSize().GetWidth()])[0] - collapsed
1290
1291
1292 # go reposition
1293
1294 for panels in self._panels:
1295 if not panels.IsExpanded():
1296 offset = offset + panels.Reposition(offset)
1297
1298 return value
1299
1300
1301 def GetPanelsLength(self, collapsed, expanded):
1302 """
1303 Returns the length of the panels that are expanded and
1304 collapsed.
1305
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
1308 collapsed panels.
1309 """
1310
1311 value = 0
1312
1313 # assumed here that all the panels that are expanded
1314 # are positioned after each other from 0,0 to end.
1315
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
1321 else:
1322 collapsed = collapsed + offset
1323
1324 return collapsed, expanded, value
1325
1326
1327 def Collapse(self, foldpanel):
1328 """
1329 Collapses the given FoldPanel reference, and updates the
1330 foldpanel bar.
1331
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.
1335 """
1336
1337 try:
1338 item = self._panels.index(foldpanel)
1339 except:
1340 raise "ERROR: Invalid Panel Passed To Collapse: " + repr(foldpanel)
1341
1342 foldpanel.Collapse()
1343 self.RefreshPanelsFrom(foldpanel)
1344
1345
1346 def Expand(self, foldpanel):
1347 """
1348 Expands the given FoldPanel reference, and updates the
1349 foldpanel bar.
1350
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
1353 is restored.
1354 """
1355
1356 foldpanel.Expand()
1357 self.RefreshPanelsFrom(foldpanel)
1358
1359
1360 def ApplyCaptionStyle(self, foldpanel, cbstyle):
1361 """
1362 Sets the style of the caption bar (`CaptionBar`) of the
1363 FoldPanel.
1364
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`
1370 """
1371 foldpanel.ApplyCaptionStyle(cbstyle)
1372
1373
1374 def ApplyCaptionStyleAll(self, cbstyle):
1375 """
1376 Sets the style of all the caption bars of the FoldPanel.
1377
1378 The changes are applied immediately.
1379 """
1380 for panels in self._panels:
1381 self.ApplyCaptionStyle(panels, cbstyle)
1382
1383
1384 def GetCaptionStyle(self, foldpanel):
1385 """
1386 Returns the currently used caption style for the FoldPanel.
1387
1388 It is returned as a CaptionBarStyle class. After modifying it,
1389 it can be set again.
1390 """
1391 return foldpanel.GetCaptionStyle()
1392
1393
1394 def IsVertical(self):
1395 """
1396 Returns whether the CaptionBar has default orientation or not.
1397
1398 Default is vertical.
1399 """
1400 return self._isVertical
1401
1402
1403 def GetFoldPanel(self, item):
1404 """
1405 Returns the panel associated with the index "item".
1406
1407 See the example at the bottom of the module, especially the events
1408 for the "Collapse Me" and "Expand Me" buttons.
1409 """
1410 try:
1411 ind = self._panels[item]
1412 return self._panels[item]
1413 except:
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))
1417
1418
1419 def GetCount(self):
1420 """ Returns the number of panels in the FoldPanelBar. """
1421
1422 try:
1423 return len(self._panels)
1424 except:
1425 raise "ERROR: No Panels Have Been Added To FoldPanelBar"
1426
1427
1428
1429 # --------------------------------------------------------------------------------- #
1430 # class FoldPanelItem
1431 # --------------------------------------------------------------------------------- #
1432
1433 class FoldPanelItem(wx.Panel):
1434 """
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.
1438 """
1439 # Define Empty CaptionBar Style
1440 EmptyCaptionBarStyle = CaptionBarStyle()
1441
1442 def __init__(self, parent, id=wx.ID_ANY, caption="", foldIcons=None,
1443 collapsed=False, cbstyle=EmptyCaptionBarStyle):
1444 """ Default Class Constructor. """
1445
1446 wx.Panel.__init__(self, parent, id, style=wx.CLIP_CHILDREN)
1447 self._controlCreated = False
1448 self._UserSize = 0
1449 self._PanelSize = 0
1450 self._LastInsertPos = 0
1451 self._itemPos = 0
1452 self._userSized = False
1453
1454 if foldIcons is None:
1455 foldIcons = wx.ImageList(16, 16)
1456
1457 bmp = GetExpandedIconBitmap()
1458 foldIcons.Add(bmp)
1459 bmp = GetCollapsedIconBitmap()
1460 foldIcons.Add(bmp)
1461
1462 self._foldIcons = foldIcons
1463
1464 # create the caption bar, in collapsed or expanded state
1465
1466 self._captionBar = CaptionBar(self, wx.ID_ANY, wx.Point(0,0),
1467 size=wx.DefaultSize, caption=caption,
1468 foldIcons=foldIcons, cbstyle=cbstyle)
1469
1470 if collapsed:
1471 self._captionBar.Collapse()
1472
1473 self._controlCreated = True
1474
1475 # make initial size for component, if collapsed, the
1476 # size is determined on the panel height and won't change
1477
1478 size = self._captionBar.GetSize()
1479
1480 self._PanelSize = (self.IsVertical() and [size.GetHeight()] or \
1481 [size.GetWidth()])[0]
1482
1483 self._LastInsertPos = self._PanelSize
1484 self._items = []
1485
1486 self.Bind(EVT_CAPTIONBAR, self.OnPressCaption)
1487 self.Bind(wx.EVT_PAINT, self.OnPaint)
1488
1489
1490 def AddWindow(self, window, flags=FPB_ALIGN_WIDTH, Spacing=FPB_DEFAULT_SPACING,
1491 leftSpacing=FPB_DEFAULT_LEFTLINESPACING,
1492 rightSpacing=FPB_DEFAULT_RIGHTLINESPACING):
1493 """
1494 Adds a window item to the list of items on this panel.
1495
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.
1501 """
1502
1503 wi = FoldWindowItem(self, window, Type="WINDOW", flags=flags, Spacing=Spacing,
1504 leftSpacing=leftSpacing, rightSpacing=rightSpacing)
1505
1506 self._items.append(wi)
1507
1508 vertical = self.IsVertical()
1509
1510 self._Spacing = Spacing
1511 self._leftSpacing = leftSpacing
1512 self._rightSpacing = rightSpacing
1513
1514 xpos = (vertical and [leftSpacing] or [self._LastInsertPos + Spacing])[0]
1515 ypos = (vertical and [self._LastInsertPos + Spacing] or [leftSpacing])[0]
1516
1517 window.SetDimensions(xpos, ypos, -1, -1, wx.SIZE_USE_EXISTING)
1518
1519 self._LastInsertPos = self._LastInsertPos + wi.GetWindowLength(vertical)
1520 self.ResizePanel()
1521
1522
1523 def AddSeparator(self, colour=wx.BLACK, Spacing=FPB_DEFAULT_SPACING,
1524 leftSpacing=FPB_DEFAULT_LEFTSPACING,
1525 rightSpacing=FPB_DEFAULT_RIGHTSPACING):
1526 """
1527 Adds a separator item to the list of items on this panel. """
1528
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)
1533
1534 self._items.append(wi)
1535 self._LastInsertPos = self._LastInsertPos + \
1536 wi.GetWindowLength(self.IsVertical())
1537
1538 self.ResizePanel()
1539
1540
1541 def Reposition(self, pos):
1542 """
1543 Repositions this FoldPanelBar and reports the length occupied
1544 for the next FoldPanelBar in the list.
1545 """
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
1549
1550 self.Freeze()
1551
1552 vertical = self.IsVertical()
1553 xpos = (vertical and [-1] or [pos])[0]
1554 ypos = (vertical and [pos] or [-1])[0]
1555
1556 self.SetDimensions(xpos, ypos, -1, -1, wx.SIZE_USE_EXISTING)
1557 self._itemPos = pos
1558
1559 self.Thaw()
1560
1561 return self.GetPanelLength()
1562
1563
1564 def OnPressCaption(self, event):
1565 """ Handles the EVT_CAPTIONBAR event in the FoldPanelItem. """
1566
1567 # tell the upper container we are responsible
1568 # for this event, so it can fold the panel item
1569 # and do a refresh
1570
1571 event.SetTag(self)
1572 event.Skip()
1573
1574
1575 def ResizePanel(self):
1576 """ Resizes the panel. """
1577
1578 # prevent unnecessary updates by blocking repaints for a sec
1579
1580 self.Freeze()
1581
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)
1585
1586
1587 if self._captionBar.IsCollapsed():
1588 size = self._captionBar.GetSize()
1589 self._PanelSize = (vertical and [size.GetHeight()] or [size.GetWidth()])[0]
1590 else:
1591 size = self.GetBestSize()
1592 self._PanelSize = (vertical and [size.GetHeight()] or [size.GetWidth()])[0]
1593
1594 if self._UserSize:
1595 if vertical:
1596 size.SetHeight(self._UserSize)
1597 else:
1598 size.SetWidth(self._UserSize)
1599
1600 pnlsize = self.GetParent().GetSize()
1601
1602 if vertical:
1603 size.SetWidth(pnlsize.GetWidth())
1604 else:
1605 size.SetHeight(pnlsize.GetHeight())
1606
1607 # resize caption bar
1608 xsize = (vertical and [size.GetWidth()] or [-1])[0]
1609 ysize = (vertical and [-1] or [size.GetHeight()])[0]
1610
1611 self._captionBar.SetSize((xsize, ysize))
1612
1613 # resize the panel
1614 self.SetSize(size)
1615
1616 # go by all the controls and call Layout
1617
1618 for items in self._items:
1619 items.ResizeItem((vertical and [size.GetWidth()] or \
1620 [size.GetHeight()])[0], vertical)
1621
1622 self.Thaw()
1623
1624
1625 def OnPaint(self, event):
1626 """ Handles the EVT_PAINT event in the FoldPanelItem. """
1627
1628 # draw all the items that are lines
1629
1630 dc = wx.PaintDC(self)
1631 vertical = self.IsVertical()
1632
1633 for item in self._items:
1634
1635 if item.GetType() == "SEPARATOR":
1636 pen = wx.Pen(item.GetLineColour(), 1, wx.SOLID)
1637 dc.SetPen(pen)
1638 a = item.GetLeftSpacing()
1639 b = item.GetLineY() + item.GetSpacing()
1640 c = item.GetLineLength()
1641 d = a + c
1642
1643 if vertical:
1644 dc.DrawLine(a, b, d, b)
1645 else:
1646 dc.DrawLine(b, a, b, d)
1647
1648 event.Skip()
1649
1650
1651 def IsVertical(self):
1652 """
1653 Returns wether the CaptionBar Has Default Orientation Or Not.
1654
1655 Default is vertical.
1656 """
1657
1658 # grandparent of FoldPanelItem is FoldPanelBar
1659 # default is vertical
1660
1661 if isinstance(self.GetGrandParent(), FoldPanelBar):
1662 return self.GetGrandParent().IsVertical()
1663 else:
1664 raise "ERROR: Wrong Parent " + repr(self.GetGrandParent())
1665
1666
1667 def IsExpanded(self):
1668 """
1669 Returns expanded or collapsed status. If the panel is
1670 expanded, True is returned.
1671 """
1672
1673 return not self._captionBar.IsCollapsed()
1674
1675
1676 def GetItemPos(self):
1677 """ Returns item's position. """
1678
1679 return self._itemPos
1680
1681
1682 def Collapse(self):
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
1685 # visual state
1686
1687 self._captionBar.Collapse()
1688 self.ResizePanel()
1689
1690
1691 def Expand(self):
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
1694 # visual state
1695
1696 self._captionBar.Expand()
1697 self.ResizePanel()
1698
1699
1700 def GetPanelLength(self):
1701 """ Returns size of panel. """
1702
1703 if self._captionBar.IsCollapsed():
1704 return self.GetCaptionLength()
1705 elif self._userSized:
1706 return self._UserSize
1707
1708 return self._PanelSize
1709
1710
1711 def GetCaptionLength(self):
1712 """
1713 Returns height of caption only. This is for folding
1714 calculation purposes.
1715 """
1716
1717 size = self._captionBar.GetSize()
1718 return (self.IsVertical() and [size.GetHeight()] or [size.GetWidth()])[0]
1719
1720
1721 def ApplyCaptionStyle(self, cbstyle):
1722 """ Applies the style defined in cbstyle to the CaptionBar."""
1723
1724 self._captionBar.SetCaptionStyle(cbstyle)
1725
1726
1727 def GetCaptionStyle(self):
1728 """
1729 Returns the current style of the captionbar in a
1730 CaptionBarStyle class.
1731
1732 This can be used to change and set back the changes.
1733 """
1734
1735 return self._captionBar.GetCaptionStyle()
1736
1737
1738 # ----------------------------------------------------------------------------------- #
1739 # class FoldWindowItem
1740 # ----------------------------------------------------------------------------------- #
1741
1742 class FoldWindowItem:
1743 """
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...).
1748 """
1749 def __init__(self, parent, window=None, **kw):
1750 """
1751 Default Class Constructor
1752
1753 Initialize with::
1754
1755 Type = "WINDOW", flags = FPB_ALIGN_WIDTH,
1756 Spacing = FPB_DEFAULT_SPACING,
1757 leftSpacing = FPB_DEFAULT_LEFTSPACING,
1758 rightSpacing = FPB_DEFAULT_RIGHTSPACING
1759
1760 or::
1761
1762 Type = "SEPARATOR"
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
1768 """
1769
1770
1771 if not kw.has_key("Type"):
1772 raise 'ERROR: Missing Window Type Information. This Should Be "WINDOW" Or "SEPARATOR"'
1773
1774 if kw.get("Type") == "WINDOW":
1775 # Window constructor. This initialises the class as a wx.Window Type
1776
1777 if kw.has_key("flags"):
1778 self._flags = kw.get("flags")
1779 else:
1780 self._flags = FPB_ALIGN_WIDTH
1781 if kw.has_key("Spacing"):
1782 self._Spacing = kw.get("Spacing")
1783 else:
1784 self._Spacing = FPB_DEFAULT_SPACING
1785 if kw.has_key("leftSpacing"):
1786 self._leftSpacing = kw.get("leftSpacing")
1787 else:
1788 self._leftSpacing = FPB_DEFAULT_LEFTSPACING
1789 if kw.has_key("rightSpacing"):
1790 self._rightSpacing = kw.get("rightSpacing")
1791 else:
1792 self._rightSpacing = FPB_DEFAULT_RIGHTSPACING
1793
1794 self._lineY = 0
1795 self._sepLineColour = None
1796 self._wnd = window
1797
1798
1799 elif kw.get("Type") == "SEPARATOR":
1800 # separator constructor. This initialises the class as a separator type
1801
1802 if kw.has_key("y"):
1803 self._lineY = kw.get("y")
1804 else:
1805 raise "ERROR: Undefined Y Position For The Separator"
1806 if kw.has_key("lineColour"):
1807 self._sepLineColour = kw.get("lineColour")
1808 else:
1809 self._sepLineColour = wx.BLACK
1810 if kw.has_key("flags"):
1811 self._flags = kw.get("flags")
1812 else:
1813 self._flags = FPB_ALIGN_WIDTH
1814 if kw.has_key("Spacing"):
1815 self._Spacing = kw.get("Spacing")
1816 else:
1817 self._Spacing = FPB_DEFAULT_SPACING
1818 if kw.has_key("leftSpacing"):
1819 self._leftSpacing = kw.get("leftSpacing")
1820 else:
1821 self._leftSpacing = FPB_DEFAULT_LEFTSPACING
1822 if kw.has_key("rightSpacing"):
1823 self._rightSpacing = kw.get("rightSpacing")
1824 else:
1825 self._rightSpacing = FPB_DEFAULT_RIGHTSPACING
1826
1827 self._wnd = window
1828
1829 else:
1830 raise "ERROR: Undefined Window Type Selected: " + repr(kw.get("Type"))
1831
1832 self._type = kw.get("Type")
1833 self._lineLength = 0
1834
1835
1836 def GetType(self):
1837 return self._type
1838
1839 def GetLineY(self):
1840 return self._lineY
1841
1842 def GetLineLength(self):
1843 return self._lineLength
1844
1845 def GetLineColour(self):
1846 return self._sepLineColour
1847
1848 def GetLeftSpacing(self):
1849 return self._leftSpacing
1850
1851 def GetRightSpacing(self):
1852 return self._rightSpacing
1853
1854 def GetSpacing(self):
1855 return self._Spacing
1856
1857
1858 def GetWindowLength(self, vertical=True):
1859 """
1860 Returns space needed by the window if type is FoldWindowItem
1861 "WINDOW" and returns the total size plus the extra spacing.
1862 """
1863
1864 value = 0
1865 if self._type == "WINDOW":
1866 size = self._wnd.GetSize()
1867 value = (vertical and [size.GetHeight()] or [size.GetWidth()])[0] + \
1868 self._Spacing
1869
1870 elif self._type == "SEPARATOR":
1871 value = 1 + self._Spacing
1872
1873 return value
1874
1875
1876 def ResizeItem(self, size, vertical=True):
1877 """
1878 Resizes the element, whatever it is.
1879
1880 A separator or line will be always aligned by width or height
1881 depending on orientation of the whole panel.
1882 """
1883
1884 if self._flags & FPB_ALIGN_WIDTH:
1885 # align by taking full width
1886 mySize = size - self._leftSpacing - self._rightSpacing
1887
1888 if mySize < 0:
1889 mySize = 10 # can't have negative width
1890
1891 if self._type == "SEPARATOR":
1892 self._lineLength = mySize
1893 else:
1894 xsize = (vertical and [mySize] or [-1])[0]
1895 ysize = (vertical and [-1] or [mySize])[0]
1896
1897 self._wnd.SetSize((xsize, ysize))
1898