tree.py 63 KB

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