17 |
#__BuildDate__ = "$Date$" |
#__BuildDate__ = "$Date$" |
18 |
|
|
19 |
import os |
import os |
20 |
|
import copy |
21 |
|
|
22 |
from wxPython.wx import * |
from wxPython.wx import * |
23 |
from wxPython.wx import __version__ as wxPython_version |
from wxPython.wx import __version__ as wxPython_version |
24 |
|
|
25 |
|
from wxPython.lib.dialogs import wxMultipleChoiceDialog |
26 |
|
|
27 |
import Thuban |
import Thuban |
28 |
import Thuban.version |
import Thuban.version |
29 |
|
|
30 |
from Thuban import _ |
from Thuban import _ |
31 |
from Thuban.Model.session import create_empty_session |
from Thuban.Model.session import create_empty_session |
32 |
from Thuban.Model.layer import Layer, RasterLayer |
from Thuban.Model.layer import Layer, RasterLayer |
|
from Thuban.Model.color import Color |
|
|
from Thuban.Model.proj import Projection |
|
33 |
|
|
34 |
import view |
import view |
35 |
import tree |
import tree |
|
import proj4dialog |
|
36 |
import tableview, identifyview |
import tableview, identifyview |
37 |
from Thuban.UI.classifier import Classifier |
from Thuban.UI.classifier import Classifier |
38 |
import legend |
import legend |
65 |
# implemented in the __getattr__ method. |
# implemented in the __getattr__ method. |
66 |
delegated_methods = {"SelectLayer": "canvas", |
delegated_methods = {"SelectLayer": "canvas", |
67 |
"SelectShapes": "canvas", |
"SelectShapes": "canvas", |
68 |
|
"SelectedLayer": "canvas", |
69 |
"SelectedShapes": "canvas", |
"SelectedShapes": "canvas", |
70 |
} |
} |
71 |
|
|
105 |
|
|
106 |
self.init_dialogs() |
self.init_dialogs() |
107 |
|
|
108 |
EVT_CLOSE(self, self._OnClose) |
EVT_CLOSE(self, self.OnClose) |
109 |
|
|
110 |
def Subscribe(self, channel, *args): |
def Subscribe(self, channel, *args): |
111 |
"""Subscribe a function to a message channel. |
"""Subscribe a function to a message channel. |
396 |
def Exit(self): |
def Exit(self): |
397 |
self.Close(False) |
self.Close(False) |
398 |
|
|
399 |
def _OnClose(self, event): |
def OnClose(self, event): |
400 |
result = self.save_modified_session(can_veto = event.CanVeto()) |
result = self.save_modified_session(can_veto = event.CanVeto()) |
401 |
if result == wxID_CANCEL: |
if result == wxID_CANCEL: |
402 |
event.Veto() |
event.Veto() |
405 |
# wx's destroy event, but that isn't implemented for wxGTK |
# wx's destroy event, but that isn't implemented for wxGTK |
406 |
# yet. |
# yet. |
407 |
self.canvas.Unsubscribe(VIEW_POSITION, self.view_position_changed) |
self.canvas.Unsubscribe(VIEW_POSITION, self.view_position_changed) |
408 |
DockFrame._OnClose(self, event) |
DockFrame.OnClose(self, event) |
409 |
|
for dlg in self.dialogs.values(): |
410 |
|
dlg.Destroy() |
411 |
|
self.canvas.Destroy() |
412 |
self.Destroy() |
self.Destroy() |
413 |
|
|
414 |
def SetMap(self, map): |
def SetMap(self, map): |
463 |
if dlg.ShowModal() == wxID_OK: |
if dlg.ShowModal() == wxID_OK: |
464 |
filename = dlg.GetPath() |
filename = dlg.GetPath() |
465 |
title = os.path.splitext(os.path.basename(filename))[0] |
title = os.path.splitext(os.path.basename(filename))[0] |
|
store = self.application.Session().OpenShapefile(filename) |
|
|
layer = Layer(title, store) |
|
466 |
map = self.canvas.Map() |
map = self.canvas.Map() |
467 |
has_layers = map.HasLayers() |
has_layers = map.HasLayers() |
468 |
try: |
try: |
469 |
map.AddLayer(layer) |
store = self.application.Session().OpenShapefile(filename) |
470 |
except IOError: |
except IOError: |
471 |
# the layer couldn't be opened |
# the layer couldn't be opened |
472 |
self.RunMessageBox(_("Add Layer"), |
self.RunMessageBox(_("Add Layer"), |
473 |
_("Can't open the file '%s'.") % filename) |
_("Can't open the file '%s'.") % filename) |
474 |
else: |
else: |
475 |
|
layer = Layer(title, store) |
476 |
|
map.AddLayer(layer) |
477 |
if not has_layers: |
if not has_layers: |
478 |
# if we're adding a layer to an empty map, fit the |
# if we're adding a layer to an empty map, fit the |
479 |
# new map to the window |
# new map to the window |
486 |
if dlg.ShowModal() == wxID_OK: |
if dlg.ShowModal() == wxID_OK: |
487 |
filename = dlg.GetPath() |
filename = dlg.GetPath() |
488 |
title = os.path.splitext(os.path.basename(filename))[0] |
title = os.path.splitext(os.path.basename(filename))[0] |
|
layer = RasterLayer(title, filename) |
|
489 |
map = self.canvas.Map() |
map = self.canvas.Map() |
490 |
has_layers = map.HasLayers() |
has_layers = map.HasLayers() |
491 |
try: |
try: |
492 |
map.AddLayer(layer) |
layer = RasterLayer(title, filename) |
493 |
except IOError: |
except IOError: |
494 |
# the layer couldn't be opened |
# the layer couldn't be opened |
495 |
self.RunMessageBox(_("Add Image Layer"), |
self.RunMessageBox(_("Add Image Layer"), |
496 |
_("Can't open the file '%s'.") % filename) |
_("Can't open the file '%s'.") % filename) |
497 |
else: |
else: |
498 |
|
map.AddLayer(layer) |
499 |
if not has_layers: |
if not has_layers: |
500 |
# if we're adding a layer to an empty map, fit the |
# if we're adding a layer to an empty map, fit the |
501 |
# new map to the window |
# new map to the window |
549 |
layer = self.current_layer() |
layer = self.current_layer() |
550 |
if layer is not None: |
if layer is not None: |
551 |
layer.SetVisible(0) |
layer.SetVisible(0) |
552 |
|
|
553 |
def ShowLayer(self): |
def ShowLayer(self): |
554 |
layer = self.current_layer() |
layer = self.current_layer() |
555 |
if layer is not None: |
if layer is not None: |
556 |
layer.SetVisible(1) |
layer.SetVisible(1) |
557 |
|
|
558 |
|
def DuplicateLayer(self): |
559 |
|
"""Ceate a new layer above the selected layer with the same shapestore |
560 |
|
""" |
561 |
|
layer = self.current_layer() |
562 |
|
if layer is not None and hasattr(layer, "ShapeStore"): |
563 |
|
new_layer = Layer(_("Copy of `%s'") % layer.Title(), |
564 |
|
layer.ShapeStore(), |
565 |
|
projection = layer.GetProjection()) |
566 |
|
new_classification = copy.deepcopy(layer.GetClassification()) |
567 |
|
new_layer.SetClassification(new_classification) |
568 |
|
self.Map().AddLayer(new_layer) |
569 |
|
|
570 |
|
def CanDuplicateLayer(self): |
571 |
|
"""Return whether the DuplicateLayer method can create a duplicate""" |
572 |
|
layer = self.current_layer() |
573 |
|
return layer is not None and hasattr(layer, "ShapeStore") |
574 |
|
|
575 |
def LayerShowTable(self): |
def LayerShowTable(self): |
576 |
layer = self.current_layer() |
layer = self.current_layer() |
577 |
if layer is not None: |
if layer is not None: |
580 |
dialog = self.get_open_dialog(name) |
dialog = self.get_open_dialog(name) |
581 |
if dialog is None: |
if dialog is None: |
582 |
dialog = tableview.LayerTableFrame(self, name, |
dialog = tableview.LayerTableFrame(self, name, |
583 |
_("Table: %s") % layer.Title(), |
_("Layer Table: %s") % layer.Title(), |
584 |
layer, table) |
layer, table) |
585 |
self.add_dialog(name, dialog) |
self.add_dialog(name, dialog) |
586 |
dialog.Show(true) |
dialog.Show(True) |
587 |
else: |
else: |
588 |
# FIXME: bring dialog to front here |
# FIXME: bring dialog to front here |
589 |
pass |
pass |
638 |
dialog.Raise() |
dialog.Raise() |
639 |
|
|
640 |
def LayerJoinTable(self): |
def LayerJoinTable(self): |
641 |
print "LayerJoinTable" |
layer = self.canvas.SelectedLayer() |
642 |
|
if layer is not None: |
643 |
|
dlg = JoinDialog(self, _("Join Layer with Table"), |
644 |
|
self.application.session, |
645 |
|
layer = layer) |
646 |
|
dlg.ShowModal() |
647 |
|
|
648 |
def LayerUnjoinTable(self): |
def LayerUnjoinTable(self): |
649 |
print "LayerUnjoinTable" |
layer = self.canvas.SelectedLayer() |
650 |
|
if layer is not None: |
651 |
|
orig_store = layer.ShapeStore().OrigShapeStore() |
652 |
|
if orig_store: |
653 |
|
layer.SetShapeStore(orig_store) |
654 |
|
|
655 |
def ShowLegend(self): |
def ShowLegend(self): |
656 |
if not self.LegendShown(): |
if not self.LegendShown(): |
670 |
else: |
else: |
671 |
dialog.Show(not dialog.IsShown()) |
dialog.Show(not dialog.IsShown()) |
672 |
|
|
673 |
|
self.canvas.FitMapToWindow() |
674 |
|
|
675 |
def LegendShown(self): |
def LegendShown(self): |
676 |
"""Return true iff the legend is currently open""" |
"""Return true iff the legend is currently open""" |
677 |
dialog = self.FindRegisteredDock("legend") |
dialog = self.FindRegisteredDock("legend") |
678 |
return dialog is not None and dialog.IsShown() |
return dialog is not None and dialog.IsShown() |
679 |
|
|
680 |
def TableOpen(self): |
def TableOpen(self): |
|
print "TableOpen" |
|
681 |
dlg = wxFileDialog(self, _("Open Table"), ".", "", |
dlg = wxFileDialog(self, _("Open Table"), ".", "", |
682 |
"DBF Files (*.dbf)|*.dbf|" + |
_("DBF Files (*.dbf)") + "|*.dbf|" + |
683 |
"CSV Files (*.csv)|*.csv|" + |
#_("CSV Files (*.csv)") + "|*.csv|" + |
684 |
"All Files (*.*)|*.*", |
_("All Files (*.*)") + "|*.*", |
685 |
wxOPEN) |
wxOPEN) |
686 |
if dlg.ShowModal() == wxID_OK: |
if dlg.ShowModal() == wxID_OK: |
687 |
#self.application.session.OpenTable(dlg.GetPath()) |
filename = dlg.GetPath() |
688 |
pass |
dlg.Destroy() |
689 |
|
try: |
690 |
dlg.Destroy() |
table = self.application.session.OpenTableFile(filename) |
691 |
|
except IOError: |
692 |
|
# the layer couldn't be opened |
693 |
|
self.RunMessageBox(_("Open Table"), |
694 |
|
_("Can't open the file '%s'.") % filename) |
695 |
|
else: |
696 |
|
self.ShowTableView(table) |
697 |
|
|
698 |
def TableClose(self): |
def TableClose(self): |
699 |
print "TableClose" |
tables = self.application.session.UnreferencedTables() |
700 |
|
|
701 |
|
lst = [(t.Title(), t) for t in tables] |
702 |
|
lst.sort() |
703 |
|
titles = [i[0] for i in lst] |
704 |
|
dlg = wxMultipleChoiceDialog(self, _("Pick the tables to close:"), |
705 |
|
_("Close Table"), titles, |
706 |
|
size = (400, 300), |
707 |
|
style = wxDEFAULT_DIALOG_STYLE | |
708 |
|
wxRESIZE_BORDER) |
709 |
|
if dlg.ShowModal() == wxID_OK: |
710 |
|
for i in dlg.GetValue(): |
711 |
|
self.application.session.RemoveTable(lst[i][1]) |
712 |
|
|
713 |
|
|
714 |
def TableShow(self): |
def TableShow(self): |
715 |
print "TableShow" |
"""Offer a multi-selection dialog for tables to be displayed |
716 |
|
|
717 |
def TableHide(self): |
The windows for the selected tables are opened or brought to |
718 |
print "TableHide" |
the front. |
719 |
|
""" |
720 |
|
tables = self.application.session.Tables() |
721 |
|
|
722 |
|
lst = [(t.Title(), t) for t in tables] |
723 |
|
lst.sort() |
724 |
|
titles = [i[0] for i in lst] |
725 |
|
dlg = wxMultipleChoiceDialog(self, _("Pick the table to show:"), |
726 |
|
_("Show Table"), titles, |
727 |
|
size = (400,300), |
728 |
|
style = wxDEFAULT_DIALOG_STYLE | |
729 |
|
wxRESIZE_BORDER) |
730 |
|
if (dlg.ShowModal() == wxID_OK): |
731 |
|
for i in dlg.GetValue(): |
732 |
|
# XXX: if the table belongs to a layer, open a |
733 |
|
# LayerTableFrame instead of QueryTableFrame |
734 |
|
self.ShowTableView(lst[i][1]) |
735 |
|
|
736 |
def TableJoin(self): |
def TableJoin(self): |
|
print "TableJoin" |
|
737 |
dlg = JoinDialog(self, _("Join Tables"), self.application.session) |
dlg = JoinDialog(self, _("Join Tables"), self.application.session) |
738 |
if dlg.ShowModal() == wxID_OK: |
dlg.ShowModal() |
739 |
print "OK" |
|
740 |
|
def ShowTableView(self, table): |
741 |
|
"""Open a table view for the table and optionally""" |
742 |
|
name = "table_view%d" % id(table) |
743 |
|
dialog = self.get_open_dialog(name) |
744 |
|
if dialog is None: |
745 |
|
dialog = tableview.QueryTableFrame(self, name, |
746 |
|
_("Table: %s") % table.Title(), |
747 |
|
table) |
748 |
|
self.add_dialog(name, dialog) |
749 |
|
dialog.Show(True) |
750 |
|
# FIXME: else bring dialog to front |
751 |
|
|
752 |
|
def TableRename(self): |
753 |
|
"""Let the user rename a table""" |
754 |
|
|
755 |
|
# First, let the user select a table |
756 |
|
tables = self.application.session.Tables() |
757 |
|
lst = [(t.Title(), t) for t in tables] |
758 |
|
lst.sort() |
759 |
|
titles = [i[0] for i in lst] |
760 |
|
dlg = wxMultipleChoiceDialog(self, _("Pick the table to rename:"), |
761 |
|
_("Rename Table"), titles, |
762 |
|
size = (400,300), |
763 |
|
style = wxDEFAULT_DIALOG_STYLE | |
764 |
|
wxRESIZE_BORDER) |
765 |
|
if (dlg.ShowModal() == wxID_OK): |
766 |
|
to_rename = [lst[i][1] for i in dlg.GetValue()] |
767 |
|
dlg.Destroy() |
768 |
|
else: |
769 |
|
to_rename = [] |
770 |
|
|
771 |
|
# Second, let the user rename the layers |
772 |
|
for table in to_rename: |
773 |
|
dlg = wxTextEntryDialog(self, "Table Title: ", "Rename Table", |
774 |
|
table.Title()) |
775 |
|
try: |
776 |
|
if dlg.ShowModal() == wxID_OK: |
777 |
|
title = dlg.GetValue() |
778 |
|
if title != "": |
779 |
|
table.SetTitle(title) |
780 |
|
|
781 |
|
# Make sure the session is marked as modified. |
782 |
|
# FIXME: This should be handled automatically, |
783 |
|
# but that requires more changes to the tables |
784 |
|
# than I have time for currently. |
785 |
|
self.application.session.changed() |
786 |
|
finally: |
787 |
|
dlg.Destroy() |
788 |
|
|
789 |
|
|
790 |
def ZoomInTool(self): |
def ZoomInTool(self): |
791 |
self.canvas.ZoomInTool() |
self.canvas.ZoomInTool() |
829 |
|
|
830 |
dlg.Destroy() |
dlg.Destroy() |
831 |
|
|
832 |
|
def RenameLayer(self): |
833 |
|
"""Let the user rename the currently selected layer""" |
834 |
|
layer = self.current_layer() |
835 |
|
if layer is not None: |
836 |
|
dlg = wxTextEntryDialog(self, "Layer Title: ", "Rename Layer", |
837 |
|
layer.Title()) |
838 |
|
try: |
839 |
|
if dlg.ShowModal() == wxID_OK: |
840 |
|
title = dlg.GetValue() |
841 |
|
if title != "": |
842 |
|
layer.SetTitle(title) |
843 |
|
finally: |
844 |
|
dlg.Destroy() |
845 |
|
|
846 |
def identify_view_on_demand(self, layer, shapes): |
def identify_view_on_demand(self, layer, shapes): |
847 |
"""Subscribed to the canvas' SHAPES_SELECTED message |
"""Subscribed to the canvas' SHAPES_SELECTED message |
848 |
|
|
997 |
# Layer menu |
# Layer menu |
998 |
_method_command("layer_projection", _("Pro&jection..."), "LayerProjection", |
_method_command("layer_projection", _("Pro&jection..."), "LayerProjection", |
999 |
sensitive = _has_selected_layer) |
sensitive = _has_selected_layer) |
1000 |
|
_method_command("layer_duplicate", _("&Duplicate"), "DuplicateLayer", |
1001 |
|
helptext = _("Duplicate selected layer(s)"), |
1002 |
|
sensitive = lambda context: context.mainwindow.CanDuplicateLayer()) |
1003 |
|
_method_command("layer_rename", _("Re&name ..."), "RenameLayer", |
1004 |
|
helptext = _("Rename selected layer"), |
1005 |
|
sensitive = _has_selected_layer) |
1006 |
_method_command("layer_raise", _("&Raise"), "RaiseLayer", |
_method_command("layer_raise", _("&Raise"), "RaiseLayer", |
1007 |
helptext = _("Raise selected layer(s)"), |
helptext = _("Raise selected layer(s)"), |
1008 |
sensitive = _has_selected_layer) |
sensitive = _has_selected_layer) |
1022 |
sensitive = _has_selected_layer) |
sensitive = _has_selected_layer) |
1023 |
_method_command("layer_jointable", _("&Join Table..."), "LayerJoinTable", |
_method_command("layer_jointable", _("&Join Table..."), "LayerJoinTable", |
1024 |
sensitive = _has_selected_layer) |
sensitive = _has_selected_layer) |
1025 |
|
|
1026 |
|
def _can_unjoin(context): |
1027 |
|
"""Return whether the Layer/Unjoin command can be executed. |
1028 |
|
|
1029 |
|
This is the case if a layer is selected and that layer has a |
1030 |
|
shapestore that has an original shapestore. |
1031 |
|
""" |
1032 |
|
layer = context.mainwindow.SelectedLayer() |
1033 |
|
if layer is None: |
1034 |
|
return 0 |
1035 |
|
getstore = getattr(layer, "ShapeStore", None) |
1036 |
|
if getstore is not None: |
1037 |
|
return getstore().OrigShapeStore() is not None |
1038 |
|
else: |
1039 |
|
return 0 |
1040 |
_method_command("layer_unjointable", _("&Unjoin Table..."), "LayerUnjoinTable", |
_method_command("layer_unjointable", _("&Unjoin Table..."), "LayerUnjoinTable", |
1041 |
sensitive = _has_selected_layer) |
sensitive = _can_unjoin) |
1042 |
|
|
1043 |
|
|
1044 |
|
def _has_tables(context): |
1045 |
|
return bool(context.session.Tables()) |
1046 |
|
|
1047 |
# Table menu |
# Table menu |
1048 |
_method_command("table_open", _("&Open..."), "TableOpen") |
_method_command("table_open", _("&Open..."), "TableOpen") |
1049 |
_method_command("table_close", _("&Close"), "TableClose") |
_method_command("table_close", _("&Close"), "TableClose", |
1050 |
_method_command("table_show", _("&Show"), "TableShow") |
sensitive = lambda context: bool(context.session.UnreferencedTables())) |
1051 |
_method_command("table_hide", _("&Hide"), "TableHide") |
_method_command("table_rename", _("&Rename..."), "TableRename", |
1052 |
_method_command("table_join", _("&Join..."), "TableJoin") |
sensitive = _has_tables) |
1053 |
|
_method_command("table_show", _("&Show"), "TableShow", |
1054 |
|
sensitive = _has_tables) |
1055 |
|
_method_command("table_join", _("&Join..."), "TableJoin", |
1056 |
|
sensitive = _has_tables) |
1057 |
|
|
1058 |
# Export only under Windows ... |
# Export only under Windows ... |
1059 |
map_menu = ["layer_add", "rasterlayer_add", "layer_remove", "map_rename", |
map_menu = ["layer_add", "rasterlayer_add", "layer_remove", "map_rename", |
1083 |
"exit"]), |
"exit"]), |
1084 |
Menu("map", _("&Map"), map_menu), |
Menu("map", _("&Map"), map_menu), |
1085 |
Menu("layer", _("&Layer"), |
Menu("layer", _("&Layer"), |
1086 |
["layer_raise", "layer_lower", |
["layer_rename", "layer_duplicate", |
1087 |
|
None, |
1088 |
|
"layer_raise", "layer_lower", |
1089 |
None, |
None, |
1090 |
"layer_show", "layer_hide", |
"layer_show", "layer_hide", |
1091 |
None, |
None, |
1097 |
None, |
None, |
1098 |
"layer_properties"]), |
"layer_properties"]), |
1099 |
Menu("table", _("&Table"), |
Menu("table", _("&Table"), |
1100 |
["table_open", "table_close", |
["table_open", "table_close", "table_rename", |
1101 |
None, |
None, |
1102 |
"table_show", "table_hide", |
"table_show", |
1103 |
None, |
None, |
1104 |
"table_join"]), |
"table_join"]), |
1105 |
Menu("help", _("&Help"), |
Menu("help", _("&Help"), |