tree.py 47 KB

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