tree.py 58 KB

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