tree.py 62 KB

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