tree.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. """
  2. @package datacatalog::tree
  3. @brief Data catalog tree classes
  4. Classes:
  5. - datacatalog::LocationMapTree
  6. - datacatalog::DataCatalogTree
  7. (C) 2014-2015 by Tereza Fiedlerova, and the GRASS Development Team
  8. This program is free software under the GNU General Public
  9. License (>=v2). Read the file COPYING that comes with GRASS
  10. for details.
  11. @author Tereza Fiedlerova
  12. @author Anna Petrasova (kratochanna gmail com)
  13. """
  14. import os
  15. from multiprocessing import Process, Queue, cpu_count
  16. import wx
  17. from core.gcmd import RunCommand, GError, GMessage, GWarning
  18. from core.utils import GetListOfLocations
  19. from core.debug import Debug
  20. from gui_core.dialogs import TextEntryDialog
  21. from core.giface import StandaloneGrassInterface
  22. from core.treemodel import TreeModel, DictNode
  23. from gui_core.treeview import TreeView
  24. from grass.pydispatch.signal import Signal
  25. import grass.script as gscript
  26. from grass.exceptions import CalledModuleError
  27. def getEnvironment(gisdbase, location, mapset):
  28. """Creates environment to be passed in run_command for example.
  29. Returns tuple with temporary file path and the environment. The user
  30. of this function is responsile for deleting the file."""
  31. tmp_gisrc_file = gscript.tempfile()
  32. with open(tmp_gisrc_file, 'w') as f:
  33. f.write('MAPSET: {mapset}\n'.format(mapset=mapset))
  34. f.write('GISDBASE: {g}\n'.format(g=gisdbase))
  35. f.write('LOCATION_NAME: {l}\n'.format(l=location))
  36. f.write('GUI: text\n')
  37. env = os.environ.copy()
  38. env['GISRC'] = tmp_gisrc_file
  39. return tmp_gisrc_file, env
  40. def getLocationTree(gisdbase, location, queue):
  41. """Creates dictionary with mapsets, elements, layers for given location.
  42. Returns tuple with the dictionary and error (or None)"""
  43. tmp_gisrc_file, env = getEnvironment(gisdbase, location, 'PERMANENT')
  44. env['GRASS_SKIP_MAPSET_OWNER_CHECK'] = '1'
  45. maps_dict = {}
  46. elements = ['raster', 'raster_3d', 'vector']
  47. try:
  48. mapsets = gscript.read_command('g.mapsets', flags='l', quiet=True, env=env).strip()
  49. except CalledModuleError:
  50. queue.put((maps_dict, _("Failed to read mapsets from location <{l}>.").format(l=location)))
  51. gscript.try_remove(tmp_gisrc_file)
  52. return
  53. else:
  54. listOfMapsets = mapsets.split()
  55. Debug.msg(4, "Location <{}>: {} mapsets found".format(location, len(listOfMapsets)))
  56. for each in listOfMapsets:
  57. maps_dict[each] = {}
  58. for elem in elements:
  59. maps_dict[each][elem] = []
  60. try:
  61. maplist = gscript.read_command('g.list', flags='mt', type=elements,
  62. mapset=','.join(listOfMapsets), quiet=True, env=env).strip()
  63. except CalledModuleError:
  64. queue.put((maps_dict, _("Failed to read maps from location <{l}>.").format(l=location)))
  65. gscript.try_remove(tmp_gisrc_file)
  66. return
  67. else:
  68. # fill dictionary
  69. listOfMaps = maplist.splitlines()
  70. Debug.msg(4, "Location <{}>: {} maps found".format(location, len(listOfMaps)))
  71. for each in listOfMaps:
  72. ltype, wholename = each.split('/')
  73. name, mapset = wholename.split('@')
  74. maps_dict[mapset][ltype].append(name)
  75. queue.put((maps_dict, None))
  76. gscript.try_remove(tmp_gisrc_file)
  77. class DataCatalogNode(DictNode):
  78. """Node representing item in datacatalog."""
  79. def __init__(self, label, data=None):
  80. super(DataCatalogNode, self).__init__(label=label, data=data)
  81. def match(self, **kwargs):
  82. """Method used for searching according to given parameters.
  83. :param value: dictionary value to be matched
  84. :param key: data dictionary key
  85. """
  86. if not kwargs:
  87. return False
  88. for key in kwargs:
  89. if not (key in self.data and self.data[key] == kwargs[key]):
  90. return False
  91. return True
  92. class LocationMapTree(TreeView):
  93. def __init__(self, parent, model=None, style=wx.TR_HIDE_ROOT | wx.TR_EDIT_LABELS | wx.TR_LINES_AT_ROOT |
  94. wx.TR_HAS_BUTTONS | wx.TR_FULL_ROW_HIGHLIGHT | wx.TR_SINGLE):
  95. """Location Map Tree constructor."""
  96. self._model = TreeModel(DataCatalogNode)
  97. super(LocationMapTree, self).__init__(parent=parent, model=self._model, id=wx.ID_ANY, style=style)
  98. self.showNotification = Signal('Tree.showNotification')
  99. self.parent = parent
  100. self.contextMenu.connect(self.OnRightClick)
  101. self.itemActivated.connect(self.OnDoubleClick)
  102. self._initVariables()
  103. def _initTreeItems(self, locations=None, mapsets=None):
  104. """Add locations, mapsets and layers to the tree.
  105. Runs in multiple processes. Saves resulting data and error."""
  106. # mapsets param currently unused
  107. if not locations:
  108. locations = GetListOfLocations(self.gisdbase)
  109. loc_count = proc_count = 0
  110. queue_list = []
  111. proc_list = []
  112. loc_list = []
  113. nprocs = 4
  114. try:
  115. nprocs = cpu_count()
  116. except NotImplementedError:
  117. nprocs = 4
  118. results = dict()
  119. errors = []
  120. location_nodes = []
  121. nlocations = len(locations)
  122. grassdata_node = self._model.AppendNode(parent=self._model.root,
  123. label=_('GRASS locations ({})').format(self.gisdbase),
  124. data=dict(type='grassdata'))
  125. for location in locations:
  126. results[location] = dict()
  127. varloc = self._model.AppendNode(parent=grassdata_node, label=location,
  128. data=dict(type='location', name=location))
  129. location_nodes.append(varloc)
  130. loc_count += 1
  131. Debug.msg(3, "Scanning location <{}> ({}/{})".format(location, loc_count, nlocations))
  132. q = Queue()
  133. p = Process(target=getLocationTree,
  134. args=(self.gisdbase, location, q))
  135. p.start()
  136. queue_list.append(q)
  137. proc_list.append(p)
  138. loc_list.append(location)
  139. proc_count += 1
  140. # Wait for all running processes
  141. if proc_count == nprocs or loc_count == nlocations:
  142. Debug.msg(4, "Process subresults")
  143. for i in range(len(loc_list)):
  144. maps, error = queue_list[i].get()
  145. proc_list[i].join()
  146. if error:
  147. errors.append(error)
  148. for key in sorted(maps.keys()):
  149. mapset_node = self._model.AppendNode(parent=location_nodes[i], label=key,
  150. data=dict(type='mapset', name=key))
  151. for elem in maps[key]:
  152. if maps[key][elem]:
  153. element_node = self._model.AppendNode(parent=mapset_node, label=elem,
  154. data=dict(type='element', name=elem))
  155. for layer in maps[key][elem]:
  156. self._model.AppendNode(parent=element_node, label=layer,
  157. data=dict(type=elem, name=layer))
  158. proc_count = 0
  159. proc_list = []
  160. queue_list = []
  161. loc_list = []
  162. location_nodes = []
  163. if errors:
  164. GWarning('\n'.join(errors))
  165. Debug.msg(1, "Tree filled")
  166. self.RefreshItems()
  167. def InitTreeItems(self):
  168. """Create popup menu for layers"""
  169. raise NotImplementedError()
  170. def _popupMenuLayer(self):
  171. """Create popup menu for layers"""
  172. raise NotImplementedError()
  173. def _popupMenuMapset(self):
  174. """Create popup menu for mapsets"""
  175. raise NotImplementedError()
  176. def _initVariables(self):
  177. """Init variables."""
  178. self.selected_layer = None
  179. self.selected_type = None
  180. self.selected_mapset = None
  181. self.selected_location = None
  182. gisenv = gscript.gisenv()
  183. self.gisdbase = gisenv['GISDBASE']
  184. self.glocation = gisenv['LOCATION_NAME']
  185. self.gmapset = gisenv['MAPSET']
  186. def GetControl(self):
  187. """Returns control itself."""
  188. return self
  189. def DefineItems(self, item):
  190. """Set selected items."""
  191. self.selected_layer = None
  192. self.selected_type = None
  193. self.selected_mapset = None
  194. self.selected_location = None
  195. type = item.data['type']
  196. if type in ('raster', 'raster_3d', 'vector'):
  197. self.selected_layer = item
  198. type = 'element'
  199. item = item.parent
  200. if type == 'element':
  201. self.selected_type = item
  202. type = 'mapset'
  203. item = item.parent
  204. if type == 'mapset':
  205. self.selected_mapset = item
  206. type = 'location'
  207. item = item.parent
  208. if type == 'location':
  209. self.selected_location = item
  210. def OnSelChanged(self, event):
  211. self.selected_layer = None
  212. def OnRightClick(self, node):
  213. """Display popup menu."""
  214. self.DefineItems(node)
  215. if self.selected_layer:
  216. self._popupMenuLayer(self.selected_mapset.label == self.gmapset)
  217. elif self.selected_mapset and not self.selected_type:
  218. self._popupMenuMapset()
  219. def OnDoubleClick(self, node):
  220. """Expand/Collapse node."""
  221. if self.IsNodeExpanded(node):
  222. self.CollapseNode(node, recursive=False)
  223. else:
  224. self.ExpandNode(node, recursive=False)
  225. def ExpandCurrentLocation(self):
  226. """Expand current location"""
  227. location = gscript.gisenv()['LOCATION_NAME']
  228. item = self._model.SearchNodes(name=location, type='location')
  229. if item:
  230. self.Select(item[0], select=True)
  231. self.ExpandNode(item[0], recursive=False)
  232. else:
  233. Debug.msg(1, "Location <%s> not found" % location)
  234. def ExpandCurrentMapset(self):
  235. """Expand current mapset"""
  236. gisenv = gscript.gisenv()
  237. location = gisenv['LOCATION_NAME']
  238. mapset = gisenv['MAPSET']
  239. locationItem = self._model.SearchNodes(name=location, type='location')
  240. mapsetItem = None
  241. if locationItem:
  242. mapsetItem = self._model.SearchNodes(parent=locationItem[0], name=mapset, type='mapset')
  243. if mapsetItem:
  244. self.Select(mapsetItem[0], select=True)
  245. self.ExpandNode(mapsetItem[0], recursive=True)
  246. else:
  247. Debug.msg(1, "Mapset <%s> not found" % mapset)
  248. class DataCatalogTree(LocationMapTree):
  249. def __init__(self, parent, giface=None):
  250. """Data Catalog Tree constructor."""
  251. super(DataCatalogTree, self).__init__(parent)
  252. self._giface = giface
  253. self._initVariablesCatalog()
  254. self.beginDrag = Signal('DataCatalogTree.beginDrag')
  255. self.endDrag = Signal('DataCatalogTree.endDrag')
  256. self.startEdit = Signal('DataCatalogTree.startEdit')
  257. self.endEdit = Signal('DataCatalogTree.endEdit')
  258. self.Bind(wx.EVT_TREE_BEGIN_DRAG, lambda evt:
  259. self._emitSignal(evt.GetItem(), self.beginDrag, event=evt))
  260. self.Bind(wx.EVT_TREE_END_DRAG, lambda evt:
  261. self._emitSignal(evt.GetItem(), self.endDrag, event=evt))
  262. self.beginDrag.connect(self.OnBeginDrag)
  263. self.endDrag.connect(self.OnEndDrag)
  264. self.Bind(wx.EVT_TREE_BEGIN_LABEL_EDIT, lambda evt:
  265. self._emitSignal(evt.GetItem(), self.startEdit, event=evt))
  266. self.Bind(wx.EVT_TREE_END_LABEL_EDIT, lambda evt:
  267. self._emitSignal(evt.GetItem(), self.endEdit, event=evt))
  268. ###self.startEdit.connect(self.OnStartEditLabel)
  269. ###self.endEdit.connect(self.OnEditLabel)
  270. def _initVariablesCatalog(self):
  271. """Init variables."""
  272. self.copy_layer = None
  273. self.copy_type = None
  274. self.copy_mapset = None
  275. self.copy_location = None
  276. def _runCommand(self, prog, **kwargs):
  277. cmdString = ' '.join(gscript.make_command(prog, **kwargs))
  278. ret = RunCommand(prog, parent=self, **kwargs)
  279. return ret, cmdString
  280. def InitTreeItems(self):
  281. """Add locations, mapsets and layers to the tree."""
  282. self._initTreeItems()
  283. def OnCopy(self, event):
  284. """Copy layer or mapset (just save it temporarily, copying is done by paste)"""
  285. self.copy_layer = self.selected_layer
  286. self.copy_type = self.selected_type
  287. self.copy_mapset = self.selected_mapset
  288. self.copy_location = self.selected_location
  289. label = _("Map <{layer}> marked for copying. "
  290. "You can paste it to the current mapset "
  291. "<{mapset}>.".format(layer=self.copy_layer.label, mapset=self.gmapset))
  292. self.showNotification.emit(message=label)
  293. def OnRename(self, event):
  294. """Rename levent with dialog"""
  295. if self.selected_layer:
  296. self.old_name = self.selected_layer.label
  297. self.new_name = self._getUserEntry(_('New name'), _('Rename map'), self.old_name)
  298. self.Rename()
  299. def OnStartEditLabel(self, node, event):
  300. """Start label editing"""
  301. self.DefineItems(node)
  302. Debug.msg(1, "Start label edit {name}".format(name=node.label))
  303. label = _("Editing {name}").format(name=node.label)
  304. self.showNotification.emit(message=label)
  305. if not self.selected_layer:
  306. event.Veto()
  307. def OnEditLabel(self, node, event):
  308. """End label editing"""
  309. if self.selected_layer and not event.IsEditCancelled():
  310. self.old_name = node.label
  311. Debug.msg(1, "End label edit {name}".format(name=self.old_name))
  312. self.new_name = event.GetLabel()
  313. self.Rename()
  314. def Rename(self):
  315. """Rename layer"""
  316. if self.selected_layer and self.new_name:
  317. string = self.old_name + ',' + self.new_name
  318. gisrc, env = getEnvironment(self.gisdbase, self.selected_location.label, self.selected_mapset.label)
  319. renamed = 0
  320. label = _("Renaming map <{name}>...").format(name=string)
  321. self.showNotification.emit(message=label)
  322. if self.selected_type.label == 'vector':
  323. renamed, cmd = self._runCommand('g.rename', vector=string, env=env)
  324. elif self.selected_type.label == 'raster':
  325. renamed, cmd = self._runCommand('g.rename', raster=string, env=env)
  326. else:
  327. renamed, cmd = self._runCommand('g.rename', raster3d=string, env=env)
  328. if renamed == 0:
  329. self.selected_layer.label = self.new_name
  330. self.selected_layer.data['name'] = self.new_name
  331. self.RefreshNode(self.selected_layer)
  332. self.showNotification.emit(message=_("{cmd} -- completed").format(cmd=cmd))
  333. Debug.msg(1, "LAYER RENAMED TO: " + self.new_name)
  334. gscript.try_remove(gisrc)
  335. def OnPaste(self, event):
  336. """Paste layer or mapset"""
  337. # copying between mapsets of one location
  338. if not self.copy_layer:
  339. GMessage(_("No map selected for copying."), parent=self)
  340. return
  341. if self.selected_location == self.copy_location and \
  342. self.selected_mapset.data['name'] == gscript.gisenv()['MAPSET']:
  343. if self.selected_type:
  344. if self.copy_type.label != self.selected_type.label: # copy raster to vector or vice versa
  345. GError(_("Failed to copy map: invalid map type "
  346. "({} vs. {}).".format(self.copy_type.label, self.selected_type.label)), parent=self)
  347. return
  348. self.new_name = self._getUserEntry(_('New name'), _('Copy map'),
  349. self.copy_layer.label + '_copy')
  350. if not self.new_name:
  351. return
  352. if self.copy_layer.label == self.new_name:
  353. GMessage(_("Failed to copy map: new map has the same name"), parent=self)
  354. return
  355. if not self.selected_type:
  356. found = self._model.SearchNodes(parent=self.selected_mapset, type='element', name=self.copy_type.label)
  357. self.selected_type = found[0] if found else None
  358. overwrite = False
  359. if self.selected_type:
  360. found = self._model.SearchNodes(parent=self.selected_type, type=self.copy_type.label, name=self.new_name)
  361. if found and found[0]:
  362. dlg = wx.MessageDialog(parent=self,
  363. message = _("Map <{map}> already exists "
  364. "in the current mapset. "
  365. "Do you want to overwrite it?").format(map=self.new_name),
  366. caption = _("Overwrite?"),
  367. style = wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  368. ret = dlg.ShowModal()
  369. dlg.Destroy()
  370. if ret == wx.ID_YES:
  371. overwrite = True
  372. string = self.copy_layer.label + '@' + self.copy_mapset.label + ',' + self.new_name
  373. gisrc, env = getEnvironment(self.gisdbase, self.selected_location.label, self.selected_mapset.label)
  374. pasted = 0
  375. label = _("Copying <{name}>...").format(name=string)
  376. self.showNotification.emit(message=label)
  377. if self.copy_type.label == 'vector':
  378. pasted, cmd = self._runCommand('g.copy', vector=string, overwrite=overwrite, env=env)
  379. node = 'vector'
  380. elif self.copy_type.label == 'raster':
  381. pasted, cmd = self._runCommand('g.copy', raster=string, overwrite=overwrite, env=env)
  382. node = 'raster'
  383. else:
  384. pasted, cmd = self._runCommand('g.copy', raster_3d=string, overwrite=overwrite, env=env)
  385. node = 'raster_3d'
  386. if pasted == 0:
  387. if not self.selected_type:
  388. # add type node if not exists
  389. self.selected_type = self._model.AppendNode(parent=self.selected_mapset, label=node,
  390. data=dict(type='element', name=node))
  391. if not overwrite:
  392. self._model.AppendNode(parent=self.selected_type, label=self.new_name,
  393. data=dict(type=node, name=self.new_name))
  394. self._model.SortChildren(self.selected_type)
  395. self.RefreshNode(self.selected_type, recursive=True)
  396. Debug.msg(1, "COPIED TO: " + self.new_name)
  397. self.showNotification.emit(message= _("{cmd} -- completed").format(cmd=cmd))
  398. gscript.try_remove(gisrc)
  399. else:
  400. if self.selected_location != self.copy_location:
  401. GError(_("Failed to copy map: action is allowed only within the same location."),
  402. parent=self)
  403. else:
  404. GError(_("Failed to copy map: action is allowed only within the current mapset."),
  405. parent=self)
  406. # expand selected mapset
  407. self.ExpandNode(self.selected_mapset, recursive=True)
  408. def OnDelete(self, event):
  409. """Delete layer or mapset"""
  410. if self.selected_layer:
  411. string = self.selected_layer.label
  412. gisrc, env = getEnvironment(self.gisdbase, self.selected_location.label, self.selected_mapset.label)
  413. removed = 0
  414. # TODO: rewrite this that it will tell map type in the dialog
  415. if self._confirmDialog(question=_('Do you really want to delete map <{m}>?').format(m=string),
  416. title=_('Delete map')) == wx.ID_YES:
  417. label = _("Deleting {name}...").format(name=string)
  418. self.showNotification.emit(message=label)
  419. if self.selected_type.label == 'vector':
  420. removed, cmd = self._runCommand('g.remove', flags='f', type='vector',
  421. name=string, env=env)
  422. elif self.selected_type.label == 'raster':
  423. removed, cmd = self._runCommand('g.remove', flags='f', type='raster',
  424. name=string, env=env)
  425. else:
  426. removed, cmd = self._runCommand('g.remove', flags='f', type='raster_3d',
  427. name=string, env=env)
  428. if removed == 0:
  429. self._model.RemoveNode(self.selected_layer)
  430. self.RefreshNode(self.selected_type, recursive=True)
  431. Debug.msg(1, "LAYER " + string + " DELETED")
  432. self.showNotification.emit(message= _("{cmd} -- completed").format(cmd=cmd))
  433. gscript.try_remove(gisrc)
  434. def OnDisplayLayer(self, event):
  435. """Display layer in current graphics view"""
  436. layerName = []
  437. if self.selected_location.label == self.glocation and self.selected_mapset:
  438. string = self.selected_layer.label + '@' + self.selected_mapset.label
  439. layerName.append(string)
  440. label = _("Displaying {name}...").format(name=string)
  441. self.showNotification.emit(message=label)
  442. label = "d." + self.selected_type.label[:4] + " --q map=" + string + \
  443. _(" -- completed. Go to Layers tab for further operations.")
  444. if self.selected_type.label == 'vector':
  445. self._giface.lmgr.AddMaps(layerName, 'vector', True)
  446. elif self.selected_type.label == 'raster':
  447. self._giface.lmgr.AddMaps(layerName, 'raster', True)
  448. else:
  449. self._giface.lmgr.AddMaps(layerName, 'raster_3d', True)
  450. label = "d.rast --q map=" + string + _(" -- completed. Go to Layers tab for further operations.") # generate this message (command) automatically?
  451. self.showNotification.emit(message=label)
  452. Debug.msg(1, "LAYER " + self.selected_layer.label + " DISPLAYED")
  453. else:
  454. GError(_("Failed to display layer: not in current mapset or invalid layer"),
  455. parent=self)
  456. def OnBeginDrag(self, node, event):
  457. """Just copy necessary data"""
  458. if wx.GetMouseState().ControlDown():
  459. #cursor = wx.StockCursor(wx.CURSOR_HAND)
  460. #self.SetCursor(cursor)
  461. event.Allow()
  462. self.DefineItems(node)
  463. self.OnCopy(event)
  464. Debug.msg(1, "DRAG")
  465. else:
  466. event.Veto()
  467. Debug.msg(1, "DRAGGING without ctrl key does nothing")
  468. def OnEndDrag(self, node, event):
  469. """Copy layer into target"""
  470. #cursor = wx.StockCursor(wx.CURSOR_ARROW)
  471. #self.SetCursor(cursor)
  472. if node:
  473. self.DefineItems(node)
  474. if self.selected_location == self.copy_location and self.selected_mapset:
  475. event.Allow()
  476. self.OnPaste(event)
  477. #cursor = wx.StockCursor(wx.CURSOR_DEFAULT)
  478. #self.SetCursor(cursor) # TODO: change cursor while dragging and then back, this is not working
  479. Debug.msg(1, "DROP DONE")
  480. else:
  481. event.Veto()
  482. def _getUserEntry(self, message, title, value):
  483. """Dialog for simple text entry"""
  484. dlg = TextEntryDialog(self, message, title)
  485. dlg.SetValue(value)
  486. if dlg.ShowModal() == wx.ID_OK:
  487. name = dlg.GetValue()
  488. else:
  489. name = None
  490. dlg.Destroy()
  491. return name
  492. def _confirmDialog(self, question, title):
  493. """Confirm dialog"""
  494. dlg = wx.MessageDialog(self, question, title, wx.YES_NO)
  495. res = dlg.ShowModal()
  496. dlg.Destroy()
  497. return res
  498. def _popupMenuLayer(self, current_mapset):
  499. """Create popup menu for layers"""
  500. menu = wx.Menu()
  501. item = wx.MenuItem(menu, wx.NewId(), _("&Copy"))
  502. menu.AppendItem(item)
  503. self.Bind(wx.EVT_MENU, self.OnCopy, item)
  504. item = wx.MenuItem(menu, wx.NewId(), _("&Delete"))
  505. menu.AppendItem(item)
  506. self.Bind(wx.EVT_MENU, self.OnDelete, item)
  507. if not current_mapset:
  508. item.Enable(False)
  509. item = wx.MenuItem(menu, wx.NewId(), _("&Rename"))
  510. menu.AppendItem(item)
  511. self.Bind(wx.EVT_MENU, self.OnRename, item)
  512. if not current_mapset:
  513. item.Enable(False)
  514. if not isinstance(self._giface, StandaloneGrassInterface):
  515. item = wx.MenuItem(menu, wx.NewId(), _("&Display"))
  516. menu.AppendItem(item)
  517. self.Bind(wx.EVT_MENU, self.OnDisplayLayer, item)
  518. self.PopupMenu(menu)
  519. menu.Destroy()
  520. def _popupMenuMapset(self):
  521. """Create popup menu for mapsets"""
  522. menu = wx.Menu()
  523. item = wx.MenuItem(menu, wx.NewId(), _("&Paste"))
  524. menu.AppendItem(item)
  525. self.Bind(wx.EVT_MENU, self.OnPaste, item)
  526. self.PopupMenu(menu)
  527. menu.Destroy()