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

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

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 518 by jonathan, Tue Mar 11 17:34:21 2003 UTC revision 2446 by frank, Mon Dec 13 11:52:34 2004 UTC
# Line 1  Line 1 
1  # Copyright (C) 2001, 2002, 2003 by Intevation GmbH  # Copyright (C) 2001, 2002, 2003, 2004 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]>
# Line 13  Thuban's application object. Line 13  Thuban's application object.
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 _  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, LoadCancelled
29  from Thuban.Model.messages import MAPS_CHANGED  from Thuban.Model.messages import MAPS_CHANGED
30    from Thuban.Model.layer import RasterLayer
31    import Thuban.Model.resource
32    
33  import view  import view
34  import tree  import tree
 from interactor import Interactor  
35  import mainwindow  import mainwindow
36    import dbdialog
37    import altpathdialog
38    import exceptiondialog
39    
40  from messages import SESSION_REPLACED  from messages import SESSION_REPLACED
41    
42    
   
43  class ThubanApplication(wxApp, Publisher):  class ThubanApplication(wxApp, Publisher):
44    
45      """      """
# Line 41  class ThubanApplication(wxApp, Publisher Line 47  class ThubanApplication(wxApp, Publisher
47    
48      All wxWindows programs have to have an instance of an application      All wxWindows programs have to have an instance of an application
49      class derived from wxApp. In Thuban the application class holds      class derived from wxApp. In Thuban the application class holds
50      references to the main window, the session and the interactor.      references to the main window and the session.
51      """      """
52    
53      def OnInit(self):      def OnInit(self):
54            sys.excepthook = self.ShowExceptionDialog
55    
56            # Initialize instance variables before trying to create any
57            # windows.  Creating windows can start an event loop if
58            # e.g. message boxes are popped up for some reason, and event
59            # handlers, especially EVT_UPDATE_UI may want to access things
60            # from the application.
61    
62            # Defaults for the directories used in file dialogs
63            self.path={"data":".", "projection":".", "alt_path":""}
64    
65            self.session = None
66            self.top = None
67            self.create_session()
68    
69            # Create an optional splash screen and then the mainwindow
70          self.splash = self.splash_screen()          self.splash = self.splash_screen()
71          if self.splash is not None:          if self.splash is not None:
72              self.splash.Show()              self.splash.Show()
73          self.read_startup_files()          self.read_startup_files()
         self.interactor = Interactor(None)  
74          self.top = self.CreateMainWindow()          self.top = self.CreateMainWindow()
75            # The session was alredy created above and we need to get the
76            # map into the mainwindow.  maps_changed does that.
77            self.maps_changed()
78          self.SetTopWindow(self.top)          self.SetTopWindow(self.top)
79          if self.splash is None:          if self.splash is None:
80              self.ShowMainWindow()              self.ShowMainWindow()
81          self.session = None  
         self.create_session()  
82          return True          return True
83    
84      def OnExit(self):      def OnExit(self):
# Line 64  class ThubanApplication(wxApp, Publisher Line 87  class ThubanApplication(wxApp, Publisher
87          Extend this in derived classes if needed.          Extend this in derived classes if needed.
88          """          """
89          self.session.Destroy()          self.session.Destroy()
90          self.interactor.Destroy()          self.session = None
91          Publisher.Destroy(self)          Publisher.Destroy(self)
92    
     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()  
   
93      def read_startup_files(self):      def read_startup_files(self):
94          """Read the startup files."""          """Read the startup files."""
95          # for now the startup file is ~/.thuban/thubanstart.py          # for now the startup file is ~/.thuban/thubanstart.py
96          dir =os.path.expanduser("~/.thuban")          dir = get_application_dir()
97          if os.path.isdir(dir):          if os.path.isdir(dir):
98              sys.path.append(dir)              sys.path.append(dir)
99              try:              try:
# Line 91  class ThubanApplication(wxApp, Publisher Line 105  class ThubanApplication(wxApp, Publisher
105                          # The ImportError exception was raised from                          # The ImportError exception was raised from
106                          # inside the thubanstart module.                          # inside the thubanstart module.
107                          sys.stderr.write(_("Cannot import the thubanstart"                          sys.stderr.write(_("Cannot import the thubanstart"
108                                           "module\n"))                                           " module\n"))
109                          traceback.print_exc(None, sys.stderr)                          traceback.print_exc(None, sys.stderr)
110                      else:                      else:
111                          # There's no thubanstart module.                          # There's no thubanstart module.
# Line 133  class ThubanApplication(wxApp, Publisher Line 147  class ThubanApplication(wxApp, Publisher
147          time.          time.
148          """          """
149          self.top.Show(True)          self.top.Show(True)
150          
151      def CreateMainWindow(self):      def CreateMainWindow(self):
152          """Create and return the main window for the application.          """Create and return the main window for the application.
153    
154          Override this in subclasses to instantiate the Thuban mainwindow          Override this in subclasses to instantiate the Thuban mainwindow
155          with different parameters or to use a different class for the          with different parameters or to use a different class for the
156          main window.          main window.
   
         when this method is called by OnInit self.interactor (to be used  
         for the interactor argument of the standard Thuban main window  
         class) has already been instantiated.  
