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

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

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

revision 1070 by bh, Tue May 27 15:09:31 2003 UTC revision 2534 by jan, Fri Jan 21 08:31:16 2005 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 12  The main window Line 12  The main window
12  """  """
13    
14  __version__ = "$Revision$"  __version__ = "$Revision$"
15    # $Source$
16  __ThubanVersion__ = "0.2" #"$THUBAN_0_2$"  # $Id$
 #__BuildDate__ = "$Date$"  
17    
18  import os  import os
19    import copy
20    
21  from wxPython.wx import *  from wxPython.wx import *
 from wxPython.wx import __version__ as wxPython_version  
   
 from wxPython.lib.dialogs import wxMultipleChoiceDialog  
22    
23  import Thuban  import Thuban
 import Thuban.version  
24    
25  from Thuban import _  from Thuban import _
26    from Thuban.Model.messages import TITLE_CHANGED
27  from Thuban.Model.session import create_empty_session  from Thuban.Model.session import create_empty_session
28  from Thuban.Model.layer import Layer, RasterLayer  from Thuban.Model.layer import Layer, RasterLayer
29    from Thuban.Model.postgisdb import PostGISShapeStore, has_postgis_support
30    # XXX: replace this by
31    # from wxPython.lib.dialogs import wxMultipleChoiceDialog
32    # when Thuban does not support wxPython 2.4.0 any more.
33    from Thuban.UI.multiplechoicedialog import wxMultipleChoiceDialog
34    
35  import view  import view
36  import tree  import tree
37  import tableview, identifyview  import tableview, identifyview
 from Thuban.UI.classifier import Classifier  
38  import legend  import legend
39  from menu import Menu  from menu import Menu
40    
41  from context import Context  from context import Context
42  from command import registry, Command, ToolCommand  from command import registry, Command, ToolCommand
43  from messages import LAYER_SELECTED, SHAPES_SELECTED, VIEW_POSITION  from messages import LAYER_SELECTED, SHAPES_SELECTED, VIEW_POSITION, \
44         MAP_REPLACED
45    from about import About
46    
47  from Thuban.UI.dock import DockFrame  from Thuban.UI.dock import DockFrame
48  from Thuban.UI.join import JoinDialog  from Thuban.UI.join import JoinDialog
49    from Thuban.UI.dbdialog import DBFrame, DBDialog, ChooseDBTableDialog
50  import resource  import resource
51    import Thuban.Model.resource
52    
53  import projdialog  import projdialog
54    
55    from Thuban.Lib.classmapper import ClassMapper
56    
57    layer_properties_dialogs = ClassMapper()
58    
59  class MainWindow(DockFrame):  class MainWindow(DockFrame):
60    
# Line 58  class MainWindow(DockFrame): Line 64  class MainWindow(DockFrame):
64      # actually come from. This delegation is implemented in the      # actually come from. This delegation is implemented in the
65      # Subscribe and unsubscribed methods      # Subscribe and unsubscribed methods
66      delegated_messages = {LAYER_SELECTED: "canvas",      delegated_messages = {LAYER_SELECTED: "canvas",
67                            SHAPES_SELECTED: "canvas"}                            SHAPES_SELECTED: "canvas",
68                              MAP_REPLACED: "canvas"}
69    
70      # Methods delegated to some instance variables. The delegation is      # Methods delegated to some instance variables. The delegation is
71      # implemented in the __getattr__ method.      # implemented in the __getattr__ method.
72      delegated_methods = {"SelectLayer": "canvas",      delegated_methods = {"SelectLayer": "canvas",
73                           "SelectShapes": "canvas",                           "SelectShapes": "canvas",
74                             "SelectedLayer": "canvas",
75                           "SelectedShapes": "canvas",                           "SelectedShapes": "canvas",
76                           }                           }
77    
# Line 96  class MainWindow(DockFrame): Line 104  class MainWindow(DockFrame):
104          canvas.Subscribe(VIEW_POSITION, self.view_position_changed)          canvas.Subscribe(VIEW_POSITION, self.view_position_changed)
105          canvas.Subscribe(SHAPES_SELECTED, self.identify_view_on_demand)          canvas.Subscribe(SHAPES_SELECTED, self.identify_view_on_demand)
106          self.canvas = canvas          self.canvas = canvas
107            self.canvas.Subscribe(TITLE_CHANGED, self.title_changed)
108    
109          self.SetMainWindow(self.canvas)          self.SetMainWindow(self.canvas)
110    
# Line 103  class MainWindow(DockFrame): Line 112  class MainWindow(DockFrame):
112    
113          self.init_dialogs()          self.init_dialogs()
114    
115            self.ShowLegend()
116    
117          EVT_CLOSE(self, self.OnClose)          EVT_CLOSE(self, self.OnClose)
118    
119      def Subscribe(self, channel, *args):      def Subscribe(self, channel, *args):
# Line 125  class MainWindow(DockFrame): Line 136  class MainWindow(DockFrame):
136          """          """
137          if channel in self.delegated_messages:          if channel in self.delegated_messages:
138              object = getattr(self, self.delegated_messages[channel])              object = getattr(self, self.delegated_messages[channel])
139              object.Unsubscribe(channel, *args)              try:
140                    object.Unsubscribe(channel, *args)
141                except wxPyDeadObjectError:
142                    # The object was a wxObject and has already been
143                    # destroyed. Hopefully it has unsubscribed all its
144                    # subscribers already so that it's OK if we do nothing
145                    # here
146                    pass
147    
148      def __getattr__(self, attr):      def __getattr__(self, attr):
149          """If attr is one of the delegated methods return that method          """If attr is one of the delegated methods return that method
# Line 317  class MainWindow(DockFrame): Line 335  class MainWindow(DockFrame):
335          return self.dialogs.get(name)          return self.dialogs.get(name)
336    
337      def view_position_changed(self):      def view_position_changed(self):
338            """Put current position in status bar after a mouse move."""
339          pos = self.canvas.CurrentPosition()          pos = self.canvas.CurrentPosition()
340          if pos is not None:          if pos is not None:
341              text = "(%10.10g, %10.10g)" % pos              text = "(%10.10g, %10.10g)" % pos
342          else:          else:
343              text = ""              text = ""
344                # XXX This is a hack until we find a better place for this code.
345                # (BER 20050120)
346                # BH wrote (20050120):
347                # this branch is only executed when the mouse
348                # leaves the canvas window, so it's not that often [..]
349                # [Here] not the right place to put this code.  
350                # I don't have a better solution at hand,
351                # but the view_position_changed is only there to update
352                # the current position.  If other information is to
353                # be shown in the status bar it should
354                # be handled in a different way and
355                # by other methods.
356                #
357                # The status bar widget supports some kind of stack of texts.
358                # maybe that can be used to distinguis
359                # between short-lived information such as the mouse position
360                # and more permanent information such as the hint
361                # about the projections.
362                map = self.canvas.Map()
363                for layer in map.layers:
364                    bbox = layer.LatLongBoundingBox()
365                    if bbox:
366                        left, bottom, right, top = bbox
367                        if not (-180 <= left <= 180 and
368                            -180 <= right <= 180 and
369                            -90 <= top <= 90 and
370                            -90 <= bottom <= 90):
371                            text = _("Select layer '%s' and pick a projection "
372                                     "using Layer/Projection...") % layer.title
373                            break
374    
375          self.set_position_text(text)          self.set_position_text(text)
376    
377      def set_position_text(self, text):      def set_position_text(self, text):
# Line 333  class MainWindow(DockFrame): Line 383  class MainWindow(DockFrame):
383          """          """
384          self.SetStatusText(text)          self.SetStatusText(text)
385    
386        def OpenOrRaiseDialog(self, name, dialog_class, *args, **kw):
387            """
388            Open or raise a dialog.
389    
390            If a dialog with the denoted name does already exist it is
391            raised.  Otherwise a new dialog, an instance of dialog_class,
392            is created, inserted into the main list and displayed.
393            """
394            dialog = self.get_open_dialog(name)
395    
396            if dialog is None:
397                dialog = dialog_class(self, name, *args, **kw)
398                self.add_dialog(name, dialog)
399                dialog.Show(True)
400            else:
401                dialog.Raise()
402                              
403      def save_modified_session(self, can_veto = 1):      def save_modified_session(self, can_veto = 1):
404          """If the current session has been modified, ask the user          """If the current session has been modified, ask the user
405          whether to save it and do so if requested. Return the outcome of          whether to save it and do so if requested. Return the outcome of
# Line 356  class MainWindow(DockFrame): Line 423  class MainWindow(DockFrame):
423              result = wxID_NO              result = wxID_NO
424          return result          return result
425    
     def prepare_new_session(self):  
         for d in self.dialogs.values():  
             if not isinstance(d, tree.SessionTreeView):  
                 d.Close()  
   
