tree.py 51 KB

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