1 |
# Copyright (C) 2001, 2002 by Intevation GmbH |
# Copyright (C) 2001, 2002, 2003 by Intevation GmbH |
2 |
# Authors: |
# Authors: |
3 |
# Jan-Oliver Wagner <[email protected]> |
# Jan-Oliver Wagner <[email protected]> |
4 |
# Bernhard Herzog <[email protected]> |
# Bernhard Herzog <[email protected]> |
13 |
__version__ = "$Revision$" |
__version__ = "$Revision$" |
14 |
|
|
15 |
import sys, os |
import sys, os |
16 |
|
import os.path |
17 |
|
|
18 |
import traceback |
import traceback |
19 |
|
|
20 |
from wxPython.wx import * |
from wxPython.wx import * |
21 |
|
|
22 |
from Thuban.Lib.connector import Publisher |
from Thuban.Lib.connector import Publisher |
23 |
|
from Thuban.Lib.fileutil import get_application_dir |
24 |
|
|
25 |
|
from Thuban import _ |
26 |
from Thuban.Model.session import create_empty_session |
from Thuban.Model.session import create_empty_session |
27 |
from Thuban.Model.save import save_session |
from Thuban.Model.save import save_session |
28 |
from Thuban.Model.load import load_session |
from Thuban.Model.load import load_session |
30 |
|
|
31 |
import view |
import view |
32 |
import tree |
import tree |
|
from interactor import Interactor |
|
33 |
import mainwindow |
import mainwindow |
34 |
|
|
35 |
from messages import SESSION_CHANGED |
from messages import SESSION_REPLACED |
36 |
|
|
37 |
|
|
38 |
|
|
43 |
|
|
44 |
All wxWindows programs have to have an instance of an application |
All wxWindows programs have to have an instance of an application |
45 |
class derived from wxApp. In Thuban the application class holds |
class derived from wxApp. In Thuban the application class holds |
46 |
references to the main window, the session and the interactor. |
references to the main window and the session. |
47 |
""" |
""" |
48 |
|
|
49 |
def OnInit(self): |
def OnInit(self): |
50 |
|
self.splash = self.splash_screen() |
51 |
|
if self.splash is not None: |
52 |
|
self.splash.Show() |
53 |
self.read_startup_files() |
self.read_startup_files() |
54 |
self.interactor = Interactor(None) |
self.top = self.CreateMainWindow() |
55 |
top = self.CreateMainWindow() |
self.SetTopWindow(self.top) |
56 |
top.Show(true) |
if self.splash is None: |
57 |
self.top = top |
self.ShowMainWindow() |
|
self.SetTopWindow(top) |
|
58 |
self.session = None |
self.session = None |
59 |
self.create_session() |
self.create_session() |
60 |
return true |
return True |
61 |
|
|
62 |
def OnExit(self): |
def OnExit(self): |
63 |
"""Clean up code. |
"""Clean up code. |
65 |
Extend this in derived classes if needed. |
Extend this in derived classes if needed. |
66 |
""" |
""" |
67 |
self.session.Destroy() |
self.session.Destroy() |
68 |
self.interactor.Destroy() |
self.session = None |
69 |
Publisher.Destroy(self) |
Publisher.Destroy(self) |
70 |
|
|
|
def MainLoop(self): |
|
|
"""Call the inherited MainLoop method and then call OnExit. |
|
|
|
|
|
In wxPython OnExit isn't called automatically, unfortunately, so |
|
|
we do it here. |
|
|
""" |
|
|
wxApp.MainLoop(self) |
|
|
self.OnExit() |
|
|
|
|
71 |
def read_startup_files(self): |
def read_startup_files(self): |
72 |
"""Read the startup files.""" |
"""Read the startup files.""" |
73 |
# for now the startup file is ~/.thuban/thubanstart.py |
# for now the startup file is ~/.thuban/thubanstart.py |
74 |
dir =os.path.expanduser("~/.thuban") |
dir = get_application_dir() |
75 |
if os.path.isdir(dir): |
if os.path.isdir(dir): |
76 |
sys.path.append(dir) |
sys.path.append(dir) |
77 |
try: |
try: |
82 |
if tb.tb_next is not None: |
if tb.tb_next is not None: |
83 |
# The ImportError exception was raised from |
# The ImportError exception was raised from |
84 |
# inside the thubanstart module. |
# inside the thubanstart module. |
85 |
sys.stderr.write("Cannot import the thubanstart" |
sys.stderr.write(_("Cannot import the thubanstart" |
86 |
"module\n") |
" module\n")) |
87 |
traceback.print_exc(None, sys.stderr) |
traceback.print_exc(None, sys.stderr) |
88 |
else: |
else: |
89 |
# There's no thubanstart module. |
# There's no thubanstart module. |
90 |
sys.stderr.write("No thubanstart module available\n") |
sys.stderr.write(_("No thubanstart module available\n")) |
91 |
finally: |
finally: |
92 |
# make sure we delete the traceback object, |
# make sure we delete the traceback object, |
93 |
# otherwise there's be circular references involving |
# otherwise there's be circular references involving |
94 |
# the current stack frame |
# the current stack frame |
95 |
del tb |
del tb |
96 |
except: |
except: |
97 |
sys.stderr.write("Cannot import the thubanstart module\n") |
sys.stderr.write(_("Cannot import the thubanstart module\n")) |
98 |
traceback.print_exc(None, sys.stderr) |
traceback.print_exc(None, sys.stderr) |
99 |
else: |
else: |
100 |
# There's no .thuban directory |
# There's no .thuban directory |
101 |
sys.stderr.write("No ~/.thuban directory\n") |
sys.stderr.write(_("No ~/.thuban directory\n")) |
102 |
|
|
103 |
|
def splash_screen(self): |
104 |
|
"""Create and return a splash screen. |
105 |
|
|
106 |
|
This method is called by OnInit to determine whether the |
107 |
|
application should have a splashscreen. If the application |
108 |
|
should display a splash screen override this method in a derived |
109 |
|
class and have it create and return the wxSplashScreen instance. |
110 |
|
The implementation of this method in the derived class should |
111 |
|
also arranged for ShowMainWindow to be called. |
112 |
|
|
113 |
|
The default implementation simply returns None so that no splash |
114 |
|
screen is shown and ShowMainWindow will be called automatically. |
115 |
|
""" |
116 |
|
return None |
117 |
|
|
118 |
|
def ShowMainWindow(self): |
119 |
|
"""Show the main window |
120 |
|
|
121 |
|
Normally this method is automatically called by OnInit to show |
122 |
|
the main window. However, if the splash_screen method has |
123 |
|
returned a splashscreen it is expected that the derived class |
124 |
|
also arranges for ShowMainWindow to be called at the appropriate |
125 |
|
time. |
126 |
|
""" |
127 |
|
self.top.Show(True) |
128 |
|
|
129 |
def CreateMainWindow(self): |
def CreateMainWindow(self): |
130 |
"""Create and return the main window for the application. |
"""Create and return the main window for the application. |
132 |
Override this in subclasses to instantiate the Thuban mainwindow |
Override this in subclasses to instantiate the Thuban mainwindow |
133 |
with different parameters or to use a different class for the |
with different parameters or to use a different class for the |
134 |
main window. |
main window. |
135 |
|
""" |
136 |
when this method is called by OnInit self.interactor (to be used |
msg = (_("This is the wxPython-based Graphical User Interface" |
137 |
for the interactor argument of the standard Thuban main window |
" for exploring geographic data")) |
138 |
class) has already been instantiated. |
return mainwindow.MainWindow(NULL, -1, "Thuban", self, None, |
139 |
""" |
initial_message = msg, |
140 |
msg = ("This is the wxPython-based Graphical User Interface" |
size = (600, 400)) |
|
" for exploring geographic data") |
|
|
return mainwindow.MainWindow(NULL, -1, "Thuban", self, self.interactor, |
|
|
initial_message = msg) |
|
141 |
|
|
142 |
def Session(self): |
def Session(self): |
143 |
"""Return the application's session object""" |
"""Return the application's session object""" |
146 |
def SetSession(self, session): |
def SetSession(self, session): |
147 |
"""Make session the new session. |
"""Make session the new session. |
148 |
|
|
149 |
Issue SESSION_CHANGED after self.session has become the new |
Issue SESSION_REPLACED after self.session has become the new |
150 |
session. After the session has been assigned call |
session. After the session has been assigned call |
151 |
self.subscribe_session() with the new session and |
self.subscribe_session() with the new session and |
152 |
self.unsubscribe_session with the old one. |
self.unsubscribe_session with the old one. |
154 |
oldsession = self.session |
oldsession = self.session |
155 |
self.session = session |
self.session = session |
156 |
self.subscribe_session(self.session) |
self.subscribe_session(self.session) |
157 |
self.issue(SESSION_CHANGED) |
self.issue(SESSION_REPLACED) |
|
self.interactor.SetSession(session) |
|
158 |
self.maps_changed() |
self.maps_changed() |
159 |
if oldsession is not None: |
if oldsession is not None: |
160 |
self.unsubscribe_session(oldsession) |
self.unsubscribe_session(oldsession) |
187 |
self.SetSession(create_empty_session()) |
self.SetSession(create_empty_session()) |
188 |
|
|
189 |
def OpenSession(self, filename): |
def OpenSession(self, filename): |
190 |
|
"""Open the session in the file named filename""" |
191 |
|
# Make sure we deal with an absolute pathname. Otherwise we can |
192 |
|
# get problems when saving because the saving code expects an |
193 |
|
# absolute directory name |
194 |
|
filename = os.path.abspath(filename) |
195 |
session = load_session(filename) |
session = load_session(filename) |
196 |
session.SetFilename(filename) |
session.SetFilename(filename) |
197 |
session.UnsetModified() |
session.UnsetModified() |