426      def NewSession(self):      def NewSession(self):
427          if self.save_modified_session() != wxID_CANCEL:          if self.save_modified_session() != wxID_CANCEL:
             self.prepare_new_session()  
428              self.application.SetSession(create_empty_session())              self.application.SetSession(create_empty_session())
429    
430      def OpenSession(self):      def OpenSession(self):
431          if self.save_modified_session() != wxID_CANCEL:          if self.save_modified_session() != wxID_CANCEL:
432              dlg = wxFileDialog(self, _("Open Session"), ".", "",              dlg = wxFileDialog(self, _("Open Session"),
433                                   self.application.Path("data"), "",
434                                 "Thuban Session File (*.thuban)|*.thuban",                                 "Thuban Session File (*.thuban)|*.thuban",
435                                 wxOPEN)                                 wxOPEN)
436              if dlg.ShowModal() == wxID_OK:              if dlg.ShowModal() == wxID_OK:
437                  self.prepare_new_session()                  self.application.OpenSession(dlg.GetPath(),
438                  self.application.OpenSession(dlg.GetPath())                                               self.run_db_param_dialog)
439                    self.application.SetPath("data", dlg.GetPath())
440              dlg.Destroy()              dlg.Destroy()
441    
442        def run_db_param_dialog(self, parameters, message):
443            dlg = DBDialog(self, _("DB Connection Parameters"), parameters,
444                           message)
445            return dlg.RunDialog()
446    
447      def SaveSession(self):      def SaveSession(self):
448          if self.application.session.filename == None:          if self.application.session.filename == None:
449              self.SaveSessionAs()              self.SaveSessionAs()
# Line 383  class MainWindow(DockFrame): Line 451  class MainWindow(DockFrame):
451              self.application.SaveSession()              self.application.SaveSession()
452    
453      def SaveSessionAs(self):      def SaveSessionAs(self):
454          dlg = wxFileDialog(self, _("Save Session As"), ".", "",          dlg = wxFileDialog(self, _("Save Session As"),
455                               self.application.Path("data"), "",
456                             "Thuban Session File (*.thuban)|*.thuban",                             "Thuban Session File (*.thuban)|*.thuban",
457                             wxSAVE|wxOVERWRITE_PROMPT)                             wxSAVE|wxOVERWRITE_PROMPT)
458          if dlg.ShowModal() == wxID_OK:          if dlg.ShowModal() == wxID_OK:
459              self.application.session.SetFilename(dlg.GetPath())              self.application.session.SetFilename(dlg.GetPath())
460              self.application.SaveSession()              self.application.SaveSession()
461                self.application.SetPath("data",dlg.GetPath())
462          dlg.Destroy()          dlg.Destroy()
463    
464      def Exit(self):      def Exit(self):
# Line 411  class MainWindow(DockFrame): Line 481  class MainWindow(DockFrame):
481    
482      def SetMap(self, map):      def SetMap(self, map):
483          self.canvas.SetMap(map)          self.canvas.SetMap(map)
484          self.__SetTitle(map.Title())          self.update_title()
485    
486          dialog = self.FindRegisteredDock("legend")          dialog = self.FindRegisteredDock("legend")
487          if dialog is not None:          if dialog is not None:
# Line 438  class MainWindow(DockFrame): Line 508  class MainWindow(DockFrame):
508          return self.get_open_dialog("session_tree") is not None          return self.get_open_dialog("session_tree") is not None
509    
510      def About(self):      def About(self):
511          self.RunMessageBox(_("About"),          dlg = About(self)
512                             _("Thuban %s\n"          dlg.ShowModal()
513                              #"Build Date: %s\n"          dlg.Destroy()
514                              "using:\n"  
515                              "  %s\n"      def DatabaseManagement(self):
516                              "  %s\n\n"          name = "dbmanagement"
517                              "Thuban is a program for\n"          dialog = self.get_open_dialog(name)
518                              "exploring geographic data.\n"          if dialog is None:
519                              "Copyright (C) 2001-2003 Intevation GmbH.\n"              map = self.canvas.Map()
520                              "Thuban is licensed under the GNU GPL"              dialog = DBFrame(self, name, self.application.Session())
521                              % (Thuban.version.longversion,              self.add_dialog(name, dialog)
522                                 "wxPython %s" % wxPython_version,              dialog.Show()
523                                 "Python %d.%d.%d" % sys.version_info[:3]          dialog.Raise()
                               )),  
 #                           % __ThubanVersion__), #__BuildDate__)),  
                            wxOK | wxICON_INFORMATION)  
