tree.py 51 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304
  1. """
  2. @package datacatalog::tree
  3. @brief Data catalog tree classes
  4. Classes:
  5. - datacatalog::NameEntryDialog
  6. - datacatalog::DataCatalogNode
  7. - datacatalog::DataCatalogTree
  8. (C) 2014-2018 by Tereza Fiedlerova, and the GRASS Development Team
  9. This program is free software under the GNU General Public
  10. License (>=v2). Read the file COPYING that comes with GRASS
  11. for details.
  12. @author Tereza Fiedlerova
  13. @author Anna Petrasova (kratochanna gmail com)
  14. @author Linda Kladivova (l.kladivova@seznam.cz)
  15. """
  16. import re
  17. import copy
  18. from multiprocessing import Process, Queue, cpu_count
  19. import wx
  20. from core.gcmd import RunCommand, GError, GMessage, GWarning
  21. from core.utils import GetListOfLocations
  22. from core.debug import Debug
  23. from gui_core.dialogs import TextEntryDialog
  24. from core.giface import StandaloneGrassInterface
  25. from core.treemodel import TreeModel, DictNode
  26. from gui_core.treeview import TreeView
  27. from gui_core.wrap import Menu
  28. from datacatalog.dialogs import CatalogReprojectionDialog
  29. from icons.icon import MetaIcon
  30. from startup.guiutils import (create_mapset_interactively,
  31. rename_mapset_interactively,
  32. rename_location_interactively,
  33. delete_mapset_interactively,
  34. delete_location_interactively)
  35. from grass.pydispatch.signal import Signal
  36. import grass.script as gscript
  37. from grass.script import gisenv
  38. from grass.exceptions import CalledModuleError
  39. def filterModel(model, element=None, name=None):
  40. """Filter tree model based on type or name of map using regular expressions.
  41. Copies tree and remove nodes which don't match."""
  42. fmodel = copy.deepcopy(model)
  43. nodesToRemove = []
  44. if name:
  45. try:
  46. regex = re.compile(name)
  47. except:
  48. return fmodel
  49. for gisdbase in fmodel.root.children:
  50. for location in gisdbase.children:
  51. for mapset in location.children:
  52. for layer in mapset.children:
  53. if element and layer.data['type'] != element:
  54. nodesToRemove.append(layer)
  55. continue
  56. if name and regex.search(layer.data['name']) is None:
  57. nodesToRemove.append(layer)
  58. for node in reversed(nodesToRemove):
  59. fmodel.RemoveNode(node)
  60. cleanUpTree(fmodel)
  61. return fmodel
  62. def cleanUpTree(model):
  63. """Removes empty element/mapsets/locations nodes.
  64. It first removes empty elements, then mapsets, then locations"""
  65. # removes empty mapsets
  66. nodesToRemove = []
  67. for gisdbase in model.root.children:
  68. for location in gisdbase.children:
  69. for mapset in location.children:
  70. if not mapset.children:
  71. nodesToRemove.append(mapset)
  72. for node in reversed(nodesToRemove):
  73. model.RemoveNode(node)
  74. # removes empty locations
  75. nodesToRemove = []
  76. for gisdbase in model.root.children:
  77. for location in gisdbase.children:
  78. if not location.children:
  79. nodesToRemove.append(location)
  80. for node in reversed(nodesToRemove):
  81. model.RemoveNode(node)
  82. def getLocationTree(gisdbase, location, queue, mapsets=None):
  83. """Creates dictionary with mapsets, elements, layers for given location.
  84. Returns tuple with the dictionary and error (or None)"""
  85. tmp_gisrc_file, env = gscript.create_environment(gisdbase, location, 'PERMANENT')
  86. env['GRASS_SKIP_MAPSET_OWNER_CHECK'] = '1'
  87. maps_dict = {}
  88. elements = ['raster', 'raster_3d', 'vector']
  89. try:
  90. if not mapsets:
  91. mapsets = gscript.read_command(
  92. 'g.mapsets',
  93. flags='l',
  94. separator='comma',
  95. quiet=True,
  96. env=env).strip()
  97. except CalledModuleError:
  98. queue.put(
  99. (maps_dict,
  100. _("Failed to read mapsets from location <{l}>.").format(
  101. l=location)))
  102. gscript.try_remove(tmp_gisrc_file)
  103. return
  104. else:
  105. listOfMapsets = mapsets.split(',')
  106. Debug.msg(
  107. 4, "Location <{0}>: {1} mapsets found".format(
  108. location, len(listOfMapsets)))
  109. for each in listOfMapsets:
  110. maps_dict[each] = {}
  111. for elem in elements:
  112. maps_dict[each][elem] = []
  113. try:
  114. maplist = gscript.read_command(
  115. 'g.list', flags='mt', type=elements,
  116. mapset=','.join(listOfMapsets),
  117. quiet=True, env=env).strip()
  118. except CalledModuleError:
  119. queue.put(
  120. (maps_dict,
  121. _("Failed to read maps from location <{l}>.").format(
  122. l=location)))
  123. gscript.try_remove(tmp_gisrc_file)
  124. return
  125. else:
  126. # fill dictionary
  127. listOfMaps = maplist.splitlines()
  128. Debug.msg(
  129. 4, "Location <{0}>: {1} maps found".format(
  130. location, len(listOfMaps)))
  131. for each in listOfMaps:
  132. ltype, wholename = each.split('/')
  133. name, mapset = wholename.split('@')
  134. maps_dict[mapset][ltype].append(name)
  135. queue.put((maps_dict, None))
  136. gscript.try_remove(tmp_gisrc_file)
  137. def map_exists(name, element, env, mapset=None):
  138. """Check is map is present in the mapset given in the environment
  139. :param name: name of the map
  140. :param element: data type ('raster', 'raster_3d', and 'vector')
  141. :param env environment created by function gscript.create_environment
  142. """
  143. if not mapset:
  144. mapset = gscript.run_command('g.mapset', flags='p', env=env).strip()
  145. # change type to element used by find file
  146. if element == 'raster':
  147. element = 'cell'
  148. elif element == 'raster_3d':
  149. element = 'grid3'
  150. # g.findfile returns non-zero when file was not found
  151. # se we ignore return code and just focus on stdout
  152. process = gscript.start_command(
  153. 'g.findfile',
  154. flags='n',
  155. element=element,
  156. file=name,
  157. mapset=mapset,
  158. stdout=gscript.PIPE,
  159. stderr=gscript.PIPE,
  160. env=env)
  161. output, errors = process.communicate()
  162. info = gscript.parse_key_val(output, sep='=')
  163. # file is the key questioned in grass.script.core find_file()
  164. # return code should be equivalent to checking the output
  165. if info['file']:
  166. return True
  167. else:
  168. return False
  169. class NameEntryDialog(TextEntryDialog):
  170. def __init__(self, element, mapset, env, **kwargs):
  171. TextEntryDialog.__init__(self, **kwargs)
  172. self._element = element
  173. self._mapset = mapset
  174. self._env = env
  175. id_OK = self.GetAffirmativeId()
  176. self.Bind(wx.EVT_BUTTON, self.OnOK, self.FindWindowById(id_OK))
  177. def OnOK(self, event):
  178. new = self.GetValue()
  179. if not new:
  180. return
  181. if map_exists(new, self._element, self._env, self._mapset):
  182. dlg = wx.MessageDialog(
  183. self,
  184. message=_(
  185. "Map of type {elem} <{name}> already exists in mapset <{mapset}>. "
  186. "Do you want to overwrite it?").format(
  187. elem=self._element,
  188. name=new,
  189. mapset=self._mapset),
  190. caption=_("Overwrite?"),
  191. style=wx.YES_NO)
  192. if dlg.ShowModal() == wx.ID_YES:
  193. dlg.Destroy()
  194. self._env['GRASS_OVERWRITE'] = '1'
  195. self.EndModal(wx.ID_OK)
  196. else:
  197. dlg.Destroy()
  198. return
  199. else:
  200. self.EndModal(wx.ID_OK)
  201. class DataCatalogNode(DictNode):
  202. """Node representing item in datacatalog."""
  203. def __init__(self, data=None):
  204. super(DataCatalogNode, self).__init__(data=data)
  205. @property
  206. def label(self):
  207. return self.data["name"]
  208. def match(self, **kwargs):
  209. """Method used for searching according to given parameters.
  210. :param value: dictionary value to be matched
  211. :param key: data dictionary key
  212. """
  213. if not kwargs:
  214. return False
  215. for key in kwargs:
  216. if not (key in self.data and self.data[key] == kwargs[key]):
  217. return False
  218. return True
  219. class DataCatalogTree(TreeView):
  220. def __init__(
  221. self, parent, model=None, giface=None,
  222. style=wx.TR_HIDE_ROOT | wx.TR_EDIT_LABELS |
  223. wx.TR_LINES_AT_ROOT | wx.TR_HAS_BUTTONS |
  224. wx.TR_FULL_ROW_HIGHLIGHT | wx.TR_MULTIPLE):
  225. """Location Map Tree constructor."""
  226. self._model = TreeModel(DataCatalogNode)
  227. self._orig_model = self._model
  228. super(
  229. DataCatalogTree,
  230. self).__init__(
  231. parent=parent,
  232. model=self._model,
  233. id=wx.ID_ANY,
  234. style=style)
  235. self._giface = giface
  236. self._restricted = True
  237. self.showNotification = Signal('Tree.showNotification')
  238. self.changeMapset = Signal('Tree.changeMapset')
  239. self.changeLocation = Signal('Tree.changeLocation')
  240. self.parent = parent
  241. self.contextMenu.connect(self.OnRightClick)
  242. self.itemActivated.connect(self.OnDoubleClick)
  243. self._iconTypes = ['grassdb', 'location', 'mapset', 'raster',
  244. 'vector', 'raster_3d']
  245. self._initImages()
  246. self._initVariables()
  247. self._initVariablesCatalog()
  248. self.UpdateCurrentDbLocationMapsetNode()
  249. self.grassdatabases = []
  250. self.grassdatabases.append(gisenv()['GISDBASE'])
  251. self.beginDrag = Signal('DataCatalogTree.beginDrag')
  252. self.endDrag = Signal('DataCatalogTree.endDrag')
  253. self.startEdit = Signal('DataCatalogTree.startEdit')
  254. self.endEdit = Signal('DataCatalogTree.endEdit')
  255. self.Bind(wx.EVT_TREE_BEGIN_DRAG, lambda evt:
  256. self._emitSignal(evt.GetItem(), self.beginDrag, event=evt))
  257. self.Bind(wx.EVT_TREE_END_DRAG, lambda evt:
  258. self._emitSignal(evt.GetItem(), self.endDrag, event=evt))
  259. self.beginDrag.connect(self.OnBeginDrag)
  260. self.endDrag.connect(self.OnEndDrag)
  261. self.Bind(wx.EVT_TREE_BEGIN_LABEL_EDIT, lambda evt:
  262. self._emitSignal(evt.GetItem(), self.startEdit, event=evt))
  263. self.Bind(wx.EVT_TREE_END_LABEL_EDIT, lambda evt:
  264. self._emitSignal(evt.GetItem(), self.endEdit, event=evt))
  265. self.startEdit.connect(self.OnStartEditLabel)
  266. self.endEdit.connect(self.OnEditLabel)
  267. def _initTreeItems(self, locations=None, mapsets=None):
  268. """Add grass databases, locations, mapsets and layers to the tree.
  269. Runs in multiple processes. Saves resulting data and error."""
  270. # mapsets param currently unused
  271. for grassdatabase in self.grassdatabases:
  272. locations = GetListOfLocations(grassdatabase)
  273. loc_count = proc_count = 0
  274. queue_list = []
  275. proc_list = []
  276. loc_list = []
  277. nprocs = 4
  278. try:
  279. nprocs = cpu_count()
  280. except NotImplementedError:
  281. nprocs = 4
  282. results = dict()
  283. errors = []
  284. location_nodes = []
  285. nlocations = len(locations)
  286. grassdata_node = self._model.AppendNode(
  287. parent=self._model.root,
  288. data=dict(type='grassdb', name=grassdatabase))
  289. for location in locations:
  290. results[location] = dict()
  291. varloc = self._model.AppendNode(
  292. parent=grassdata_node, data=dict(
  293. type='location', name=location))
  294. location_nodes.append(varloc)
  295. loc_count += 1
  296. Debug.msg(
  297. 3, "Scanning location <{0}> ({1}/{2})".format(location, loc_count, nlocations))
  298. q = Queue()
  299. p = Process(target=getLocationTree,
  300. args=(grassdatabase, location, q))
  301. p.start()
  302. queue_list.append(q)
  303. proc_list.append(p)
  304. loc_list.append(location)
  305. proc_count += 1
  306. # Wait for all running processes
  307. if proc_count == nprocs or loc_count == nlocations:
  308. Debug.msg(4, "Process subresults")
  309. for i in range(len(loc_list)):
  310. maps, error = queue_list[i].get()
  311. proc_list[i].join()
  312. if error:
  313. errors.append(error)
  314. for key in sorted(maps.keys()):
  315. mapset_node = self._model.AppendNode(
  316. parent=location_nodes[i],
  317. data=dict(type='mapset', name=key))
  318. self._populateMapsetItem(mapset_node, maps[key])
  319. proc_count = 0
  320. proc_list = []
  321. queue_list = []
  322. loc_list = []
  323. location_nodes = []
  324. if errors:
  325. wx.CallAfter(GWarning, '\n'.join(errors))
  326. Debug.msg(1, "Tree filled")
  327. self.UpdateCurrentDbLocationMapsetNode()
  328. self.RefreshItems()
  329. def UpdateCurrentDbLocationMapsetNode(self):
  330. self.current_grassdb_node, self.current_location_node, self.current_mapset_node = \
  331. self.GetCurrentDbLocationMapsetNode()
  332. def ReloadTreeItems(self):
  333. """Reload dbs, locations, mapsets and layers in the tree."""
  334. self._orig_model = self._model
  335. self._model.RemoveNode(self._model.root)
  336. self.InitTreeItems()
  337. def ReloadCurrentMapset(self):
  338. """Reload current mapset tree only."""
  339. def get_first_child(node):
  340. try:
  341. child = self.current_mapset_node.children[0]
  342. except IndexError:
  343. child = None
  344. return child
  345. self.UpdateCurrentDbLocationMapsetNode()
  346. if not self.current_grassdb_node or not self.current_location_node or not self.current_mapset_node:
  347. return
  348. if self.current_mapset_node.children:
  349. node = get_first_child(self.current_mapset_node)
  350. while node:
  351. self._model.RemoveNode(node)
  352. node = get_first_child(self.current_mapset_node)
  353. q = Queue()
  354. p = Process(
  355. target=getLocationTree,
  356. args=(
  357. self.current_grassdb_node.data['name'],
  358. self.current_location_node.data['name'],
  359. q,
  360. self.current_mapset_node.data['name']))
  361. p.start()
  362. maps, error = q.get()
  363. if error:
  364. raise CalledModuleError(error)
  365. self._populateMapsetItem(self.current_mapset_node,
  366. maps[self.current_mapset_node.data['name']])
  367. self._orig_model = copy.deepcopy(self._model)
  368. self.RefreshNode(self.current_mapset_node)
  369. self.RefreshItems()
  370. def _populateMapsetItem(self, mapset_node, data):
  371. for elem in data:
  372. if data[elem]:
  373. for layer in data[elem]:
  374. self._model.AppendNode(parent=mapset_node,
  375. data=dict(type=elem, name=layer))
  376. self._model.SortChildren(mapset_node)
  377. def _initVariables(self):
  378. """Init variables."""
  379. self.selected_grassdb = []
  380. self.selected_layer = []
  381. self.selected_mapset = []
  382. self.selected_location = []
  383. self.mixed = False
  384. def _initImages(self):
  385. bmpsize = (16, 16)
  386. icons = {
  387. 'grassdb': MetaIcon(img='grassdb').GetBitmap(bmpsize),
  388. 'location': MetaIcon(img='location').GetBitmap(bmpsize),
  389. 'mapset': MetaIcon(img='mapset').GetBitmap(bmpsize),
  390. 'raster': MetaIcon(img='raster').GetBitmap(bmpsize),
  391. 'vector': MetaIcon(img='vector').GetBitmap(bmpsize),
  392. 'raster_3d': MetaIcon(img='raster3d').GetBitmap(bmpsize)
  393. }
  394. il = wx.ImageList(bmpsize[0], bmpsize[1], mask=False)
  395. for each in self._iconTypes:
  396. il.Add(icons[each])
  397. self.AssignImageList(il)
  398. def GetControl(self):
  399. """Returns control itself."""
  400. return self
  401. def DefineItems(self, selected):
  402. """Set selected items."""
  403. self._initVariables()
  404. mixed = []
  405. for item in selected:
  406. type = item.data['type']
  407. if type in ('raster', 'raster_3d', 'vector'):
  408. self.selected_layer.append(item)
  409. self.selected_mapset.append(item.parent)
  410. self.selected_location.append(item.parent.parent)
  411. self.selected_grassdb.append(item.parent.parent.parent)
  412. mixed.append('layer')
  413. elif type == 'mapset':
  414. self.selected_layer.append(None)
  415. self.selected_mapset.append(item)
  416. self.selected_location.append(item.parent)
  417. self.selected_grassdb.append(item.parent.parent)
  418. mixed.append('mapset')
  419. elif type == 'location':
  420. self.selected_layer.append(None)
  421. self.selected_mapset.append(None)
  422. self.selected_location.append(item)
  423. self.selected_grassdb.append(item.parent)
  424. mixed.append('location')
  425. elif type == 'grassdb':
  426. self.selected_layer.append(None)
  427. self.selected_mapset.append(None)
  428. self.selected_location.append(None)
  429. self.selected_grassdb.append(item)
  430. mixed.append('grassdb')
  431. self.mixed = False
  432. if len(set(mixed)) > 1:
  433. self.mixed = True
  434. def OnSelChanged(self, event):
  435. self.selected_layer = None
  436. def OnRightClick(self, node):
  437. """Display popup menu."""
  438. self.DefineItems(self.GetSelected())
  439. if self.mixed:
  440. self._popupMenuEmpty()
  441. return
  442. if not self.selected_layer:
  443. self._popupMenuEmpty()
  444. elif self.selected_layer[0]:
  445. self._popupMenuLayer()
  446. elif self.selected_mapset[0] and len(self.selected_mapset) == 1:
  447. self._popupMenuMapset()
  448. elif self.selected_location[0] and not self.selected_mapset[0] and len(self.selected_location) == 1:
  449. self._popupMenuLocation()
  450. elif self.selected_grassdb[0] and not self.selected_location[0] and len(self.selected_grassdb) == 1:
  451. self._popupMenuGrassDb()
  452. else:
  453. self._popupMenuEmpty()
  454. def OnDoubleClick(self, node):
  455. """Double click on item/node.
  456. Display selected layer if node is a map layer otherwise
  457. expand/collapse node.
  458. """
  459. if not isinstance(self._giface, StandaloneGrassInterface):
  460. self.DefineItems([node])
  461. if self.selected_layer[0] is not None:
  462. # display selected layer and return
  463. self.DisplayLayer()
  464. return
  465. # expand/collapse location/mapset...
  466. if self.IsNodeExpanded(node):
  467. self.CollapseNode(node, recursive=False)
  468. else:
  469. self.ExpandNode(node, recursive=False)
  470. def ExpandCurrentLocation(self):
  471. """Expand current location"""
  472. location = gscript.gisenv()['LOCATION_NAME']
  473. item = self._model.SearchNodes(name=location, type='location')
  474. if item:
  475. self.Select(item[0], select=True)
  476. self.ExpandNode(item[0], recursive=False)
  477. else:
  478. Debug.msg(1, "Location <%s> not found" % location)
  479. def GetCurrentDbLocationMapsetNode(self):
  480. """Get current mapset node"""
  481. genv = gisenv()
  482. gisdbase = genv['GISDBASE']
  483. location = genv['LOCATION_NAME']
  484. mapset = genv['MAPSET']
  485. grassdbItem = self._model.SearchNodes(
  486. name=gisdbase, type='grassdb')
  487. if not grassdbItem:
  488. return None, None, None
  489. locationItem = self._model.SearchNodes(
  490. parent=grassdbItem[0],
  491. name=location, type='location')
  492. if not locationItem:
  493. return grassdbItem[0], None, None
  494. mapsetItem = self._model.SearchNodes(
  495. parent=locationItem[0],
  496. name=mapset, type='mapset')
  497. if not mapsetItem:
  498. return grassdbItem[0], locationItem[0], None
  499. return grassdbItem[0], locationItem[0], mapsetItem[0]
  500. def OnGetItemImage(self, index, which=wx.TreeItemIcon_Normal, column=0):
  501. """Overriden method to return image for each item."""
  502. node = self._model.GetNodeByIndex(index)
  503. try:
  504. return self._iconTypes.index(node.data['type'])
  505. except ValueError:
  506. return 0
  507. def OnGetItemFont(self, index):
  508. """Overriden method to return font for each item.
  509. Used to highlight current db/loc/mapset."""
  510. node = self._model.GetNodeByIndex(index)
  511. font = self.GetFont()
  512. if node.data['type'] in ('grassdb', 'location', 'mapset'):
  513. if node in (self.current_grassdb_node, self.current_location_node, self.current_mapset_node):
  514. font.SetWeight(wx.FONTWEIGHT_BOLD)
  515. else:
  516. font.SetWeight(wx.FONTWEIGHT_NORMAL)
  517. return font
  518. def ExpandCurrentMapset(self):
  519. """Expand current mapset"""
  520. if self.current_mapset_node:
  521. self.Select(self.current_mapset_node, select=True)
  522. self.ExpandNode(self.current_mapset_node, recursive=True)
  523. def _initVariablesCatalog(self):
  524. """Init variables."""
  525. self.copy_mode = False
  526. self.copy_layer = None
  527. self.copy_mapset = None
  528. self.copy_location = None
  529. self.copy_grassdb = None
  530. def SetRestriction(self, restrict):
  531. self._restricted = restrict
  532. def _runCommand(self, prog, **kwargs):
  533. cmdString = ' '.join(gscript.make_command(prog, **kwargs))
  534. ret = RunCommand(prog, parent=self, **kwargs)
  535. return ret, cmdString
  536. def InitTreeItems(self):
  537. """Add locations, mapsets and layers to the tree."""
  538. self._initTreeItems()
  539. def OnMoveMap(self, event):
  540. """Move layer or mapset (just save it temporarily, copying is done by paste)"""
  541. self.copy_mode = False
  542. self.copy_layer = self.selected_layer[:]
  543. self.copy_mapset = self.selected_mapset[:]
  544. self.copy_location = self.selected_location[:]
  545. self.copy_grassdb = self.selected_grassdb[:]
  546. if len(self.copy_layer) > 1:
  547. label = _("{c} maps marked for moving.").format(c=len(self.selected_layer))
  548. else:
  549. label = _("Map <{layer}> marked for moving.").format(layer=self.copy_layer[0].data['name'])
  550. self.showNotification.emit(message=label)
  551. def OnCopyMap(self, event):
  552. """Copy layer or mapset (just save it temporarily, copying is done by paste)"""
  553. self.copy_mode = True
  554. self.copy_layer = self.selected_layer[:]
  555. self.copy_mapset = self.selected_mapset[:]
  556. self.copy_location = self.selected_location[:]
  557. self.copy_grassdb = self.selected_grassdb[:]
  558. if len(self.copy_layer) > 1:
  559. label = _("{c} maps marked for copying.").format(c=len(self.selected_layer))
  560. else:
  561. label = _("Map <{layer}> marked for copying.").format(layer=self.copy_layer[0].data['name'])
  562. self.showNotification.emit(message=label)
  563. def OnRenameMap(self, event):
  564. """Rename layer with dialog"""
  565. old_name = self.selected_layer[0].data['name']
  566. gisrc, env = gscript.create_environment(
  567. self.selected_grassdb[0].data['name'],
  568. self.selected_location[0].data['name'],
  569. self.selected_mapset[0].data['name'])
  570. new_name = self._getNewMapName(
  571. _('New name'),
  572. _('Rename map'),
  573. old_name,
  574. env=env,
  575. mapset=self.selected_mapset[0].data['name'],
  576. element=self.selected_layer[0].data['type'])
  577. if new_name:
  578. self.Rename(old_name, new_name)
  579. self.ReloadTreeItems()
  580. def OnCreateMapset(self, event):
  581. """Create new mapset"""
  582. gisdbase = self.selected_grassdb[0]
  583. location = self.selected_location[0]
  584. try:
  585. mapset = create_mapset_interactively(self,
  586. gisdbase.data['name'],
  587. location.data['name'],)
  588. if mapset:
  589. self.InsertMapset(name=mapset,
  590. location_node=location)
  591. self.ReloadTreeItems()
  592. except Exception as e:
  593. GError(parent=self,
  594. message=_("Unable to create new mapset: %s") % e,
  595. showTraceback=False)
  596. def OnRenameMapset(self, event):
  597. """
  598. Rename selected mapset
  599. """
  600. try:
  601. newmapset = rename_mapset_interactively(
  602. self,
  603. self.selected_grassdb[0].data['name'],
  604. self.selected_location[0].data['name'],
  605. self.selected_mapset[0].data['name'])
  606. if newmapset:
  607. self.ReloadTreeItems()
  608. except Exception as e:
  609. GError(parent=self,
  610. message=_("Unable to rename mapset: %s") % e,
  611. showTraceback=False)
  612. def OnRenameLocation(self, event):
  613. """
  614. Rename selected location
  615. """
  616. try:
  617. newlocation = rename_location_interactively(
  618. self,
  619. self.selected_grassdb[0].data['name'],
  620. self.selected_location[0].data['name'])
  621. if newlocation:
  622. self.ReloadTreeItems()
  623. except Exception as e:
  624. GError(parent=self,
  625. message=_("Unable to rename location: %s") % e,
  626. showTraceback=False)
  627. def OnStartEditLabel(self, node, event):
  628. """Start label editing"""
  629. self.DefineItems([node])
  630. Debug.msg(1, "Start label edit {name}".format(name=node.data['name']))
  631. label = _("Editing {name}").format(name=node.data['name'])
  632. self.showNotification.emit(message=label)
  633. if not self.selected_layer:
  634. event.Veto()
  635. def OnEditLabel(self, node, event):
  636. """End label editing"""
  637. if self.selected_layer and not event.IsEditCancelled():
  638. old_name = node.data['name']
  639. Debug.msg(1, "End label edit {name}".format(name=old_name))
  640. new_name = event.GetLabel()
  641. self.Rename(old_name, new_name)
  642. def Rename(self, old, new):
  643. """Rename layer"""
  644. string = old + ',' + new
  645. gisrc, env = gscript.create_environment(
  646. self.selected_grassdb[0].data['name'],
  647. self.selected_location[0].data['name'],
  648. self.selected_mapset[0].data['name'])
  649. label = _("Renaming map <{name}>...").format(name=string)
  650. self.showNotification.emit(message=label)
  651. if self.selected_layer[0].data['type'] == 'vector':
  652. renamed, cmd = self._runCommand('g.rename', vector=string, env=env)
  653. elif self.selected_layer[0].data['type'] == 'raster':
  654. renamed, cmd = self._runCommand('g.rename', raster=string, env=env)
  655. else:
  656. renamed, cmd = self._runCommand(
  657. 'g.rename', raster3d=string, env=env)
  658. if renamed == 0:
  659. self.selected_layer[0].data['name'] = new
  660. self.RefreshNode(self.selected_layer[0])
  661. self.showNotification.emit(
  662. message=_("{cmd} -- completed").format(cmd=cmd))
  663. Debug.msg(1, "LAYER RENAMED TO: " + new)
  664. gscript.try_remove(gisrc)
  665. def OnPasteMap(self, event):
  666. # copying between mapsets of one location
  667. if not self.copy_layer:
  668. if self.copy_mode:
  669. GMessage(_("No map selected for copying."), parent=self)
  670. else:
  671. GMessage(_("No map selected for moving."), parent=self)
  672. return
  673. for i in range(len(self.copy_layer)):
  674. gisrc, env = gscript.create_environment(self.selected_grassdb[0].data['name'],
  675. self.selected_location[0].data['name'],
  676. self.selected_mapset[0].data['name'])
  677. gisrc2, env2 = gscript.create_environment(self.copy_grassdb[i].data['name'],
  678. self.copy_location[i].data['name'],
  679. self.copy_mapset[i].data['name'])
  680. new_name = self.copy_layer[i].data['name']
  681. if self.selected_location[0] == self.copy_location[i]:
  682. # within one mapset
  683. if self.selected_mapset[0] == self.copy_mapset[i]:
  684. # ignore when just moves map
  685. if self.copy_mode is False:
  686. return
  687. new_name = self._getNewMapName(_('New name for <{n}>').format(n=self.copy_layer[i].data['name']),
  688. _('Select new name'),
  689. self.copy_layer[i].data['name'], env=env,
  690. mapset=self.selected_mapset[0].data['name'],
  691. element=self.copy_layer[i].data['type'])
  692. if not new_name:
  693. return
  694. # within one location, different mapsets
  695. else:
  696. if map_exists(new_name, element=self.copy_layer[i].data['type'], env=env,
  697. mapset=self.selected_mapset[0].data['name']):
  698. new_name = self._getNewMapName(_('New name for <{n}>').format(n=self.copy_layer[i].data['name']),
  699. _('Select new name'),
  700. self.copy_layer[i].data['name'], env=env,
  701. mapset=self.selected_mapset[0].data['name'],
  702. element=self.copy_layer[i].data['type'])
  703. if not new_name:
  704. return
  705. string = self.copy_layer[i].data['name'] + '@' + self.copy_mapset[i].data['name'] + ',' + new_name
  706. pasted = 0
  707. if self.copy_mode:
  708. label = _("Copying <{name}>...").format(name=string)
  709. else:
  710. label = _("Moving <{name}>...").format(name=string)
  711. self.showNotification.emit(message=label)
  712. if self.copy_layer[i].data['type'] == 'vector':
  713. pasted, cmd = self._runCommand('g.copy', vector=string, env=env)
  714. node = 'vector'
  715. elif self.copy_layer[i].data['type'] == 'raster':
  716. pasted, cmd = self._runCommand('g.copy', raster=string, env=env)
  717. node = 'raster'
  718. else:
  719. pasted, cmd = self._runCommand('g.copy', raster_3d=string, env=env)
  720. node = 'raster_3d'
  721. if pasted == 0:
  722. self.InsertLayer(name=new_name, mapset_node=self.selected_mapset[0],
  723. element_name=node)
  724. Debug.msg(1, "COPIED TO: " + new_name)
  725. if self.copy_mode:
  726. self.showNotification.emit(message=_("g.copy completed"))
  727. else:
  728. self.showNotification.emit(message=_("g.copy completed"))
  729. # remove old
  730. if not self.copy_mode:
  731. self._removeMapAfterCopy(self.copy_layer[i], self.copy_mapset[i], env2)
  732. gscript.try_remove(gisrc)
  733. gscript.try_remove(gisrc2)
  734. # expand selected mapset
  735. else:
  736. if self.copy_layer[i].data['type'] == 'raster_3d':
  737. GError(_("Reprojection is not implemented for 3D rasters"), parent=self)
  738. return
  739. if map_exists(new_name, element=self.copy_layer[i].data['type'], env=env,
  740. mapset=self.selected_mapset[0].data['name']):
  741. new_name = self._getNewMapName(_('New name'), _('Select new name'),
  742. self.copy_layer[i].data['name'], env=env,
  743. mapset=self.selected_mapset[0].data['name'],
  744. element=self.copy_layer[i].data['type'])
  745. if not new_name:
  746. continue
  747. callback = lambda gisrc2=gisrc2, gisrc=gisrc, cLayer=self.copy_layer[i], \
  748. cMapset=self.copy_mapset[i], cMode=self.copy_mode, name=new_name: \
  749. self._onDoneReprojection(env2, gisrc2, gisrc, cLayer, cMapset, cMode, name)
  750. dlg = CatalogReprojectionDialog(self, self._giface,
  751. self.copy_grassdb[i].data['name'],
  752. self.copy_location[i].data['name'],
  753. self.copy_mapset[i].data['name'],
  754. self.copy_layer[i].data['name'],
  755. env2,
  756. self.selected_grassdb[0].data['name'],
  757. self.selected_location[0].data['name'],
  758. self.selected_mapset[0].data['name'],
  759. new_name,
  760. self.copy_layer[i].data['type'],
  761. env, callback)
  762. dlg.ShowModal()
  763. self.ExpandNode(self.selected_mapset[0], recursive=True)
  764. self._initVariablesCatalog()
  765. def _onDoneReprojection(self, iEnv, iGisrc, oGisrc, cLayer, cMapset, cMode, name):
  766. self.InsertLayer(name=name, mapset_node=self.selected_mapset[0],
  767. element_name=cLayer.data['type'])
  768. if not cMode:
  769. self._removeMapAfterCopy(cLayer, cMapset, iEnv)
  770. gscript.try_remove(iGisrc)
  771. gscript.try_remove(oGisrc)
  772. self.ExpandNode(self.selected_mapset[0], recursive=True)
  773. def _removeMapAfterCopy(self, cLayer, cMapset, env):
  774. removed, cmd = self._runCommand('g.remove', type=cLayer.data['type'],
  775. name=cLayer.data['name'], flags='f', env=env)
  776. if removed == 0:
  777. self._model.RemoveNode(cLayer)
  778. self.RefreshNode(cMapset, recursive=True)
  779. Debug.msg(1, "LAYER " + cLayer.data['name'] + " DELETED")
  780. self.showNotification.emit(message=_("g.remove completed"))
  781. def InsertLayer(self, name, mapset_node, element_name):
  782. """Insert layer into model and refresh tree"""
  783. self._model.AppendNode(parent=mapset_node,
  784. data=dict(type=element_name, name=name))
  785. self._model.SortChildren(mapset_node)
  786. self.RefreshNode(mapset_node, recursive=True)
  787. def InsertMapset(self, name, location_node):
  788. """Insert mapset into model and refresh tree"""
  789. self._model.AppendNode(parent=location_node,
  790. data=dict(type="mapset", name=name))
  791. self._model.SortChildren(location_node)
  792. self.RefreshNode(location_node, recursive=True)
  793. def InsertGrassDb(self, name):
  794. """Insert grass db into model and refresh tree"""
  795. self.grassdatabases.append(name)
  796. self._model.AppendNode(parent=self._model.root,
  797. data=dict(type="grassdb", name=name))
  798. self._model.SortChildren(self._model.root)
  799. self.ReloadTreeItems()
  800. def OnDeleteMap(self, event):
  801. """Delete layer or mapset"""
  802. names = [self.selected_layer[i].data['name'] + '@' + self.selected_mapset[i].data['name']
  803. for i in range(len(self.selected_layer))]
  804. if len(names) < 10:
  805. question = _("Do you really want to delete map(s) <{m}>?").format(m=', '.join(names))
  806. else:
  807. question = _("Do you really want to delete {n} maps?").format(n=len(names))
  808. if self._confirmDialog(question, title=_('Delete map')) == wx.ID_YES:
  809. label = _("Deleting {name}...").format(name=names)
  810. self.showNotification.emit(message=label)
  811. for i in range(len(self.selected_layer)):
  812. gisrc, env = gscript.create_environment(
  813. self.selected_grassdb[i].data['name'],
  814. self.selected_location[i].data['name'],
  815. self.selected_mapset[i].data['name'])
  816. removed, cmd = self._runCommand(
  817. 'g.remove', flags='f', type=self.selected_layer[i].data['type'],
  818. name=self.selected_layer[i].data['name'], env=env)
  819. if removed == 0:
  820. self._model.RemoveNode(self.selected_layer[i])
  821. self.RefreshNode(self.selected_mapset[i], recursive=True)
  822. Debug.msg(1, "LAYER " + self.selected_layer[i].data['name'] + " DELETED")
  823. # remove map layer from layer tree if exists
  824. if not isinstance(self._giface, StandaloneGrassInterface):
  825. name = self.selected_layer[i].data['name'] + '@' + self.selected_mapset[i].data['name']
  826. layers = self._giface.GetLayerList().GetLayersByName(name)
  827. for layer in layers:
  828. self._giface.GetLayerList().DeleteLayer(layer)
  829. gscript.try_remove(gisrc)
  830. self.UnselectAll()
  831. self.showNotification.emit(message=_("g.remove completed"))
  832. def OnDeleteMapset(self, event):
  833. """
  834. Delete selected mapset
  835. """
  836. try:
  837. delete_mapset_interactively(self,
  838. self.selected_grassdb[0].data['name'],
  839. self.selected_location[0].data['name'],
  840. self.selected_mapset[0].data['name'])
  841. except Exception as e:
  842. GError(parent=self,
  843. message=_("Unable to delete mapset: %s") % e,
  844. showTraceback=False)
  845. self.ReloadTreeItems()
  846. def OnDeleteLocation(self, event):
  847. """
  848. Delete selected location
  849. """
  850. try:
  851. delete_location_interactively(self,
  852. self.selected_grassdb[0].data['name'],
  853. self.selected_location[0].data['name'])
  854. except Exception as e:
  855. GError(parent=self,
  856. message=_("Unable to delete location: %s") % e,
  857. showTraceback=False)
  858. self.ReloadTreeItems()
  859. def OnDisplayLayer(self, event):
  860. """Display layer in current graphics view"""
  861. self.DisplayLayer()
  862. def DisplayLayer(self):
  863. """Display selected layer in current graphics view"""
  864. all_names = []
  865. names = {'raster': [], 'vector': [], 'raster3d': []}
  866. for i in range(len(self.selected_layer)):
  867. name = self.selected_layer[i].data['name'] + '@' + self.selected_mapset[i].data['name']
  868. names[self.selected_layer[i].data['type']].append(name)
  869. all_names.append(name)
  870. #if self.selected_location[0].data['name'] == gisenv()['LOCATION_NAME'] and self.selected_mapset[0]:
  871. for ltype in names:
  872. if names[ltype]:
  873. self._giface.lmgr.AddMaps(list(reversed(names[ltype])), ltype, True)
  874. if len(self._giface.GetLayerList()) == 1:
  875. # zoom to map if there is only one map layer
  876. self._giface.GetMapWindow().ZoomToMap()
  877. Debug.msg(1, "Displayed layer(s): " + str(all_names))
  878. def OnBeginDrag(self, node, event):
  879. """Just copy necessary data"""
  880. self.DefineItems(self.GetSelected())
  881. if self.selected_layer and not (self._restricted and gisenv()[
  882. 'LOCATION_NAME'] != self.selected_location[0].data['name']):
  883. event.Allow()
  884. self.OnCopyMap(event)
  885. Debug.msg(1, "DRAG")
  886. else:
  887. event.Veto()
  888. def OnEndDrag(self, node, event):
  889. """Copy layer into target"""
  890. self.copy_mode = wx.GetMouseState().ControlDown()
  891. if node:
  892. self.DefineItems([node])
  893. if self._restricted and gisenv()['MAPSET'] != self.selected_mapset[0].data['name']:
  894. GMessage(_("To move or copy maps to other mapsets, unlock editing of other mapsets"),
  895. parent=self)
  896. event.Veto()
  897. return
  898. event.Allow()
  899. Debug.msg(1, "DROP DONE")
  900. self.OnPasteMap(event)
  901. def OnSwitchDbLocationMapset(self, event):
  902. """Switch to location and mapset"""
  903. self._SwitchDbLocationMapset()
  904. def _SwitchDbLocationMapset(self):
  905. """Switch to location and mapset"""
  906. genv = gisenv()
  907. # Distinguish when only part of db/location/mapset is changed.
  908. if (
  909. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  910. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  911. ):
  912. self.changeMapset.emit(mapset=self.selected_mapset[0].data['name'])
  913. elif self.selected_grassdb[0].data['name'] == genv['GISDBASE']:
  914. self.changeLocation.emit(mapset=self.selected_mapset[0].data['name'],
  915. location=self.selected_location[0].data['name'],
  916. dbase=None)
  917. else:
  918. self.changeLocation.emit(mapset=self.selected_mapset[0].data['name'],
  919. location=self.selected_location[0].data['name'],
  920. dbase=self.selected_grassdb[0].data['name'])
  921. self.UpdateCurrentDbLocationMapsetNode()
  922. self.ExpandCurrentMapset()
  923. self.RefreshItems()
  924. def OnMetadata(self, event):
  925. """Show metadata of any raster/vector/3draster"""
  926. def done(event):
  927. gscript.try_remove(event.userData)
  928. for i in range(len(self.selected_layer)):
  929. if self.selected_layer[i].data['type'] == 'raster':
  930. cmd = ['r.info']
  931. elif self.selected_layer[i].data['type'] == 'vector':
  932. cmd = ['v.info']
  933. elif self.selected_layer[i].data['type'] == 'raster_3d':
  934. cmd = ['r3.info']
  935. cmd.append('map=%s@%s' % (self.selected_layer[i].data['name'], self.selected_mapset[i].data['name']))
  936. gisrc, env = gscript.create_environment(
  937. self.selected_grassdb[i].data['name'],
  938. self.selected_location[i].data['name'],
  939. self.selected_mapset[i].data['name'])
  940. # print output to command log area
  941. # temp gisrc file must be deleted onDone
  942. self._giface.RunCmd(cmd, env=env, onDone=done, userData=gisrc)
  943. def OnCopyName(self, event):
  944. """Copy layer name to clipboard"""
  945. if wx.TheClipboard.Open():
  946. do = wx.TextDataObject()
  947. text = []
  948. for i in range(len(self.selected_layer)):
  949. text.append('%s@%s' % (self.selected_layer[i].data['name'], self.selected_mapset[i].data['name']))
  950. do.SetText(','.join(text))
  951. wx.TheClipboard.SetData(do)
  952. wx.TheClipboard.Close()
  953. def Filter(self, text):
  954. """Filter tree based on name and type."""
  955. text = text.strip()
  956. if len(text.split(':')) > 1:
  957. name = text.split(':')[1].strip()
  958. elem = text.split(':')[0].strip()
  959. if 'r' == elem:
  960. element = 'raster'
  961. elif 'r3' == elem:
  962. element = 'raster_3d'
  963. elif 'v' == elem:
  964. element = 'vector'
  965. else:
  966. element = None
  967. else:
  968. element = None
  969. name = text.strip()
  970. self._model = filterModel(self._orig_model, name=name, element=element)
  971. self.UpdateCurrentDbLocationMapsetNode()
  972. self.RefreshItems()
  973. self.ExpandCurrentMapset()
  974. def _getNewMapName(self, message, title, value, element, mapset, env):
  975. """Dialog for simple text entry"""
  976. dlg = NameEntryDialog(parent=self, message=message, caption=title,
  977. element=element, env=env, mapset=mapset)
  978. dlg.SetValue(value)
  979. if dlg.ShowModal() == wx.ID_OK:
  980. name = dlg.GetValue()
  981. else:
  982. name = None
  983. dlg.Destroy()
  984. return name
  985. def _confirmDialog(self, question, title):
  986. """Confirm dialog"""
  987. dlg = wx.MessageDialog(self, question, title, wx.YES_NO)
  988. res = dlg.ShowModal()
  989. dlg.Destroy()
  990. return res
  991. def _isCurrent(self, genv):
  992. if self._restricted:
  993. currentMapset = currentLocation = currentGrassDb = True
  994. for i in range(len(self.selected_grassdb)):
  995. if self.selected_grassdb[i].data['name'] != genv['GISDBASE']:
  996. currentGrassDb = False
  997. currentLocation = False
  998. currentMapset = False
  999. break
  1000. if currentLocation:
  1001. for i in range(len(self.selected_location)):
  1002. if self.selected_location[i].data['name'] != genv['LOCATION_NAME']:
  1003. currentLocation = False
  1004. currentMapset = False
  1005. break
  1006. if currentMapset:
  1007. for i in range(len(self.selected_mapset)):
  1008. if self.selected_mapset[i].data['name'] != genv['MAPSET']:
  1009. currentMapset = False
  1010. break
  1011. return currentGrassDb, currentLocation, currentMapset
  1012. else:
  1013. return True, True, True
  1014. def _popupMenuLayer(self):
  1015. """Create popup menu for layers"""
  1016. menu = Menu()
  1017. genv = gisenv()
  1018. currentGrassDb, currentLocation, currentMapset = self._isCurrent(genv)
  1019. item = wx.MenuItem(menu, wx.ID_ANY, _("&Cut"))
  1020. menu.AppendItem(item)
  1021. self.Bind(wx.EVT_MENU, self.OnMoveMap, item)
  1022. if not currentMapset:
  1023. item.Enable(False)
  1024. item = wx.MenuItem(menu, wx.ID_ANY, _("&Copy"))
  1025. menu.AppendItem(item)
  1026. self.Bind(wx.EVT_MENU, self.OnCopyMap, item)
  1027. item = wx.MenuItem(menu, wx.ID_ANY, _("Copy &name"))
  1028. menu.AppendItem(item)
  1029. self.Bind(wx.EVT_MENU, self.OnCopyName, item)
  1030. item = wx.MenuItem(menu, wx.ID_ANY, _("&Paste"))
  1031. menu.AppendItem(item)
  1032. self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
  1033. if not(currentMapset and self.copy_layer):
  1034. item.Enable(False)
  1035. item = wx.MenuItem(menu, wx.ID_ANY, _("&Delete"))
  1036. menu.AppendItem(item)
  1037. self.Bind(wx.EVT_MENU, self.OnDeleteMap, item)
  1038. item.Enable(currentMapset)
  1039. item = wx.MenuItem(menu, wx.ID_ANY, _("&Rename"))
  1040. menu.AppendItem(item)
  1041. self.Bind(wx.EVT_MENU, self.OnRenameMap, item)
  1042. item.Enable(currentMapset and len(self.selected_layer) == 1)
  1043. menu.AppendSeparator()
  1044. if not isinstance(self._giface, StandaloneGrassInterface):
  1045. if all([each.data['name'] == genv['LOCATION_NAME'] for each in self.selected_location]):
  1046. if len(self.selected_layer) > 1:
  1047. item = wx.MenuItem(menu, wx.ID_ANY, _("&Display layers"))
  1048. else:
  1049. item = wx.MenuItem(menu, wx.ID_ANY, _("&Display layer"))
  1050. menu.AppendItem(item)
  1051. self.Bind(wx.EVT_MENU, self.OnDisplayLayer, item)
  1052. item = wx.MenuItem(menu, wx.ID_ANY, _("Show &metadata"))
  1053. menu.AppendItem(item)
  1054. self.Bind(wx.EVT_MENU, self.OnMetadata, item)
  1055. self.PopupMenu(menu)
  1056. menu.Destroy()
  1057. def _popupMenuMapset(self):
  1058. """Create popup menu for mapsets"""
  1059. menu = Menu()
  1060. genv = gisenv()
  1061. currentGrassDb, currentLocation, currentMapset = self._isCurrent(genv)
  1062. item = wx.MenuItem(menu, wx.ID_ANY, _("&Paste"))
  1063. menu.AppendItem(item)
  1064. self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
  1065. if not(currentMapset and self.copy_layer):
  1066. item.Enable(False)
  1067. item = wx.MenuItem(menu, wx.ID_ANY, _("&Switch mapset"))
  1068. menu.AppendItem(item)
  1069. self.Bind(wx.EVT_MENU, self.OnSwitchDbLocationMapset, item)
  1070. if (
  1071. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  1072. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  1073. and self.selected_mapset[0].data['name'] == genv['MAPSET']
  1074. ):
  1075. item.Enable(False)
  1076. item = wx.MenuItem(menu, wx.ID_ANY, _("&Delete mapset"))
  1077. menu.AppendItem(item)
  1078. self.Bind(wx.EVT_MENU, self.OnDeleteMapset, item)
  1079. if (
  1080. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  1081. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  1082. and self.selected_mapset[0].data['name'] == genv['MAPSET']
  1083. ):
  1084. item.Enable(False)
  1085. item = wx.MenuItem(menu, wx.ID_ANY, _("&Rename mapset"))
  1086. menu.AppendItem(item)
  1087. self.Bind(wx.EVT_MENU, self.OnRenameMapset, item)
  1088. if (
  1089. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  1090. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  1091. and self.selected_mapset[0].data['name'] == genv['MAPSET']
  1092. ):
  1093. item.Enable(False)
  1094. self.PopupMenu(menu)
  1095. menu.Destroy()
  1096. def _popupMenuLocation(self):
  1097. """Create popup menu for locations"""
  1098. menu = Menu()
  1099. genv = gisenv()
  1100. item = wx.MenuItem(menu, wx.ID_ANY, _("&Create mapset"))
  1101. menu.AppendItem(item)
  1102. self.Bind(wx.EVT_MENU, self.OnCreateMapset, item)
  1103. item = wx.MenuItem(menu, wx.ID_ANY, _("&Delete location"))
  1104. menu.AppendItem(item)
  1105. self.Bind(wx.EVT_MENU, self.OnDeleteLocation, item)
  1106. if (
  1107. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  1108. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  1109. ):
  1110. item.Enable(False)
  1111. item = wx.MenuItem(menu, wx.ID_ANY, _("&Rename location"))
  1112. menu.AppendItem(item)
  1113. self.Bind(wx.EVT_MENU, self.OnRenameLocation, item)
  1114. if (
  1115. self.selected_grassdb[0].data['name'] == genv['GISDBASE']
  1116. and self.selected_location[0].data['name'] == genv['LOCATION_NAME']
  1117. ):
  1118. item.Enable(False)
  1119. self.PopupMenu(menu)
  1120. menu.Destroy()
  1121. def _popupMenuGrassDb(self):
  1122. """Create popup menu for grass db"""
  1123. menu = Menu()
  1124. self.PopupMenu(menu)
  1125. menu.Destroy()
  1126. def _popupMenuElement(self):
  1127. """Create popup menu for elements"""
  1128. menu = Menu()
  1129. item = wx.MenuItem(menu, wx.ID_ANY, _("&Paste"))
  1130. menu.AppendItem(item)
  1131. self.Bind(wx.EVT_MENU, self.OnPasteMap, item)
  1132. genv = gisenv()
  1133. currentGrassDb, currentLocation, currentMapset = self._isCurrent(genv)
  1134. if not(currentMapset and self.copy_layer):
  1135. item.Enable(False)
  1136. self.PopupMenu(menu)
  1137. menu.Destroy()
  1138. def _popupMenuEmpty(self):
  1139. """Create empty popup when multiple different types of items are selected"""
  1140. menu = Menu()
  1141. item = wx.MenuItem(menu, wx.ID_ANY, _("No available options"))
  1142. menu.AppendItem(item)
  1143. item.Enable(False)
  1144. self.PopupMenu(menu)
  1145. menu.Destroy()