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 from wxPython
.wx
import *
16 from wxPython
.lib
.splashscreen
import SplashScreen
17 from wxPython
.html
import wxHtmlWindow
21 #---------------------------------------------------------------------------
25 ('New since last release', ['ContextHelp',
32 'wxFindReplaceDialog',
37 ('Windows', ['wxFrame', 'wxDialog', 'wxMiniFrame',
38 'wxGrid', 'wxSashWindow',
39 'wxScrolledWindow', 'wxSplitterWindow',
40 'wxStatusBar', 'wxNotebook',
42 'wxStyledTextCtrl_1', 'wxStyledTextCtrl_2',
46 ('Common Dialogs', ['wxColourDialog', 'wxDirDialog', 'wxFileDialog',
47 'wxSingleChoiceDialog', 'wxTextEntryDialog',
48 'wxFontDialog', 'wxPageSetupDialog', 'wxPrintDialog',
49 'wxMessageDialog', 'wxProgressDialog', 'wxFindReplaceDialog',
52 ('Controls', ['wxButton', 'wxCheckBox', 'wxCheckListBox', 'wxChoice',
53 'wxComboBox', 'wxGauge', 'wxListBox', 'wxListCtrl', 'wxTextCtrl',
54 'wxTreeCtrl', 'wxSpinButton', 'wxSpinCtrl', 'wxStaticText',
55 'wxStaticBitmap', 'wxRadioBox', 'wxSlider', 'wxToolBar',
56 'wxCalendarCtrl', 'wxToggleButton',
59 ('Window Layout', ['wxLayoutConstraints', 'LayoutAnchors', 'Sizers', 'XML_Resource']),
61 ('Miscellaneous', [ 'DragAndDrop', 'CustomDragAndDrop', 'FontEnumerator',
62 'wxTimer', 'wxValidator', 'wxGLCanvas', 'DialogUnits',
63 'wxImage', 'wxMask', 'PrintFramework', 'wxOGL',
64 'PythonEvents', 'Threads',
65 'ActiveXWrapper_Acrobat', 'ActiveXWrapper_IE',
66 'wxDragImage', "wxProcess", "FancyText", "OOR", "wxWave",
67 'wxJoystick', 'DrawXXXList',
70 ('wxPython Library', ['Layoutf', 'wxScrolledMessageDialog',
71 'wxMultipleChoiceDialog', 'wxPlotCanvas', 'wxFloatBar',
72 'wxCalendar', 'wxMVCTree', 'wxVTKRenderWindow',
73 'FileBrowseButton', 'GenericButtons', 'wxEditor',
74 'ColourSelect', 'ImageBrowser',
75 'infoframe', 'ColourDB', 'PyCrust', 'TablePrint',
78 ('Cool Contribs', ['pyTree', 'hangman', 'SlashDot', 'XMLtreeview']),
82 #---------------------------------------------------------------------------
85 def __init__(self
, textCtrl
, logTime
=0):
86 wxPyLog
.__init
__(self
)
88 self
.logTime
= logTime
90 def DoLogString(self
, message
, timeStamp
):
92 message
= time
.strftime("%X", time
.localtime(timeStamp
)) + \
94 self
.tc
.AppendText(message
+ '\n')
97 #---------------------------------------------------------------------------
99 class wxPythonDemo(wxFrame
):
101 def __init__(self
, parent
, id, title
):
102 wxFrame
.__init
__(self
, parent
, -1, title
, size
= (800, 600),
103 style
=wxDEFAULT_FRAME_STYLE|wxNO_FULL_REPAINT_ON_RESIZE
)
105 self
.cwd
= os
.getcwd()
106 self
.curOverview
= ""
109 icon
= wxIconFromXPMData(images
.getMondrianData())
111 # another way to do it
112 bmp
= images
.getMondrianBitmap()
114 icon
.CopyFromBitmap(bmp
)
118 if wxPlatform
== '__WXMSW__':
119 # setup a taskbar icon, and catch some events from it
120 self
.tbicon
= wxTaskBarIcon()
121 self
.tbicon
.SetIcon(icon
, "wxPython Demo")
122 EVT_TASKBAR_LEFT_DCLICK(self
.tbicon
, self
.OnTaskBarActivate
)
123 EVT_TASKBAR_RIGHT_UP(self
.tbicon
, self
.OnTaskBarMenu
)
124 EVT_MENU(self
.tbicon
, self
.TBMENU_RESTORE
, self
.OnTaskBarActivate
)
125 EVT_MENU(self
.tbicon
, self
.TBMENU_CLOSE
, self
.OnTaskBarClose
)
129 EVT_IDLE(self
, self
.OnIdle
)
130 EVT_CLOSE(self
, self
.OnCloseWindow
)
131 EVT_ICONIZE(self
, self
.OnIconfiy
)
132 EVT_MAXIMIZE(self
, self
.OnMaximize
)
135 self
.CreateStatusBar(1, wxST_SIZEGRIP
)
137 splitter
= wxSplitterWindow(self
, -1, style
=wxNO_3D|wxSP_3D
)
138 splitter2
= wxSplitterWindow(splitter
, -1, style
=wxNO_3D|wxSP_3D
)
140 def EmptyHandler(evt
): pass
141 EVT_ERASE_BACKGROUND(splitter
, EmptyHandler
)
142 EVT_ERASE_BACKGROUND(splitter2
, EmptyHandler
)
144 # Prevent TreeCtrl from displaying all items after destruction when true
148 self
.mainmenu
= wxMenuBar()
151 menu
.Append(exitID
, 'E&xit\tAlt-X', 'Get the heck outta here!')
152 EVT_MENU(self
, exitID
, self
.OnFileExit
)
153 self
.mainmenu
.Append(menu
, '&File')
157 for item
in _treeList
:
159 for childItem
in item
[1]:
161 submenu
.Append(mID
, childItem
)
162 EVT_MENU(self
, mID
, self
.OnDemoMenu
)
163 menu
.AppendMenu(wxNewId(), item
[0], submenu
)
164 self
.mainmenu
.Append(menu
, '&Demo')
170 menu
.Append(helpID
, '&About\tCtrl-H', 'wxPython RULES!!!')
171 EVT_MENU(self
, helpID
, self
.OnHelpAbout
)
172 self
.mainmenu
.Append(menu
, '&Help')
173 self
.SetMenuBar(self
.mainmenu
)
175 # set the menu accellerator table...
176 aTable
= wxAcceleratorTable([(wxACCEL_ALT
, ord('X'), exitID
),
177 (wxACCEL_CTRL
, ord('H'), helpID
)])
178 self
.SetAcceleratorTable(aTable
)
184 self
.tree
= wxTreeCtrl(splitter
, tID
,
185 style
=wxTR_HAS_BUTTONS |
187 wxTR_HAS_VARIABLE_ROW_HEIGHT |
189 #self.tree.SetBackgroundColour(wxNamedColour("Pink"))
190 root
= self
.tree
.AddRoot("Overview")
192 for item
in _treeList
:
193 child
= self
.tree
.AppendItem(root
, item
[0])
194 if not firstChild
: firstChild
= child
195 for childItem
in item
[1]:
196 theDemo
= self
.tree
.AppendItem(child
, childItem
)
197 self
.treeMap
[childItem
] = theDemo
199 self
.tree
.Expand(root
)
200 self
.tree
.Expand(firstChild
)
201 EVT_TREE_ITEM_EXPANDED (self
.tree
, tID
, self
.OnItemExpanded
)
202 EVT_TREE_ITEM_COLLAPSED (self
.tree
, tID
, self
.OnItemCollapsed
)
203 EVT_TREE_SEL_CHANGED (self
.tree
, tID
, self
.OnSelChanged
)
204 EVT_LEFT_DOWN (self
.tree
, self
.OnTreeLeftDown
)
207 self
.nb
= wxNotebook(splitter2
, -1, style
=wxCLIP_CHILDREN
)
209 # Set up a wxHtmlWindow on the Overview Notebook page
210 # we put it in a panel first because there seems to be a
211 # refresh bug of some sort (wxGTK) when it is directly in
214 self
.ovr
= wxHtmlWindow(self
.nb
, -1, size
=(400, 400))
215 self
.nb
.AddPage(self
.ovr
, "Overview")
217 else: # hopefully I can remove this hacky code soon, see bug #216861
218 panel
= wxPanel(self
.nb
, -1, style
=wxCLIP_CHILDREN
)
219 self
.ovr
= wxHtmlWindow(panel
, -1, size
=(400, 400))
220 self
.nb
.AddPage(panel
, "Overview")
222 def OnOvrSize(evt
, ovr
=self
.ovr
):
223 ovr
.SetSize(evt
.GetSize())
225 EVT_SIZE(panel
, OnOvrSize
)
226 EVT_ERASE_BACKGROUND(panel
, EmptyHandler
)
229 self
.SetOverview("Overview", overview
)
232 # Set up a TextCtrl on the Demo Code Notebook page
233 self
.txt
= wxTextCtrl(self
.nb
, -1,
234 style
= wxTE_MULTILINE|wxTE_READONLY|wxHSCROLL
)
235 self
.nb
.AddPage(self
.txt
, "Demo Code")
238 # Set up a log on the View Log Notebook page
239 self
.log
= wxTextCtrl(splitter2
, -1,
240 style
= wxTE_MULTILINE|wxTE_READONLY|wxHSCROLL
)
241 # Set the wxWindows log target to be this textctrl
242 #wxLog_SetActiveTarget(wxLogTextCtrl(self.log))
243 wxLog_SetActiveTarget(MyLog(self
.log
))
249 # add the windows to the splitter and split it.
250 splitter2
.SplitHorizontally(self
.nb
, self
.log
)
251 splitter2
.SetSashPosition(450, true
)
252 splitter2
.SetMinimumPaneSize(20)
254 splitter
.SplitVertically(self
.tree
, splitter2
)
255 splitter
.SetSashPosition(180, true
)
256 splitter
.SetMinimumPaneSize(20)
259 # select initial items
260 self
.nb
.SetSelection(0)
261 self
.tree
.SelectItem(root
)
263 if len(sys
.argv
) == 2:
265 selectedDemo
= self
.treeMap
[sys
.argv
[1]]
269 self
.tree
.SelectItem(selectedDemo
)
270 self
.tree
.EnsureVisible(selectedDemo
)
273 wxLogMessage('window handle: %s' % self
.GetHandle())
276 #---------------------------------------------
277 def WriteText(self
, text
):
278 if text
[-1:] == '\n':
283 def write(self
, txt
):
286 #---------------------------------------------
287 def OnItemExpanded(self
, event
):
288 item
= event
.GetItem()
289 wxLogMessage("OnItemExpanded: %s" % self
.tree
.GetItemText(item
))
292 #---------------------------------------------
293 def OnItemCollapsed(self
, event
):
294 item
= event
.GetItem()
295 wxLogMessage("OnItemCollapsed: %s" % self
.tree
.GetItemText(item
))
298 #---------------------------------------------
299 def OnTreeLeftDown(self
, event
):
300 pt
= event
.GetPosition();
301 item
, flags
= self
.tree
.HitTest(pt
)
302 if item
== self
.tree
.GetSelection():
303 self
.SetOverview(self
.tree
.GetItemText(item
), self
.curOverview
)
306 #---------------------------------------------
307 def OnSelChanged(self
, event
):
311 item
= event
.GetItem()
312 itemText
= self
.tree
.GetItemText(item
)
313 self
.RunDemo(itemText
)
316 #---------------------------------------------
317 def RunDemo(self
, itemText
):
319 if self
.nb
.GetPageCount() == 3:
320 if self
.nb
.GetSelection() == 2:
321 self
.nb
.SetSelection(0)
322 self
.nb
.DeletePage(2)
324 if itemText
== 'Overview':
325 self
.GetDemoFile('Main.py')
326 self
.SetOverview('Overview', overview
)
331 if os
.path
.exists(itemText
+ '.py'):
333 wxLogMessage("Running demo %s.py..." % itemText
)
335 self
.GetDemoFile(itemText
+ '.py')
336 module
= __import__(itemText
, globals())
337 self
.SetOverview(itemText
, module
.overview
)
341 # in case runTest is modal, make sure things look right...
345 self
.window
= module
.runTest(self
, self
.nb
, self
) ###
347 self
.nb
.AddPage(self
.window
, 'Demo')
349 self
.nb
.SetSelection(2)
358 #---------------------------------------------
360 def GetDemoFile(self
, filename
):
363 self
.txt
.SetValue(open(filename
).read())
365 self
.txt
.WriteText("Cannot open %s file." % filename
)
367 self
.txt
.SetInsertionPoint(0)
368 self
.txt
.ShowPosition(0)
370 #---------------------------------------------
371 def SetOverview(self
, name
, text
):
372 self
.curOverview
= text
374 if lead
!= '<html>' and lead
!= '<HTML>':
375 text
= string
.join(string
.split(text
, '\n'), '<br>')
376 #text = '<font size="-1"><pre>' + text + '</pre></font>'
377 self
.ovr
.SetPage(text
)
378 self
.nb
.SetPageText(0, name
)
380 #---------------------------------------------
382 def OnFileExit(self
, *event
):
386 def OnHelpAbout(self
, event
):
387 from About
import MyAboutBox
388 about
= MyAboutBox(self
)
393 #---------------------------------------------
394 def OnCloseWindow(self
, event
):
398 if hasattr(self
, "tbicon"):
403 #---------------------------------------------
404 def OnIdle(self
, event
):
406 self
.otherWin
.Raise()
407 self
.window
= self
.otherWin
410 #---------------------------------------------
411 def OnDemoMenu(self
, event
):
413 selectedDemo
= self
.treeMap
[self
.mainmenu
.GetLabel(event
.GetId())]
417 self
.tree
.SelectItem(selectedDemo
)
418 self
.tree
.EnsureVisible(selectedDemo
)
421 #---------------------------------------------
422 def OnTaskBarActivate(self
, evt
):
423 if self
.IsIconized():
425 if not self
.IsShown():
429 #---------------------------------------------
431 TBMENU_RESTORE
= 1000
434 def OnTaskBarMenu(self
, evt
):
436 menu
.Append(self
.TBMENU_RESTORE
, "Restore wxPython Demo")
437 menu
.Append(self
.TBMENU_CLOSE
, "Close")
438 self
.tbicon
.PopupMenu(menu
)
441 #---------------------------------------------
442 def OnTaskBarClose(self
, evt
):
445 # because of the way wxTaskBarIcon.PopupMenu is implemented we have to
446 # prod the main idle handler a bit to get the window to actually close
447 wxGetApp().ProcessIdle()
450 #---------------------------------------------
451 def OnIconfiy(self
, evt
):
452 wxLogMessage("OnIconfiy")
455 #---------------------------------------------
456 def OnMaximize(self
, evt
):
457 wxLogMessage("OnMaximize")
463 #---------------------------------------------------------------------------
464 #---------------------------------------------------------------------------
468 wxInitAllImageHandlers()
470 self
.splash
= SplashScreen(None, bitmapfile
='bitmaps/splash.gif',
471 duration
=4000, callback
=self
.AfterSplash
)
472 self
.splash
.Show(true
)
477 def AfterSplash(self
):
478 self
.splash
.Close(true
)
479 frame
= wxPythonDemo(None, -1, "wxPython: (A Demonstration)")
481 self
.SetTopWindow(frame
)
485 def ShowTip(self
, frame
):
487 showTipText
= open("data/showTips").read()
488 showTip
, index
= eval(showTipText
)
490 showTip
, index
= (1, 0)
491 #print showTip, index
493 tp
= wxCreateFileTipProvider("data/tips.txt", index
)
494 showTip
= wxShowTip(frame
, tp
)
495 index
= tp
.GetCurrentTip()
496 open("data/showTips", "w").write(str( (showTip
, index
) ))
499 #---------------------------------------------------------------------------
503 demoPath
= os
.path
.dirname(__file__
)
511 #---------------------------------------------------------------------------
515 overview
= """<html><body>
518 Python is an interpreted, interactive, object-oriented programming
519 language often compared to Tcl, Perl, Scheme, or Java.
521 <p> Python combines remarkable power with very clear syntax. It has
522 modules, classes, exceptions, very high level dynamic data types, and
523 dynamic typing. There are interfaces to many system calls and
524 libraries, and new built-in modules are easily written in C or
525 C++. Python is also usable as an extension language for applications
526 that need a programmable interface. <p>
530 wxWindows is a free C++ framework designed to make cross-platform
531 programming child's play. Well, almost. wxWindows 2 supports Windows
532 3.1/95/98/NT, Unix with GTK/Motif/Lesstif, with a Mac version
533 underway. Other ports are under consideration. <p>
535 wxWindows is a set of libraries that allows C++ applications to
536 compile and run on several different types of computers, with minimal
537 source code changes. There is one library per supported GUI (such as
538 Motif, or Windows). As well as providing a common API (Application
539 Programming Interface) for GUI functionality, it provides
540 functionality for accessing some commonly-used operating system
541 facilities, such as copying or deleting files. wxWindows is a
542 'framework' in the sense that it provides a lot of built-in
543 functionality, which the application can use or replace as required,
544 thus saving a great deal of coding effort. Basic data structures such
545 as strings, linked lists and hash tables are also supported.
550 wxPython is a Python extension module that encapsulates the wxWindows
551 GUI classes. Currently it is only available for the Win32 and GTK
552 ports of wxWindows, but as soon as the other ports are brought up to
553 the same level as Win32 and GTK, it should be fairly trivial to
554 enable wxPython to be used with the new GUI.
558 The wxPython extension module attempts to mirror the class heiarchy
559 of wxWindows as closely as possible. This means that there is a
560 wxFrame class in wxPython that looks, smells, tastes and acts almost
561 the same as the wxFrame class in the C++ version. Unfortunately,
562 because of differences in the languages, wxPython doesn't match
563 wxWindows exactly, but the differences should be easy to absorb
564 because they are natural to Python. For example, some methods that
565 return multiple values via argument pointers in C++ will return a
566 tuple of values in Python.
570 There is still much to be done for wxPython, many classes still need
571 to be mirrored. Also, wxWindows is still somewhat of a moving target
572 so it is a bit of an effort just keeping wxPython up to date. On the
573 other hand, there are enough of the core classes completed that
574 useful applications can be written.
578 wxPython is close enough to the C++ version that the majority of
579 the wxPython documentation is actually just notes attached to the C++
580 documents that describe the places where wxPython is different. There
581 is also a series of sample programs included, and a series of
582 documentation pages that assist the programmer in getting started
588 #----------------------------------------------------------------------------
589 #----------------------------------------------------------------------------
591 if __name__
== '__main__':
594 #----------------------------------------------------------------------------