12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085 |
- """
- @package datacatalog::tree
- @brief Data catalog tree classes
- Classes:
- - datacatalog::LocationMapTree
- - datacatalog::DataCatalogTree
- (C) 2014-2018 by Tereza Fiedlerova, and the GRASS Development Team
- This program is free software under the GNU General Public
- License (>=v2). Read the file COPYING that comes with GRASS
- for details.
- @author Tereza Fiedlerova
- @author Anna Petrasova (kratochanna gmail com)
- """
- import os
- import re
- import copy
- from multiprocessing import Process, Queue, cpu_count
- import wx
- from core.gcmd import RunCommand, GError, GMessage, GWarning
- from core.utils import GetListOfLocations
- from core.debug import Debug
- from gui_core.dialogs import TextEntryDialog
- from core.giface import StandaloneGrassInterface
- from core.treemodel import TreeModel, DictNode
- from gui_core.treeview import TreeView
- from gui_core.wrap import Menu
- from datacatalog.dialogs import CatalogReprojectionDialog
- from grass.pydispatch.signal import Signal
- import grass.script as gscript
- from grass.script import gisenv
- from grass.exceptions import CalledModuleError
- def filterModel(model, element=None, name=None):
- """Filter tree model based on type or name of map using regular expressions.
- Copies tree and remove nodes which don't match."""
- fmodel = copy.deepcopy(model)
- nodesToRemove = []
- if name:
- try:
- regex = re.compile(name)
- except:
- return fmodel
- for gisdbase in fmodel.root.children:
- for location in gisdbase.children:
- for mapset in location.children:
- for elem in mapset.children:
- if element and elem.label != element:
- nodesToRemove.append(elem)
- continue
- for node in elem.children:
- if name and regex.search(node.label) is None:
- nodesToRemove.append(node)
- for node in reversed(nodesToRemove):
- fmodel.RemoveNode(node)
- cleanUpTree(fmodel)
- return fmodel
- def cleanUpTree(model):
- """Removes empty element/mapsets/locations nodes.
- It first removes empty elements, then mapsets, then locations"""
- # removes empty elements
- nodesToRemove = []
- for gisdbase in model.root.children:
- for location in gisdbase.children:
- for mapset in location.children:
- for element in mapset.children:
- if not element.children:
- nodesToRemove.append(element)
- for node in reversed(nodesToRemove):
- model.RemoveNode(node)
- # removes empty mapsets
- nodesToRemove = []
- for gisdbase in model.root.children:
- for location in gisdbase.children:
- for mapset in location.children:
- if not mapset.children:
- nodesToRemove.append(mapset)
- for node in reversed(nodesToRemove):
- model.RemoveNode(node)
- # removes empty locations
- nodesToRemove = []
- for gisdbase in model.root.children:
- for location in gisdbase.children:
- if not location.children:
- nodesToRemove.append(location)
- for node in reversed(nodesToRemove):
- model.RemoveNode(node)
- def getLocationTree(gisdbase, location, queue, mapsets=None):
- """Creates dictionary with mapsets, elements, layers for given location.
- Returns tuple with the dictionary and error (or None)"""
- tmp_gisrc_file, env = gscript.create_environment(gisdbase, location, 'PERMANENT')
- env['GRASS_SKIP_MAPSET_OWNER_CHECK'] = '1'
- maps_dict = {}
- elements = ['raster', 'raster_3d', 'vector']
- try:
- if not mapsets:
- mapsets = gscript.read_command(
- 'g.mapsets',
- flags='l',
- separator='comma',
- quiet=True,
- env=env).strip()
- except CalledModuleError:
- queue.put(
- (maps_dict,
- _("Failed to read mapsets from location <{l}>.").format(
- l=location)))
- gscript.try_remove(tmp_gisrc_file)
- return
- else:
- listOfMapsets = mapsets.split(',')
- Debug.msg(
- 4, "Location <{0}>: {1} mapsets found".format(
- location, len(listOfMapsets)))
- for each in listOfMapsets:
- maps_dict[each] = {}
- for elem in elements:
- maps_dict[each][elem] = []
- try:
- maplist = gscript.read_command(
- 'g.list', flags='mt', type=elements,
- mapset=','.join(listOfMapsets),
- quiet=True, env=env).strip()
- except CalledModuleError:
- queue.put(
- (maps_dict,
- _("Failed to read maps from location <{l}>.").format(
- l=location)))
- gscript.try_remove(tmp_gisrc_file)
- return
- else:
- # fill dictionary
- listOfMaps = maplist.splitlines()
- Debug.msg(
- 4, "Location <{0}>: {1} maps found".format(
- location, len(listOfMaps)))
- for each in listOfMaps:
- ltype, wholename = each.split('/')
- name, mapset = wholename.split('@')
- maps_dict[mapset][ltype].append(name)
- queue.put((maps_dict, None))
- gscript.try_remove(tmp_gisrc_file)
- def map_exists(name, element, env, mapset=None):
- """Check is map is present in the mapset given in the environment
- :param name: name of the map
- :param element: data type ('raster', 'raster_3d', and 'vector')
- :param env environment created by function gscript.create_environment
- """
- if not mapset:
- mapset = gscript.run_command('g.mapset', flags='p', env=env).strip()
- # change type to element used by find file
- if element == 'raster':
- element = 'cell'
- elif element == 'raster_3d':
- element = 'grid3'
- # g.findfile returns non-zero when file was not found
- # se we ignore return code and just focus on stdout
- process = gscript.start_command(
- 'g.findfile',
- flags='n',
- element=element,
- file=name,
- mapset=mapset,
- stdout=gscript.PIPE,
- stderr=gscript.PIPE,
- env=env)
- output, errors = process.communicate()
- info = gscript.parse_key_val(output, sep='=')
- # file is the key questioned in grass.script.core find_file()
- # return code should be equivalent to checking the output
- if info['file']:
- return True
- else:
- return False
- class NameEntryDialog(TextEntryDialog):
- def __init__(self, element, mapset, env, **kwargs):
- TextEntryDialog.__init__(self, **kwargs)
- self._element = element
- self._mapset = mapset
- self._env = env
- id_OK = self.GetAffirmativeId()
- self.Bind(wx.EVT_BUTTON, self.OnOK, self.FindWindowById(id_OK))
- def OnOK(self, event):
- new = self.GetValue()
- if not new:
- return
- if map_exists(new, self._element, self._env, self._mapset):
- dlg = wx.MessageDialog(
- self,
- message=_(
- "Map of type {elem} <{name}> already exists in mapset <{mapset}>. "
- "Do you want to overwrite it?").format(
- elem=self._element,
- name=new,
- mapset=self._mapset),
- caption=_("Overwrite?"),
- style=wx.YES_NO)
- if dlg.ShowModal() == wx.ID_YES:
- dlg.Destroy()
- self._env['GRASS_OVERWRITE'] = '1'
- self.EndModal(wx.ID_OK)
- else:
- dlg.Destroy()
- return
- else:
- self.EndModal(wx.ID_OK)
- class DataCatalogNode(DictNode):
- """Node representing item in datacatalog."""
- def __init__(self, label, data=None):
- super(DataCatalogNode, self).__init__(label=label, data=data)
- def match(self, **kwargs):
- """Method used for searching according to given parameters.
- :param value: dictionary value to be matched
- :param key: data dictionary key
- """
- if not kwargs:
- return False
- for key in kwargs:
- if not (key in self.data and self.data[key] == kwargs[key]):
- return False
- return True
- class LocationMapTree(TreeView):
- def __init__(
- self, parent, model=None, style=wx.TR_HIDE_ROOT | wx.TR_EDIT_LABELS
- | wx.TR_LINES_AT_ROOT | wx.TR_HAS_BUTTONS | wx.TR_FULL_ROW_HIGHLIGHT | wx.TR_MULTIPLE):
- """Location Map Tree constructor."""
- self._model = TreeModel(DataCatalogNode)
- self._orig_model = self._model
- super(
- LocationMapTree,
- self).__init__(
- parent=parent,
- model=self._model,
- id=wx.ID_ANY,
- style=style)
- self.showNotification = Signal('Tree.showNotification')
- self.changeMapset = Signal('Tree.changeMapset')
- self.changeLocation = Signal('Tree.changeLocation')
- self.parent = parent
- self.contextMenu.connect(self.OnRightClick)
- self.itemActivated.connect(self.OnDoubleClick)
- self._initVariables()
- def _initTreeItems(self, locations=None, mapsets=None):
- """Add locations, mapsets and layers to the tree.
- Runs in multiple processes. Saves resulting data and error."""
- # mapsets param currently unused
- genv = gisenv()
- if not locations:
- locations = GetListOfLocations(genv['GISDBASE'])
- loc_count = proc_count = 0
- queue_list = []
- proc_list = []
- loc_list = []
- nprocs = 4
- try:
- nprocs = cpu_count()
- except NotImplementedError:
- nprocs = 4
- results = dict()
- errors = []
- location_nodes = []
- nlocations = len(locations)
- grassdata_node = self._model.AppendNode(
- parent=self._model.root, label=_('GRASS locations in {0}').format(
- genv['GISDBASE']), data=dict(
- type='grassdata'))
- for location in locations:
- results[location] = dict()
- varloc = self._model.AppendNode(
- parent=grassdata_node, label=location, data=dict(
- type='location', name=location))
- location_nodes.append(varloc)
- loc_count += 1
- Debug.msg(
- 3, "Scanning location <{0}> ({1}/{2})".format(location, loc_count, nlocations))
- q = Queue()
- p = Process(target=getLocationTree,
- args=(genv['GISDBASE'], location, q))
- p.start()
- queue_list.append(q)
- proc_list.append(p)
- loc_list.append(location)
- proc_count += 1
- # Wait for all running processes
- if proc_count == nprocs or loc_count == nlocations:
- Debug.msg(4, "Process subresults")
- for i in range(len(loc_list)):
- maps, error = queue_list[i].get()
- proc_list[i].join()
- if error:
- errors.append(error)
- for key in sorted(maps.keys()):
- mapset_node = self._model.AppendNode(
- parent=location_nodes[i],
- label=key, data=dict(
- type='mapset', name=key))
- self._populateMapsetItem(mapset_node, maps[key])
- proc_count = 0
- proc_list = []
- queue_list = []
- loc_list = []
- location_nodes = []
- if errors:
- wx.CallAfter(GWarning, '\n'.join(errors))
- Debug.msg(1, "Tree filled")
- self.RefreshItems()
- def InitTreeItems(self):
- """Load locations, mapsets and layers in the tree."""
- raise NotImplementedError()
- def ReloadTreeItems(self):
- """Reload locations, mapsets and layers in the tree."""
- self._orig_model = self._model
- self._model.RemoveNode(self._model.root)
- self.InitTreeItems()
- def ReloadCurrentMapset(self):
- """Reload current mapset tree only."""
- def get_first_child(node):
- try:
- child = mapsetItem.children[0]
- except IndexError:
- child = None
- return child
- genv = gisenv()
- locationItem, mapsetItem = self.GetCurrentLocationMapsetNode()
- if not locationItem or not mapsetItem:
- return
- if mapsetItem.children:
- node = get_first_child(mapsetItem)
- while node:
- self._model.RemoveNode(node)
- node = get_first_child(mapsetItem)
- q = Queue()
- p = Process(
- target=getLocationTree,
- args=(
- genv['GISDBASE'],
- locationItem.data['name'],
- q,
- mapsetItem.data['name']))
- p.start()
- maps, error = q.get()
- if error:
- raise CalledModuleError(error)
- self._populateMapsetItem(mapsetItem, maps[mapsetItem.data['name']])
- self._orig_model = copy.deepcopy(self._model)
- self.RefreshNode(mapsetItem)
- self.RefreshItems()
- def _populateMapsetItem(self, mapset_node, data):
- for elem in data:
- if data[elem]:
- element_node = self._model.AppendNode(
- parent=mapset_node, label=elem,
- data=dict(type='element', name=elem))
- for layer in data[elem]:
- self._model.AppendNode(parent=element_node, label=layer,
- data=dict(type=elem, name=layer))
- def _popupMenuLayer(self):
- """Create popup menu for layers"""
- raise NotImplementedError()
- def _popupMenuMapset(self):
- """Create popup menu for mapsets"""
- raise NotImplementedError()
- def _popupMenuElement(self):
- """Create popup menu for elements"""
- raise NotImplementedError()
- def _initVariables(self):
- """Init variables."""
- self.selected_layer = []
- self.selected_type = []
- self.selected_mapset = []
- self.selected_location = []
- self.mixed = False
- def GetControl(self):
- """Returns control itself."""
- return self
- def DefineItems(self, selected):
- """Set selected items."""
- self._initVariables()
- mixed = []
- for item in selected:
- type = item.data['type']
- if type in ('raster', 'raster_3d', 'vector'):
- self.selected_layer.append(item)
- self.selected_type.append(item.parent)
- self.selected_mapset.append(item.parent.parent)
- self.selected_location.append(item.parent.parent.parent)
- mixed.append('layer')
- elif type == 'element':
- self.selected_layer.append(None)
- self.selected_type.append(item)
- self.selected_mapset.append(item.parent)
- self.selected_location.append(item.parent.parent)
- mixed.append('element')
- elif type == 'mapset':
- self.selected_layer.append(None)
- self.selected_type.append(None)
- self.selected_mapset.append(item)
- self.selected_location.append(item.parent)
- mixed.append('mapset')
- elif type == 'location':
- self.selected_layer.append(None)
- self.selected_type.append(None)
- self.selected_mapset.append(None)
- self.selected_location.append(item)
- mixed.append('location')
- self.mixed = False
- if len(set(mixed)) > 1:
- self.mixed = True
- def OnSelChanged(self, event):
- self.selected_layer = None
- def OnRightClick(self, node):
- """Display popup menu."""
- self.DefineItems(self.GetSelected())
- if self.mixed:
- self._popupMenuEmpty()
- return
- if self.selected_layer[0]:
- self._popupMenuLayer()
- elif self.selected_mapset[0] and not self.selected_type[0] and len(self.selected_mapset) == 1:
- self._popupMenuMapset()
- elif self.selected_type[0] and len(self.selected_type) == 1:
- self._popupMenuElement()
- else:
- self._popupMenuEmpty()
- def OnDoubleClick(self, node):
- """Double click on item/node.
- Display selected layer if node is a map layer otherwise
- expand/collapse node.
- """
- if not isinstance(self._giface, StandaloneGrassInterface):
- self.DefineItems([node])
- if self.selected_layer[0] is not None:
- # display selected layer and return
- self.DisplayLayer()
- return
- # expand/collapse location/mapset...
- if self.IsNodeExpanded(node):
- self.CollapseNode(node, recursive=False)
- else:
- self.ExpandNode(node, recursive=False)
- def ExpandCurrentLocation(self):
- """Expand current location"""
- location = gscript.gisenv()['LOCATION_NAME']
- item = self._model.SearchNodes(name=location, type='location')
- if item:
- self.Select(item[0], select=True)
- self.ExpandNode(item[0], recursive=False)
- else:
- Debug.msg(1, "Location <%s> not found" % location)
- def GetCurrentLocationMapsetNode(self):
- """Get current mapset node"""
- genv = gisenv()
- location = genv['LOCATION_NAME']
- mapset = genv['MAPSET']
- locationItem = self._model.SearchNodes(name=location, type='location')
- if not locationItem:
- return None, None
- mapsetItem = self._model.SearchNodes(
- parent=locationItem[0],
- name=mapset, type='mapset')
- if not mapsetItem:
- return locationItem[0], None
- return locationItem[0], mapsetItem[0]
- def ExpandCurrentMapset(self):
- """Expand current mapset"""
- locationItem, mapsetItem = self.GetCurrentLocationMapsetNode()
- if mapsetItem:
- self.Select(mapsetItem, select=True)
- self.ExpandNode(mapsetItem, recursive=True)
- class DataCatalogTree(LocationMapTree):
- def __init__(self, parent, giface=None):
- """Data Catalog Tree constructor."""
- super(DataCatalogTree, self).__init__(parent)
- self._giface = giface
- self._restricted = True
- self._initVariablesCatalog()
- self.beginDrag = Signal('DataCatalogTree.beginDrag')
- self.endDrag = Signal('DataCatalogTree.endDrag')
- self.startEdit = Signal('DataCatalogTree.startEdit')
- self.endEdit = Signal('DataCatalogTree.endEdit')
- self.Bind(wx.EVT_TREE_BEGIN_DRAG, lambda evt:
- self._emitSignal(evt.GetItem(), self.beginDrag, event=evt))
- self.Bind(wx.EVT_TREE_END_DRAG, lambda evt:
- self._emitSignal(evt.GetItem(), self.endDrag, event=evt))
- self.beginDrag.connect(self.OnBeginDrag)
- self.endDrag.connect(self.OnEndDrag)
- self.Bind(wx.EVT_TREE_BEGIN_LABEL_EDIT, lambda evt:
- self._emitSignal(evt.GetItem(), self.startEdit, event=evt))
- self.Bind(wx.EVT_TREE_END_LABEL_EDIT, lambda evt:
- self._emitSignal(evt.GetItem(), self.endEdit, event=evt))
- self.startEdit.connect(self.OnStartEditLabel)
- self.endEdit.connect(self.OnEditLabel)
- def _initVariablesCatalog(self):
- """Init variables."""
- self.copy_mode = False
- self.copy_layer = None
- self.copy_type = None
- self.copy_mapset = None
- self.copy_location = None
- def SetRestriction(self, restrict):
- self._restricted = restrict
- def _runCommand(self, prog, **kwargs):
- cmdString = ' '.join(gscript.make_command(prog, **kwargs))
- ret = RunCommand(prog, parent=self, **kwargs)
- return ret, cmdString
- def InitTreeItems(self):
- """Add locations, mapsets and layers to the tree."""
- self._initTreeItems()
- def OnMoveMap(self, event):
- """Move layer or mapset (just save it temporarily, copying is done by paste)"""
- self.copy_mode = False
- self.copy_layer = self.selected_layer[:]
- self.copy_type = self.selected_type[:]
- self.copy_mapset = self.selected_mapset[:]
- self.copy_location = self.selected_location[:]
- if len(self.copy_layer) > 1:
- label = _("{c} maps marked for moving.").format(c=len(self.selected_layer))
- else:
- label = _("Map <{layer}> marked for moving.").format(layer=self.copy_layer[0].label)
- self.showNotification.emit(message=label)
- def OnCopyMap(self, event):
- """Copy layer or mapset (just save it temporarily, copying is done by paste)"""
- self.copy_mode = True
- self.copy_layer = self.selected_layer[:]
- self.copy_type = self.selected_type[:]
- self.copy_mapset = self.selected_mapset[:]
- self.copy_location = self.selected_location[:]
- if len(self.copy_layer) > 1:
- label = _("{c} maps marked for copying.").format(c=len(self.selected_layer))
- else:
- label = _("Map <{layer}> marked for copying.").format(layer=self.copy_layer[0].label)
- self.showNotification.emit(message=label)
- def OnRenameMap(self, event):
- """Rename layer with dialog"""
- old_name = self.selected_layer[0].label
- gisrc, env = gscript.create_environment(
- gisenv()['GISDBASE'],
- self.selected_location[0].label, mapset=self.selected_mapset[0].label)
- new_name = self._getNewMapName(
- _('New name'),
- _('Rename map'),
- old_name,
- env=env,
- mapset=self.selected_mapset[0].label,
- element=self.selected_type[0].label)
- if new_name:
- self.Rename(old_name, new_name)
- def OnStartEditLabel(self, node, event):
- """Start label editing"""
- self.DefineItems([node])
- Debug.msg(1, "Start label edit {name}".format(name=node.label))
- label = _("Editing {name}").format(name=node.label)
- self.showNotification.emit(message=label)
- if not self.selected_layer:
- event.Veto()
- def OnEditLabel(self, node, event):
- """End label editing"""
- if self.selected_layer and not event.IsEditCancelled():
- old_name = node.label
- Debug.msg(1, "End label edit {name}".format(name=old_name))
- new_name = event.GetLabel()
- self.Rename(old_name, new_name)
- def Rename(self, old, new):
- """Rename layer"""
- string = old + ',' + new
- gisrc, env = gscript.create_environment(
- gisenv()['GISDBASE'],
- self.selected_location[0].label, self.selected_mapset[0].label)
- label = _("Renaming map <{name}>...").format(name=string)
- self.showNotification.emit(message=label)
- if self.selected_type[0].label == 'vector':
- renamed, cmd = self._runCommand('g.rename', vector=string, env=env)
- elif self.selected_type[0].label == 'raster':
- renamed, cmd = self._runCommand('g.rename', raster=string, env=env)
- else:
- renamed, cmd = self._runCommand(
- 'g.rename', raster3d=string, env=env)
- if renamed == 0:
- self.selected_layer[0].label = new
- self.selected_layer[0].data['name'] = new
- self.RefreshNode(self.selected_layer[0])
- self.showNotification.emit(
- message=_("{cmd} -- completed").format(cmd=cmd))
- Debug.msg(1, "LAYER RENAMED TO: " + new)
- gscript.try_remove(gisrc)
- def OnPasteMap(self, event):
- # copying between mapsets of one location
- if not self.copy_layer:
- if self.copy_mode:
- GMessage(_("No map selected for copying."), parent=self)
- else:
- GMessage(_("No map selected for moving."), parent=self)
- return
- for i in range(len(self.copy_layer)):
- gisrc, env = gscript.create_environment(
- gisenv()['GISDBASE'], self.selected_location[0].label, mapset=self.selected_mapset[0].label)
- gisrc2, env2 = gscript.create_environment(
- gisenv()['GISDBASE'], self.copy_location[i].label, mapset=self.copy_mapset[i].label)
- new_name = self.copy_layer[i].label
- if self.selected_location[0] == self.copy_location[i]:
- # within one mapset
- if self.selected_mapset[0] == self.copy_mapset[i]:
- # ignore when just moves map
- if self.copy_mode is False:
- return
- new_name = self._getNewMapName(_('New name for <{n}>').format(n=self.copy_layer[i].label),
- _('Select new name'),
- self.copy_layer[i].label, env=env,
- mapset=self.selected_mapset[0].label,
- element=self.copy_type[i].label)
- if not new_name:
- return
- # within one location, different mapsets
- else:
- if map_exists(new_name, element=self.copy_type[i].label, env=env,
- mapset=self.selected_mapset[0].label):
- new_name = self._getNewMapName(_('New name for <{n}>').format(n=self.copy_layer[i].label),
- _('Select new name'),
- self.copy_layer[i].label, env=env,
- mapset=self.selected_mapset[0].label,
- element=self.copy_type[i].label)
- if not new_name:
- return
-
- string = self.copy_layer[i].label + '@' + self.copy_mapset[i].label + ',' + new_name
- pasted = 0
- if self.copy_mode:
- label = _("Copying <{name}>...").format(name=string)
- else:
- label = _("Moving <{name}>...").format(name=string)
- self.showNotification.emit(message=label)
- if self.copy_type[i].label == 'vector':
- pasted, cmd = self._runCommand('g.copy', vector=string, env=env)
- node = 'vector'
- elif self.copy_type[i].label == 'raster':
- pasted, cmd = self._runCommand('g.copy', raster=string, env=env)
- node = 'raster'
- else:
- pasted, cmd = self._runCommand('g.copy', raster_3d=string, env=env)
- node = 'raster_3d'
- if pasted == 0:
- self.InsertLayer(name=new_name, mapset_node=self.selected_mapset[0],
- element_name=node)
- Debug.msg(1, "COPIED TO: " + new_name)
- if self.copy_mode:
- self.showNotification.emit(message=_("g.copy completed"))
- else:
- self.showNotification.emit(message=_("g.copy completed"))
-
- # remove old
- if not self.copy_mode:
- self._removeMapAfterCopy(self.copy_layer[i], self.copy_type[i], env2)
-
- gscript.try_remove(gisrc)
- gscript.try_remove(gisrc2)
- # expand selected mapset
- else:
- if self.copy_type[i].label == 'raster_3d':
- GError(_("Reprojection is not implemented for 3D rasters"), parent=self)
- return
- if map_exists(new_name, element=self.copy_type[i].label, env=env,
- mapset=self.selected_mapset[0].label):
- new_name = self._getNewMapName(_('New name'), _('Select new name'),
- self.copy_layer[i].label, env=env,
- mapset=self.selected_mapset[0].label,
- element=self.copy_type[i].label)
- if not new_name:
- continue
- gisdbase = gisenv()['GISDBASE']
- callback = lambda gisrc2=gisrc2, gisrc=gisrc, cLayer=self.copy_layer[i], \
- cType=self.copy_type[i], cMode=self.copy_mode, name=new_name: \
- self._onDoneReprojection(env2, gisrc2, gisrc, cLayer, cType, cMode, name)
- dlg = CatalogReprojectionDialog(self, self._giface, gisdbase, self.copy_location[i].label,
- self.copy_mapset[i].label, self.copy_layer[i].label, env2,
- gisdbase, self.selected_location[0].label, self.selected_mapset[0].label,
- new_name, self.copy_type[i].label, env, callback)
- dlg.ShowModal()
- self.ExpandNode(self.selected_mapset[0], recursive=True)
- self._initVariablesCatalog()
- def _onDoneReprojection(self, iEnv, iGisrc, oGisrc, cLayer, cType, cMode, name):
- self.InsertLayer(name=name, mapset_node=self.selected_mapset[0],
- element_name=cType.label)
- if not cMode:
- self._removeMapAfterCopy(cLayer, cType, iEnv)
- gscript.try_remove(iGisrc)
- gscript.try_remove(oGisrc)
- self.ExpandNode(self.selected_mapset[0], recursive=True)
- def _removeMapAfterCopy(self, cLayer, cType, env):
- removed, cmd = self._runCommand('g.remove', type=cType.label,
- name=cLayer.label, flags='f', env=env)
- if removed == 0:
- self._model.RemoveNode(cLayer)
- self.RefreshNode(cType, recursive=True)
- Debug.msg(1, "LAYER " + cLayer.label + " DELETED")
- self.showNotification.emit(message=_("g.remove completed"))
- def InsertLayer(self, name, mapset_node, element_name):
- """Insert layer into model and refresh tree"""
- found_element = self._model.SearchNodes(
- parent=mapset_node, type='element', name=element_name)
- found_element = found_element[0] if found_element else None
- if not found_element:
- # add type node if not exists
- found_element = self._model.AppendNode(
- parent=mapset_node, label=element_name,
- data=dict(type='element', name=element_name))
- found = self._model.SearchNodes(parent=found_element, name=name)
- if len(found) == 0:
- self._model.AppendNode(parent=found_element, label=name,
- data=dict(type=element_name, name=name))
- self._model.SortChildren(found_element)
- self.RefreshNode(mapset_node, recursive=True)
- def OnDeleteMap(self, event):
- """Delete layer or mapset"""
- names = [self.selected_layer[i].label + '@' + self.selected_mapset[i].label
- for i in range(len(self.selected_layer))]
- if len(names) < 10:
- question = _("Do you really want to delete map(s) <{m}>?").format(m=', '.join(names))
- else:
- question = _("Do you really want to delete {n} maps?").format(n=len(names))
- if self._confirmDialog(question, title=_('Delete map')) == wx.ID_YES:
- label = _("Deleting {name}...").format(name=names)
- self.showNotification.emit(message=label)
- for i in range(len(self.selected_layer)):
- gisrc, env = gscript.create_environment(
- gisenv()['GISDBASE'],
- self.selected_location[i].label, self.selected_mapset[i].label)
- removed, cmd = self._runCommand(
- 'g.remove', flags='f', type=self.selected_type[i].label,
- name=self.selected_layer[i].label, env=env)
- if removed == 0:
- self._model.RemoveNode(self.selected_layer[i])
- self.RefreshNode(self.selected_type[i], recursive=True)
- Debug.msg(1, "LAYER " + self.selected_layer[i].label + " DELETED")
- # remove map layer from layer tree if exists
- if not isinstance(self._giface, StandaloneGrassInterface):
- name = self.selected_layer[i].label + '@' + self.selected_mapset[i].label
- layers = self._giface.GetLayerList().GetLayersByName(name)
- for layer in layers:
- self._giface.GetLayerList().DeleteLayer(layer)
- gscript.try_remove(gisrc)
- self.UnselectAll()
- self.showNotification.emit(message=_("g.remove completed"))
- def OnDisplayLayer(self, event):
- """Display layer in current graphics view"""
- self.DisplayLayer()
- def DisplayLayer(self):
- """Display selected layer in current graphics view"""
- all_names = []
- names = {'raster': [], 'vector': [], 'raster3d': []}
- for i in range(len(self.selected_layer)):
- name = self.selected_layer[i].label + '@' + self.selected_mapset[i].label
- names[self.selected_type[i].label].append(name)
- all_names.append(name)
- #if self.selected_location[0].label == gisenv()['LOCATION_NAME'] and self.selected_mapset[0]:
- for ltype in names:
- if names[ltype]:
- self._giface.lmgr.AddMaps(list(reversed(names[ltype])), ltype, True)
- if len(self._giface.GetLayerList()) == 1:
- # zoom to map if there is only one map layer
- self._giface.GetMapWindow().ZoomToMap()
- Debug.msg(1, "Displayed layer(s): " + str(all_names))
- def OnBeginDrag(self, node, event):
- """Just copy necessary data"""
- self.DefineItems(self.GetSelected())
- if self.selected_layer and not (self._restricted and gisenv()[
- 'LOCATION_NAME'] != self.selected_location[0].label):
- event.Allow()
- self.OnCopyMap(event)
- Debug.msg(1, "DRAG")
- else:
- event.Veto()
- def OnEndDrag(self, node, event):
- """Copy layer into target"""
- self.copy_mode = wx.GetMouseState().ControlDown()
- if node:
- self.DefineItems([node])
- if self._restricted and gisenv()['MAPSET'] != self.selected_mapset[0].label:
- GMessage(_("To move or copy maps to other mapsets, unlock editing of other mapsets"),
- parent=self)
- event.Veto()
- return
- event.Allow()
- Debug.msg(1, "DROP DONE")
- self.OnPasteMap(event)
- def OnSwitchLocationMapset(self, event):
- genv = gisenv()
- if self.selected_location[0].label == genv['LOCATION_NAME']:
- self.changeMapset.emit(mapset=self.selected_mapset[0].label)
- else:
- self.changeLocation.emit(mapset=self.selected_mapset[0].label, location=self.selected_location[0].label)
- self.ExpandCurrentMapset()
- def OnMetadata(self, event):
- """Show metadata of any raster/vector/3draster"""
- def done(event):
- gscript.try_remove(event.userData)
- for i in range(len(self.selected_layer)):
- if self.selected_type[i].label == 'raster':
- cmd = ['r.info']
- elif self.selected_type[i].label == 'vector':
- cmd = ['v.info']
- elif self.selected_type[i].label == 'raster_3d':
- cmd = ['r3.info']
- cmd.append('map=%s@%s' % (self.selected_layer[i].label, self.selected_mapset[i].label))
- gisrc, env = gscript.create_environment(
- gisenv()['GISDBASE'],
- self.selected_location[i].label, self.selected_mapset[i].label)
- # print output to command log area
- # temp gisrc file must be deleted onDone
- self._giface.RunCmd(cmd, env=env, onDone=done, userData=gisrc)
- def OnCopyName(self, event):
- """Copy layer name to clipboard"""
- if wx.TheClipboard.Open():
- do = wx.TextDataObject()
- text = []
- for i in range(len(self.selected_layer)):
- text.append('%s@%s' % (self.selected_layer[i].label, self.selected_mapset[i].label))
- do.SetText(','.join(text))
- wx.TheClipboard.SetData(do)
- wx.TheClipboard.Close()
- def Filter(self, text):
- """Filter tree based on name and type."""
- text = text.strip()
- if len(text.split(':')) > 1:
- name = text.split(':')[1].strip()
- elem = text.split(':')[0].strip()
- if 'r' == elem:
- element = 'raster'
- elif 'r3' == elem:
- element = 'raster_3d'
- elif 'v' == elem:
- element = 'vector'
- else:
- element = None
- else:
- element = None
- name = text.strip()
- self._model = filterModel(self._orig_model, name=name, element=element)
- self.RefreshItems()
- self.ExpandCurrentMapset()
- def _getNewMapName(self, message, title, value, element, mapset, env):
- """Dialog for simple text entry"""
- dlg = NameEntryDialog(parent=self, message=message, caption=title,
- element=element, env=env, mapset=mapset)
- dlg.SetValue(value)
- if dlg.ShowModal() == wx.ID_OK:
- name = dlg.GetValue()
- else:
- name = None
- dlg.Destroy()
- return name
- def _confirmDialog(self, question, title):
- """Confirm dialog"""
- dlg = wx.MessageDialog(self, question, title, wx.YES_NO)
- res = dlg.ShowModal()
- dlg.Destroy()
- return res
- def _isCurrent(self, genv):
- if self._restricted:
- currentMapset = currentLocation = True
- for i in range(len(self.selected_location)):
- if self.selected_location[i].label != genv['LOCATION_NAME']:
- currentLocation = False
- currentMapset = False
- break
- if currentMapset:
- for i in range(len(self.selected_mapset)):
- if self.selected_mapset[i].label != genv['MAPSET']:
- currentMapset = False
- break
- return currentLocation, currentMapset
- else:
- return True, True
- def _popupMenuLayer(self):
- """Create popup menu for layers"""
- menu = Menu()
- genv = gisenv()
- currentLocation, currentMapset = self._isCurrent(genv)
- item = wx.MenuItem(menu, wx.NewId(), _("&Cut"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnMoveMap, item)
- if not currentMapset:
- item.Enable(False)
- item = wx.MenuItem(menu, wx.NewId(), _("&Copy"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnCopyMap, item)
- item = wx.MenuItem(menu, wx.NewId(), _("Copy &name"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnCopyName, item)
- item = wx.MenuItem(menu, wx.NewId(), _("&Paste"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
- if not(currentMapset and self.copy_layer):
- item.Enable(False)
- item = wx.MenuItem(menu, wx.NewId(), _("&Delete"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnDeleteMap, item)
- item.Enable(currentMapset)
- item = wx.MenuItem(menu, wx.NewId(), _("&Rename"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnRenameMap, item)
- item.Enable(currentMapset and len(self.selected_layer) == 1)
- menu.AppendSeparator()
- if not isinstance(self._giface, StandaloneGrassInterface):
- if all([each.label == genv['LOCATION_NAME'] for each in self.selected_location]):
- if len(self.selected_layer) > 1:
- item = wx.MenuItem(menu, wx.NewId(), _("&Display layers"))
- else:
- item = wx.MenuItem(menu, wx.NewId(), _("&Display layer"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnDisplayLayer, item)
- item = wx.MenuItem(menu, wx.NewId(), _("Show &metadata"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnMetadata, item)
- self.PopupMenu(menu)
- menu.Destroy()
- def _popupMenuMapset(self):
- """Create popup menu for mapsets"""
- menu = Menu()
- genv = gisenv()
- currentLocation, currentMapset = self._isCurrent(genv)
- item = wx.MenuItem(menu, wx.NewId(), _("&Paste"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
- if not(currentMapset and self.copy_layer):
- item.Enable(False)
- item = wx.MenuItem(menu, wx.NewId(), _("&Switch mapset"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnSwitchLocationMapset, item)
- if (self.selected_location[0].label == genv['LOCATION_NAME']
- and self.selected_mapset[0].label == genv['MAPSET']):
- item.Enable(False)
- self.PopupMenu(menu)
- menu.Destroy()
- def _popupMenuElement(self):
- """Create popup menu for elements"""
- menu = Menu()
- item = wx.MenuItem(menu, wx.NewId(), _("&Paste"))
- menu.AppendItem(item)
- self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
- genv = gisenv()
- currentLocation, currentMapset = self._isCurrent(genv)
- if not(currentMapset and self.copy_layer):
- item.Enable(False)
- self.PopupMenu(menu)
- menu.Destroy()
- def _popupMenuEmpty(self):
- """Create empty popup when multiple different types of items are selected"""
- menu = Menu()
- item = wx.MenuItem(menu, wx.NewId(), _("No available options"))
- menu.AppendItem(item)
- item.Enable(False)
- self.PopupMenu(menu)
- menu.Destroy()
|