524    
525      def AddLayer(self):      def AddLayer(self):
526          dlg = wxFileDialog(self, _("Select a data file"), ".", "", "*.*",          dlg = wxFileDialog(self, _("Select one or more data files"),
527                             wxOPEN)                             self.application.Path("data"), "",
528                               _("Shapefiles (*.shp)") + "|*.shp;*.SHP|" +
529                               _("All Files (*.*)") + "|*.*",
530                               wxOPEN | wxMULTIPLE)
531          if dlg.ShowModal() == wxID_OK:          if dlg.ShowModal() == wxID_OK:
532              filename = dlg.GetPath()              filenames = dlg.GetPaths()
533              title = os.path.splitext(os.path.basename(filename))[0]              for filename in filenames:
534              map = self.canvas.Map()                  title = os.path.splitext(os.path.basename(filename))[0]
535              has_layers = map.HasLayers()                  map = self.canvas.Map()
536              try:                  has_layers = map.HasLayers()
537                  store = self.application.Session().OpenShapefile(filename)                  try:
538              except IOError:                      store = self.application.Session().OpenShapefile(filename)
539                  # the layer couldn't be opened                  except IOError:
540                  self.RunMessageBox(_("Add Layer"),                      # the layer couldn't be opened
541                                     _("Can't open the file '%s'.") % filename)                      self.RunMessageBox(_("Add Layer"),
542              else:                                         _("Can't open the file '%s'.")%filename)
543                  layer = Layer(title, store)                  else:
544                  map.AddLayer(layer)                      layer = Layer(title, store)
545                  if not has_layers:                      map.AddLayer(layer)
546                      # if we're adding a layer to an empty map, fit the                      if not has_layers:
547                      # new map to the window                          # if we're adding a layer to an empty map, fit the
548                      self.canvas.FitMapToWindow()                          # new map to the window
549                            self.canvas.FitMapToWindow()
550                        self.application.SetPath("data",filename)
551          dlg.Destroy()          dlg.Destroy()
552    
553      def AddRasterLayer(self):      def AddRasterLayer(self):
554          dlg = wxFileDialog(self, _("Select an image file"), ".", "", "*.*",          dlg = wxFileDialog(self, _("Select an image file"),
555                               self.application.Path("data"), "", "*.*",
556                             wxOPEN)                             wxOPEN)
557          if dlg.ShowModal() == wxID_OK:          if dlg.ShowModal() == wxID_OK:
558              filename = dlg.GetPath()              filename = dlg.GetPath()
# Line 498  class MainWindow(DockFrame): Line 571  class MainWindow(DockFrame):
571                      # if we're adding a layer to an empty map, fit the                      # if we're adding a layer to an empty map, fit the
572                      # new map to the window                      # new map to the window
573                      self.canvas.FitMapToWindow()                      self.canvas.FitMapToWindow()
574                    self.application.SetPath("data", filename)
575            dlg.Destroy()
576    
577        def AddDBLayer(self):
578            """Add a layer read from a database"""
579            session = self.application.Session()
580            dlg = ChooseDBTableDialog(self, self.application.Session())
581    
582            if dlg.ShowModal() == wxID_OK:
583                dbconn, dbtable, id_column, geo_column = dlg.GetTable()
584                try:
585                    title = str(dbtable)
586    
587                    # Chose the correct Interface for the database type
588                    store = session.OpenDBShapeStore(dbconn, dbtable,
589                                                     id_column = id_column,
590                                                     geometry_column = geo_column)
591                    layer = Layer(title, store)
592                except:
593                    # Some error occured while initializing the layer
594                    self.RunMessageBox(_("Add Layer from database"),
595                                       _("Can't open the database table '%s'")
596                                       % dbtable)
597                    return
598    
599                map = self.canvas.Map()
600    
601                has_layers = map.HasLayers()
602                map.AddLayer(layer)
603                if not has_layers:
604                    self.canvas.FitMapToWindow()
605    
606          dlg.Destroy()          dlg.Destroy()
607    
608      def RemoveLayer(self):      def RemoveLayer(self):
# Line 518  class MainWindow(DockFrame): Line 623  class MainWindow(DockFrame):
623              return self.canvas.Map().CanRemoveLayer(layer)              return self.canvas.Map().CanRemoveLayer(layer)
624          return False          return False
625    
626        def LayerToTop(self):
627            layer = self.current_layer()
628            if layer is not None:
629                self.canvas.Map().MoveLayerToTop(layer)
630    
631      def RaiseLayer(self):      def RaiseLayer(self):
632          layer = self.current_layer()          layer = self.current_layer()
633          if layer is not None:          if layer is not None:
# Line 528  class MainWindow(DockFrame): Line 638  class MainWindow(DockFrame):
638          if layer is not None:          if layer is not None:
639              self.canvas.Map().LowerLayer(layer)              self.canvas.Map().LowerLayer(layer)
640    
641        def LayerToBottom(self):
642            layer = self.current_layer()
643            if layer is not None:
644                self.canvas.Map().MoveLayerToBottom(layer)
645    
646      def current_layer(self):      def current_layer(self):
647          """Return the currently selected layer.          """Return the currently selected layer.
648    
# Line 539  class MainWindow(DockFrame): Line 654  class MainWindow(DockFrame):
654          """Return true if a layer is currently selected"""          """Return true if a layer is currently selected"""
655          return self.canvas.HasSelectedLayer()          return self.canvas.HasSelectedLayer()
656    
657        def has_selected_shape_layer(self):
658            """Return true if a shape layer is currently selected"""
659            return isinstance(self.current_layer(), Layer)
660    
661      def has_selected_shapes(self):      def has_selected_shapes(self):
662          """Return true if a shape is currently selected"""          """Return true if a shape is currently selected"""
663          return self.canvas.HasSelectedShapes()          return self.canvas.HasSelectedShapes()
# Line 546  class MainWindow(DockFrame): Line 665  class MainWindow(DockFrame):
665      def HideLayer(self):      def HideLayer(self):
666          layer = self.current_layer()          layer = self.current_layer()
667          if layer is not None:          if layer is not None:
668              layer.SetVisible(0)              layer.SetVisible(False)
669            
670      def ShowLayer(self):      def ShowLayer(self):
671          layer = self.current_layer()          layer = self.current_layer()
672          if layer is not None:          if layer is not None:
673              layer.SetVisible(1)              layer.SetVisible(True)
674    
675        def ToggleLayerVisibility(self):
676            layer = self.current_layer()
677            layer.SetVisible(not layer.Visible())
678    
679        def DuplicateLayer(self):
680            """Ceate a new layer above the selected layer with the same shapestore
681            """
682            layer = self.current_layer()
683            if layer is not None and hasattr(layer, "ShapeStore"):
684                new_layer = Layer(_("Copy of `%s'") % layer.Title(),
685                                  layer.ShapeStore(),
686                                  projection = layer.GetProjection())
687                new_classification = copy.deepcopy(layer.GetClassification())
688                new_layer.SetClassificationColumn(
689                        layer.GetClassificationColumn())
690                new_layer.SetClassification(new_classification)
691                self.Map().AddLayer(new_layer)
692    
693        def CanDuplicateLayer(self):
694            """Return whether the DuplicateLayer method can create a duplicate"""
695            layer = self.current_layer()
696            return layer is not None and hasattr(layer, "ShapeStore")
697    
698      def LayerShowTable(self):      def LayerShowTable(self):
699            """
700            Present a TableView Window for the current layer.
701            In case the window is already open, bring it to the front.
702            In case, there is no active layer, do nothing.
703            In case, the layer has no ShapeStore, do nothing.
704            """
705          layer = self.current_layer()          layer = self.current_layer()
706          if layer is not None:          if layer is not None:
707              table = layer.table              if not hasattr(layer, "ShapeStore"):
708                    return
709                table = layer.ShapeStore().Table()
710              name = "table_view" + str(id(table))              name = "table_view" + str(id(table))
711              dialog = self.get_open_dialog(name)              dialog = self.get_open_dialog(name)
712              if dialog is None:              if dialog is None:
# Line 566  class MainWindow(DockFrame): Line 716  class MainWindow(DockFrame):
716                  self.add_dialog(name, dialog)                  self.add_dialog(name, dialog)
717                  dialog.Show(True)                  dialog.Show(True)
718              else:              else:
719                  # FIXME: bring dialog to front here                  dialog.Raise()
                 pass  
