2 #----------------------------------------------------------------------------
4 # Purpose: Testing lots of stuff, controls, window types, etc.
8 # Created: A long time ago, in a galaxy far, far away...
10 # Copyright: (c) 1999 by Total Control Software
11 # Licence: wxWindows license
12 #----------------------------------------------------------------------------
15 # * Problems with flickering related to ERASE_BACKGROUND
16 # and the splitters. Might be a problem with this 2.5 beta...?
17 # UPDATE: can't see on 2.5.2 GTK - maybe just a faster machine :)
19 # * Annoying switching between tabs and resulting flicker
20 # how to replace a page in the notebook without deleting/adding?
21 # Where is SetPage!? tried freeze...tried reparent of dummy panel....
24 # * UI design more prefessional
25 # * save file positions (new field in demoModules) (@ LoadDemoSource)
26 # * Update main overview
28 # * Why don't we move _treeList into a separate module
30 import sys
, os
, time
, traceback
, types
32 import wx
# This module uses the new wx namespace
39 ##print "wx.VERSION_STRING = ", wx.VERSION_STRING
41 ##raw_input("Press Enter...")
44 #---------------------------------------------------------------------------
49 ('Recent Additions/Updates', [
54 # managed windows == things with a (optional) caption you can close
55 ('Base Frames and Dialogs', [
79 # dialogs from libraries
82 'MultipleChoiceDialog',
83 'ScrolledMessageDialog',
87 ('Core Windows/Controls', [
123 ('Custom Controls', [
136 # controls coming from other libraries
137 ('More Windows/Controls', [
138 'ActiveX_FlashWindow',
139 'ActiveX_IEHtmlWindow',
141 #'RightTextCtrl', deprecated as we have wxTE_RIGHT now.
154 'MaskedEditControls',
170 # How to lay out the controls in a frame/dialog
180 'XmlResourceHandler',
181 'XmlResourceSubclass',
185 ('Process and Events', [
196 ('Clipboard and DnD', [
229 # need libs not coming with the demo
230 ('Samples using an external library', [
235 ('Check out the samples dir too', [
242 #---------------------------------------------------------------------------
243 # Show how to derive a custom wxLog class
245 class MyLog(wx
.PyLog
):
246 def __init__(self
, textCtrl
, logTime
=0):
247 wx
.PyLog
.__init
__(self
)
249 self
.logTime
= logTime
251 def DoLogString(self
, message
, timeStamp
):
253 message
= time
.strftime("%X", time
.localtime(timeStamp
)) + \
256 self
.tc
.AppendText(message
+ '\n')
259 class MyTP(wx
.PyTipProvider
):
261 return "This is my tip"
264 #---------------------------------------------------------------------------
265 # A class to be used to display source code in the demo. Try using the
266 # wxSTC in the StyledTextCtrl_2 sample first, fall back to wxTextCtrl
267 # if there is an error, such as the stc module not being present.
271 ##raise ImportError # for testing the alternate implementation
273 from StyledTextCtrl_2
import PythonSTC
275 class DemoCodeEditor(PythonSTC
):
276 def __init__(self
, parent
):
277 PythonSTC
.__init
__(self
, parent
, -1, wx
.BORDER_NONE
)
280 # Some methods to make it compatible with how the wxTextCtrl is used
281 def SetValue(self
, value
):
283 value
= value
.decode('iso8859_1')
285 self
.EmptyUndoBuffer()
288 def IsModified(self
):
289 return self
.GetModify()
294 def SetInsertionPoint(self
, pos
):
295 self
.SetCurrentPos(pos
)
298 def ShowPosition(self
, pos
):
299 line
= self
.LineFromPosition(pos
)
300 self
.EnsureVisible(line
)
302 def GetLastPosition(self
):
303 return self
.GetLength()
305 def GetPositionFromLine(self
, line
):
306 return self
.PositionFromLine(line
)
308 def GetRange(self
, start
, end
):
309 return self
.GetTextRange(start
, end
)
311 def GetSelection(self
):
312 return self
.GetAnchor(), self
.GetCurrentPos()
314 def SetSelection(self
, start
, end
):
315 self
.SetSelectionStart(start
)
316 self
.SetSelectionEnd(end
)
318 def SelectLine(self
, line
):
319 start
= self
.PositionFromLine(line
)
320 end
= self
.GetLineEndPosition(line
)
321 self
.SetSelection(start
, end
)
323 def SetUpEditor(self
):
325 This method carries out the work of setting up the demo editor.
326 It's seperate so as not to clutter up the init code.
330 self
.SetLexer(stc
.STC_LEX_PYTHON
)
331 self
.SetKeyWords(0, " ".join(keyword
.kwlist
))
334 self
.SetProperty("fold", "1" )
336 # Highlight tab/space mixing (shouldn't be any)
337 self
.SetProperty("tab.timmy.whinge.level", "1")
339 # Set left and right margins
342 # Set up the numbers in the margin for margin #1
343 self
.SetMarginType(1, wx
.stc
.STC_MARGIN_NUMBER
)
344 # Reasonable value for, say, 4-5 digits using a mono font (40 pix)
345 self
.SetMarginWidth(1, 40)
347 # Indentation and tab stuff
348 self
.SetIndent(4) # Proscribed indent size for wx
349 self
.SetIndentationGuides(True) # Show indent guides
350 self
.SetBackSpaceUnIndents(True)# Backspace unindents rather than delete 1 space
351 self
.SetTabIndents(True) # Tab key indents
352 self
.SetTabWidth(4) # Proscribed tab size for wx
353 self
.SetUseTabs(False) # Use spaces rather than tabs, or
354 # TabTimmy will complain!
356 self
.SetViewWhiteSpace(False) # Don't view white space
359 #self.SetEOLMode(wx.stc.STC_EOL_CRLF) # Just leave it at the default (autosense)
360 self
.SetViewEOL(False)
361 # No right-edge mode indicator
362 self
.SetEdgeMode(stc
.STC_EDGE_NONE
)
364 # Setup a margin to hold fold markers
365 self
.SetMarginType(2, stc
.STC_MARGIN_SYMBOL
)
366 self
.SetMarginMask(2, stc
.STC_MASK_FOLDERS
)
367 self
.SetMarginSensitive(2, True)
368 self
.SetMarginWidth(2, 12)
370 # and now set up the fold markers
371 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEREND
, stc
.STC_MARK_BOXPLUSCONNECTED
, "white", "black")
372 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEROPENMID
, stc
.STC_MARK_BOXMINUSCONNECTED
, "white", "black")
373 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERMIDTAIL
, stc
.STC_MARK_TCORNER
, "white", "black")
374 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERTAIL
, stc
.STC_MARK_LCORNER
, "white", "black")
375 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERSUB
, stc
.STC_MARK_VLINE
, "white", "black")
376 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDER
, stc
.STC_MARK_BOXPLUS
, "white", "black")
377 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEROPEN
, stc
.STC_MARK_BOXMINUS
, "white", "black")
379 # Global default style
380 if wx
.Platform
== '__WXMSW__':
381 self
.StyleSetSpec(stc
.STC_STYLE_DEFAULT
,
382 'fore:#000000,back:#FFFFFF,face:Courier New,size:9')
384 self
.StyleSetSpec(stc
.STC_STYLE_DEFAULT
,
385 'fore:#000000,back:#FFFFFF,face:Courier,size:12')
387 # Clear styles and revert to default.
390 # Following style specs only indicate differences from default.
391 # The rest remains unchanged.
393 # Line numbers in margin
394 self
.StyleSetSpec(wx
.stc
.STC_STYLE_LINENUMBER
,'fore:#000000,back:#99A9C2')
397 self
.StyleSetSpec(wx
.stc
.STC_STYLE_BRACELIGHT
,'fore:#00009D,back:#FFFF00')
399 self
.StyleSetSpec(wx
.stc
.STC_STYLE_BRACEBAD
,'fore:#00009D,back:#FF0000')
401 self
.StyleSetSpec(wx
.stc
.STC_STYLE_INDENTGUIDE
, "fore:#CDCDCD")
404 self
.StyleSetSpec(wx
.stc
.STC_P_DEFAULT
, 'fore:#000000')
406 self
.StyleSetSpec(wx
.stc
.STC_P_COMMENTLINE
, 'fore:#008000,back:#F0FFF0')
407 self
.StyleSetSpec(wx
.stc
.STC_P_COMMENTBLOCK
, 'fore:#008000,back:#F0FFF0')
409 self
.StyleSetSpec(wx
.stc
.STC_P_NUMBER
, 'fore:#008080')
410 # Strings and characters
411 self
.StyleSetSpec(wx
.stc
.STC_P_STRING
, 'fore:#800080')
412 self
.StyleSetSpec(wx
.stc
.STC_P_CHARACTER
, 'fore:#800080')
414 self
.StyleSetSpec(wx
.stc
.STC_P_WORD
, 'fore:#000080,bold')
416 self
.StyleSetSpec(wx
.stc
.STC_P_TRIPLE
, 'fore:#800080,back:#FFFFEA')
417 self
.StyleSetSpec(wx
.stc
.STC_P_TRIPLEDOUBLE
, 'fore:#800080,back:#FFFFEA')
419 self
.StyleSetSpec(wx
.stc
.STC_P_CLASSNAME
, 'fore:#0000FF,bold')
421 self
.StyleSetSpec(wx
.stc
.STC_P_DEFNAME
, 'fore:#008080,bold')
423 self
.StyleSetSpec(wx
.stc
.STC_P_OPERATOR
, 'fore:#800000,bold')
424 # Identifiers. I leave this as not bold because everything seems
425 # to be an identifier if it doesn't match the above criterae
426 self
.StyleSetSpec(wx
.stc
.STC_P_IDENTIFIER
, 'fore:#000000')
429 self
.SetCaretForeground("BLUE")
430 # Selection background
431 self
.SetSelBackground(1, '#66CCFF')
433 self
.SetSelBackground(True, wx
.SystemSettings_GetColour(wx
.SYS_COLOUR_HIGHLIGHT
))
434 self
.SetSelForeground(True, wx
.SystemSettings_GetColour(wx
.SYS_COLOUR_HIGHLIGHTTEXT
))
436 def RegisterModifiedEvent(self
, eventHandler
):
437 self
.Bind(wx
.stc
.EVT_STC_CHANGE
, eventHandler
)
441 class DemoCodeEditor(wx
.TextCtrl
):
442 def __init__(self
, parent
):
443 wx
.TextCtrl
.__init
__(self
, parent
, -1, style
= wx
.TE_MULTILINE |
444 wx
.HSCROLL | wx
.TE_RICH2 | wx
.TE_NOHIDESEL
)
446 def RegisterModifiedEvent(self
, eventHandler
):
447 self
.Bind(wx
.EVT_TEXT
, eventHandler
)
449 def SetReadOnly(self
, flag
):
450 self
.SetEditable(not flag
)
451 # NOTE: STC already has this method
454 return self
.GetValue()
456 def GetPositionFromLine(line
):
457 return self
.XYToPosition(0,line
)
459 def GotoLine(self
, line
):
460 pos
= self
.editor
.GetPositionFromLine(line
)
461 self
.editor
.SetInsertionPoint(pos
)
462 self
.editor
.ShowPosition(pos
)
464 def SelectLine(self
, line
):
465 start
= self
.GetPositionFromLine(line
)
466 end
= start
+ self
.GetLineLength(line
)
467 self
.SetSelection(start
, end
)
470 #---------------------------------------------------------------------------
471 # Constants for module versions
475 modDefault
= modOriginal
477 #---------------------------------------------------------------------------
479 class DemoCodePanel(wx
.Panel
):
480 """Panel for the 'Demo Code' tab"""
481 def __init__(self
, parent
, mainFrame
):
482 wx
.Panel
.__init
__(self
, parent
)
483 self
.mainFrame
= mainFrame
484 self
.editor
= DemoCodeEditor(self
)
485 self
.editor
.RegisterModifiedEvent(self
.OnCodeModified
)
487 self
.btnSave
= wx
.Button(self
, -1, "Save Changes")
488 self
.btnRestore
= wx
.Button(self
, -1, "Delete Modified")
489 self
.btnSave
.Enable(False)
490 self
.btnSave
.Bind(wx
.EVT_BUTTON
, self
.OnSave
)
491 self
.btnRestore
.Bind(wx
.EVT_BUTTON
, self
.OnRestore
)
493 self
.radioButtons
= { modOriginal
: wx
.RadioButton(self
, -1, "Original", style
= wx
.RB_GROUP
),
494 modModified
: wx
.RadioButton(self
, -1, "Modified") }
496 self
.controlBox
= wx
.BoxSizer(wx
.HORIZONTAL
)
497 self
.controlBox
.Add(wx
.StaticText(self
, -1, "Active Version:"), 0,
498 wx
.RIGHT | wx
.LEFT | wx
.ALIGN_CENTER_VERTICAL
, 5)
499 for modID
, radioButton
in self
.radioButtons
.items():
500 self
.controlBox
.Add(radioButton
, 0, wx
.EXPAND | wx
.RIGHT
, 5)
501 radioButton
.modID
= modID
# makes it easier for the event handler
502 radioButton
.Bind(wx
.EVT_RADIOBUTTON
, self
.OnRadioButton
)
504 self
.controlBox
.Add(self
.btnSave
, 0, wx
.RIGHT
, 5)
505 self
.controlBox
.Add(self
.btnRestore
, 0)
507 self
.box
= wx
.BoxSizer(wx
.VERTICAL
)
508 self
.box
.Add(self
.controlBox
, 0, wx
.EXPAND
)
509 self
.box
.Add(self
.editor
, 1, wx
.EXPAND
)
512 self
.SetSizer(self
.box
)
515 # Loads a demo from a DemoModules object
516 def LoadDemo(self
, demoModules
):
517 self
.demoModules
= demoModules
518 if (modDefault
== modModified
) and demoModules
.Exists(modModified
):
519 demoModules
.SetActive(modModified
)
521 demoModules
.SetActive(modOriginal
)
522 self
.radioButtons
[demoModules
.GetActiveID()].Enable(True)
523 self
.ActiveModuleChanged()
526 def ActiveModuleChanged(self
):
527 self
.LoadDemoSource(self
.demoModules
.GetSource())
528 self
.UpdateControlState()
532 def LoadDemoSource(self
, source
):
534 self
.editor
.SetValue(source
)
536 self
.btnSave
.Enable(False)
539 def JumpToLine(self
, line
, highlight
=False):
540 self
.editor
.GotoLine(line
)
541 self
.editor
.SetFocus()
543 self
.editor
.SelectLine(line
)
546 def UpdateControlState(self
):
547 active
= self
.demoModules
.GetActiveID()
548 # Update the radio/restore buttons
549 for moduleID
in self
.radioButtons
:
550 btn
= self
.radioButtons
[moduleID
]
551 if moduleID
== active
:
556 if self
.demoModules
.Exists(moduleID
):
558 if moduleID
== modModified
:
559 self
.btnRestore
.Enable(True)
562 if moduleID
== modModified
:
563 self
.btnRestore
.Enable(False)
566 def OnRadioButton(self
, event
):
567 radioSelected
= event
.GetEventObject()
568 modSelected
= radioSelected
.modID
569 if modSelected
!= self
.demoModules
.GetActiveID():
570 busy
= wx
.BusyInfo("Reloading demo module...")
571 self
.demoModules
.SetActive(modSelected
)
572 self
.ActiveModuleChanged()
575 def ReloadDemo(self
):
576 if self
.demoModules
.name
!= __name__
:
577 self
.mainFrame
.RunModule()
580 def OnCodeModified(self
, event
):
581 self
.btnSave
.Enable(self
.editor
.IsModified())
584 def OnSave(self
, event
):
585 if self
.demoModules
.Exists(modModified
):
586 if self
.demoModules
.GetActiveID() == modOriginal
:
587 overwriteMsg
= "You are about to overwrite an already existing modified copy\n" + \
588 "Do you want to continue?"
589 dlg
= wx
.MessageDialog(self
, overwriteMsg
, "wxPython Demo",
590 wx
.YES_NO | wx
.NO_DEFAULT| wx
.ICON_EXCLAMATION
)
591 result
= dlg
.ShowModal()
592 if result
== wx
.ID_NO
:
596 self
.demoModules
.SetActive(modModified
)
597 modifiedFilename
= GetModifiedFilename(self
.demoModules
.name
)
599 # Create the demo directory if one doesn't already exist
600 if not os
.path
.exists(GetModifiedDirectory()):
602 os
.makedirs(GetModifiedDirectory())
603 if not os
.path
.exists(GetModifiedDirectory()):
604 wx
.LogMessage("BUG: Created demo directory but it still doesn't exit")
607 wx
.LogMessage("Error creating demo directory: %s" % GetModifiedDirectory())
610 wx
.LogMessage("Created directory for modified demos: %s" % GetModifiedDirectory())
613 f
= open(modifiedFilename
, "w")
614 source
= self
.editor
.GetText()
620 busy
= wx
.BusyInfo("Reloading demo module...")
621 self
.demoModules
.LoadFromFile(modModified
, modifiedFilename
)
622 self
.ActiveModuleChanged()
625 def OnRestore(self
, event
): # Handles the "Delete Modified" button
626 modifiedFilename
= GetModifiedFilename(self
.demoModules
.name
)
627 self
.demoModules
.Delete(modModified
)
628 os
.unlink(modifiedFilename
) # Delete the modified copy
629 busy
= wx
.BusyInfo("Reloading demo module...")
630 self
.ActiveModuleChanged()
633 #---------------------------------------------------------------------------
636 """Convert paths to the platform-specific separator"""
637 str = apply(os
.path
.join
, tuple(path
.split('/')))
638 # HACK: on Linux, a leading / gets lost...
639 if path
.startswith('/'):
644 def GetModifiedDirectory():
646 Returns the directory where modified versions of the demo files
649 return opj(wx
.GetHomeDir() + "/.wxPyDemo/modified/")
652 def GetModifiedFilename(name
):
654 Returns the filename of the modified version of the specified demo
656 if not name
.endswith(".py"):
658 return GetModifiedDirectory() + name
661 def GetOriginalFilename(name
):
663 Returns the filename of the original version of the specified demo
665 if not name
.endswith(".py"):
670 def DoesModifiedExist(name
):
671 """Returns whether the specified demo has a modified copy"""
672 if os
.path
.exists(GetModifiedFilename(name
)):
678 #---------------------------------------------------------------------------
680 class ModuleDictWrapper
:
681 """Emulates a module with a dynamically compiled __dict__"""
682 def __init__(self
, dict):
685 def __getattr__(self
, name
):
686 if name
in self
.dict:
687 return self
.dict[name
]
693 Dynamically manages the original/modified versions of a demo
696 def __init__(self
, name
):
700 # (dict , source , filename , description , error information )
701 # ( 0 , 1 , 2 , 3 , 4 )
702 self
.modules
= [[None, "" , "" , "<original>" , None],
703 [None, "" , "" , "<modified>" , None]]
705 # load original module
706 self
.LoadFromFile(modOriginal
, GetOriginalFilename(name
))
708 # load modified module (if one exists)
709 if DoesModifiedExist(name
):
710 self
.LoadFromFile(modModified
, GetModifiedFilename(name
))
713 def LoadFromFile(self
, modID
, filename
):
714 self
.modules
[modID
][2] = filename
715 file = open(filename
, "r")
716 self
.LoadFromSource(modID
, file.read())
720 def LoadFromSource(self
, modID
, source
):
721 self
.modules
[modID
][1] = source
725 def LoadDict(self
, modID
):
726 if self
.name
!= __name__
:
727 source
= self
.modules
[modID
][1]
728 description
= self
.modules
[modID
][3]
731 self
.modules
[modID
][0] = {}
732 code
= compile(source
, description
, "exec")
733 exec code
in self
.modules
[modID
][0]
735 self
.modules
[modID
][4] = DemoError(sys
.exc_info())
736 self
.modules
[modID
][0] = None
738 self
.modules
[modID
][4] = None
741 def SetActive(self
, modID
):
742 if modID
!= modOriginal
and modID
!= modModified
:
745 self
.modActive
= modID
749 dict = self
.modules
[self
.modActive
][0]
753 return ModuleDictWrapper(dict)
756 def GetActiveID(self
):
757 return self
.modActive
760 def GetSource(self
, modID
= None):
762 modID
= self
.modActive
763 return self
.modules
[modID
][1]
766 def GetFilename(self
, modID
= None):
768 modID
= self
.modActive
769 return self
.modules
[self
.modActive
][2]
772 def GetErrorInfo(self
, modID
= None):
774 modID
= self
.modActive
775 return self
.modules
[self
.modActive
][4]
778 def Exists(self
, modID
):
779 return self
.modules
[modID
][1] != ""
782 def UpdateFile(self
, modID
= None):
783 """Updates the file from which a module was loaded
784 with (possibly updated) source"""
786 modID
= self
.modActive
788 source
= self
.modules
[modID
][1]
789 filename
= self
.modules
[modID
][2]
792 file = open(filename
, "w")
798 def Delete(self
, modID
):
799 if self
.modActive
== modID
:
802 self
.modules
[modID
][0] = None
803 self
.modules
[modID
][1] = ""
804 self
.modules
[modID
][2] = ""
807 #---------------------------------------------------------------------------
808 class ReloadDemoPanel(wx
.Panel
):
810 Panel put into the demo tab when the demo just shows some
811 top-level window. Enables the demo to be reloaded after being
815 infoText
= "This demo runs outside the main window"
817 def __init__(self
, parent
, codePanel
, log
):
818 wx
.Panel
.__init
__(self
, parent
, -1)
819 self
.codePanel
= codePanel
822 self
.label
= wx
.StaticText(self
, -1, self
.infoText
)
823 self
.btnReload
= wx
.Button(self
, -1, "Reload Demo")
824 self
.btnReload
.Bind(wx
.EVT_BUTTON
, self
.OnReload
)
826 self
.box
= wx
.BoxSizer(wx
.VERTICAL
)
827 self
.box
.Add(self
.label
, 0, wx
.ALIGN_CENTER | wx
.ALL
, 10)
828 self
.box
.Add(self
.btnReload
, 0, wx
.ALIGN_CENTER | wx
.ALL
, 10)
831 self
.SetSizer(self
.box
)
833 def OnReload(self
, event
):
834 self
.codePanel
.ReloadDemo()
836 #---------------------------------------------------------------------------
839 """Wraps and stores information about the current exception"""
840 def __init__(self
, exc_info
):
843 excType
, excValue
= exc_info
[:2]
844 # traceback list entries: (filename, line number, function name, text)
845 self
.traceback
= traceback
.extract_tb(exc_info
[2])
847 # --Based on traceback.py::format_exception_only()--
848 if type(excType
) == types
.ClassType
:
849 self
.exception_type
= excType
.__name
__
851 self
.exception_type
= excType
853 # If it's a syntax error, extra information needs
854 # to be added to the traceback
855 if excType
is SyntaxError:
857 msg
, (filename
, lineno
, self
.offset
, line
) = excValue
862 filename
= "<string>"
864 self
.traceback
.append( (filename
, lineno
, "", line
) )
867 self
.exception_details
= str(excValue
)
869 self
.exception_details
= "<unprintable %s object>" & type(excValue
).__name
__
876 Details : %s" % ( str(self
.exception_type
), str(self
.traceback
), self
.exception_details
)
879 #---------------------------------------------------------------------------
881 class DemoErrorPanel(wx
.Panel
):
882 """Panel put into the demo tab when the demo fails to run due to errors"""
884 def __init__(self
, parent
, codePanel
, demoError
, log
):
885 wx
.Panel
.__init
__(self
, parent
, -1)#, style=wx.NO_FULL_REPAINT_ON_RESIZE)
886 self
.codePanel
= codePanel
890 self
.box
= wx
.BoxSizer(wx
.VERTICAL
)
893 self
.box
.Add(wx
.StaticText(self
, -1, "An error has occured while trying to run the demo")
894 , 0, wx
.ALIGN_CENTER | wx
.TOP
, 10)
896 # Exception Information
897 boxInfo
= wx
.StaticBox(self
, -1, "Exception Info" )
898 boxInfoSizer
= wx
.StaticBoxSizer(boxInfo
, wx
.VERTICAL
) # Used to center the grid within the box
899 boxInfoGrid
= wx
.FlexGridSizer(0, 2, 0, 0)
900 textFlags
= wx
.ALIGN_RIGHT | wx
.LEFT | wx
.RIGHT | wx
.TOP
901 boxInfoGrid
.Add(wx
.StaticText(self
, -1, "Type: "), 0, textFlags
, 5 )
902 boxInfoGrid
.Add(wx
.StaticText(self
, -1, demoError
.exception_type
) , 0, textFlags
, 5 )
903 boxInfoGrid
.Add(wx
.StaticText(self
, -1, "Details: ") , 0, textFlags
, 5 )
904 boxInfoGrid
.Add(wx
.StaticText(self
, -1, demoError
.exception_details
) , 0, textFlags
, 5 )
905 boxInfoSizer
.Add(boxInfoGrid
, 0, wx
.ALIGN_CENTRE | wx
.ALL
, 5 )
906 self
.box
.Add(boxInfoSizer
, 0, wx
.ALIGN_CENTER | wx
.ALL
, 5)
908 # Set up the traceback list
909 # This one automatically resizes last column to take up remaining space
910 from ListCtrl
import TestListCtrl
911 self
.list = TestListCtrl(self
, -1, style
=wx
.LC_REPORT | wx
.SUNKEN_BORDER
)
912 self
.list.Bind(wx
.EVT_LEFT_DCLICK
, self
.OnDoubleClick
)
913 self
.list.Bind(wx
.EVT_LIST_ITEM_SELECTED
, self
.OnItemSelected
)
914 self
.list.InsertColumn(0, "Filename")
915 self
.list.InsertColumn(1, "Line", wx
.LIST_FORMAT_RIGHT
)
916 self
.list.InsertColumn(2, "Function")
917 self
.list.InsertColumn(3, "Code")
918 self
.InsertTraceback(self
.list, demoError
.traceback
)
919 self
.list.SetColumnWidth(0, wx
.LIST_AUTOSIZE
)
920 self
.list.SetColumnWidth(2, wx
.LIST_AUTOSIZE
)
921 self
.box
.Add(wx
.StaticText(self
, -1, "Traceback:")
922 , 0, wx
.ALIGN_CENTER | wx
.TOP
, 5)
923 self
.box
.Add(self
.list, 1, wx
.GROW | wx
.ALIGN_CENTER | wx
.ALL
, 5)
924 self
.box
.Add(wx
.StaticText(self
, -1, "Entries from the demo module are shown in blue\n"
925 + "Double-click on them to go to the offending line")
926 , 0, wx
.ALIGN_CENTER | wx
.BOTTOM
, 5)
929 self
.SetSizer(self
.box
)
932 def InsertTraceback(self
, list, traceback
):
933 #Add the traceback data
934 for x
in range(len(traceback
)):
936 list.InsertStringItem(x
, os
.path
.basename(data
[0])) # Filename
937 list.SetStringItem(x
, 1, str(data
[1])) # Line
938 list.SetStringItem(x
, 2, str(data
[2])) # Function
939 list.SetStringItem(x
, 3, str(data
[3])) # Code
941 # Check whether this entry is from the demo module
942 if data
[0] == "<original>" or data
[0] == "<modified>": # FIXME: make more generalised
943 self
.list.SetItemData(x
, int(data
[1])) # Store line number for easy access
944 # Give it a blue colour
945 item
= self
.list.GetItem(x
)
946 item
.SetTextColour(wx
.BLUE
)
947 self
.list.SetItem(item
)
949 self
.list.SetItemData(x
, -1) # Editor can't jump into this one's code
952 def OnItemSelected(self
, event
):
953 # This occurs before OnDoubleClick and can be used to set the
954 # currentItem. OnDoubleClick doesn't get a wxListEvent....
955 self
.currentItem
= event
.m_itemIndex
959 def OnDoubleClick(self
, event
):
960 # If double-clicking on a demo's entry, jump to the line number
961 line
= self
.list.GetItemData(self
.currentItem
)
963 self
.nb
.SetSelection(1) # Switch to the code viewer tab
964 wx
.CallAfter(self
.codePanel
.JumpToLine
, line
-1, True)
968 #---------------------------------------------------------------------------
970 class wxPythonDemo(wx
.Frame
):
971 overviewText
= "wxPython Overview"
973 def __init__(self
, parent
, title
):
974 wx
.Frame
.__init
__(self
, parent
, -1, title
, size
= (950, 750),
975 style
=wx
.DEFAULT_FRAME_STYLE | wx
.NO_FULL_REPAINT_ON_RESIZE
)
978 self
.cwd
= os
.getcwd()
979 self
.curOverview
= ""
982 self
.useModified
= False
984 icon
= images
.getMondrianIcon()
987 if wx
.Platform
!= '__WXMAC__':
988 # setup a taskbar icon, and catch some events from it
989 dim
= 16 # (may want to use 22 on wxGTK, but 16 looks okay too)
990 icon
= wx
.IconFromBitmap(
991 images
.getMondrianImage().Scale(dim
,dim
).ConvertToBitmap() )
992 #icon = wx.Icon('bmp_source/mondrian.ico', wx.BITMAP_TYPE_ICO)
993 #icon = images.getMondrianIcon()
994 self
.tbicon
= wx
.TaskBarIcon()
995 self
.tbicon
.SetIcon(icon
, "wxPython Demo")
996 self
.tbicon
.Bind(wx
.EVT_TASKBAR_LEFT_DCLICK
, self
.OnTaskBarActivate
)
997 self
.tbicon
.Bind(wx
.EVT_TASKBAR_RIGHT_UP
, self
.OnTaskBarMenu
)
998 self
.tbicon
.Bind(wx
.EVT_MENU
, self
.OnTaskBarActivate
, id=self
.TBMENU_RESTORE
)
999 self
.tbicon
.Bind(wx
.EVT_MENU
, self
.OnTaskBarClose
, id=self
.TBMENU_CLOSE
)
1001 wx
.CallAfter(self
.ShowTip
)
1003 self
.otherWin
= None
1004 self
.Bind(wx
.EVT_IDLE
, self
.OnIdle
)
1005 self
.Bind(wx
.EVT_CLOSE
, self
.OnCloseWindow
)
1006 self
.Bind(wx
.EVT_ICONIZE
, self
.OnIconfiy
)
1007 self
.Bind(wx
.EVT_MAXIMIZE
, self
.OnMaximize
)
1009 self
.Centre(wx
.BOTH
)
1010 self
.CreateStatusBar(1, wx
.ST_SIZEGRIP
)
1012 splitter
= wx
.SplitterWindow(self
, -1, style
=wx
.CLIP_CHILDREN | wx
.SP_LIVE_UPDATE | wx
.SP_3D
)
1013 splitter2
= wx
.SplitterWindow(splitter
, -1, style
=wx
.CLIP_CHILDREN | wx
.SP_LIVE_UPDATE | wx
.SP_3D
)
1015 def EmptyHandler(evt
): pass
1016 #splitter.Bind(wx.EVT_ERASE_BACKGROUND, EmptyHandler)
1017 #splitter2.Bind(wx.EVT_ERASE_BACKGROUND, EmptyHandler)
1019 # Prevent TreeCtrl from displaying all items after destruction when True
1023 self
.nb
= wx
.Notebook(splitter2
, -1, style
=wx
.CLIP_CHILDREN
)
1026 self
.mainmenu
= wx
.MenuBar()
1028 item
= menu
.Append(-1, '&Redirect Output',
1029 'Redirect print statements to a window',
1031 self
.Bind(wx
.EVT_MENU
, self
.OnToggleRedirect
, item
)
1033 item
= menu
.Append(-1, 'E&xit\tAlt-X', 'Get the heck outta here!')
1034 self
.Bind(wx
.EVT_MENU
, self
.OnFileExit
, item
)
1035 wx
.App_SetMacExitMenuItemId(item
.GetId())
1036 self
.mainmenu
.Append(menu
, '&File')
1040 for item
in _treeList
:
1042 for childItem
in item
[1]:
1043 mi
= submenu
.Append(-1, childItem
)
1044 self
.Bind(wx
.EVT_MENU
, self
.OnDemoMenu
, mi
)
1045 menu
.AppendMenu(wx
.NewId(), item
[0], submenu
)
1046 self
.mainmenu
.Append(menu
, '&Demo')
1048 # Make a Demo Code menu
1049 #TODO: Add new menu items
1050 # Like the option-enabled entries to select the
1052 #TODO: should we bother?
1055 #saveID = wx.NewId()
1056 #restoreID = wx.NewId()
1058 #menu.Append(saveID, '&Save\tCtrl-S', 'Save edited demo')
1059 #menu.Append(restoreID, '&Delete Modified\tCtrl-R', 'Delete modified copy')
1060 #self.Bind(wx.EVT_MENU, self.codePage.OnSave, id=saveID)
1061 #self.Bind(wx.EVT_MENU, self.codePage.OnRestore, id=restoreID)
1062 #self.mainmenu.Append(menu, 'Demo &Code')
1068 findnextID
= wx
.NewId()
1070 findItem
= menu
.Append(-1, '&Find\tCtrl-F', 'Find in the Demo Code')
1071 findnextItem
= menu
.Append(-1, 'Find &Next\tF3', 'Find Next')
1072 menu
.AppendSeparator()
1073 helpItem
= menu
.Append(-1, '&About\tCtrl-H', 'wxPython RULES!!!')
1074 wx
.App_SetMacAboutMenuItemId(helpItem
.GetId())
1075 self
.Bind(wx
.EVT_MENU
, self
.OnHelpAbout
, helpItem
)
1076 self
.Bind(wx
.EVT_MENU
, self
.OnHelpFind
, findItem
)
1077 self
.Bind(wx
.EVT_MENU
, self
.OnFindNext
, findnextItem
)
1078 self
.Bind(wx
.EVT_COMMAND_FIND
, self
.OnFind
)
1079 self
.Bind(wx
.EVT_COMMAND_FIND_NEXT
, self
.OnFind
)
1080 self
.Bind(wx
.EVT_COMMAND_FIND_CLOSE
, self
.OnFindClose
)
1081 self
.mainmenu
.Append(menu
, '&Help')
1082 self
.SetMenuBar(self
.mainmenu
)
1084 self
.finddata
= wx
.FindReplaceData()
1087 # This is another way to set Accelerators, in addition to
1088 # using the '\t<key>' syntax in the menu items.
1089 aTable
= wx
.AcceleratorTable([(wx
.ACCEL_ALT
, ord('X'), exitID
),
1090 (wx
.ACCEL_CTRL
, ord('H'), helpID
),
1091 (wx
.ACCEL_CTRL
, ord('F'), findID
),
1092 (wx
.ACCEL_NORMAL
, WXK_F3
, findnextID
)
1094 self
.SetAcceleratorTable(aTable
)
1100 self
.tree
= wx
.TreeCtrl(splitter
, tID
, style
=
1101 wx
.TR_DEFAULT_STYLE
#| wx.TR_HAS_VARIABLE_ROW_HEIGHT
1104 root
= self
.tree
.AddRoot("wxPython Overview")
1106 for item
in _treeList
:
1107 child
= self
.tree
.AppendItem(root
, item
[0])
1108 if not firstChild
: firstChild
= child
1109 for childItem
in item
[1]:
1110 theDemo
= self
.tree
.AppendItem(child
, childItem
)
1111 self
.treeMap
[childItem
] = theDemo
1113 self
.tree
.Expand(root
)
1114 self
.tree
.Expand(firstChild
)
1115 self
.tree
.Bind(wx
.EVT_TREE_ITEM_EXPANDED
, self
.OnItemExpanded
, id=tID
)
1116 self
.tree
.Bind(wx
.EVT_TREE_ITEM_COLLAPSED
, self
.OnItemCollapsed
, id=tID
)
1117 self
.tree
.Bind(wx
.EVT_TREE_SEL_CHANGED
, self
.OnSelChanged
, id=tID
)
1118 self
.tree
.Bind(wx
.EVT_LEFT_DOWN
, self
.OnTreeLeftDown
)
1120 # Set up a wx.html.HtmlWindow on the Overview Notebook page
1121 # we put it in a panel first because there seems to be a
1122 # refresh bug of some sort (wxGTK) when it is directly in
1125 self
.ovr
= wx
.html
.HtmlWindow(self
.nb
, -1, size
=(400, 400))
1126 self
.nb
.AddPage(self
.ovr
, self
.overviewText
)
1128 else: # hopefully I can remove this hacky code soon, see SF bug #216861
1129 panel
= wx
.Panel(self
.nb
, -1, style
=wx
.CLIP_CHILDREN
)
1130 self
.ovr
= wx
.html
.HtmlWindow(panel
, -1, size
=(400, 400))
1131 self
.nb
.AddPage(panel
, self
.overviewText
)
1133 def OnOvrSize(evt
, ovr
=self
.ovr
):
1134 ovr
.SetSize(evt
.GetSize())
1135 panel
.Bind(wx
.EVT_SIZE
, OnOvrSize
)
1136 panel
.Bind(wx
.EVT_ERASE_BACKGROUND
, EmptyHandler
)
1138 if "gtk2" in wx
.PlatformInfo
:
1139 self
.ovr
.NormalizeFontSizes()
1140 self
.SetOverview(self
.overviewText
, mainOverview
)
1143 # Set up a log window
1144 self
.log
= wx
.TextCtrl(splitter2
, -1,
1145 style
= wx
.TE_MULTILINE|wx
.TE_READONLY|wx
.HSCROLL
)
1147 # Set the wxWindows log target to be this textctrl
1148 #wx.Log_SetActiveTarget(wx.LogTextCtrl(self.log))
1150 # But instead of the above we want to show how to use our own wx.Log class
1151 wx
.Log_SetActiveTarget(MyLog(self
.log
))
1153 # for serious debugging
1154 #wx.Log_SetActiveTarget(wx.LogStderr())
1155 #wx.Log_SetTraceMask(wx.TraceMessages)
1158 # add the windows to the splitter and split it.
1159 splitter2
.SplitHorizontally(self
.nb
, self
.log
, -160)
1160 splitter
.SplitVertically(self
.tree
, splitter2
, 200)
1162 splitter
.SetMinimumPaneSize(20)
1163 splitter2
.SetMinimumPaneSize(20)
1165 # Make the splitter on the right expand the top window when resized
1166 def SplitterOnSize(evt
):
1167 splitter
= evt
.GetEventObject()
1168 sz
= splitter
.GetSize()
1169 splitter
.SetSashPosition(sz
.height
- 160, False)
1172 splitter2
.Bind(wx
.EVT_SIZE
, SplitterOnSize
)
1174 # select initial items
1175 self
.nb
.SetSelection(0)
1176 self
.tree
.SelectItem(root
)
1178 # Load 'Main' module
1179 self
.LoadDemo(self
.overviewText
)
1182 # select some other initial module?
1183 if len(sys
.argv
) > 1:
1185 if arg
.endswith('.py'):
1187 selectedDemo
= self
.treeMap
.get(arg
, None)
1189 self
.tree
.SelectItem(selectedDemo
)
1190 self
.tree
.EnsureVisible(selectedDemo
)
1193 #---------------------------------------------
1194 def WriteText(self
, text
):
1195 if text
[-1:] == '\n':
1199 def write(self
, txt
):
1202 #---------------------------------------------
1203 def OnItemExpanded(self
, event
):
1204 item
= event
.GetItem()
1205 wx
.LogMessage("OnItemExpanded: %s" % self
.tree
.GetItemText(item
))
1208 #---------------------------------------------
1209 def OnItemCollapsed(self
, event
):
1210 item
= event
.GetItem()
1211 wx
.LogMessage("OnItemCollapsed: %s" % self
.tree
.GetItemText(item
))
1214 #---------------------------------------------
1215 def OnTreeLeftDown(self
, event
):
1216 # reset the overview text if the tree item is clicked on again
1217 pt
= event
.GetPosition();
1218 item
, flags
= self
.tree
.HitTest(pt
)
1219 if item
== self
.tree
.GetSelection():
1220 self
.SetOverview(self
.tree
.GetItemText(item
)+" Overview", self
.curOverview
)
1223 #---------------------------------------------
1224 def OnSelChanged(self
, event
):
1225 if self
.dying
or not self
.loaded
:
1228 item
= event
.GetItem()
1229 itemText
= self
.tree
.GetItemText(item
)
1230 self
.LoadDemo(itemText
)
1232 #---------------------------------------------
1233 def LoadDemo(self
, demoName
):
1235 wx
.BeginBusyCursor()
1238 self
.ShutdownDemoModule()
1240 if demoName
== self
.overviewText
:
1241 # User selected the "wxPython Overview" node
1243 # Changing the main window at runtime not yet supported...
1244 self
.demoModules
= DemoModules(__name__
)
1245 self
.SetOverview(self
.overviewText
, mainOverview
)
1246 self
.LoadDemoSource()
1247 self
.UpdateNotebook(0)
1249 if os
.path
.exists(GetOriginalFilename(demoName
)):
1250 wx
.LogMessage("Loading demo %s.py..." % demoName
)
1251 self
.demoModules
= DemoModules(demoName
)
1252 self
.LoadDemoSource()
1255 self
.SetOverview("wxPython", mainOverview
)
1256 self
.codePage
= None
1257 self
.UpdateNotebook(0)
1261 #---------------------------------------------
1262 def LoadDemoSource(self
):
1263 self
.codePage
= None
1264 self
.codePage
= DemoCodePanel(self
.nb
, self
)
1265 self
.codePage
.LoadDemo(self
.demoModules
)
1267 #---------------------------------------------
1268 def RunModule(self
):
1269 """Runs the active module"""
1271 module
= self
.demoModules
.GetActive()
1272 self
.ShutdownDemoModule()
1275 # o If the demo returns a window it is placed in a tab.
1276 # o Otherwise, a placeholder tab is created, informing the user that the
1277 # demo runs outside the main window, and allowing it to be reloaded.
1278 # o If an error occurs (or has occured before) an error tab is created.
1280 if module
is not None:
1281 wx
.LogMessage("Running demo module...")
1282 if hasattr(module
, "overview"):
1283 overviewText
= module
.overview
1285 # in case runTest is modal, make sure things look right...
1289 self
.demoPage
= module
.runTest(self
, self
.nb
, self
)
1290 if self
.demoPage
is None:
1291 self
.demoPage
= ReloadDemoPanel(self
.nb
, self
.codePage
, self
)
1293 self
.demoPage
= DemoErrorPanel(self
.nb
, self
.codePage
, DemoError(sys
.exc_info()), self
)
1295 # There was a previous error in compiling or exec-ing
1296 self
.demoPage
= DemoErrorPanel(self
.nb
, self
.codePage
, self
.demoModules
.GetErrorInfo(), self
)
1297 self
.SetOverview(self
.demoModules
.name
+ " Overview", overviewText
)
1298 self
.UpdateNotebook()
1300 #---------------------------------------------
1301 def ShutdownDemoModule(self
):
1303 # inform the window that it's time to quit if it cares
1304 if hasattr(self
.demoPage
, "ShutdownDemo"):
1305 self
.demoPage
.ShutdownDemo()
1306 wx
.YieldIfNeeded() # in case the page has pending events
1307 self
.demoPage
= None
1309 #---------------------------------------------
1310 def UpdateNotebook(self
, select
= -1):
1314 def UpdatePage(page
, pageText
):
1317 for i
in range(nb
.GetPageCount()):
1318 if nb
.GetPageText(i
) == pageText
:
1327 # panel = wx.Panel(nb, -1)
1328 # page.Reparent(panel)
1330 # nb.AddPage(panel, pageText)
1331 nb
.AddPage(page
, pageText
)
1332 if debug
: wx
.LogMessage("DBG: ADDED %s" % pageText
)
1334 # if nb.GetPage(pagePos).page != page:
1335 if nb
.GetPage(pagePos
) != page
:
1336 # Reload an existing page
1339 # panel = nb.GetPage(pagePos)
1341 # page.Reparent(panel)
1343 nb
.DeletePage(pagePos
)
1344 nb
.InsertPage(pagePos
, page
, pageText
)
1346 if debug
: wx
.LogMessage("DBG: RELOADED %s" % pageText
)
1348 # Excellent! No redraw/flicker
1349 if debug
: wx
.LogMessage("DBG: SAVED from reloading %s" % pageText
)
1352 nb
.DeletePage(pagePos
)
1353 if debug
: wx
.LogMessage("DBG: DELETED %s" % pageText
)
1355 if debug
: wx
.LogMessage("DBG: STILL GONE - %s" % pageText
)
1358 select
= nb
.GetSelection()
1360 UpdatePage(self
.codePage
, "Demo Code")
1361 UpdatePage(self
.demoPage
, "Demo")
1364 nb
.SetSelection(select
)
1366 #---------------------------------------------
1367 def SetOverview(self
, name
, text
):
1368 self
.curOverview
= text
1370 if lead
!= '<html>' and lead
!= '<HTML>':
1371 text
= '<br>'.join(text
.split('\n'))
1373 text
= text
.decode('iso8859_1')
1374 self
.ovr
.SetPage(text
)
1375 self
.nb
.SetPageText(0, name
)
1377 #---------------------------------------------
1379 def OnFileExit(self
, *event
):
1382 def OnToggleRedirect(self
, event
):
1386 print "Print statements and other standard output will now be directed to this window."
1389 print "Print statements and other standard output will now be sent to the usual location."
1391 def OnHelpAbout(self
, event
):
1392 from About
import MyAboutBox
1393 about
= MyAboutBox(self
)
1397 def OnHelpFind(self
, event
):
1398 self
.nb
.SetSelection(1)
1399 self
.finddlg
= wx
.FindReplaceDialog(self
, self
.finddata
, "Find",
1403 self
.finddlg
.Show(True)
1405 def OnFind(self
, event
):
1406 editor
= self
.codePage
.editor
1407 self
.nb
.SetSelection(1)
1408 end
= editor
.GetLastPosition()
1409 textstring
= editor
.GetRange(0, end
).lower()
1410 start
= editor
.GetSelection()[1]
1411 findstring
= self
.finddata
.GetFindString().lower()
1412 loc
= textstring
.find(findstring
, start
)
1413 if loc
== -1 and start
!= 0:
1414 # string not found, start at beginning
1416 loc
= textstring
.find(findstring
, start
)
1418 dlg
= wx
.MessageDialog(self
, 'Find String Not Found',
1419 'Find String Not Found in Demo File',
1420 wx
.OK | wx
.ICON_INFORMATION
)
1425 self
.finddlg
.SetFocus()
1428 self
.finddlg
.Destroy()
1429 editor
.ShowPosition(loc
)
1430 editor
.SetSelection(loc
, loc
+ len(findstring
))
1434 def OnFindNext(self
, event
):
1435 if self
.finddata
.GetFindString():
1438 self
.OnHelpFind(event
)
1440 def OnFindClose(self
, event
):
1441 event
.GetDialog().Destroy()
1444 #---------------------------------------------
1445 def OnCloseWindow(self
, event
):
1447 self
.demoPage
= None
1448 self
.codePage
= None
1449 self
.mainmenu
= None
1453 #---------------------------------------------
1454 def OnIdle(self
, event
):
1456 self
.otherWin
.Raise()
1457 self
.demoPage
= self
.otherWin
1458 self
.otherWin
= None
1461 #---------------------------------------------
1464 showTipText
= open(opj("data/showTips")).read()
1465 showTip
, index
= eval(showTipText
)
1467 showTip
, index
= (1, 0)
1469 tp
= wx
.CreateFileTipProvider(opj("data/tips.txt"), index
)
1471 showTip
= wx
.ShowTip(self
, tp
)
1472 index
= tp
.GetCurrentTip()
1473 open(opj("data/showTips"), "w").write(str( (showTip
, index
) ))
1476 #---------------------------------------------
1477 def OnDemoMenu(self
, event
):
1479 selectedDemo
= self
.treeMap
[self
.mainmenu
.GetLabel(event
.GetId())]
1483 self
.tree
.SelectItem(selectedDemo
)
1484 self
.tree
.EnsureVisible(selectedDemo
)
1487 #---------------------------------------------
1488 def OnTaskBarActivate(self
, evt
):
1489 if self
.IsIconized():
1491 if not self
.IsShown():
1495 #---------------------------------------------
1497 TBMENU_RESTORE
= 1000
1500 def OnTaskBarMenu(self
, evt
):
1502 menu
.Append(self
.TBMENU_RESTORE
, "Restore wxPython Demo")
1503 menu
.Append(self
.TBMENU_CLOSE
, "Close")
1504 self
.tbicon
.PopupMenu(menu
)
1507 #---------------------------------------------
1508 def OnTaskBarClose(self
, evt
):
1511 # because of the way wx.TaskBarIcon.PopupMenu is implemented we have to
1512 # prod the main idle handler a bit to get the window to actually close
1513 wx
.GetApp().ProcessIdle()
1516 #---------------------------------------------
1517 def OnIconfiy(self
, evt
):
1518 wx
.LogMessage("OnIconfiy: %d" % evt
.Iconized())
1521 #---------------------------------------------
1522 def OnMaximize(self
, evt
):
1523 wx
.LogMessage("OnMaximize")
1529 #---------------------------------------------------------------------------
1530 #---------------------------------------------------------------------------
1532 class MySplashScreen(wx
.SplashScreen
):
1534 bmp
= wx
.Image(opj("bitmaps/splash.gif")).ConvertToBitmap()
1535 wx
.SplashScreen
.__init
__(self
, bmp
,
1536 wx
.SPLASH_CENTRE_ON_SCREEN | wx
.SPLASH_TIMEOUT
,
1538 self
.Bind(wx
.EVT_CLOSE
, self
.OnClose
)
1540 def OnClose(self
, evt
):
1542 frame
= wxPythonDemo(None, "wxPython: (A Demonstration)")
1544 evt
.Skip() # Make sure the default handler runs too...
1547 class MyApp(wx
.App
):
1550 Create and show the splash screen. It will then create and show
1551 the main frame when it is time to do so.
1555 #self.SetAssertMode(wx.PYAPP_ASSERT_DIALOG)
1557 # Normally when using a SplashScreen you would create it, show
1558 # it and then continue on with the applicaiton's
1559 # initialization, finally creating and showing the main
1560 # application window(s). In this case we have nothing else to
1561 # do so we'll delay showing the main frame until later (see
1562 # OnClose above) so the users can see the SplashScreen effect.
1563 splash
= MySplashScreen()
1570 #---------------------------------------------------------------------------
1574 demoPath
= os
.path
.dirname(__file__
)
1578 app
= MyApp(0) ##wx.Platform == "__WXMAC__")
1581 #---------------------------------------------------------------------------
1584 mainOverview
= """<html><body>
1587 <p> wxPython is a <b>GUI toolkit</b> for the Python programming
1588 language. It allows Python programmers to create programs with a
1589 robust, highly functional graphical user interface, simply and easily.
1590 It is implemented as a Python extension module (native code) that
1591 wraps the popular wxWindows cross platform GUI library, which is
1594 <p> Like Python and wxWindows, wxPython is <b>Open Source</b> which
1595 means that it is free for anyone to use and the source code is
1596 available for anyone to look at and modify. Or anyone can contribute
1597 fixes or enhancements to the project.
1599 <p> wxPython is a <b>cross-platform</b> toolkit. This means that the
1600 same program will run on multiple platforms without modification.
1601 Currently supported platforms are 32-bit Microsoft Windows, most Unix
1602 or unix-like systems, and Macintosh OS X. Since the language is
1603 Python, wxPython programs are <b>simple, easy</b> to write and easy to
1606 <p> <b>This demo</b> is not only a collection of test cases for
1607 wxPython, but is also designed to help you learn about and how to use
1608 wxPython. Each sample is listed in the tree control on the left.
1609 When a sample is selected in the tree then a module is loaded and run
1610 (usually in a tab of this notebook,) and the source code of the module
1611 is loaded in another tab for you to browse and learn from.
1616 #----------------------------------------------------------------------------
1617 #----------------------------------------------------------------------------
1619 if __name__
== '__main__':
1623 #----------------------------------------------------------------------------