/[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 6 by bh, Tue Aug 28 15:41:52 2001 UTC revision 1501 by jonathan, Tue Jul 29 13:50:35 2003 UTC
# Line 1  Line 1 
1  # Copyright (C) 2001 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]>
5  #  #
6  # This program is free software under the GPL (>=v2)  # This program is free software under the GPL (>=v2)
7  # Read the file COPYING coming with Thuban for details.  # Read the file COPYING coming with Thuban for details.
# Line 11  Thuban's application object. Line 12  Thuban's application object.
12    
13  __version__ = "$Revision$"  __version__ = "$Revision$"
14    
15    import sys, os
16    import os.path
17    
18    import traceback
19    
20  from wxPython.wx import *  from wxPython.wx import *
21    from wxPython.lib.dialogs import wxScrolledMessageDialog
22    
23  from Thuban.Lib.connector import Publisher  from Thuban.Lib.connector import Publisher
24    from Thuban.Lib.fileutil import get_application_dir
25    
26    from Thuban import _
27  from Thuban.Model.session import create_empty_session  from Thuban.Model.session import create_empty_session
28  from Thuban.Model.save import save_session  from Thuban.Model.save import save_session
29  from Thuban.Model.load import load_session  from Thuban.Model.load import load_session
30    from Thuban.Model.messages import MAPS_CHANGED
31    from Thuban.Model.layer import RasterLayer
32    import Thuban.Model.resource
33    
34  import view  import view
35  import tree  import tree
36  from interactor import Interactor  import mainwindow
 from mainwindow import MainWindow  
37    
38  from messages import SESSION_CHANGED  from messages import SESSION_REPLACED
39    
40    
41    
# Line 35  class ThubanApplication(wxApp, Publisher Line 46  class ThubanApplication(wxApp, Publisher
46    
47      All wxWindows programs have to have an instance of an application      All wxWindows programs have to have an instance of an application
48      class derived from wxApp. In Thuban the application class holds      class derived from wxApp. In Thuban the application class holds
49      references to the main window, the session and the interactor.      references to the main window and the session.
50      """      """
51    
52      def OnInit(self):      def OnInit(self):
53          # ugly hack to set the global app object before window-classes          self.splash = self.splash_screen()
54          # are instantiated          if self.splash is not None:
55          import main              self.splash.Show()
56          main.app = self          self.read_startup_files()
57          self.interactor = Interactor(None)          self.top = self.CreateMainWindow()
58          top = MainWindow(NULL, -1)          self.SetTopWindow(self.top)
59          top.Show(true)          if self.splash is None:
60          self.top = top              self.ShowMainWindow()
         self.SetTopWindow(top)  
61          self.session = None          self.session = None
62          self.create_session()          self.create_session()
63            return True
64    
65        def OnExit(self):
66            """Clean up code.
67    
68            Extend this in derived classes if needed.
69            """
70            self.session.Destroy()
71            self.session = None
72            Publisher.Destroy(self)
73    
74          f = wxFrame(top, -1, 'test', wxDefaultPosition, wxSize(300, 300))      def read_startup_files(self):
75          self.tree = tree.myTreeCtrlPanel(f, self)          """Read the startup files."""
76          f.Show(true)          # for now the startup file is ~/.thuban/thubanstart.py
77          return true          dir = get_application_dir()
78            if os.path.isdir(dir):
79                sys.path.append(dir)
80                try:
81                    import thubanstart
82                except ImportError:
83                    tb = sys.exc_info()[2]
84                    try:
85                        if tb.tb_next is not None:
86                            # The ImportError exception was raised from
87                            # inside the thubanstart module.
88                            sys.stderr.write(_("Cannot import the thubanstart"
89                                             " module\n"))
90                            traceback.print_exc(None, sys.stderr)
91                        else:
92                            # There's no thubanstart module.
93                            sys.stderr.write(_("No thubanstart module available\n"))
94                    finally:
95                        # make sure we delete the traceback object,
96                        # otherwise there's be circular references involving
97                        # the current stack frame
98                        del tb
99                except:
100                    sys.stderr.write(_("Cannot import the thubanstart module\n"))
101                    traceback.print_exc(None, sys.stderr)
102            else:
103                # There's no .thuban directory
104                sys.stderr.write(_("No ~/.thuban directory\n"))
105    
106        def splash_screen(self):
107            """Create and return a splash screen.
108    
109            This method is called by OnInit to determine whether the
110            application should have a splashscreen. If the application
111            should display a splash screen override this method in a derived
112            class and have it create and return the wxSplashScreen instance.
113            The implementation of this method in the derived class should
114            also arranged for ShowMainWindow to be called.
115    
116            The default implementation simply returns None so that no splash
117            screen is shown and ShowMainWindow will be called automatically.
118            """
119            return None
120    
121        def ShowMainWindow(self):
122            """Show the main window
123    
124            Normally this method is automatically called by OnInit to show
125            the main window. However, if the splash_screen method has
126            returned a splashscreen it is expected that the derived class
127            also arranges for ShowMainWindow to be called at the appropriate
128            time.
129            """
130            self.top.Show(True)
131    
132        def CreateMainWindow(self):
133            """Create and return the main window for the application.
134    
135            Override this in subclasses to instantiate the Thuban mainwindow
136            with different parameters or to use a different class for the
137            main window.
138            """
139            msg = (_("This is the wxPython-based Graphical User Interface"
140                   " for exploring geographic data"))
141            return mainwindow.MainWindow(NULL, -1, "Thuban", self, None,
142                                         initial_message = msg,
143                                         size = (600, 400))
144    
145        def Session(self):
146            """Return the application's session object"""
147            return self.session
148    
149      def SetSession(self, session):      def SetSession(self, session):
150            """Make session the new session.
151    
152            Issue SESSION_REPLACED after self.session has become the new
153            session. After the session has been assigned call
154            self.subscribe_session() with the new session and
155            self.unsubscribe_session with the old one.
156            """
157          oldsession = self.session          oldsession = self.session
158          self.session = session          self.session = session
159          self.issue(SESSION_CHANGED)          self.subscribe_session(self.session)
160          self.interactor.SetSession(session)          self.issue(SESSION_REPLACED)
161          self.set_map()          self.maps_changed()
162          if oldsession is not None:          if oldsession is not None:
163                self.unsubscribe_session(oldsession)
164              oldsession.Destroy()              oldsession.Destroy()
165    
166        def subscribe_session(self, session):
167            """Subscribe to some of the sessions channels.
168    
169            Extend this method in derived classes if you need additional
170            channels.
171            """
172            session.Subscribe(MAPS_CHANGED, self.maps_changed)
173    
174        def unsubscribe_session(self, session):
175            """Unsubscribe from the sessions channels.
176    
177            Extend this method in derived classes if you subscribed to
178            additional channels in subscribe_session().
179            """
180            session.Unsubscribe(MAPS_CHANGED, self.maps_changed)
181    
182      def create_session(self):      def create_session(self):
183          # Create a simple default session          """Create a default session.
184    
185            Override this method in derived classes to instantiate the
186            session differently or to use a different session class. Don't
187            subscribe to channels here yet. Do that in the
188            subscribe_session() method.
189            """
190          self.SetSession(create_empty_session())          self.SetSession(create_empty_session())
191    
192      def OpenSession(self, filename):      def OpenSession(self, filename):
193            """Open the session in the file named filename"""
194            # Make sure we deal with an absolute pathname. Otherwise we can
195            # get problems when saving because the saving code expects an
196            # absolute directory name
197            filename = os.path.abspath(filename)
198          session = load_session(filename)          session = load_session(filename)
199          session.SetFilename(filename)          session.SetFilename(filename)
200          session.UnsetModified()          session.UnsetModified()
201          self.SetSession(session)          self.SetSession(session)
202    
203            for map in session.Maps():
204                for layer in map.Layers():
205                    if isinstance(layer, RasterLayer) \
206                        and not Thuban.Model.resource.has_gdal_support():
207                        msg = _("The current session contains Image layers,\n" +
208                                "but the GDAL library is not available to " +
209                                "draw them.")
210                        dlg = wx.wxMessageDialog(None,
211                                                 msg,
212                                                 _("Library not available"),
213                                                 wx.wxOK | wx.wxICON_INFORMATION)
214                        print msg
215                        dlg.ShowModal()
216                        dlg.Destroy()
217                        break
218    
219      def SaveSession(self):      def SaveSession(self):
220          save_session(self.session, self.session.filename)          save_session(self.session, self.session.filename)
221    
222      def set_map(self):      def maps_changed(self, *args):
223          if self.session.HasMaps():          if self.session.HasMaps():
224              self.top.SetMap(self.session.Maps()[0])              self.top.SetMap(self.session.Maps()[0])
225          else:          else:
226              self.top.SetMap(None)              self.top.SetMap(None)
227    
228    in_exception_dialog = 0 # flag: are we already inside the exception dialog?
229    
230    def show_exception_dialog(exc_type, exc_value, exc_traceback):
231        """Show a message box with information about an exception.
232    
233        The parameters are the usual values describing an exception in
234        Python, the exception type, the value and the traceback.
235    
236        This method can be used as a value for the sys.excepthook.
237        """
238        global in_exception_dialog
239    
240        if in_exception_dialog:
241            return
242        in_exception_dialog = 1
243        while wxIsBusy():
244            wxEndBusyCursor() # reset the mouse cursor
245    
246        try:
247            lines = traceback.format_exception(exc_type, exc_value,
248                                               exc_traceback)
249            message = "An unhandled exception occurred:\n%s\n" % exc_value+\
250                      "(please report to " \
251                      "http://thuban.intevation.org/bugtracker.html)"\
252                      "\n\n\n"+\
253                      "".join(lines)
254            print message
255    
256            # We don't use an explicit parent here because this method might
257            # be called in circumstances where the main window doesn't exist
258            # anymore.
259            dlg = wxScrolledMessageDialog(None, message,
260                                          "Thuban: Internal Error")
261            dlg.ShowModal()
262            dlg.Destroy()
263    
264        finally:
265            in_exception_dialog = 0
266            # delete the last exception info that python keeps in
267            # sys.last_* because especially last_traceback keeps
268            # indirect references to all objects bound to local
269            # variables and this might prevent some object from being
270            # collected early enough.
271            sys.last_type = sys.last_value = sys.last_traceback = None
272    

Legend:
Removed from v.6  
changed lines
  Added in v.1501

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26