720    
721      def MapProjection(self):      def MapProjection(self):
722    
# Line 609  class MainWindow(DockFrame): Line 758  class MainWindow(DockFrame):
758          self.OpenLayerProperties(layer)          self.OpenLayerProperties(layer)
759    
760      def OpenLayerProperties(self, layer, group = None):      def OpenLayerProperties(self, layer, group = None):
761          name = "layer_properties" + str(id(layer))          """
762          dialog = self.get_open_dialog(name)          Open or raise the properties dialog.
763    
764          if dialog is None:          This method opens or raises the properties dialog for the
765              dialog = Classifier(self, name, layer, group)          currently selected layer if one is defined for this layer
766              self.add_dialog(name, dialog)          type.
767              dialog.Show()          """
768          dialog.Raise()          dialog_class = layer_properties_dialogs.get(layer)
769    
770            if dialog_class is not None:
771                name = "layer_properties" + str(id(layer))
772                self.OpenOrRaiseDialog(name, dialog_class, layer, group = group)
773    
774      def LayerJoinTable(self):      def LayerJoinTable(self):
775          print "LayerJoinTable: Not implemented."          layer = self.canvas.SelectedLayer()
776            if layer is not None:
777                dlg = JoinDialog(self, _("Join Layer with Table"),
778                                 self.application.session,
779                                 layer = layer)
780                dlg.ShowModal()
781    
782      def LayerUnjoinTable(self):      def LayerUnjoinTable(self):
783          print "LayerUnjoinTable: Not implemented."          layer = self.canvas.SelectedLayer()
784            if layer is not None:
785                orig_store = layer.ShapeStore().OrigShapeStore()
786                if orig_store:
787                    layer.SetShapeStore(orig_store)
788    
789      def ShowLegend(self):      def ShowLegend(self):
790          if not self.LegendShown():          if not self.LegendShown():
# Line 648  class MainWindow(DockFrame): Line 810  class MainWindow(DockFrame):
810          return dialog is not None and dialog.IsShown()          return dialog is not None and dialog.IsShown()
811    
812      def TableOpen(self):      def TableOpen(self):
813          dlg = wxFileDialog(self, _("Open Table"), ".", "",          dlg = wxFileDialog(self, _("Open Table"),
814                               self.application.Path("data"), "",
815                             _("DBF Files (*.dbf)") + "|*.dbf|" +                             _("DBF Files (*.dbf)") + "|*.dbf|" +
816                             #_("CSV Files (*.csv)") + "|*.csv|" +                             #_("CSV Files (*.csv)") + "|*.csv|" +
817                             _("All Files (*.*)") + "|*.*",                             _("All Files (*.*)") + "|*.*",
# Line 664  class MainWindow(DockFrame): Line 827  class MainWindow(DockFrame):
827                                     _("Can't open the file '%s'.") % filename)                                     _("Can't open the file '%s'.") % filename)
828              else:              else:
829                  self.ShowTableView(table)                  self.ShowTableView(table)
830                    self.application.SetPath("data",filename)
831    
832      def TableClose(self):      def TableClose(self):
833          tables = self.application.session.UnreferencedTables()          tables = self.application.session.UnreferencedTables()
834    
835            lst = [(t.Title(), t) for t in tables]
836            lst.sort()
837            titles = [i[0] for i in lst]
838          dlg = wxMultipleChoiceDialog(self, _("Pick the tables to close:"),          dlg = wxMultipleChoiceDialog(self, _("Pick the tables to close:"),
839                                       _("Close Table"),                                       _("Close Table"), titles,
840                                       [t.Title() for t in tables],                                       size = (400, 300),
841                                       size = (400, 300), style=wxRESIZE_BORDER)                                       style = wxDEFAULT_DIALOG_STYLE |
842                                                 wxRESIZE_BORDER)
843          if dlg.ShowModal() == wxID_OK:          if dlg.ShowModal() == wxID_OK:
844              for i in dlg.GetValue():              for i in dlg.GetValue():
845                  self.application.session.RemoveTable(tables[i])                  self.application.session.RemoveTable(lst[i][1])
846    
847    
848      def TableShow(self):      def TableShow(self):
# Line 685  class MainWindow(DockFrame): Line 853  class MainWindow(DockFrame):
853          """          """
854          tables = self.application.session.Tables()          tables = self.application.session.Tables()
855    
856            lst = [(t.Title(), t) for t in tables]
857            lst.sort()
858            titles = [i[0] for i in lst]
859          dlg = wxMultipleChoiceDialog(self, _("Pick the table to show:"),          dlg = wxMultipleChoiceDialog(self, _("Pick the table to show:"),
860                                       _("Show Table"),                                       _("Show Table"), titles,
861                                       [t.Title() for t in tables],                                       size = (400,300),
862                                       size = (400,300), style = wxRESIZE_BORDER)                                       style = wxDEFAULT_DIALOG_STYLE |
863                                                 wxRESIZE_BORDER)
864          if (dlg.ShowModal() == wxID_OK):          if (dlg.ShowModal() == wxID_OK):
865              for i in dlg.GetValue():              for i in dlg.GetValue():
866                  # XXX: if the table belongs to a layer, open a                  # XXX: if the table belongs to a layer, open a
867                  # LayerTableFrame instead of QueryTableFrame                  # LayerTableFrame instead of QueryTableFrame
868                  self.ShowTableView(tables[i])                  self.ShowTableView(lst[i][1])
869    
870      def TableJoin(self):      def TableJoin(self):
871          dlg = JoinDialog(self, _("Join Tables"), self.application.session)          dlg = JoinDialog(self, _("Join Tables"), self.application.session)
# Line 709  class MainWindow(DockFrame): Line 881  class MainWindow(DockFrame):
881                                                 table)                                                 table)
882              self.add_dialog(name, dialog)              self.add_dialog(name, dialog)
883              dialog.Show(True)              dialog.Show(True)
884          # FIXME: else bring dialog to front          dialog.Raise()
885    
886        def TableRename(self):
887            """Let the user rename a table"""
888    
889            # First, let the user select a table
890            tables = self.application.session.Tables()
891            lst = [(t.Title(), t) for t in tables]
892            lst.sort()
893            titles = [i[0] for i in lst]
894            dlg = wxMultipleChoiceDialog(self, _("Pick the table to rename:"),
895                                         _("Rename Table"), titles,
896                                         size = (400,300),
897                                         style = wxDEFAULT_DIALOG_STYLE |
898                                                 wxRESIZE_BORDER)
899            if (dlg.ShowModal() == wxID_OK):
900                to_rename = [lst[i][1] for i in dlg.GetValue()]
901                dlg.Destroy()
902            else:
903                to_rename = []
904    
905            # Second, let the user rename the layers
906            for table in to_rename:
907                dlg = wxTextEntryDialog(self, _("Table Title:"), _("Rename Table"),
908                                        table.Title())
909                try:
910                    if dlg.ShowModal() == wxID_OK:
911                        title = dlg.GetValue()
912                        if title != "":
913                            table.SetTitle(title)
914    
915                            # Make sure the session is marked as modified.
916                            # FIXME: This should be handled automatically,
917                            # but that requires more changes to the tables
918                            # than I have time for currently.
919                            self.application.session.changed()
920                finally:
921                    dlg.Destroy()
922    
923    
924      def ZoomInTool(self):      def ZoomInTool(self):
925          self.canvas.ZoomInTool()          self.canvas.ZoomInTool()
# Line 743  class MainWindow(DockFrame): Line 953  class MainWindow(DockFrame):
953          self.canvas.Print()          self.canvas.Print()
954    
955      def RenameMap(self):      def RenameMap(self):
956          dlg = wxTextEntryDialog(self, "Map Title: ", "Rename Map",          dlg = wxTextEntryDialog(self, _("Map Title:"), _("Rename Map"),
957                                  self.Map().Title())                                  self.Map().Title())
958          if dlg.ShowModal() == wxID_OK:          if dlg.ShowModal() == wxID_OK:
959              title = dlg.GetValue()              title = dlg.GetValue()
960              if title != "":              if title != "":
961                  self.Map().SetTitle(title)                  self.Map().SetTitle(title)
                 self.__SetTitle(title)  
