tree.py 55 KB

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