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 ('Frames and Dialogs', [
78 # dialogs from libraries
81 'MultipleChoiceDialog',
82 'ScrolledMessageDialog',
86 ('Core Windows/Controls', [
122 ('Custom Controls', [
135 # controls coming from other libraries
136 ('More Windows/Controls', [
137 'ActiveX_FlashWindow',
138 'ActiveX_IEHtmlWindow',
140 #'RightTextCtrl', deprecated as we have wxTE_RIGHT now.
153 'MaskedEditControls',
169 # How to lay out the controls in a frame/dialog
179 'XmlResourceHandler',
180 'XmlResourceSubclass',
184 ('Process and Events', [
191 ##'infoframe', # needs better explaination and some fixing
195 ('Clipboard and DnD', [
216 ##'DialogUnits', # needs more explainations
228 # need libs not coming with the demo
229 ('Samples using an external library', [
234 ('Check out the samples dir too', [
241 #---------------------------------------------------------------------------
242 # Show how to derive a custom wxLog class
244 class MyLog(wx
.PyLog
):
245 def __init__(self
, textCtrl
, logTime
=0):
246 wx
.PyLog
.__init
__(self
)
248 self
.logTime
= logTime
250 def DoLogString(self
, message
, timeStamp
):
252 message
= time
.strftime("%X", time
.localtime(timeStamp
)) + \
255 self
.tc
.AppendText(message
+ '\n')
258 class MyTP(wx
.PyTipProvider
):
260 return "This is my tip"
263 #---------------------------------------------------------------------------
264 # A class to be used to display source code in the demo. Try using the
265 # wxSTC in the StyledTextCtrl_2 sample first, fall back to wxTextCtrl
266 # if there is an error, such as the stc module not being present.
270 ##raise ImportError # for testing the alternate implementation
272 from StyledTextCtrl_2
import PythonSTC
274 class DemoCodeEditor(PythonSTC
):
275 def __init__(self
, parent
):
276 PythonSTC
.__init
__(self
, parent
, -1, style
=wx
.BORDER_NONE
)
279 # Some methods to make it compatible with how the wxTextCtrl is used
280 def SetValue(self
, value
):
282 value
= value
.decode('iso8859_1')
284 self
.EmptyUndoBuffer()
287 def IsModified(self
):
288 return self
.GetModify()
293 def SetInsertionPoint(self
, pos
):
294 self
.SetCurrentPos(pos
)
297 def ShowPosition(self
, pos
):
298 line
= self
.LineFromPosition(pos
)
299 #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
358 # EOL: Since we are loading/saving ourselves, and the
359 # strings will always have \n's in them, set the STC to
360 # edit them that way.
361 self
.SetEOLMode(wx
.stc
.STC_EOL_LF
)
362 self
.SetViewEOL(False)
364 # No right-edge mode indicator
365 self
.SetEdgeMode(stc
.STC_EDGE_NONE
)
367 # Setup a margin to hold fold markers
368 self
.SetMarginType(2, stc
.STC_MARGIN_SYMBOL
)
369 self
.SetMarginMask(2, stc
.STC_MASK_FOLDERS
)
370 self
.SetMarginSensitive(2, True)
371 self
.SetMarginWidth(2, 12)
373 # and now set up the fold markers
374 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEREND
, stc
.STC_MARK_BOXPLUSCONNECTED
, "white", "black")
375 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEROPENMID
, stc
.STC_MARK_BOXMINUSCONNECTED
, "white", "black")
376 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERMIDTAIL
, stc
.STC_MARK_TCORNER
, "white", "black")
377 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERTAIL
, stc
.STC_MARK_LCORNER
, "white", "black")
378 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDERSUB
, stc
.STC_MARK_VLINE
, "white", "black")
379 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDER
, stc
.STC_MARK_BOXPLUS
, "white", "black")
380 self
.MarkerDefine(stc
.STC_MARKNUM_FOLDEROPEN
, stc
.STC_MARK_BOXMINUS
, "white", "black")
382 # Global default style
383 if wx
.Platform
== '__WXMSW__':
384 self
.StyleSetSpec(stc
.STC_STYLE_DEFAULT
,
385 'fore:#000000,back:#FFFFFF,face:Courier New,size:9')
387 self
.StyleSetSpec(stc
.STC_STYLE_DEFAULT
,
388 'fore:#000000,back:#FFFFFF,face:Courier,size:12')
390 # Clear styles and revert to default.
393 # Following style specs only indicate differences from default.
394 # The rest remains unchanged.
396 # Line numbers in margin
397 self
.StyleSetSpec(wx
.stc
.STC_STYLE_LINENUMBER
,'fore:#000000,back:#99A9C2')
400 self
.StyleSetSpec(wx
.stc
.STC_STYLE_BRACELIGHT
,'fore:#00009D,back:#FFFF00')
402 self
.StyleSetSpec(wx
.stc
.STC_STYLE_BRACEBAD
,'fore:#00009D,back:#FF0000')
404 self
.StyleSetSpec(wx
.stc
.STC_STYLE_INDENTGUIDE
, "fore:#CDCDCD")
407 self
.StyleSetSpec(wx
.stc
.STC_P_DEFAULT
, 'fore:#000000')
409 self
.StyleSetSpec(wx
.stc
.STC_P_COMMENTLINE
, 'fore:#008000,back:#F0FFF0')
410 self
.StyleSetSpec(wx
.stc
.STC_P_COMMENTBLOCK
, 'fore:#008000,back:#F0FFF0')
412 self
.StyleSetSpec(wx
.stc
.STC_P_NUMBER
, 'fore:#008080')
413 # Strings and characters
414 self
.StyleSetSpec(wx
.stc
.STC_P_STRING
, 'fore:#800080')
415 self
.StyleSetSpec(wx
.stc
.STC_P_CHARACTER
, 'fore:#800080')
417 self
.StyleSetSpec(wx
.stc
.STC_P_WORD
, 'fore:#000080,bold')
419 self
.StyleSetSpec(wx
.stc
.STC_P_TRIPLE
, 'fore:#800080,back:#FFFFEA')
420 self
.StyleSetSpec(wx
.stc
.STC_P_TRIPLEDOUBLE
, 'fore:#800080,back:#FFFFEA')
422 self
.StyleSetSpec(wx
.stc
.STC_P_CLASSNAME
, 'fore:#0000FF,bold')
424 self
.StyleSetSpec(wx
.stc
.STC_P_DEFNAME
, 'fore:#008080,bold')
426 self
.StyleSetSpec(wx
.stc
.STC_P_OPERATOR
, 'fore:#800000,bold')
427 # Identifiers. I leave this as not bold because everything seems
428 # to be an identifier if it doesn't match the above criterae
429 self
.StyleSetSpec(wx
.stc
.STC_P_IDENTIFIER
, 'fore:#000000')
432 self
.SetCaretForeground("BLUE")
433 # Selection background
434 self
.SetSelBackground(1, '#66CCFF')
436 self
.SetSelBackground(True, wx
.SystemSettings_GetColour(wx
.SYS_COLOUR_HIGHLIGHT
))
437 self
.SetSelForeground(True, wx
.SystemSettings_GetColour(wx
.SYS_COLOUR_HIGHLIGHTTEXT
))
439 def RegisterModifiedEvent(self
, eventHandler
):
440 self
.Bind(wx
.stc
.EVT_STC_CHANGE
, eventHandler
)
444 class DemoCodeEditor(wx
.TextCtrl
):
445 def __init__(self
, parent
):
446 wx
.TextCtrl
.__init
__(self
, parent
, -1, style
=
447 wx
.TE_MULTILINE | wx
.HSCROLL | wx
.TE_RICH2 | wx
.TE_NOHIDESEL
)
449 def RegisterModifiedEvent(self
, eventHandler
):
450 self
.Bind(wx
.EVT_TEXT
, eventHandler
)
452 def SetReadOnly(self
, flag
):
453 self
.SetEditable(not flag
)
454 # NOTE: STC already has this method
457 return self
.GetValue()
459 def GetPositionFromLine(line
):
460 return self
.XYToPosition(0,line
)
462 def GotoLine(self
, line
):
463 pos
= self
.editor
.GetPositionFromLine(line
)
464 self
.editor
.SetInsertionPoint(pos
)
465 self
.editor
.ShowPosition(pos
)
467 def SelectLine(self
, line
):
468 start
= self
.GetPositionFromLine(line
)
469 end
= start
+ self
.GetLineLength(line
)
470 self
.SetSelection(start
, end
)
473 #---------------------------------------------------------------------------
474 # Constants for module versions
478 modDefault
= modOriginal
480 #---------------------------------------------------------------------------
482 class DemoCodePanel(wx
.Panel
):
483 """Panel for the 'Demo Code' tab"""
484 def __init__(self
, parent
, mainFrame
):
485 wx
.Panel
.__init
__(self
, parent
, size
=(1,1))
486 if 'wxMSW' in wx
.PlatformInfo
:
488 self
.mainFrame
= mainFrame
489 self
.editor
= DemoCodeEditor(self
)
490 self
.editor
.RegisterModifiedEvent(self
.OnCodeModified
)
492 self
.btnSave
= wx
.Button(self
, -1, "Save Changes")
493 self
.btnRestore
= wx
.Button(self
, -1, "Delete Modified")
494 self
.btnSave
.Enable(False)
495 self
.btnSave
.Bind(wx
.EVT_BUTTON
, self
.OnSave
)
496 self
.btnRestore
.Bind(wx
.EVT_BUTTON
, self
.OnRestore
)
498 self
.radioButtons
= { modOriginal
: wx
.RadioButton(self
, -1, "Original", style
= wx
.RB_GROUP
),
499 modModified
: wx
.RadioButton(self
, -1, "Modified") }
501 self
.controlBox
= wx
.BoxSizer(wx
.HORIZONTAL
)
502 self
.controlBox
.Add(wx
.StaticText(self
, -1, "Active Version:"), 0,
503 wx
.RIGHT | wx
.LEFT | wx
.ALIGN_CENTER_VERTICAL
, 5)
504 for modID
, radioButton
in self
.radioButtons
.items():
505 self
.controlBox
.Add(radioButton
, 0, wx
.EXPAND | wx
.RIGHT
, 5)
506 radioButton
.modID
= modID
# makes it easier for the event handler
507 radioButton
.Bind(wx
.EVT_RADIOBUTTON
, self
.OnRadioButton
)
509 self
.controlBox
.Add(self
.btnSave
, 0, wx
.RIGHT
, 5)
510 self
.controlBox
.Add(self
.btnRestore
, 0)
512 self
.box
= wx
.BoxSizer(wx
.VERTICAL
)
513 self
.box
.Add(self
.controlBox
, 0, wx
.EXPAND
)
514 self
.box
.Add(wx
.StaticLine(self
), 0, wx
.EXPAND
)
515 self
.box
.Add(self
.editor
, 1, wx
.EXPAND
)
518 self
.SetSizer(self
.box
)
521 # Loads a demo from a DemoModules object
522 def LoadDemo(self
, demoModules
):
523 self
.demoModules
= demoModules
524 if (modDefault
== modModified
) and demoModules
.Exists(modModified
):
525 demoModules
.SetActive(modModified
)
527 demoModules
.SetActive(modOriginal
)
528 self
.radioButtons
[demoModules
.GetActiveID()].Enable(True)
529 self
.ActiveModuleChanged()
532 def ActiveModuleChanged(self
):
533 self
.LoadDemoSource(self
.demoModules
.GetSource())
534 self
.UpdateControlState()
538 def LoadDemoSource(self
, source
):
540 self
.editor
.SetValue(source
)
542 self
.btnSave
.Enable(False)
545 def JumpToLine(self
, line
, highlight
=False):
546 self
.editor
.GotoLine(line
)
547 self
.editor
.SetFocus()
549 self
.editor
.SelectLine(line
)
552 def UpdateControlState(self
):
553 active
= self
.demoModules
.GetActiveID()
554 # Update the radio/restore buttons
555 for moduleID
in self
.radioButtons
:
556 btn
= self
.radioButtons
[moduleID
]
557 if moduleID
== active
:
562 if self
.demoModules
.Exists(moduleID
):
564 if moduleID
== modModified
:
565 self
.btnRestore
.Enable(True)
568 if moduleID
== modModified
:
569 self
.btnRestore
.Enable(False)
572 def OnRadioButton(self
, event
):
573 radioSelected
= event
.GetEventObject()
574 modSelected
= radioSelected
.modID
575 if modSelected
!= self
.demoModules
.GetActiveID():
576 busy
= wx
.BusyInfo("Reloading demo module...")
577 self
.demoModules
.SetActive(modSelected
)
578 self
.ActiveModuleChanged()
581 def ReloadDemo(self
):
582 if self
.demoModules
.name
!= __name__
:
583 self
.mainFrame
.RunModule()
586 def OnCodeModified(self
, event
):
587 self
.btnSave
.Enable(self
.editor
.IsModified())
590 def OnSave(self
, event
):
591 if self
.demoModules
.Exists(modModified
):
592 if self
.demoModules
.GetActiveID() == modOriginal
:
593 overwriteMsg
= "You are about to overwrite an already existing modified copy\n" + \
594 "Do you want to continue?"
595 dlg
= wx
.MessageDialog(self
, overwriteMsg
, "wxPython Demo",
596 wx
.YES_NO | wx
.NO_DEFAULT| wx
.ICON_EXCLAMATION
)
597 result
= dlg
.ShowModal()
598 if result
== wx
.ID_NO
:
602 self
.demoModules
.SetActive(modModified
)
603 modifiedFilename
= GetModifiedFilename(self
.demoModules
.name
)
605 # Create the demo directory if one doesn't already exist
606 if not os
.path
.exists(GetModifiedDirectory()):
608 os
.makedirs(GetModifiedDirectory())
609 if not os
.path
.exists(GetModifiedDirectory()):
610 wx
.LogMessage("BUG: Created demo directory but it still doesn't exit")
613 wx
.LogMessage("Error creating demo directory: %s" % GetModifiedDirectory())
616 wx
.LogMessage("Created directory for modified demos: %s" % GetModifiedDirectory())
619 f
= open(modifiedFilename
, "wt")
620 source
= self
.editor
.GetText()
626 busy
= wx
.BusyInfo("Reloading demo module...")
627 self
.demoModules
.LoadFromFile(modModified
, modifiedFilename
)
628 self
.ActiveModuleChanged()
631 def OnRestore(self
, event
): # Handles the "Delete Modified" button
632 modifiedFilename
= GetModifiedFilename(self
.demoModules
.name
)
633 self
.demoModules
.Delete(modModified
)
634 os
.unlink(modifiedFilename
) # Delete the modified copy
635 busy
= wx
.BusyInfo("Reloading demo module...")
636 self
.ActiveModuleChanged()
639 #---------------------------------------------------------------------------
642 """Convert paths to the platform-specific separator"""
643 str = apply(os
.path
.join
, tuple(path
.split('/')))
644 # HACK: on Linux, a leading / gets lost...
645 if path
.startswith('/'):
650 def GetModifiedDirectory():
652 Returns the directory where modified versions of the demo files
655 return opj(wx
.GetHomeDir() + "/.wxPyDemo/modified/")
658 def GetModifiedFilename(name
):
660 Returns the filename of the modified version of the specified demo
662 if not name
.endswith(".py"):
664 return GetModifiedDirectory() + name
667 def GetOriginalFilename(name
):
669 Returns the filename of the original version of the specified demo
671 if not name
.endswith(".py"):
676 def DoesModifiedExist(name
):
677 """Returns whether the specified demo has a modified copy"""
678 if os
.path
.exists(GetModifiedFilename(name
)):
684 #---------------------------------------------------------------------------
686 class ModuleDictWrapper
:
687 """Emulates a module with a dynamically compiled __dict__"""
688 def __init__(self
, dict):
691 def __getattr__(self
, name
):
692 if name
in self
.dict:
693 return self
.dict[name
]
699 Dynamically manages the original/modified versions of a demo
702 def __init__(self
, name
):
706 # (dict , source , filename , description , error information )
707 # ( 0 , 1 , 2 , 3 , 4 )
708 self
.modules
= [[None, "" , "" , "<original>" , None],
709 [None, "" , "" , "<modified>" , None]]
711 # load original module
712 self
.LoadFromFile(modOriginal
, GetOriginalFilename(name
))
713 self
.SetActive(modOriginal
)
715 # load modified module (if one exists)
716 if DoesModifiedExist(name
):
717 self
.LoadFromFile(modModified
, GetModifiedFilename(name
))
720 def LoadFromFile(self
, modID
, filename
):
721 self
.modules
[modID
][2] = filename
722 file = open(filename
, "rt")
723 self
.LoadFromSource(modID
, file.read())
727 def LoadFromSource(self
, modID
, source
):
728 self
.modules
[modID
][1] = source
732 def LoadDict(self
, modID
):
733 if self
.name
!= __name__
:
734 source
= self
.modules
[modID
][1]
735 description
= self
.modules
[modID
][3]
738 self
.modules
[modID
][0] = {}
739 code
= compile(source
, description
, "exec")
740 exec code
in self
.modules
[modID
][0]
742 self
.modules
[modID
][4] = DemoError(sys
.exc_info())
743 self
.modules
[modID
][0] = None
745 self
.modules
[modID
][4] = None
748 def SetActive(self
, modID
):
749 if modID
!= modOriginal
and modID
!= modModified
:
752 self
.modActive
= modID
756 dict = self
.modules
[self
.modActive
][0]
760 return ModuleDictWrapper(dict)
763 def GetActiveID(self
):
764 return self
.modActive
767 def GetSource(self
, modID
= None):
769 modID
= self
.modActive
770 return self
.modules
[modID
][1]
773 def GetFilename(self
, modID
= None):
775 modID
= self
.modActive
776 return self
.modules
[self
.modActive
][2]
779 def GetErrorInfo(self
, modID
= None):
781 modID
= self
.modActive
782 return self
.modules
[self
.modActive
][4]
785 def Exists(self
, modID
):
786 return self
.modules
[modID
][1] != ""
789 def UpdateFile(self
, modID
= None):
790 """Updates the file from which a module was loaded
791 with (possibly updated) source"""
793 modID
= self
.modActive
795 source
= self
.modules
[modID
][1]
796 filename
= self
.modules
[modID
][2]
799 file = open(filename
, "wt")
805 def Delete(self
, modID
):
806 if self
.modActive
== modID
:
809 self
.modules
[modID
][0] = None
810 self
.modules
[modID
][1] = ""
811 self
.modules
[modID
][2] = ""
814 #---------------------------------------------------------------------------
817 """Wraps and stores information about the current exception"""
818 def __init__(self
, exc_info
):
821 excType
, excValue
= exc_info
[:2]
822 # traceback list entries: (filename, line number, function name, text)
823 self
.traceback
= traceback
.extract_tb(exc_info
[2])
825 # --Based on traceback.py::format_exception_only()--
826 if type(excType
) == types
.ClassType
:
827 self
.exception_type
= excType
.__name
__
829 self
.exception_type
= excType
831 # If it's a syntax error, extra information needs
832 # to be added to the traceback
833 if excType
is SyntaxError:
835 msg
, (filename
, lineno
, self
.offset
, line
) = excValue
840 filename
= "<string>"
842 self
.traceback
.append( (filename
, lineno
, "", line
) )
845 self
.exception_details
= str(excValue
)
847 self
.exception_details
= "<unprintable %s object>" & type(excValue
).__name
__
854 Details : %s" % ( str(self
.exception_type
), str(self
.traceback
), self
.exception_details
)
857 #---------------------------------------------------------------------------
859 class DemoErrorPanel(wx
.Panel
):
860 """Panel put into the demo tab when the demo fails to run due to errors"""
862 def __init__(self
, parent
, codePanel
, demoError
, log
):
863 wx
.Panel
.__init
__(self
, parent
, -1)#, style=wx.NO_FULL_REPAINT_ON_RESIZE)
864 self
.codePanel
= codePanel
868 self
.box
= wx
.BoxSizer(wx
.VERTICAL
)
871 self
.box
.Add(wx
.StaticText(self
, -1, "An error has occured while trying to run the demo")
872 , 0, wx
.ALIGN_CENTER | wx
.TOP
, 10)
874 # Exception Information
875 boxInfo
= wx
.StaticBox(self
, -1, "Exception Info" )
876 boxInfoSizer
= wx
.StaticBoxSizer(boxInfo
, wx
.VERTICAL
) # Used to center the grid within the box
877 boxInfoGrid
= wx
.FlexGridSizer(0, 2, 0, 0)
878 textFlags
= wx
.ALIGN_RIGHT | wx
.LEFT | wx
.RIGHT | wx
.TOP
879 boxInfoGrid
.Add(wx
.StaticText(self
, -1, "Type: "), 0, textFlags
, 5 )
880 boxInfoGrid
.Add(wx
.StaticText(self
, -1, demoError
.exception_type
) , 0, textFlags
, 5 )
881 boxInfoGrid
.Add(wx
.StaticText(self
, -1, "Details: ") , 0, textFlags
, 5 )
882 boxInfoGrid
.Add(wx
.StaticText(self
, -1, demoError
.exception_details
) , 0, textFlags
, 5 )
883 boxInfoSizer
.Add(boxInfoGrid
, 0, wx
.ALIGN_CENTRE | wx
.ALL
, 5 )
884 self
.box
.Add(boxInfoSizer
, 0, wx
.ALIGN_CENTER | wx
.ALL
, 5)
886 # Set up the traceback list
887 # This one automatically resizes last column to take up remaining space
888 from ListCtrl
import TestListCtrl
889 self
.list = TestListCtrl(self
, -1, style
=wx
.LC_REPORT | wx
.SUNKEN_BORDER
)
890 self
.list.Bind(wx
.EVT_LEFT_DCLICK
, self
.OnDoubleClick
)
891 self
.list.Bind(wx
.EVT_LIST_ITEM_SELECTED
, self
.OnItemSelected
)
892 self
.list.InsertColumn(0, "Filename")
893 self
.list.InsertColumn(1, "Line", wx
.LIST_FORMAT_RIGHT
)
894 self
.list.InsertColumn(2, "Function")
895 self
.list.InsertColumn(3, "Code")
896 self
.InsertTraceback(self
.list, demoError
.traceback
)
897 self
.list.SetColumnWidth(0, wx
.LIST_AUTOSIZE
)
898 self
.list.SetColumnWidth(2, wx
.LIST_AUTOSIZE
)
899 self
.box
.Add(wx
.StaticText(self
, -1, "Traceback:")
900 , 0, wx
.ALIGN_CENTER | wx
.TOP
, 5)
901 self
.box
.Add(self
.list, 1, wx
.GROW | wx
.ALIGN_CENTER | wx
.ALL
, 5)
902 self
.box
.Add(wx
.StaticText(self
, -1, "Entries from the demo module are shown in blue\n"
903 + "Double-click on them to go to the offending line")
904 , 0, wx
.ALIGN_CENTER | wx
.BOTTOM
, 5)
907 self
.SetSizer(self
.box
)
910 def InsertTraceback(self
, list, traceback
):
911 #Add the traceback data
912 for x
in range(len(traceback
)):
914 list.InsertStringItem(x
, os
.path
.basename(data
[0])) # Filename
915 list.SetStringItem(x
, 1, str(data
[1])) # Line
916 list.SetStringItem(x
, 2, str(data
[2])) # Function
917 list.SetStringItem(x
, 3, str(data
[3])) # Code
919 # Check whether this entry is from the demo module
920 if data
[0] == "<original>" or data
[0] == "<modified>": # FIXME: make more generalised
921 self
.list.SetItemData(x
, int(data
[1])) # Store line number for easy access
922 # Give it a blue colour
923 item
= self
.list.GetItem(x
)
924 item
.SetTextColour(wx
.BLUE
)
925 self
.list.SetItem(item
)
927 self
.list.SetItemData(x
, -1) # Editor can't jump into this one's code
930 def OnItemSelected(self
, event
):
931 # This occurs before OnDoubleClick and can be used to set the
932 # currentItem. OnDoubleClick doesn't get a wxListEvent....
933 self
.currentItem
= event
.m_itemIndex
937 def OnDoubleClick(self
, event
):
938 # If double-clicking on a demo's entry, jump to the line number
939 line
= self
.list.GetItemData(self
.currentItem
)
941 self
.nb
.SetSelection(1) # Switch to the code viewer tab
942 wx
.CallAfter(self
.codePanel
.JumpToLine
, line
-1, True)
946 #---------------------------------------------------------------------------
948 class wxPythonDemo(wx
.Frame
):
949 overviewText
= "wxPython Overview"
951 def __init__(self
, parent
, title
):
952 wx
.Frame
.__init
__(self
, parent
, -1, title
, size
= (950, 720),
953 style
=wx
.DEFAULT_FRAME_STYLE | wx
.NO_FULL_REPAINT_ON_RESIZE
)
956 self
.cwd
= os
.getcwd()
957 self
.curOverview
= ""
961 self
.firstTime
= True
963 icon
= images
.getMondrianIcon()
966 if wx
.Platform
!= '__WXMAC__':
967 # setup a taskbar icon, and catch some events from it
968 dim
= 16 # (may want to use 22 on wxGTK, but 16 looks okay too)
969 icon
= wx
.IconFromBitmap(
970 images
.getMondrianImage().Scale(dim
,dim
).ConvertToBitmap() )
971 #icon = wx.Icon('bmp_source/mondrian.ico', wx.BITMAP_TYPE_ICO)
972 #icon = images.getMondrianIcon()
973 self
.tbicon
= wx
.TaskBarIcon()
974 self
.tbicon
.SetIcon(icon
, "wxPython Demo")
975 self
.tbicon
.Bind(wx
.EVT_TASKBAR_LEFT_DCLICK
, self
.OnTaskBarActivate
)
976 self
.tbicon
.Bind(wx
.EVT_TASKBAR_RIGHT_UP
, self
.OnTaskBarMenu
)
977 self
.tbicon
.Bind(wx
.EVT_MENU
, self
.OnTaskBarActivate
, id=self
.TBMENU_RESTORE
)
978 self
.tbicon
.Bind(wx
.EVT_MENU
, self
.OnTaskBarClose
, id=self
.TBMENU_CLOSE
)
980 wx
.CallAfter(self
.ShowTip
)
983 self
.Bind(wx
.EVT_IDLE
, self
.OnIdle
)
984 self
.Bind(wx
.EVT_CLOSE
, self
.OnCloseWindow
)
985 self
.Bind(wx
.EVT_ICONIZE
, self
.OnIconfiy
)
986 self
.Bind(wx
.EVT_MAXIMIZE
, self
.OnMaximize
)
989 self
.CreateStatusBar(1, wx
.ST_SIZEGRIP
)
991 splitter
= wx
.SplitterWindow(self
, -1, style
=wx
.CLIP_CHILDREN | wx
.SP_LIVE_UPDATE | wx
.SP_3D
)
992 splitter2
= wx
.SplitterWindow(splitter
, -1, style
=wx
.CLIP_CHILDREN | wx
.SP_LIVE_UPDATE | wx
.SP_3D
)
994 def EmptyHandler(evt
): pass
995 #splitter.Bind(wx.EVT_ERASE_BACKGROUND, EmptyHandler)
996 #splitter2.Bind(wx.EVT_ERASE_BACKGROUND, EmptyHandler)
998 # Prevent TreeCtrl from displaying all items after destruction when True
1002 self
.nb
= wx
.Notebook(splitter2
, -1, style
=wx
.CLIP_CHILDREN
)
1005 self
.mainmenu
= wx
.MenuBar()
1007 item
= menu
.Append(-1, '&Redirect Output',
1008 'Redirect print statements to a window',
1010 self
.Bind(wx
.EVT_MENU
, self
.OnToggleRedirect
, item
)
1012 item
= menu
.Append(-1, 'E&xit\tAlt-X', 'Get the heck outta here!')
1013 self
.Bind(wx
.EVT_MENU
, self
.OnFileExit
, item
)
1014 wx
.App_SetMacExitMenuItemId(item
.GetId())
1015 self
.mainmenu
.Append(menu
, '&File')
1019 for item
in _treeList
:
1021 for childItem
in item
[1]:
1022 mi
= submenu
.Append(-1, childItem
)
1023 self
.Bind(wx
.EVT_MENU
, self
.OnDemoMenu
, mi
)
1024 menu
.AppendMenu(wx
.NewId(), item
[0], submenu
)
1025 self
.mainmenu
.Append(menu
, '&Demo')
1027 # Make a Demo Code menu
1028 #TODO: Add new menu items
1029 # Like the option-enabled entries to select the
1031 #TODO: should we bother?
1034 #saveID = wx.NewId()
1035 #restoreID = wx.NewId()
1037 #menu.Append(saveID, '&Save\tCtrl-S', 'Save edited demo')
1038 #menu.Append(restoreID, '&Delete Modified\tCtrl-R', 'Delete modified copy')
1039 #self.Bind(wx.EVT_MENU, self.codePage.OnSave, id=saveID)
1040 #self.Bind(wx.EVT_MENU, self.codePage.OnRestore, id=restoreID)
1041 #self.mainmenu.Append(menu, 'Demo &Code')
1046 findItem
= menu
.Append(-1, '&Find\tCtrl-F', 'Find in the Demo Code')
1047 findnextItem
= menu
.Append(-1, 'Find &Next\tF3', 'Find Next')
1048 menu
.AppendSeparator()
1050 shellItem
= menu
.Append(-1, 'Open Py&Shell Window\tF5',
1051 'An interactive interpreter window with the demo app and frame objects in the namesapce')
1052 menu
.AppendSeparator()
1053 helpItem
= menu
.Append(-1, '&About\tCtrl-H', 'wxPython RULES!!!')
1054 wx
.App_SetMacAboutMenuItemId(helpItem
.GetId())
1056 self
.Bind(wx
.EVT_MENU
, self
.OnOpenShellWindow
, shellItem
)
1057 self
.Bind(wx
.EVT_MENU
, self
.OnHelpAbout
, helpItem
)
1058 self
.Bind(wx
.EVT_MENU
, self
.OnHelpFind
, findItem
)
1059 self
.Bind(wx
.EVT_MENU
, self
.OnFindNext
, findnextItem
)
1060 self
.Bind(wx
.EVT_COMMAND_FIND
, self
.OnFind
)
1061 self
.Bind(wx
.EVT_COMMAND_FIND_NEXT
, self
.OnFind
)
1062 self
.Bind(wx
.EVT_COMMAND_FIND_CLOSE
, self
.OnFindClose
)
1063 self
.mainmenu
.Append(menu
, '&Help')
1064 self
.SetMenuBar(self
.mainmenu
)
1066 self
.finddata
= wx
.FindReplaceData()
1069 # This is another way to set Accelerators, in addition to
1070 # using the '\t<key>' syntax in the menu items.
1071 aTable
= wx
.AcceleratorTable([(wx
.ACCEL_ALT
, ord('X'), exitID
),
1072 (wx
.ACCEL_CTRL
, ord('H'), helpID
),
1073 (wx
.ACCEL_CTRL
, ord('F'), findID
),
1074 (wx
.ACCEL_NORMAL
, WXK_F3
, findnextID
)
1076 self
.SetAcceleratorTable(aTable
)
1082 self
.tree
= wx
.TreeCtrl(splitter
, tID
, style
=
1083 wx
.TR_DEFAULT_STYLE
#| wx.TR_HAS_VARIABLE_ROW_HEIGHT
1086 root
= self
.tree
.AddRoot("wxPython Overview")
1088 for item
in _treeList
:
1089 child
= self
.tree
.AppendItem(root
, item
[0])
1090 if not firstChild
: firstChild
= child
1091 for childItem
in item
[1]:
1092 theDemo
= self
.tree
.AppendItem(child
, childItem
)
1093 self
.treeMap
[childItem
] = theDemo
1095 self
.tree
.Expand(root
)
1096 self
.tree
.Expand(firstChild
)
1097 self
.tree
.Bind(wx
.EVT_TREE_ITEM_EXPANDED
, self
.OnItemExpanded
, id=tID
)
1098 self
.tree
.Bind(wx
.EVT_TREE_ITEM_COLLAPSED
, self
.OnItemCollapsed
, id=tID
)
1099 self
.tree
.Bind(wx
.EVT_TREE_SEL_CHANGED
, self
.OnSelChanged
, id=tID
)
1100 self
.tree
.Bind(wx
.EVT_LEFT_DOWN
, self
.OnTreeLeftDown
)
1102 # Set up a wx.html.HtmlWindow on the Overview Notebook page
1103 # we put it in a panel first because there seems to be a
1104 # refresh bug of some sort (wxGTK) when it is directly in
1107 self
.ovr
= wx
.html
.HtmlWindow(self
.nb
, -1, size
=(400, 400))
1108 self
.nb
.AddPage(self
.ovr
, self
.overviewText
)
1110 else: # hopefully I can remove this hacky code soon, see SF bug #216861
1111 panel
= wx
.Panel(self
.nb
, -1, style
=wx
.CLIP_CHILDREN
)
1112 self
.ovr
= wx
.html
.HtmlWindow(panel
, -1, size
=(400, 400))
1113 self
.nb
.AddPage(panel
, self
.overviewText
)
1115 def OnOvrSize(evt
, ovr
=self
.ovr
):
1116 ovr
.SetSize(evt
.GetSize())
1117 panel
.Bind(wx
.EVT_SIZE
, OnOvrSize
)
1118 panel
.Bind(wx
.EVT_ERASE_BACKGROUND
, EmptyHandler
)
1120 if "gtk2" in wx
.PlatformInfo
:
1121 self
.ovr
.NormalizeFontSizes()
1122 self
.SetOverview(self
.overviewText
, mainOverview
)
1125 # Set up a log window
1126 self
.log
= wx
.TextCtrl(splitter2
, -1,
1127 style
= wx
.TE_MULTILINE|wx
.TE_READONLY|wx
.HSCROLL
)
1129 # Set the wxWindows log target to be this textctrl
1130 #wx.Log_SetActiveTarget(wx.LogTextCtrl(self.log))
1132 # But instead of the above we want to show how to use our own wx.Log class
1133 wx
.Log_SetActiveTarget(MyLog(self
.log
))
1135 # for serious debugging
1136 #wx.Log_SetActiveTarget(wx.LogStderr())
1137 #wx.Log_SetTraceMask(wx.TraceMessages)
1140 # add the windows to the splitter and split it.
1141 splitter2
.SplitHorizontally(self
.nb
, self
.log
, -160)
1142 splitter
.SplitVertically(self
.tree
, splitter2
, 200)
1144 splitter
.SetMinimumPaneSize(20)
1145 splitter2
.SetMinimumPaneSize(20)
1147 # Make the splitter on the right expand the top window when resized
1148 def SplitterOnSize(evt
):
1149 splitter
= evt
.GetEventObject()
1150 sz
= splitter
.GetSize()
1151 splitter
.SetSashPosition(sz
.height
- 160, False)
1154 splitter2
.Bind(wx
.EVT_SIZE
, SplitterOnSize
)
1156 # select initial items
1157 self
.nb
.SetSelection(0)
1158 self
.tree
.SelectItem(root
)
1160 # Load 'Main' module
1161 self
.LoadDemo(self
.overviewText
)
1164 # select some other initial module?
1165 if len(sys
.argv
) > 1:
1167 if arg
.endswith('.py'):
1169 selectedDemo
= self
.treeMap
.get(arg
, None)
1171 self
.tree
.SelectItem(selectedDemo
)
1172 self
.tree
.EnsureVisible(selectedDemo
)
1175 #---------------------------------------------
1176 def WriteText(self
, text
):
1177 if text
[-1:] == '\n':
1181 def write(self
, txt
):
1184 #---------------------------------------------
1185 def OnItemExpanded(self
, event
):
1186 item
= event
.GetItem()
1187 wx
.LogMessage("OnItemExpanded: %s" % self
.tree
.GetItemText(item
))
1190 #---------------------------------------------
1191 def OnItemCollapsed(self
, event
):
1192 item
= event
.GetItem()
1193 wx
.LogMessage("OnItemCollapsed: %s" % self
.tree
.GetItemText(item
))
1196 #---------------------------------------------
1197 def OnTreeLeftDown(self
, event
):
1198 # reset the overview text if the tree item is clicked on again
1199 pt
= event
.GetPosition();
1200 item
, flags
= self
.tree
.HitTest(pt
)
1201 if item
== self
.tree
.GetSelection():
1202 self
.SetOverview(self
.tree
.GetItemText(item
)+" Overview", self
.curOverview
)
1205 #---------------------------------------------
1206 def OnSelChanged(self
, event
):
1207 if self
.dying
or not self
.loaded
:
1210 item
= event
.GetItem()
1211 itemText
= self
.tree
.GetItemText(item
)
1212 self
.LoadDemo(itemText
)
1214 #---------------------------------------------
1215 def LoadDemo(self
, demoName
):
1217 wx
.BeginBusyCursor()
1220 self
.ShutdownDemoModule()
1222 if demoName
== self
.overviewText
:
1223 # User selected the "wxPython Overview" node
1225 # Changing the main window at runtime not yet supported...
1226 self
.demoModules
= DemoModules(__name__
)
1227 self
.SetOverview(self
.overviewText
, mainOverview
)
1228 self
.LoadDemoSource()
1229 self
.UpdateNotebook(0)
1231 if os
.path
.exists(GetOriginalFilename(demoName
)):
1232 wx
.LogMessage("Loading demo %s.py..." % demoName
)
1233 self
.demoModules
= DemoModules(demoName
)
1234 self
.LoadDemoSource()
1237 self
.SetOverview("wxPython", mainOverview
)
1238 self
.codePage
= None
1239 self
.UpdateNotebook(0)
1243 #---------------------------------------------
1244 def LoadDemoSource(self
):
1245 self
.codePage
= None
1246 self
.codePage
= DemoCodePanel(self
.nb
, self
)
1247 self
.codePage
.LoadDemo(self
.demoModules
)
1249 #---------------------------------------------
1250 def RunModule(self
):
1251 """Runs the active module"""
1253 module
= self
.demoModules
.GetActive()
1254 self
.ShutdownDemoModule()
1257 # o If the demo returns a window it is placed in a tab.
1258 # o Otherwise, a placeholder tab is created, informing the user that the
1259 # demo runs outside the main window, and allowing it to be reloaded.
1260 # o If an error occurs (or has occured before) an error tab is created.
1262 if module
is not None:
1263 wx
.LogMessage("Running demo module...")
1264 if hasattr(module
, "overview"):
1265 overviewText
= module
.overview
1268 self
.demoPage
= module
.runTest(self
, self
.nb
, self
)
1270 self
.demoPage
= DemoErrorPanel(self
.nb
, self
.codePage
,
1271 DemoError(sys
.exc_info()), self
)
1273 assert self
.demoPage
is not None, "runTest must return a window!"
1276 # There was a previous error in compiling or exec-ing
1277 self
.demoPage
= DemoErrorPanel(self
.nb
, self
.codePage
,
1278 self
.demoModules
.GetErrorInfo(), self
)
1280 self
.SetOverview(self
.demoModules
.name
+ " Overview", overviewText
)
1283 # cahnge to the demo page the first time a module is run
1284 self
.UpdateNotebook(2)
1285 self
.firstTime
= False
1287 # otherwise just stay on the same tab in case the user has changed to another one
1288 self
.UpdateNotebook()
1290 #---------------------------------------------
1291 def ShutdownDemoModule(self
):
1293 # inform the window that it's time to quit if it cares
1294 if hasattr(self
.demoPage
, "ShutdownDemo"):
1295 self
.demoPage
.ShutdownDemo()
1296 wx
.YieldIfNeeded() # in case the page has pending events
1297 self
.demoPage
= None
1299 #---------------------------------------------
1300 def UpdateNotebook(self
, select
= -1):
1304 def UpdatePage(page
, pageText
):
1307 for i
in range(nb
.GetPageCount()):
1308 if nb
.GetPageText(i
) == pageText
:
1316 nb
.AddPage(page
, pageText
)
1317 if debug
: wx
.LogMessage("DBG: ADDED %s" % pageText
)
1319 if nb
.GetPage(pagePos
) != page
:
1320 # Reload an existing page
1322 nb
.DeletePage(pagePos
)
1323 nb
.InsertPage(pagePos
, page
, pageText
)
1325 if debug
: wx
.LogMessage("DBG: RELOADED %s" % pageText
)
1327 # Excellent! No redraw/flicker
1328 if debug
: wx
.LogMessage("DBG: SAVED from reloading %s" % pageText
)
1331 nb
.DeletePage(pagePos
)
1332 if debug
: wx
.LogMessage("DBG: DELETED %s" % pageText
)
1334 if debug
: wx
.LogMessage("DBG: STILL GONE - %s" % pageText
)
1337 select
= nb
.GetSelection()
1339 UpdatePage(self
.codePage
, "Demo Code")
1340 UpdatePage(self
.demoPage
, "Demo")
1342 if select
>= 0 and select
< nb
.GetPageCount():
1343 nb
.SetSelection(select
)
1345 #---------------------------------------------
1346 def SetOverview(self
, name
, text
):
1347 self
.curOverview
= text
1349 if lead
!= '<html>' and lead
!= '<HTML>':
1350 text
= '<br>'.join(text
.split('\n'))
1352 text
= text
.decode('iso8859_1')
1353 self
.ovr
.SetPage(text
)
1354 self
.nb
.SetPageText(0, name
)
1356 #---------------------------------------------
1358 def OnFileExit(self
, *event
):
1361 def OnToggleRedirect(self
, event
):
1365 print "Print statements and other standard output will now be directed to this window."
1368 print "Print statements and other standard output will now be sent to the usual location."
1370 def OnHelpAbout(self
, event
):
1371 from About
import MyAboutBox
1372 about
= MyAboutBox(self
)
1376 def OnHelpFind(self
, event
):
1377 self
.nb
.SetSelection(1)
1378 self
.finddlg
= wx
.FindReplaceDialog(self
, self
.finddata
, "Find",
1382 self
.finddlg
.Show(True)
1384 def OnFind(self
, event
):
1385 editor
= self
.codePage
.editor
1386 self
.nb
.SetSelection(1)
1387 end
= editor
.GetLastPosition()
1388 textstring
= editor
.GetRange(0, end
).lower()
1389 start
= editor
.GetSelection()[1]
1390 findstring
= self
.finddata
.GetFindString().lower()
1391 loc
= textstring
.find(findstring
, start
)
1392 if loc
== -1 and start
!= 0:
1393 # string not found, start at beginning
1395 loc
= textstring
.find(findstring
, start
)
1397 dlg
= wx
.MessageDialog(self
, 'Find String Not Found',
1398 'Find String Not Found in Demo File',
1399 wx
.OK | wx
.ICON_INFORMATION
)
1404 self
.finddlg
.SetFocus()
1407 self
.finddlg
.Destroy()
1408 editor
.ShowPosition(loc
)
1409 editor
.SetSelection(loc
, loc
+ len(findstring
))
1413 def OnFindNext(self
, event
):
1414 if self
.finddata
.GetFindString():
1417 self
.OnHelpFind(event
)
1419 def OnFindClose(self
, event
):
1420 event
.GetDialog().Destroy()
1423 def OnOpenShellWindow(self
, evt
):
1425 # if it already exists then just make sure it's visible
1431 # Make a PyShell window
1433 namespace
= { 'wx' : wx
,
1434 'app' : wx
.GetApp(),
1437 self
.shell
= py
.shell
.ShellFrame(None, locals=namespace
)
1438 self
.shell
.SetSize((640,480))
1441 # Hook the close event of the main frame window so that we
1442 # close the shell at the same time if it still exists
1443 def CloseShell(evt
):
1447 self
.Bind(wx
.EVT_CLOSE
, CloseShell
)
1450 #---------------------------------------------
1451 def OnCloseWindow(self
, event
):
1453 self
.demoPage
= None
1454 self
.codePage
= None
1455 self
.mainmenu
= None
1459 #---------------------------------------------
1460 def OnIdle(self
, event
):
1462 self
.otherWin
.Raise()
1463 self
.demoPage
= self
.otherWin
1464 self
.otherWin
= None
1467 #---------------------------------------------
1470 showTipText
= open(opj("data/showTips")).read()
1471 showTip
, index
= eval(showTipText
)
1473 showTip
, index
= (1, 0)
1475 tp
= wx
.CreateFileTipProvider(opj("data/tips.txt"), index
)
1477 showTip
= wx
.ShowTip(self
, tp
)
1478 index
= tp
.GetCurrentTip()
1479 open(opj("data/showTips"), "w").write(str( (showTip
, index
) ))
1482 #---------------------------------------------
1483 def OnDemoMenu(self
, event
):
1485 selectedDemo
= self
.treeMap
[self
.mainmenu
.GetLabel(event
.GetId())]
1489 self
.tree
.SelectItem(selectedDemo
)
1490 self
.tree
.EnsureVisible(selectedDemo
)
1493 #---------------------------------------------
1494 def OnTaskBarActivate(self
, evt
):
1495 if self
.IsIconized():
1497 if not self
.IsShown():
1501 #---------------------------------------------
1503 TBMENU_RESTORE
= 1000
1506 def OnTaskBarMenu(self
, evt
):
1508 menu
.Append(self
.TBMENU_RESTORE
, "Restore wxPython Demo")
1509 menu
.Append(self
.TBMENU_CLOSE
, "Close")
1510 self
.tbicon
.PopupMenu(menu
)
1513 #---------------------------------------------
1514 def OnTaskBarClose(self
, evt
):
1517 # because of the way wx.TaskBarIcon.PopupMenu is implemented we have to
1518 # prod the main idle handler a bit to get the window to actually close
1519 wx
.GetApp().ProcessIdle()
1522 #---------------------------------------------
1523 def OnIconfiy(self
, evt
):
1524 wx
.LogMessage("OnIconfiy: %d" % evt
.Iconized())
1527 #---------------------------------------------
1528 def OnMaximize(self
, evt
):
1529 wx
.LogMessage("OnMaximize")
1535 #---------------------------------------------------------------------------
1536 #---------------------------------------------------------------------------
1538 class MySplashScreen(wx
.SplashScreen
):
1540 bmp
= wx
.Image(opj("bitmaps/splash.gif")).ConvertToBitmap()
1541 wx
.SplashScreen
.__init
__(self
, bmp
,
1542 wx
.SPLASH_CENTRE_ON_SCREEN | wx
.SPLASH_TIMEOUT
,
1544 self
.Bind(wx
.EVT_CLOSE
, self
.OnClose
)
1546 def OnClose(self
, evt
):
1548 frame
= wxPythonDemo(None, "wxPython: (A Demonstration)")
1550 evt
.Skip() # Make sure the default handler runs too...
1553 class MyApp(wx
.App
):
1556 Create and show the splash screen. It will then create and show
1557 the main frame when it is time to do so.
1561 #self.SetAssertMode(wx.PYAPP_ASSERT_DIALOG)
1563 # Normally when using a SplashScreen you would create it, show
1564 # it and then continue on with the applicaiton's
1565 # initialization, finally creating and showing the main
1566 # application window(s). In this case we have nothing else to
1567 # do so we'll delay showing the main frame until later (see
1568 # OnClose above) so the users can see the SplashScreen effect.
1569 splash
= MySplashScreen()
1576 #---------------------------------------------------------------------------
1580 demoPath
= os
.path
.dirname(__file__
)
1587 #---------------------------------------------------------------------------
1590 mainOverview
= """<html><body>
1593 <p> wxPython is a <b>GUI toolkit</b> for the Python programming
1594 language. It allows Python programmers to create programs with a
1595 robust, highly functional graphical user interface, simply and easily.
1596 It is implemented as a Python extension module (native code) that
1597 wraps the popular wxWindows cross platform GUI library, which is
1600 <p> Like Python and wxWindows, wxPython is <b>Open Source</b> which
1601 means that it is free for anyone to use and the source code is
1602 available for anyone to look at and modify. Or anyone can contribute
1603 fixes or enhancements to the project.
1605 <p> wxPython is a <b>cross-platform</b> toolkit. This means that the
1606 same program will run on multiple platforms without modification.
1607 Currently supported platforms are 32-bit Microsoft Windows, most Unix
1608 or unix-like systems, and Macintosh OS X. Since the language is
1609 Python, wxPython programs are <b>simple, easy</b> to write and easy to
1612 <p> <b>This demo</b> is not only a collection of test cases for
1613 wxPython, but is also designed to help you learn about and how to use
1614 wxPython. Each sample is listed in the tree control on the left.
1615 When a sample is selected in the tree then a module is loaded and run
1616 (usually in a tab of this notebook,) and the source code of the module
1617 is loaded in another tab for you to browse and learn from.
1622 #----------------------------------------------------------------------------
1623 #----------------------------------------------------------------------------
1625 if __name__
== '__main__':
1629 #----------------------------------------------------------------------------