962    
963          dlg.Destroy()          dlg.Destroy()
964    
965        def RenameLayer(self):
966            """Let the user rename the currently selected layer"""
967            layer = self.current_layer()
968            if layer is not None:
969                dlg = wxTextEntryDialog(self, _("Layer Title:"), _("Rename Layer"),
970                                        layer.Title())
971                try:
972                    if dlg.ShowModal() == wxID_OK:
973                        title = dlg.GetValue()
974                        if title != "":
975                            layer.SetTitle(title)
976                finally:
977                    dlg.Destroy()
978    
979      def identify_view_on_demand(self, layer, shapes):      def identify_view_on_demand(self, layer, shapes):
980          """Subscribed to the canvas' SHAPES_SELECTED message          """Subscribed to the canvas' SHAPES_SELECTED message
981    
# Line 777  class MainWindow(DockFrame): Line 1000  class MainWindow(DockFrame):
1000                  # FIXME: bring dialog to front?                  # FIXME: bring dialog to front?
1001                  pass                  pass
1002    
1003      def __SetTitle(self, title):      def title_changed(self, map):
1004          self.SetTitle("Thuban - " + title)          """Subscribed to the canvas' TITLE_CHANGED messages"""
1005            self.update_title()
1006    
1007        def update_title(self):
1008            """Update the window's title according to it's current state.
1009    
1010            In this default implementation the title is 'Thuban - ' followed
1011            by the map's title or simply 'Thuban' if there is not map.
1012            Derived classes should override this method to get different
1013            titles.
1014    
1015            This method is called automatically by other methods when the
1016            title may have to change. For the methods implemented in this
1017            class this usually only means that a different map has been set
1018            or the current map's title has changed.
1019            """
1020            map = self.Map()
1021            if map is not None:
1022                title = _("Thuban - %s") % (map.Title(),)
1023            else:
1024                title = _("Thuban")
1025            self.SetTitle(title)
1026    
1027    
1028  #  #
1029  # Define all the commands available in the main window  # Define all the commands available in the main window
# Line 820  def _has_selected_layer(context): Line 1065  def _has_selected_layer(context):
1065      """Return true if a layer is selected in the context"""      """Return true if a layer is selected in the context"""
1066      return context.mainwindow.has_selected_layer()      return context.mainwindow.has_selected_layer()
1067    
1068    def _has_selected_layer_visible(context):
1069        """Return true if a layer is selected in the context which is
1070        visible."""
1071        if context.mainwindow.has_selected_layer():
1072            layer = context.mainwindow.current_layer()
1073            if layer.Visible(): return True
1074        return False
1075    
1076    def _has_selected_shape_layer(context):
1077        """Return true if a shape layer is selected in the context"""
1078        return context.mainwindow.has_selected_shape_layer()
1079    
1080  def _has_selected_shapes(context):  def _has_selected_shapes(context):
1081      """Return true if a layer is selected in the context"""      """Return true if a layer is selected in the context"""
1082      return context.mainwindow.has_selected_shapes()      return context.mainwindow.has_selected_shapes()
# Line 839  def _has_visible_map(context): Line 1096  def _has_visible_map(context):
1096      if map is not None:      if map is not None:
1097          for layer in map.Layers():          for layer in map.Layers():
1098              if layer.Visible():              if layer.Visible():
1099                  return 1                  return True
1100      return 0      return False
1101    
1102  def _has_legend_shown(context):  def _has_legend_shown(context):
1103      """Return true if the legend window is shown"""      """Return true if the legend window is shown"""
1104      return context.mainwindow.LegendShown()      return context.mainwindow.LegendShown()
1105    
1106    def _has_gdal_support(context):
1107        """Return True if the GDAL is available"""
1108        return Thuban.Model.resource.has_gdal_support()
1109    
1110    def _has_dbconnections(context):
1111        """Return whether the the session has database connections"""
1112        return context.session.HasDBConnections()
1113    
1114    def _has_postgis_support(context):
1115        return has_postgis_support()
1116    
1117    
1118  # File menu  # File menu
1119  _method_command("new_session", _("&New Session"), "NewSession")  _method_command("new_session", _("&New Session"), "NewSession",
1120  _method_command("open_session", _("&Open Session..."), "OpenSession")                  helptext = _("Start a new session"))
1121  _method_command("save_session", _("&Save Session"), "SaveSession")  _method_command("open_session", _("&Open Session..."), "OpenSession",
1122  _method_command("save_session_as", _("Save Session &As..."), "SaveSessionAs")                  helptext = _("Open a session file"))
1123    _method_command("save_session", _("&Save Session"), "SaveSession",
1124                    helptext =_("Save this session to the file it was opened from"))
1125    _method_command("save_session_as", _("Save Session &As..."), "SaveSessionAs",
1126                    helptext = _("Save this session to a new file"))
1127  _method_command("toggle_session_tree", _("Session &Tree"), "ToggleSessionTree",  _method_command("toggle_session_tree", _("Session &Tree"), "ToggleSessionTree",
1128                  checked = _has_tree_window_shown)                  checked = _has_tree_window_shown,
1129                    helptext = _("Toggle on/off the session tree analysis window"))
1130  _method_command("toggle_legend", _("Legend"), "ToggleLegend",  _method_command("toggle_legend", _("Legend"), "ToggleLegend",
1131                  checked = _has_legend_shown)                  checked = _has_legend_shown,
1132  _method_command("exit", _("E&xit"), "Exit")                  helptext = _("Toggle Legend on/off"))
1133    _method_command("database_management", _("&Database Connections..."),
1134                    "DatabaseManagement",
1135                    sensitive = _has_postgis_support)
1136    _method_command("exit", _("E&xit"), "Exit",
1137                    helptext = _("Finish working with Thuban"))
1138    
1139  # Help menu  # Help menu
1140  _method_command("help_about", _("&About..."), "About")  _method_command("help_about", _("&About..."), "About",
1141                    helptext = _("Info about Thuban authors, version and modules"))
1142    
1143    
1144  # Map menu  # Map menu
1145  _method_command("map_projection", _("Pro&jection..."), "MapProjection")  _method_command("map_projection", _("Pro&jection..."), "MapProjection",
1146                    helptext = _("Set or change the map projection"))
1147    
1148  _tool_command("map_zoom_in_tool", _("&Zoom in"), "ZoomInTool", "ZoomInTool",  _tool_command("map_zoom_in_tool", _("&Zoom in"), "ZoomInTool", "ZoomInTool",
1149                helptext = _("Switch to map-mode 'zoom-in'"), icon = "zoom_in",                helptext = _("Switch to map-mode 'zoom-in'"), icon = "zoom_in",
# Line 882  _tool_command("map_label_tool", _("&Labe Line 1162  _tool_command("map_label_tool", _("&Labe
1162                helptext = _("Add/Remove labels"), icon = "label",                helptext = _("Add/Remove labels"), icon = "label",
1163                sensitive = _has_visible_map)                sensitive = _has_visible_map)
1164  _method_command("map_full_extent", _("&Full extent"), "FullExtent",  _method_command("map_full_extent", _("&Full extent"), "FullExtent",
1165                 helptext = _("Full Extent"), icon = "fullextent",                 helptext = _("Zoom to the full map extent"), icon = "fullextent",
1166                sensitive = _has_visible_map)                sensitive = _has_visible_map)
1167  _method_command("layer_full_extent", _("&Full layer extent"), "FullLayerExtent",  _method_command("layer_full_extent", _("&Full layer extent"), "FullLayerExtent",
1168                 helptext = _("Full Layer Extent"), icon = "fulllayerextent",                  helptext = _("Zoom to the full layer extent"),
1169                sensitive = _has_selected_layer)                  icon = "fulllayerextent", sensitive = _has_selected_layer)
1170  _method_command("selected_full_extent", _("&Full selection extent"), "FullSelectionExtent",  _method_command("selected_full_extent", _("&Full selection extent"),
1171                 helptext = _("Full Selection Extent"), icon = "fullselextent",                  "FullSelectionExtent",
1172                sensitive = _has_selected_shapes)                  helptext = _("Zoom to the full selection extent"),
1173                    icon = "fullselextent", sensitive = _has_selected_shapes)
1174  _method_command("map_export", _("E&xport"), "ExportMap",  _method_command("map_export", _("E&xport"), "ExportMap",
1175                      helptext = _("Export the map to file"))                  helptext = _("Export the map to file"))
1176  _method_command("map_print", _("Prin&t"), "PrintMap",  _method_command("map_print", _("Prin&t"), "PrintMap",
1177                  helptext = _("Print the map"))                  helptext = _("Print the map"))
1178  _method_command("map_rename", _("&Rename..."), "RenameMap",  _method_command("map_rename", _("&Rename..."), "RenameMap",
1179                  helptext = _("Rename the map"))                  helptext = _("Rename the map"))
1180  _method_command("layer_add", _("&Add Layer..."), "AddLayer",  _method_command("layer_add", _("&Add Layer..."), "AddLayer",
1181                  helptext = _("Add a new layer to active map"))                  helptext = _("Add a new layer to the map"))
1182  _method_command("rasterlayer_add", _("&Add Image Layer..."), "AddRasterLayer",  _method_command("rasterlayer_add", _("&Add Image Layer..."), "AddRasterLayer",
1183                  helptext = _("Add a new image layer to active map"))                  helptext = _("Add a new image layer to the map"),
1184                    sensitive = _has_gdal_support)
1185    _method_command("layer_add_db", _("Add &Database Layer..."), "AddDBLayer",
1186                    helptext = _("Add a new database layer to active map"),
1187                    sensitive = _has_dbconnections)
1188  _method_command("layer_remove", _("&Remove Layer"), "RemoveLayer",  _method_command("layer_remove", _("&Remove Layer"), "RemoveLayer",
1189                  helptext = _("Remove selected layer(s)"),                  helptext = _("Remove selected layer"),
1190                  sensitive = _can_remove_layer)                  sensitive = _can_remove_layer)
1191    
1192  # Layer menu  # Layer menu
1193  _method_command("layer_projection", _("Pro&jection..."), "LayerProjection",  _method_command("layer_projection", _("Pro&jection..."), "LayerProjection",
1194                    sensitive = _has_selected_layer,
1195                    helptext = _("Specify projection for selected layer"))
1196    _method_command("layer_duplicate", _("&Duplicate"), "DuplicateLayer",
1197                    helptext = _("Duplicate selected layer"),
1198              sensitive = lambda context: context.mainwindow.CanDuplicateLayer())
1199    _method_command("layer_rename", _("Re&name ..."), "RenameLayer",
1200                    helptext = _("Rename selected layer"),
1201                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1202  _method_command("layer_raise", _("&Raise"), "RaiseLayer",  _method_command("layer_raise", _("&Raise"), "RaiseLayer",
1203                  helptext = _("Raise selected layer(s)"),                  helptext = _("Raise selected layer"),
1204                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1205  _method_command("layer_lower", _("&Lower"), "LowerLayer",  _method_command("layer_lower", _("&Lower"), "LowerLayer",
1206                  helptext = _("Lower selected layer(s)"),                  helptext = _("Lower selected layer"),
1207                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1208  _method_command("layer_show", _("&Show"), "ShowLayer",  _method_command("layer_show", _("&Show"), "ShowLayer",
1209                  helptext = _("Make selected layer(s) visible"),                  helptext = _("Make selected layer visible"),
1210                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1211  _method_command("layer_hide", _("&Hide"), "HideLayer",  _method_command("layer_hide", _("&Hide"), "HideLayer",
1212                  helptext = _("Make selected layer(s) unvisible"),                  helptext = _("Make selected layer unvisible"),
1213                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1214  _method_command("layer_show_table", _("Show Ta&ble"), "LayerShowTable",  _method_command("layer_show_table", _("Show Ta&ble"), "LayerShowTable",
1215                  helptext = _("Show the selected layer's table"),                  helptext = _("Show the selected layer's table"),
1216                  sensitive = _has_selected_layer)                  sensitive = _has_selected_shape_layer)
1217  _method_command("layer_properties", _("&Properties..."), "LayerEditProperties",  _method_command("layer_properties", _("&Properties..."), "LayerEditProperties",
1218                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer,
1219                    helptext = _("Edit the properties of the selected layer"))
1220  _method_command("layer_jointable", _("&Join Table..."), "LayerJoinTable",  _method_command("layer_jointable", _("&Join Table..."), "LayerJoinTable",
1221                    sensitive = _has_selected_shape_layer,
1222                    helptext = _("Join and attach a table to the selected layer"))
1223    
1224    # further layer methods:
1225    _method_command("layer_to_top", _("&Top"), "LayerToTop",
1226                    helptext = _("Put selected layer to the top"),
1227                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1228  _method_command("layer_unjointable", _("&Unjoin Table..."), "LayerUnjoinTable",  _method_command("layer_to_bottom", _("&Bottom"), "LayerToBottom",
1229                    helptext = _("Put selected layer to the bottom"),
1230                    sensitive = _has_selected_layer)
1231    _method_command("layer_visibility", _("&Visible"), "ToggleLayerVisibility",
1232                    checked = _has_selected_layer_visible,
1233                    helptext = _("Toggle visibility of selected layer"),
1234                  sensitive = _has_selected_layer)                  sensitive = _has_selected_layer)
1235    
1236    def _can_unjoin(context):
1237        """Return whether the Layer/Unjoin command can be executed.
1238    
1239        This is the case if a layer is selected and that layer has a
1240        shapestore that has an original shapestore.
1241        """
1242        layer = context.mainwindow.SelectedLayer()
1243        if layer is None:
1244            return 0
1245        getstore = getattr(layer, "ShapeStore", None)
1246        if getstore is not None:
1247            return getstore().OrigShapeStore() is not None
1248        else:
1249            return 0
1250    _method_command("layer_unjointable", _("&Unjoin Table..."), "LayerUnjoinTable",
1251                    sensitive = _can_unjoin,
1252                    helptext = _("Undo the last join operation"))
1253    
1254    
1255    def _has_tables(context):
1256        return bool(context.session.Tables())
1257    
1258  # Table menu  # Table menu
1259  _method_command("table_open", _("&Open..."), "TableOpen")  _method_command("table_open", _("&Open..."), "TableOpen",
1260  _method_command("table_close", _("&Close"), "TableClose",                  helptext = _("Open a DBF-table from a file"))
1261         sensitive = lambda context: bool(context.session.UnreferencedTables()))  _method_command("table_close", _("&Close..."), "TableClose",
1262  _method_command("table_show", _("&Show"), "TableShow")         sensitive = lambda context: bool(context.session.UnreferencedTables()),
1263  _method_command("table_join", _("&Join..."), "TableJoin")                  helptext = _("Close one or more tables from a list"))
1264    _method_command("table_rename", _("&Rename..."), "TableRename",
1265                    sensitive = _has_tables,
1266                    helptext = _("Rename one or more tables"))
1267    _method_command("table_show", _("&Show..."), "TableShow",
1268                    sensitive = _has_tables,
1269                    helptext = _("Show one or more tables in a dialog"))
1270    _method_command("table_join", _("&Join..."), "TableJoin",
1271                    sensitive = _has_tables,
1272                    helptext = _("Join two tables creating a new one"))
1273    
1274  #  Export only under Windows ...  #  Export only under Windows ...
1275  map_menu = ["layer_add", "rasterlayer_add", "layer_remove", "map_rename",  map_menu = ["layer_add", "layer_add_db", "rasterlayer_add", "layer_remove",
1276                          None,                          None,
1277                            "map_rename",
1278                          "map_projection",                          "map_projection",
1279                          None,                          None,
1280                          "map_zoom_in_tool", "map_zoom_out_tool",                          "map_zoom_in_tool", "map_zoom_out_tool",
1281                          "map_pan_tool",                          "map_pan_tool",
1282                          "map_full_extent",                          "map_full_extent",
1283                          "layer_full_extent",                          "layer_full_extent",
1284                          "selected_full_extent",                          "selected_full_extent",
1285                          None,                          None,
# Line 960  main_menu = Menu("<main>", "<main>", Line 1296  main_menu = Menu("<main>", "<main>",
1296                   [Menu("file", _("&File"),                   [Menu("file", _("&File"),
1297                         ["new_session", "open_session", None,                         ["new_session", "open_session", None,
1298                          "save_session", "save_session_as", None,                          "save_session", "save_session_as", None,
1299                            "database_management", None,
1300                          "toggle_session_tree", None,                          "toggle_session_tree", None,
1301                          "exit"]),                          "exit"]),
1302                    Menu("map", _("&Map"), map_menu),                    Menu("map", _("&Map"), map_menu),
1303                    Menu("layer", _("&Layer"),                    Menu("layer", _("&Layer"),
1304                          ["layer_raise", "layer_lower",                         ["layer_rename", "layer_duplicate",
1305                            None,
1306                            "layer_raise", "layer_lower",
1307                          None,                          None,
1308                          "layer_show", "layer_hide",                          "layer_show", "layer_hide",
1309                          None,                          None,
# Line 976  main_menu = Menu("<main>", "<main>", Line 1315  main_menu = Menu("<main>", "<main>",
1315                          None,                          None,
1316                          "layer_properties"]),                          "layer_properties"]),
1317                    Menu("table", _("&Table"),                    Menu("table", _("&Table"),
1318                         ["table_open", "table_close",                         ["table_open", "table_close", "table_rename",
1319                         None,                         None,
1320                         "table_show",                         "table_show",
1321                         None,                         None,
# Line 993  main_toolbar = Menu("<toolbar>", "<toolb Line 1332  main_toolbar = Menu("<toolbar>", "<toolb
1332                       "selected_full_extent",                       "selected_full_extent",
1333                       None,                       None,
1334                       "map_identify_tool", "map_label_tool"])                       "map_identify_tool", "map_label_tool"])
1335    

Legend:
Removed from v.1070  
changed lines
  Added in v.2534

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26