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 #----------------------------------------------------------------------------
14 import sys
, os
, time
, string
15 from wxPython
.wx
import *
16 from wxPython
.html
import wxHtmlWindow
18 ##from wxPython.stc import *
22 #---------------------------------------------------------------------------
27 ('New since last release', [
36 # managed windows == things with a caption you can close
37 ('Base Frames and Dialogs', [
49 'wxFindReplaceDialog',
55 'wxSingleChoiceDialog',
59 # dialogs form libraries
63 'wxMultipleChoiceDialog',
64 'wxScrolledMessageDialog',
68 ('Core Windows/Controls', [
99 # controls coming from other librairies
100 ('More Windows/Controls', [
107 'PyCrustWithFilling',
112 'wxDynamicSashWindow',
118 'wxMimeTypesManager',
120 'wxStyledTextCtrl_1',
121 'wxStyledTextCtrl_2',
125 # How to lay out the controls in a frame/dialog
131 'wxLayoutConstraints',
136 ('Process and Events', [
146 ('Clipboard and DnD', [
175 # need libs not coming with the demo
176 ('Objects using an external library', [
177 'ActiveXWrapper_Acrobat',
183 # pyTree, hangman, ... in the samples dir
184 ('Check out the samples dir too', [
191 #---------------------------------------------------------------------------
193 class MyLog(wxPyLog
):
194 def __init__(self
, textCtrl
, logTime
=0):
195 wxPyLog
.__init
__(self
)
197 self
.logTime
= logTime
199 def DoLogString(self
, message
, timeStamp
):
201 message
= time
.strftime("%X", time
.localtime(timeStamp
)) + \
203 self
.tc
.AppendText(message
+ '\n')
206 #---------------------------------------------------------------------------
209 """Convert paths to the platform-specific separator"""
210 return apply(os
.path
.join
, tuple(string
.split(path
, '/')))
213 #---------------------------------------------------------------------------
215 class wxPythonDemo(wxFrame
):
216 overviewText
= "wxPython Overview"
218 def __init__(self
, parent
, id, title
):
219 wxFrame
.__init
__(self
, parent
, -1, title
, size
= (800, 600),
220 style
=wxDEFAULT_FRAME_STYLE|wxNO_FULL_REPAINT_ON_RESIZE
)
222 self
.cwd
= os
.getcwd()
223 self
.curOverview
= ""
225 icon
= images
.getMondrianIcon()
228 if wxPlatform
== '__WXMSW__':
229 # setup a taskbar icon, and catch some events from it
230 self
.tbicon
= wxTaskBarIcon()
231 self
.tbicon
.SetIcon(icon
, "wxPython Demo")
232 EVT_TASKBAR_LEFT_DCLICK(self
.tbicon
, self
.OnTaskBarActivate
)
233 EVT_TASKBAR_RIGHT_UP(self
.tbicon
, self
.OnTaskBarMenu
)
234 EVT_MENU(self
.tbicon
, self
.TBMENU_RESTORE
, self
.OnTaskBarActivate
)
235 EVT_MENU(self
.tbicon
, self
.TBMENU_CLOSE
, self
.OnTaskBarClose
)
240 EVT_IDLE(self
, self
.OnIdle
)
241 EVT_CLOSE(self
, self
.OnCloseWindow
)
242 EVT_ICONIZE(self
, self
.OnIconfiy
)
243 EVT_MAXIMIZE(self
, self
.OnMaximize
)
246 self
.CreateStatusBar(1, wxST_SIZEGRIP
)
248 splitter
= wxSplitterWindow(self
, -1, style
=wxNO_3D|wxSP_3D
)
249 splitter2
= wxSplitterWindow(splitter
, -1, style
=wxNO_3D|wxSP_3D
)
251 def EmptyHandler(evt
): pass
252 EVT_ERASE_BACKGROUND(splitter
, EmptyHandler
)
253 EVT_ERASE_BACKGROUND(splitter2
, EmptyHandler
)
255 # Prevent TreeCtrl from displaying all items after destruction when true
259 self
.mainmenu
= wxMenuBar()
262 menu
.Append(exitID
, 'E&xit\tAlt-X', 'Get the heck outta here!')
263 EVT_MENU(self
, exitID
, self
.OnFileExit
)
264 self
.mainmenu
.Append(menu
, '&File')
268 for item
in _treeList
:
270 for childItem
in item
[1]:
272 submenu
.Append(mID
, childItem
)
273 EVT_MENU(self
, mID
, self
.OnDemoMenu
)
274 menu
.AppendMenu(wxNewId(), item
[0], submenu
)
275 self
.mainmenu
.Append(menu
, '&Demo')
281 menu
.Append(helpID
, '&About\tCtrl-H', 'wxPython RULES!!!')
282 EVT_MENU(self
, helpID
, self
.OnHelpAbout
)
283 self
.mainmenu
.Append(menu
, '&Help')
284 self
.SetMenuBar(self
.mainmenu
)
286 # set the menu accellerator table...
287 aTable
= wxAcceleratorTable([(wxACCEL_ALT
, ord('X'), exitID
),
288 (wxACCEL_CTRL
, ord('H'), helpID
)])
289 self
.SetAcceleratorTable(aTable
)
295 self
.tree
= wxTreeCtrl(splitter
, tID
,
296 style
=wxTR_HAS_BUTTONS |
298 wxTR_HAS_VARIABLE_ROW_HEIGHT
)
300 #self.tree.SetBackgroundColour(wxNamedColour("Pink"))
301 root
= self
.tree
.AddRoot("wxPython Overview")
303 for item
in _treeList
:
304 child
= self
.tree
.AppendItem(root
, item
[0])
305 if not firstChild
: firstChild
= child
306 for childItem
in item
[1]:
307 theDemo
= self
.tree
.AppendItem(child
, childItem
)
308 self
.treeMap
[childItem
] = theDemo
310 self
.tree
.Expand(root
)
311 self
.tree
.Expand(firstChild
)
312 EVT_TREE_ITEM_EXPANDED (self
.tree
, tID
, self
.OnItemExpanded
)
313 EVT_TREE_ITEM_COLLAPSED (self
.tree
, tID
, self
.OnItemCollapsed
)
314 EVT_TREE_SEL_CHANGED (self
.tree
, tID
, self
.OnSelChanged
)
315 EVT_LEFT_DOWN (self
.tree
, self
.OnTreeLeftDown
)
318 self
.nb
= wxNotebook(splitter2
, -1, style
=wxCLIP_CHILDREN
)
320 # Set up a wxHtmlWindow on the Overview Notebook page
321 # we put it in a panel first because there seems to be a
322 # refresh bug of some sort (wxGTK) when it is directly in
325 self
.ovr
= wxHtmlWindow(self
.nb
, -1, size
=(400, 400))
326 self
.nb
.AddPage(self
.ovr
, self
.overviewText
)
328 else: # hopefully I can remove this hacky code soon, see bug #216861
329 panel
= wxPanel(self
.nb
, -1, style
=wxCLIP_CHILDREN
)
330 self
.ovr
= wxHtmlWindow(panel
, -1, size
=(400, 400))
331 self
.nb
.AddPage(panel
, self
.overviewText
)
333 def OnOvrSize(evt
, ovr
=self
.ovr
):
334 ovr
.SetSize(evt
.GetSize())
336 EVT_SIZE(panel
, OnOvrSize
)
337 EVT_ERASE_BACKGROUND(panel
, EmptyHandler
)
340 self
.SetOverview(self
.overviewText
, overview
)
343 # Set up a TextCtrl on the Demo Code Notebook page
344 self
.txt
= wxTextCtrl(self
.nb
, -1,
345 style
= wxTE_MULTILINE|wxTE_READONLY|wxHSCROLL
)
346 self
.nb
.AddPage(self
.txt
, "Demo Code")
349 # Set up a log on the View Log Notebook page
350 self
.log
= wxTextCtrl(splitter2
, -1,
351 style
= wxTE_MULTILINE|wxTE_READONLY|wxHSCROLL
)
353 # Set the wxWindows log target to be this textctrl
354 #wxLog_SetActiveTarget(wxLogTextCtrl(self.log))
356 # But instead of the above we want to show how to use our own wxLog class
357 wxLog_SetActiveTarget(MyLog(self
.log
))
363 # add the windows to the splitter and split it.
364 splitter2
.SplitHorizontally(self
.nb
, self
.log
)
365 splitter
.SplitVertically(self
.tree
, splitter2
)
367 splitter
.SetSashPosition(180, true
)
368 splitter
.SetMinimumPaneSize(20)
369 splitter2
.SetSashPosition(450, true
)
370 splitter2
.SetMinimumPaneSize(20)
374 # select initial items
375 self
.nb
.SetSelection(0)
376 self
.tree
.SelectItem(root
)
378 if len(sys
.argv
) == 2:
380 selectedDemo
= self
.treeMap
[sys
.argv
[1]]
384 self
.tree
.SelectItem(selectedDemo
)
385 self
.tree
.EnsureVisible(selectedDemo
)
388 wxLogMessage('window handle: %s' % self
.GetHandle())
391 #---------------------------------------------
392 def WriteText(self
, text
):
393 if text
[-1:] == '\n':
398 def write(self
, txt
):
401 #---------------------------------------------
402 def OnItemExpanded(self
, event
):
403 item
= event
.GetItem()
404 wxLogMessage("OnItemExpanded: %s" % self
.tree
.GetItemText(item
))
407 #---------------------------------------------
408 def OnItemCollapsed(self
, event
):
409 item
= event
.GetItem()
410 wxLogMessage("OnItemCollapsed: %s" % self
.tree
.GetItemText(item
))
413 #---------------------------------------------
414 def OnTreeLeftDown(self
, event
):
415 pt
= event
.GetPosition();
416 item
, flags
= self
.tree
.HitTest(pt
)
417 if item
== self
.tree
.GetSelection():
418 self
.SetOverview(self
.tree
.GetItemText(item
)+" Overview", self
.curOverview
)
421 #---------------------------------------------
422 def OnSelChanged(self
, event
):
426 item
= event
.GetItem()
427 itemText
= self
.tree
.GetItemText(item
)
428 self
.RunDemo(itemText
)
431 #---------------------------------------------
432 def RunDemo(self
, itemText
):
434 if self
.nb
.GetPageCount() == 3:
435 if self
.nb
.GetSelection() == 2:
436 self
.nb
.SetSelection(0)
437 self
.nb
.DeletePage(2)
439 if itemText
== self
.overviewText
:
440 self
.GetDemoFile('Main.py')
441 self
.SetOverview(self
.overviewText
, overview
)
446 if os
.path
.exists(itemText
+ '.py'):
448 wxLogMessage("Running demo %s.py..." % itemText
)
450 self
.GetDemoFile(itemText
+ '.py')
451 module
= __import__(itemText
, globals())
452 self
.SetOverview(itemText
+ " Overview", module
.overview
)
457 # in case runTest is modal, make sure things look right...
461 self
.window
= module
.runTest(self
, self
.nb
, self
) ###
463 self
.nb
.AddPage(self
.window
, 'Demo')
464 self
.nb
.SetSelection(2)
465 self
.nb
.Refresh() # without this wxMac has troubles showing the just added page
474 #---------------------------------------------
476 def GetDemoFile(self
, filename
):
479 self
.txt
.SetValue(open(filename
).read())
481 self
.txt
.WriteText("Cannot open %s file." % filename
)
483 self
.txt
.SetInsertionPoint(0)
484 self
.txt
.ShowPosition(0)
486 #---------------------------------------------
487 def SetOverview(self
, name
, text
):
488 self
.curOverview
= text
490 if lead
!= '<html>' and lead
!= '<HTML>':
491 text
= string
.join(string
.split(text
, '\n'), '<br>')
492 self
.ovr
.SetPage(text
)
493 self
.nb
.SetPageText(0, name
)
495 #---------------------------------------------
497 def OnFileExit(self
, *event
):
501 def OnHelpAbout(self
, event
):
502 from About
import MyAboutBox
503 about
= MyAboutBox(self
)
508 #---------------------------------------------
509 def OnCloseWindow(self
, event
):
513 if hasattr(self
, "tbicon"):
518 #---------------------------------------------
519 def OnIdle(self
, event
):
521 self
.otherWin
.Raise()
522 self
.window
= self
.otherWin
530 #---------------------------------------------
533 showTipText
= open(opj("data/showTips")).read()
534 showTip
, index
= eval(showTipText
)
536 showTip
, index
= (1, 0)
538 tp
= wxCreateFileTipProvider(opj("data/tips.txt"), index
)
539 showTip
= wxShowTip(self
, tp
)
540 index
= tp
.GetCurrentTip()
541 open(opj("data/showTips"), "w").write(str( (showTip
, index
) ))
544 #---------------------------------------------
545 def OnDemoMenu(self
, event
):
547 selectedDemo
= self
.treeMap
[self
.mainmenu
.GetLabel(event
.GetId())]
551 self
.tree
.SelectItem(selectedDemo
)
552 self
.tree
.EnsureVisible(selectedDemo
)
555 #---------------------------------------------
556 def OnTaskBarActivate(self
, evt
):
557 if self
.IsIconized():
559 if not self
.IsShown():
563 #---------------------------------------------
565 TBMENU_RESTORE
= 1000
568 def OnTaskBarMenu(self
, evt
):
570 menu
.Append(self
.TBMENU_RESTORE
, "Restore wxPython Demo")
571 menu
.Append(self
.TBMENU_CLOSE
, "Close")
572 self
.tbicon
.PopupMenu(menu
)
575 #---------------------------------------------
576 def OnTaskBarClose(self
, evt
):
579 # because of the way wxTaskBarIcon.PopupMenu is implemented we have to
580 # prod the main idle handler a bit to get the window to actually close
581 wxGetApp().ProcessIdle()
584 #---------------------------------------------
585 def OnIconfiy(self
, evt
):
586 wxLogMessage("OnIconfiy")
589 #---------------------------------------------
590 def OnMaximize(self
, evt
):
591 wxLogMessage("OnMaximize")
597 #---------------------------------------------------------------------------
598 #---------------------------------------------------------------------------
600 class MySplashScreen(wxSplashScreen
):
602 bmp
= wxImage(opj("bitmaps/splash.gif")).ConvertToBitmap()
603 wxSplashScreen
.__init
__(self
, bmp
,
604 wxSPLASH_CENTRE_ON_SCREEN|wxSPLASH_TIMEOUT
,
606 style
= wxSIMPLE_BORDER|wxFRAME_NO_TASKBAR|wxSTAY_ON_TOP
)
607 EVT_CLOSE(self
, self
.OnClose
)
609 def OnClose(self
, evt
):
610 frame
= wxPythonDemo(None, -1, "wxPython: (A Demonstration)")
612 evt
.Skip() # Make sure the default handler runs too...
618 Create and show the splash screen. It will then create and show
619 the main frame when it is time to do so.
621 wxInitAllImageHandlers()
622 splash
= MySplashScreen()
628 #---------------------------------------------------------------------------
632 demoPath
= os
.path
.dirname(__file__
)
636 app
= MyApp(wxPlatform
== "__WXMAC__")
640 #---------------------------------------------------------------------------
644 overview
= """<html><body>
647 Python is an interpreted, interactive, object-oriented programming
648 language often compared to Tcl, Perl, Scheme, or Java.
650 <p> Python combines remarkable power with very clear syntax. It has
651 modules, classes, exceptions, very high level dynamic data types, and
652 dynamic typing. There are interfaces to many system calls and
653 libraries, and new built-in modules are easily written in C or
654 C++. Python is also usable as an extension language for applications
655 that need a programmable interface. <p>
659 wxWindows is a free C++ framework designed to make cross-platform
660 programming child's play. Well, almost. wxWindows 2 supports Windows
661 3.1/95/98/NT, Unix with GTK/Motif/Lesstif, with a Mac version
662 underway. Other ports are under consideration. <p>
664 wxWindows is a set of libraries that allows C++ applications to
665 compile and run on several different types of computers, with minimal
666 source code changes. There is one library per supported GUI (such as
667 Motif, or Windows). As well as providing a common API (Application
668 Programming Interface) for GUI functionality, it provides
669 functionality for accessing some commonly-used operating system
670 facilities, such as copying or deleting files. wxWindows is a
671 'framework' in the sense that it provides a lot of built-in
672 functionality, which the application can use or replace as required,
673 thus saving a great deal of coding effort. Basic data structures such
674 as strings, linked lists and hash tables are also supported.
679 wxPython is a Python extension module that encapsulates the wxWindows
680 GUI classes. Currently it is only available for the Win32 and GTK
681 ports of wxWindows, but as soon as the other ports are brought up to
682 the same level as Win32 and GTK, it should be fairly trivial to
683 enable wxPython to be used with the new GUI.
687 The wxPython extension module attempts to mirror the class heiarchy
688 of wxWindows as closely as possible. This means that there is a
689 wxFrame class in wxPython that looks, smells, tastes and acts almost
690 the same as the wxFrame class in the C++ version. Unfortunately,
691 because of differences in the languages, wxPython doesn't match
692 wxWindows exactly, but the differences should be easy to absorb
693 because they are natural to Python. For example, some methods that
694 return multiple values via argument pointers in C++ will return a
695 tuple of values in Python.
699 There is still much to be done for wxPython, many classes still need
700 to be mirrored. Also, wxWindows is still somewhat of a moving target
701 so it is a bit of an effort just keeping wxPython up to date. On the
702 other hand, there are enough of the core classes completed that
703 useful applications can be written.
707 wxPython is close enough to the C++ version that the majority of
708 the wxPython documentation is actually just notes attached to the C++
709 documents that describe the places where wxPython is different. There
710 is also a series of sample programs included, and a series of
711 documentation pages that assist the programmer in getting started
717 #----------------------------------------------------------------------------
718 #----------------------------------------------------------------------------
720 if __name__
== '__main__':
723 #----------------------------------------------------------------------------