tree.py 54 KB

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