157          """          """
158          msg = (_("This is the wxPython-based Graphical User Interface"          msg = (_("This is the wxPython-based Graphical User Interface"
159                 " for exploring geographic data"))                 " for exploring geographic data"))
160          return mainwindow.MainWindow(NULL, -1, "Thuban", self, self.interactor,          return mainwindow.MainWindow(NULL, -1, "Thuban", self, None,
161                                       initial_message = msg)                                       initial_message = msg,
162                                         size = (600, 400))
163    
164      def Session(self):      def Session(self):
165          """Return the application's session object"""          """Return the application's session object"""
# Line 166  class ThubanApplication(wxApp, Publisher Line 177  class ThubanApplication(wxApp, Publisher
177          self.session = session          self.session = session
178          self.subscribe_session(self.session)          self.subscribe_session(self.session)
179          self.issue(SESSION_REPLACED)          self.issue(SESSION_REPLACED)
         self.interactor.SetSession(session)  
180          self.maps_changed()          self.maps_changed()
181          if oldsession is not None:          if oldsession is not None:
182              self.unsubscribe_session(oldsession)              self.unsubscribe_session(oldsession)
183              oldsession.Destroy()              oldsession.Destroy()
184    
185        def SetPath(self, group, filename):
186            """Store the application's default path for file dialogs extracted
187            from a given filename.
188            """
189            self.path[group] = os.path.dirname( filename )
190    
191        def Path(self, group):
192            """Return the application's default path for file dialogs."""
193            return self.path[group]
194    
195      def subscribe_session(self, session):      def subscribe_session(self, session):
196          """Subscribe to some of the sessions channels.          """Subscribe to some of the sessions channels.
197    
# Line 198  class ThubanApplication(wxApp, Publisher Line 218  class ThubanApplication(wxApp, Publisher
218          """          """
219          self.SetSession(create_empty_session())          self.SetSession(create_empty_session())
220    
221      def OpenSession(self, filename):      def OpenSession(self, filename, db_connection_callback = None,
222          session = load_session(filename)                                      shapefile_callback = None):
223            """Open the session in the file named filename"""
224            # Make sure we deal with an absolute pathname. Otherwise we can
225            # get problems when saving because the saving code expects an
226            # absolute directory name
227            filename = os.path.abspath(filename)
228            if db_connection_callback is None:
229                db_connection_callback = self.run_db_param_dialog
230            if shapefile_callback is None:
231                shapefile_callback = self.run_alt_path_dialog
232            try:
233                session = load_session(filename,
234                                   db_connection_callback=db_connection_callback,
235                                   shapefile_callback=shapefile_callback)
236            except LoadCancelled:
237                return
238          session.SetFilename(filename)          session.SetFilename(filename)
239          session.UnsetModified()          session.UnsetModified()
240          self.SetSession(session)          self.SetSession(session)
241    
242            for map in session.Maps():
243                for layer in map.Layers():
244                    if isinstance(layer, RasterLayer) \
245                        and not Thuban.Model.resource.has_gdal_support():
246                        msg = _("The current session contains Image layers,\n"
247                                "but the GDAL library is not available to "
248                                "draw them.")
249                        dlg = wx.wxMessageDialog(None,
250                                                 msg,
251                                                 _("Library not available"),
252                                                 wx.wxOK | wx.wxICON_INFORMATION)
253                        print msg
254                        dlg.ShowModal()
255                        dlg.Destroy()
256                        break
257    
258        def run_db_param_dialog(self, parameters, message):
259            """Implementation of the db_connection_callback for loading sessions"""
260            dlg = dbdialog.DBDialog(None, _("DB Connection Parameters"),
261                                    parameters, message)
262            return dlg.RunDialog()
263    
264        # run_alt_path_dialog: Raise a dialog to ask for an alternative path
265        # if the shapefile couldn't be found.
266        # TODO:
267        #   - Store a list of already used alternative paths and return these
268        #     iteratively (using a generator)
269        #   - How do we interact with the user to tell him we used a different
270        #     shapefile (location), mode "check"? The current approach with the
271        #     file dialog is not that comfortable.
272        #
273        def run_alt_path_dialog(self, filename, mode = None, second_try = 0):
274            """Implemetation of the shapefile_callback while loading sessions.
275            
276               This implements two modes:
277               - search: Search for an alternative path. If available from a
278                 list of alrady known paths, else interactivly by file dialog.
279                 Currently the "second_try" is important since else the user might
280                 be caught in a loop.
281               - check: Ask the user for confirmation, if a path from list has
282                 been found successful.
283    
284               Returns:
285               - fname: The full path to the (shape) file.
286               - from_list: Flags if the path was taken from list or entered
287                 manually.
288            """
289    
290            if mode == "search":
291                if self.Path('alt_path') == "" or second_try:
292                    dlg = altpathdialog.AltPathFileDialog(filename)
293                    fname = dlg.RunDialog()
294                    if fname is not None:
295                        self.SetPath('alt_path', fname)    
296                    from_list = 0
297                else:
298                    fname = os.path.join(self.Path('alt_path'),
299                                         os.path.basename(filename))
300                    from_list = 1
301            elif mode == "check":
302                    dlg = altpathdialog.AltPathConfirmDialog(filename)
303                    fname = dlg.RunDialog()
304                    if fname is not None:
305                        self.SetPath('alt_path', fname)
306                    from_list = 0
307            else:
308                fname = None
309                from_list = 0
310            return fname, from_list
311    
312    
313      def SaveSession(self):      def SaveSession(self):
314          save_session(self.session, self.session.filename)          save_session(self.session, self.session.filename)
315    
316      def maps_changed(self, *args):      def maps_changed(self, *args):
317          if self.session.HasMaps():          """Subscribed to the session's MAPS_CHANGED messages.
318              self.top.SetMap(self.session.Maps()[0])  
319          else:          Set the toplevel window's map to the map in the session. This is
320              self.top.SetMap(None)          done by calling the window's SetMap method with the map as
321            argument. If the session doesn't have any maps None is used
322            instead.
323    
324            Currently Thuban can only really handle at most one map in a
325            sessions so the first map in the session's list of maps as
326            returned by the Maps method is used.
327            """
328            # The mainwindow may not have been created yet, so check whether
329            # it has been created before calling any of its methods
330            if self.top is not None:
331                if self.session.HasMaps():
332                    self.top.SetMap(self.session.Maps()[0])
333                else:
334                    self.top.SetMap(None)
335    
336        in_exception_dialog = 0 # flag: are we already inside the exception dialog?
337    
338        def ShowExceptionDialog(self, exc_type, exc_value, exc_traceback):
339            """Show a message box with information about an exception.
340        
341            The parameters are the usual values describing an exception in
342            Python, the exception type, the value and the traceback.
343        
344            This method can be used as a value for the sys.excepthook.
345            """
346    
347            if self.in_exception_dialog:
348                return
349            self.in_exception_dialog = 1
350            while wxIsBusy():
351                wxEndBusyCursor() # reset the mouse cursor
352    
353            try:
354                lines = traceback.format_exception(exc_type, exc_value,
355                                                exc_traceback)
356                message = _("An unhandled exception occurred:\n%s\n"
357                            "(please report to"
358                            " http://thuban.intevation.org/bugtracker.html)"
359                            "\n\n%s") % (exc_value, "".join(lines))
360                print message
361    
362                # We don't use an explicit parent here because this method might
363                # be called in circumstances where the main window doesn't exist
364                # anymore.
365                exceptiondialog.run_exception_dialog(None, message)
366    
367            finally:
368                self.in_exception_dialog = 0
369                # delete the last exception info that python keeps in
370                # sys.last_* because especially last_traceback keeps
371                # indirect references to all objects bound to local
372                # variables and this might prevent some object from being
373                # collected early enough.
374                sys.last_type = sys.last_value = sys.last_traceback = None
375    

Legend:
Removed from v.518  
changed lines
  Added in v.2446

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26