tree.py 66 KB

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