/[thuban]/branches/WIP-pyshapelib-bramz/Thuban/UI/application.py
ViewVC logotype

Contents of /branches/WIP-pyshapelib-bramz/Thuban/UI/application.py

Parent Directory Parent Directory | Revision Log Revision Log


Revision 2076 - (show annotations)
Thu Feb 19 15:19:39 2004 UTC (21 years ago) by bh
Original Path: trunk/thuban/Thuban/UI/application.py
File MIME type: text/x-python
File size: 11960 byte(s)
(ThubanApplication.OnInit): Make sure
the mainwindow has a reference to the map of the initial session.
This fixes a bug introduced with the fix for RT#2245

1 # Copyright (C) 2001, 2002, 2003, 2004 by Intevation GmbH
2 # Authors:
3 # Jan-Oliver Wagner <[email protected]>
4 # Bernhard Herzog <[email protected]>
5 #
6 # This program is free software under the GPL (>=v2)
7 # Read the file COPYING coming with Thuban for details.
8
9 """
10 Thuban's application object.
11 """
12
13 __version__ = "$Revision$"
14
15 import sys, os
16 import os.path
17
18 import traceback
19
20 from wxPython.wx import *
21
22 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
27 from Thuban.Model.save import save_session
28 from Thuban.Model.load import load_session, LoadCancelled
29 from Thuban.Model.messages import MAPS_CHANGED
30 from Thuban.Model.layer import RasterLayer
31 import Thuban.Model.resource
32
33 import view
34 import tree
35 import mainwindow
36 import dbdialog
37 import exceptiondialog
38
39 from messages import SESSION_REPLACED
40
41
42 class ThubanApplication(wxApp, Publisher):
43
44 """
45 Thuban's application class.
46
47 All wxWindows programs have to have an instance of an application
48 class derived from wxApp. In Thuban the application class holds
49 references to the main window and the session.
50 """
51
52 def OnInit(self):
53 sys.excepthook = self.ShowExceptionDialog
54
55 # Initialize instance variables before trying to create any
56 # windows. Creating windows can start an event loop if
57 # e.g. message boxes are popped up for some reason, and event
58 # handlers, especially EVT_UPDATE_UI may want to access things
59 # from the application.
60
61 # Defaults for the directories used in file dialogs
62 self.path={"data":".", "projection":"."}
63
64 self.session = None
65 self.top = None
66 self.create_session()
67
68 # Create an optional splash screen and then the mainwindow
69 self.splash = self.splash_screen()
70 if self.splash is not None:
71 self.splash.Show()
72 self.read_startup_files()
73 self.top = self.CreateMainWindow()
74 # The session was alredy created above and we need to get the
75 # map into the mainwindow. maps_changed does that.
76 self.maps_changed()
77 self.SetTopWindow(self.top)
78 if self.splash is None:
79 self.ShowMainWindow()
80
81 return True
82
83 def OnExit(self):
84 """Clean up code.
85
86 Extend this in derived classes if needed.
87 """
88 self.session.Destroy()
89 self.session = None
90 Publisher.Destroy(self)
91
92 def read_startup_files(self):
93 """Read the startup files."""
94 # for now the startup file is ~/.thuban/thubanstart.py
95 dir = get_application_dir()
96 if os.path.isdir(dir):
97 sys.path.append(dir)
98 try:
99 import thubanstart
100 except ImportError:
101 tb = sys.exc_info()[2]
102 try:
103 if tb.tb_next is not None:
104 # The ImportError exception was raised from
105 # inside the thubanstart module.
106 sys.stderr.write(_("Cannot import the thubanstart"
107 " module\n"))
108 traceback.print_exc(None, sys.stderr)
109 else:
110 # There's no thubanstart module.
111 sys.stderr.write(_("No thubanstart module available\n"))
112 finally:
113 # make sure we delete the traceback object,
114 # otherwise there's be circular references involving
115 # the current stack frame
116 del tb
117 except:
118 sys.stderr.write(_("Cannot import the thubanstart module\n"))
119 traceback.print_exc(None, sys.stderr)
120 else:
121 # There's no .thuban directory
122 sys.stderr.write(_("No ~/.thuban directory\n"))
123
124 def splash_screen(self):
125 """Create and return a splash screen.
126
127 This method is called by OnInit to determine whether the
128 application should have a splashscreen. If the application
129 should display a splash screen override this method in a derived
130 class and have it create and return the wxSplashScreen instance.
131 The implementation of this method in the derived class should
132 also arranged for ShowMainWindow to be called.
133
134 The default implementation simply returns None so that no splash
135 screen is shown and ShowMainWindow will be called automatically.
136 """
137 return None
138
139 def ShowMainWindow(self):
140 """Show the main window
141
142 Normally this method is automatically called by OnInit to show
143 the main window. However, if the splash_screen method has
144 returned a splashscreen it is expected that the derived class
145 also arranges for ShowMainWindow to be called at the appropriate
146 time.
147 """
148 self.top.Show(True)
149
150 def CreateMainWindow(self):
151 """Create and return the main window for the application.
152
153 Override this in subclasses to instantiate the Thuban mainwindow
154 with different parameters or to use a different class for the
155 main window.
156 """
157 msg = (_("This is the wxPython-based Graphical User Interface"
158 " for exploring geographic data"))
159 return mainwindow.MainWindow(NULL, -1, "Thuban", self, None,
160 initial_message = msg,
161 size = (600, 400))
162
163 def Session(self):
164 """Return the application's session object"""
165 return self.session
166
167 def SetSession(self, session):
168 """Make session the new session.
169
170 Issue SESSION_REPLACED after self.session has become the new
171 session. After the session has been assigned call
172 self.subscribe_session() with the new session and
173 self.unsubscribe_session with the old one.
174 """
175 oldsession = self.session
176 self.session = session
177 self.subscribe_session(self.session)
178 self.issue(SESSION_REPLACED)
179 self.maps_changed()
180 if oldsession is not None:
181 self.unsubscribe_session(oldsession)
182 oldsession.Destroy()
183
184 def SetPath(self, group, filename):
185 """Store the application's default path for file dialogs extracted
186 from a given filename.
187 """
188 self.path[group] = os.path.dirname( filename )
189
190 def Path(self, group):
191 """Return the application's default path for file dialogs."""
192 return self.path[group]
193
194 def subscribe_session(self, session):
195 """Subscribe to some of the sessions channels.
196
197 Extend this method in derived classes if you need additional
198 channels.
199 """
200 session.Subscribe(MAPS_CHANGED, self.maps_changed)
201
202 def unsubscribe_session(self, session):
203 """Unsubscribe from the sessions channels.
204
205 Extend this method in derived classes if you subscribed to
206 additional channels in subscribe_session().
207 """
208 session.Unsubscribe(MAPS_CHANGED, self.maps_changed)
209
210 def create_session(self):
211 """Create a default session.
212
213 Override this method in derived classes to instantiate the
214 session differently or to use a different session class. Don't
215 subscribe to channels here yet. Do that in the
216 subscribe_session() method.
217 """
218 self.SetSession(create_empty_session())
219
220 def OpenSession(self, filename, db_connection_callback = None):
221 """Open the session in the file named filename"""
222 # Make sure we deal with an absolute pathname. Otherwise we can
223 # get problems when saving because the saving code expects an
224 # absolute directory name
225 filename = os.path.abspath(filename)
226 if db_connection_callback is None:
227 db_connection_callback = self.run_db_param_dialog
228 try:
229 session = load_session(filename,
230 db_connection_callback=db_connection_callback)
231 except LoadCancelled:
232 return
233 session.SetFilename(filename)
234 session.UnsetModified()
235 self.SetSession(session)
236
237 for map in session.Maps():
238 for layer in map.Layers():
239 if isinstance(layer, RasterLayer) \
240 and not Thuban.Model.resource.has_gdal_support():
241 msg = _("The current session contains Image layers,\n"
242 "but the GDAL library is not available to "
243 "draw them.")
244 dlg = wx.wxMessageDialog(None,
245 msg,
246 _("Library not available"),
247 wx.wxOK | wx.wxICON_INFORMATION)
248 print msg
249 dlg.ShowModal()
250 dlg.Destroy()
251 break
252
253 def run_db_param_dialog(self, parameters, message):
254 """Implementation of the db_connection_callback for loading sessions"""
255 dlg = dbdialog.DBDialog(None, _("DB Connection Parameters"),
256 parameters, message)
257 return dlg.RunDialog()
258
259
260 def SaveSession(self):
261 save_session(self.session, self.session.filename)
262
263 def maps_changed(self, *args):
264 """Subscribed to the session's MAPS_CHANGED messages.
265
266 Set the toplevel window's map to the map in the session. This is
267 done by calling the window's SetMap method with the map as
268 argument. If the session doesn't have any maps None is used
269 instead.
270
271 Currently Thuban can only really handle at most one map in a
272 sessions so the first map in the session's list of maps as
273 returned by the Maps method is used.
274 """
275 # The mainwindow may not have been created yet, so check whether
276 # it has been created before calling any of its methods
277 if self.top is not None:
278 if self.session.HasMaps():
279 self.top.SetMap(self.session.Maps()[0])
280 else:
281 self.top.SetMap(None)
282
283 in_exception_dialog = 0 # flag: are we already inside the exception dialog?
284
285 def ShowExceptionDialog(self, exc_type, exc_value, exc_traceback):
286 """Show a message box with information about an exception.
287
288 The parameters are the usual values describing an exception in
289 Python, the exception type, the value and the traceback.
290
291 This method can be used as a value for the sys.excepthook.
292 """
293
294 if self.in_exception_dialog:
295 return
296 self.in_exception_dialog = 1
297 while wxIsBusy():
298 wxEndBusyCursor() # reset the mouse cursor
299
300 try:
301 lines = traceback.format_exception(exc_type, exc_value,
302 exc_traceback)
303 message = _("An unhandled exception occurred:\n%s\n"
304 "(please report to"
305 " http://thuban.intevation.org/bugtracker.html)"
306 "\n\n%s") % (exc_value, "".join(lines))
307 print message
308
309 # We don't use an explicit parent here because this method might
310 # be called in circumstances where the main window doesn't exist
311 # anymore.
312 exceptiondialog.run_exception_dialog(None, message)
313
314 finally:
315 self.in_exception_dialog = 0
316 # delete the last exception info that python keeps in
317 # sys.last_* because especially last_traceback keeps
318 # indirect references to all objects bound to local
319 # variables and this might prevent some object from being
320 # collected early enough.
321 sys.last_type = sys.last_value = sys.last_traceback = None
322

Properties

Name Value
svn:eol-style native
svn:keywords Author Date Id Revision

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26