extensions.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. """
  2. @package modules.extensions
  3. @brief GRASS Addons extensions management classes
  4. Classes:
  5. - extensions::InstallExtensionWindow
  6. - extensions::ExtensionTreeModelBuilder
  7. - extensions::ManageExtensionWindow
  8. - extensions::CheckListExtension
  9. (C) 2008-2016 by the GRASS Development Team
  10. This program is free software under the GNU General Public License
  11. (>=v2). Read the file COPYING that comes with GRASS for details.
  12. @author Martin Landa <landa.martin gmail.com>
  13. @author Anna Petrasova <kratochanna gmail.com>
  14. """
  15. import os
  16. import sys
  17. import wx
  18. from grass.script import task as gtask
  19. from core import globalvar
  20. from core.gcmd import GError, RunCommand, GException, GMessage
  21. from core.utils import SetAddOnPath
  22. from core.gthread import gThread
  23. from core.menutree import TreeModel, ModuleNode
  24. from gui_core.widgets import GListCtrl
  25. from gui_core.treeview import CTreeView
  26. from core.toolboxes import toolboxesOutdated
  27. from gui_core.wrap import Button, StaticBox, TextCtrl, Menu, NewId, SearchCtrl
  28. class InstallExtensionWindow(wx.Frame):
  29. def __init__(self, parent, giface, id=wx.ID_ANY, title=_(
  30. "Fetch & install extension from GRASS Addons"), **kwargs):
  31. self.parent = parent
  32. self._giface = giface
  33. self.options = dict() # list of options
  34. wx.Frame.__init__(self, parent=parent, id=id, title=title, **kwargs)
  35. self.SetIcon(
  36. wx.Icon(
  37. os.path.join(
  38. globalvar.ICONDIR,
  39. 'grass.ico'),
  40. wx.BITMAP_TYPE_ICO))
  41. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  42. # self.repoBox = StaticBox(
  43. # parent=self.panel, id=wx.ID_ANY, label=" %s " %
  44. # _("Repository (leave empty to use the official one)"))
  45. self.treeBox = StaticBox(
  46. parent=self.panel, id=wx.ID_ANY, label=" %s " %
  47. _("List of extensions - double-click to install"))
  48. # self.repo = TextCtrl(parent=self.panel, id=wx.ID_ANY)
  49. # modelBuilder loads data into tree model
  50. self.modelBuilder = ExtensionTreeModelBuilder()
  51. # tree view displays model data
  52. self.tree = CTreeView(
  53. parent=self.panel,
  54. model=self.modelBuilder.GetModel())
  55. self.search = SearchCtrl(self.panel)
  56. self.search.SetDescriptiveText(_('Search'))
  57. self.search.ShowCancelButton(True)
  58. # load data in different thread
  59. self.thread = gThread()
  60. self.optionBox = StaticBox(parent=self.panel, id=wx.ID_ANY,
  61. label=" %s " % _("Options"))
  62. task = gtask.parse_interface('g.extension')
  63. ignoreFlags = ['l', 'c', 'g', 'a', 'f', 't', 'help', 'quiet']
  64. if sys.platform == 'win32':
  65. ignoreFlags.append('d')
  66. ignoreFlags.append('i')
  67. for f in task.get_options()['flags']:
  68. name = f.get('name', '')
  69. desc = f.get('label', '')
  70. if not desc:
  71. desc = f.get('description', '')
  72. if not name and not desc:
  73. continue
  74. if name in ignoreFlags:
  75. continue
  76. self.options[name] = wx.CheckBox(parent=self.panel, id=wx.ID_ANY,
  77. label=desc)
  78. # defaultUrl = '' # default/official one will be used when option empty
  79. # self.repo.SetValue(
  80. # task.get_param(
  81. # value='url').get(
  82. # 'default',
  83. # defaultUrl))
  84. self.statusbar = self.CreateStatusBar(number=1)
  85. # self.btnFetch = Button(parent=self.panel, id=wx.ID_ANY,
  86. # label=_("&Fetch"))
  87. # self.btnFetch.SetToolTip(_("Fetch list of available modules "
  88. # "from GRASS Addons repository"))
  89. self.btnClose = Button(parent=self.panel, id=wx.ID_CLOSE)
  90. self.btnInstall = Button(parent=self.panel, id=wx.ID_ANY,
  91. label=_("&Install"))
  92. self.btnInstall.SetToolTip(
  93. _("Install selected add-ons GRASS module"))
  94. self.btnInstall.Enable(False)
  95. self.btnHelp = Button(parent=self.panel, id=wx.ID_HELP)
  96. self.btnHelp.SetToolTip(_("Show g.extension manual page"))
  97. self.btnClose.Bind(wx.EVT_BUTTON, lambda evt: self.Close())
  98. # self.btnFetch.Bind(wx.EVT_BUTTON, self.OnFetch)
  99. self.btnInstall.Bind(wx.EVT_BUTTON, self.OnInstall)
  100. self.btnHelp.Bind(wx.EVT_BUTTON, self.OnHelp)
  101. self.search.Bind(wx.EVT_TEXT, lambda evt: self.Filter(evt.GetString()))
  102. self.search.Bind(wx.EVT_SEARCHCTRL_CANCEL_BTN,
  103. lambda evt: self.Filter(''))
  104. self.tree.selectionChanged.connect(self.OnItemSelected)
  105. self.tree.itemActivated.connect(self.OnItemActivated)
  106. self.tree.contextMenu.connect(self.OnContextMenu)
  107. wx.CallAfter(self._fetch)
  108. self._layout()
  109. def _layout(self):
  110. """Do layout"""
  111. sizer = wx.BoxSizer(wx.VERTICAL)
  112. # repoSizer = wx.StaticBoxSizer(self.repoBox, wx.VERTICAL)
  113. # repo1Sizer = wx.BoxSizer(wx.HORIZONTAL)
  114. # repo1Sizer.Add(self.repo, proportion=1,
  115. # flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL, border=1)
  116. # repo1Sizer.Add(self.btnFetch, proportion=0,
  117. # flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL, border=1)
  118. # repoSizer.Add(repo1Sizer,
  119. # flag=wx.EXPAND)
  120. sizer.Add(self.search, proportion=0, flag=wx.EXPAND | wx.ALL, border=3)
  121. treeSizer = wx.StaticBoxSizer(self.treeBox, wx.HORIZONTAL)
  122. treeSizer.Add(self.tree, proportion=1,
  123. flag=wx.ALL | wx.EXPAND, border=1)
  124. # options
  125. optionSizer = wx.StaticBoxSizer(self.optionBox, wx.VERTICAL)
  126. for key in self.options.keys():
  127. optionSizer.Add(self.options[key], proportion=0)
  128. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  129. btnSizer.Add(self.btnHelp, proportion=0)
  130. btnSizer.AddStretchSpacer()
  131. btnSizer.Add(self.btnClose, proportion=0,
  132. flag=wx.RIGHT, border=5)
  133. btnSizer.Add(self.btnInstall, proportion=0)
  134. # sizer.Add(repoSizer, proportion=0,
  135. # flag=wx.ALL | wx.EXPAND, border=3)
  136. sizer.Add(treeSizer, proportion=1,
  137. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  138. sizer.Add(optionSizer, proportion=0,
  139. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  140. sizer.Add(btnSizer, proportion=0,
  141. flag=wx.ALL | wx.EXPAND, border=5)
  142. self.panel.SetSizer(sizer)
  143. sizer.Fit(self.panel)
  144. self.Layout()
  145. def _getCmd(self):
  146. item = self.tree.GetSelected()
  147. if not item or 'command' not in item[0].data:
  148. GError(_("Extension not defined"), parent=self)
  149. return
  150. name = item[0].data['command']
  151. flags = list()
  152. for key in self.options.keys():
  153. if self.options[key].IsChecked():
  154. if len(key) == 1:
  155. flags.append('-%s' % key)
  156. else:
  157. flags.append('--%s' % key)
  158. # 'url=' + self.repo.GetValue().strip()]
  159. return ['g.extension'] + flags + ['extension={}'.format(name) ]
  160. def OnFetch(self, event):
  161. """Fetch list of available extensions"""
  162. self._fetch()
  163. def _fetch(self):
  164. """Fetch list of available extensions"""
  165. wx.BeginBusyCursor()
  166. self.SetStatusText(
  167. _("Fetching list of modules from GRASS-Addons (be patient)..."), 0)
  168. try:
  169. self.thread.Run(
  170. callable=self.modelBuilder.Load,
  171. url='', # self.repo.GetValue().strip(),
  172. ondone=lambda event: self._fetchDone())
  173. except GException as error:
  174. self._fetchDone()
  175. GError(str(error), parent=self, showTraceback=False)
  176. def _fetchDone(self):
  177. self.tree.RefreshItems()
  178. nitems = len(
  179. self.modelBuilder.GetModel().SearchNodes(
  180. key='command', value='*'))
  181. self.SetStatusText(_("%d extensions loaded") % nitems, 0)
  182. wx.EndBusyCursor()
  183. def Filter(self, text):
  184. model = self.modelBuilder.GetModel()
  185. if text:
  186. model = model.Filtered(key=['command', 'keywords', 'description'],
  187. value=text)
  188. self.tree.SetModel(model)
  189. self.tree.ExpandAll()
  190. else:
  191. self.tree.SetModel(model)
  192. def OnContextMenu(self, node):
  193. if not hasattr(self, "popupID"):
  194. self.popupID = dict()
  195. for key in ('install', 'help'):
  196. self.popupID[key] = NewId()
  197. data = node.data
  198. if data and 'command' in data:
  199. self.popupMenu = Menu()
  200. self.popupMenu.Append(self.popupID['install'], _("Install"))
  201. self.Bind(wx.EVT_MENU, self.OnInstall, id=self.popupID['install'])
  202. self.popupMenu.AppendSeparator()
  203. self.popupMenu.Append(
  204. self.popupID['help'],
  205. _("Show manual page"))
  206. self.Bind(wx.EVT_MENU, self.OnItemHelp, id=self.popupID['help'])
  207. self.PopupMenu(self.popupMenu)
  208. self.popupMenu.Destroy()
  209. def OnItemActivated(self, node):
  210. data = node.data
  211. if data and 'command' in data:
  212. self.OnInstall(event=None)
  213. def OnInstall(self, event):
  214. """Install selected extension"""
  215. log = self.parent.GetLogWindow()
  216. cmd = self._getCmd()
  217. if cmd:
  218. log.RunCmd(cmd, onDone=self.OnDone)
  219. def OnDone(self, event):
  220. if event.returncode == 0:
  221. if not os.getenv('GRASS_ADDON_BASE'):
  222. SetAddOnPath(key='BASE')
  223. globalvar.UpdateGRASSAddOnCommands()
  224. toolboxesOutdated()
  225. def OnItemHelp(self, event):
  226. item = self.tree.GetSelected()
  227. if not item or 'command' not in item[0].data:
  228. return
  229. self._giface.Help(entry=item[0].data['command'], online=True)
  230. def OnHelp(self, event):
  231. self._giface.Help(entry='g.extension')
  232. def OnItemSelected(self, node):
  233. """Item selected"""
  234. data = node.data
  235. if data is None:
  236. self.SetStatusText('', 0)
  237. self.btnInstall.Enable(False)
  238. else:
  239. self.SetStatusText(data.get('description', ''), 0)
  240. self.btnInstall.Enable(True)
  241. class ExtensionTreeModelBuilder:
  242. """Tree model of available extensions."""
  243. def __init__(self):
  244. self.mainNodes = dict()
  245. self.model = TreeModel(ModuleNode)
  246. for prefix in ('display', 'database',
  247. 'general', 'imagery',
  248. 'misc', 'postscript', 'paint',
  249. 'raster', 'raster3D', 'sites', 'temporal', 'vector', 'wxGUI', 'other'):
  250. node = self.model.AppendNode(parent=self.model.root, label=prefix)
  251. self.mainNodes[prefix] = node
  252. def GetModel(self):
  253. return self.model
  254. def _emptyTree(self):
  255. """Remove modules from tree keeping the main structure"""
  256. for node in self.mainNodes.values():
  257. for child in reversed(node.children):
  258. self.model.RemoveNode(child)
  259. def _expandPrefix(self, c):
  260. name = {'d': 'display',
  261. 'db': 'database',
  262. 'g': 'general',
  263. 'i': 'imagery',
  264. 'm': 'misc',
  265. 'ps': 'postscript',
  266. 'p': 'paint',
  267. 'r': 'raster',
  268. 'r3': 'raster3D',
  269. 's': 'sites',
  270. 't': 'temporal',
  271. 'v': 'vector',
  272. 'wx': 'wxGUI',
  273. '': 'other'}
  274. if c in name:
  275. return name[c]
  276. return c
  277. def Load(self, url, full=True):
  278. """Load list of extensions"""
  279. self._emptyTree()
  280. if full:
  281. flags = 'g'
  282. else:
  283. flags = 'l'
  284. retcode, ret, msg = RunCommand(
  285. 'g.extension', read=True, getErrorMsg=True, url=url, flags=flags, quiet=True)
  286. if retcode != 0:
  287. raise GException(_("Unable to load extensions. %s") % msg)
  288. currentNode = None
  289. for line in ret.splitlines():
  290. if full:
  291. try:
  292. key, value = line.split('=', 1)
  293. except ValueError:
  294. key = 'name'
  295. value = line
  296. if key == 'name':
  297. try:
  298. prefix, name = value.split('.', 1)
  299. except ValueError:
  300. prefix = ''
  301. name = value
  302. mainNode = self.mainNodes[self._expandPrefix(prefix)]
  303. currentNode = self.model.AppendNode(
  304. parent=mainNode, label=value)
  305. currentNode.data = {'command': value}
  306. else:
  307. if currentNode is not None:
  308. currentNode.data[key] = value
  309. else:
  310. try:
  311. prefix, name = line.strip().split('.', 1)
  312. except ValueError:
  313. prefix = ''
  314. name = line.strip()
  315. if self._expandPrefix(prefix) == prefix:
  316. prefix = ''
  317. module = prefix + '.' + name
  318. mainNode = self.mainNodes[self._expandPrefix(prefix)]
  319. currentNode = self.model.AppendNode(
  320. parent=mainNode, label=module)
  321. currentNode.data = {'command': module,
  322. 'keywords': '',
  323. 'description': ''}
  324. class ManageExtensionWindow(wx.Frame):
  325. def __init__(
  326. self, parent, id=wx.ID_ANY,
  327. title=_("Manage installed GRASS Addons extensions"),
  328. **kwargs):
  329. self.parent = parent
  330. wx.Frame.__init__(self, parent=parent, id=id, title=title, **kwargs)
  331. self.SetIcon(
  332. wx.Icon(
  333. os.path.join(
  334. globalvar.ICONDIR,
  335. 'grass.ico'),
  336. wx.BITMAP_TYPE_ICO))
  337. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  338. self.extBox = StaticBox(
  339. parent=self.panel, id=wx.ID_ANY, label=" %s " %
  340. _("List of installed extensions"))
  341. self.extList = CheckListExtension(parent=self.panel)
  342. # buttons
  343. self.btnUninstall = Button(
  344. parent=self.panel,
  345. id=wx.ID_REMOVE,
  346. label=_("Uninstall"))
  347. self.btnUninstall.SetToolTip(
  348. _("Uninstall selected Addons extensions"))
  349. self.btnUpdate = Button(
  350. parent=self.panel,
  351. id=wx.ID_REFRESH,
  352. label=_("Reinstall"))
  353. self.btnUpdate.SetToolTip(
  354. _("Reinstall selected Addons extensions"))
  355. self.btnClose = Button(parent=self.panel, id=wx.ID_CLOSE)
  356. self.btnUninstall.Bind(wx.EVT_BUTTON, self.OnUninstall)
  357. self.btnUpdate.Bind(wx.EVT_BUTTON, self.OnUpdate)
  358. self.btnClose.Bind(wx.EVT_BUTTON, lambda evt: self.Close())
  359. self._layout()
  360. def _layout(self):
  361. """Do layout"""
  362. sizer = wx.BoxSizer(wx.VERTICAL)
  363. extSizer = wx.StaticBoxSizer(self.extBox, wx.HORIZONTAL)
  364. extSizer.Add(self.extList, proportion=1,
  365. flag=wx.ALL | wx.EXPAND, border=1)
  366. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  367. btnSizer.Add(self.btnClose, proportion=0,
  368. flag=wx.RIGHT, border=5)
  369. btnSizer.Add(
  370. self.btnUpdate,
  371. proportion=0,
  372. flag=wx.RIGHT,
  373. border=5)
  374. btnSizer.Add(self.btnUninstall, proportion=0)
  375. sizer.Add(extSizer, proportion=1,
  376. flag=wx.ALL | wx.EXPAND, border=3)
  377. sizer.Add(btnSizer, proportion=0,
  378. flag=wx.ALIGN_RIGHT | wx.ALL, border=5)
  379. self.panel.SetSizer(sizer)
  380. sizer.Fit(self.panel)
  381. self.Layout()
  382. def _getSelectedExtensions(self):
  383. eList = self.extList.GetExtensions()
  384. if not eList:
  385. GMessage(_("No extension selected. "
  386. "Operation canceled."),
  387. parent=self)
  388. return []
  389. return eList
  390. def OnUninstall(self, event):
  391. """Uninstall selected extensions"""
  392. eList = self._getSelectedExtensions()
  393. if not eList:
  394. return
  395. for ext in eList:
  396. files = RunCommand(
  397. 'g.extension',
  398. parent=self,
  399. read=True,
  400. quiet=True,
  401. extension=ext,
  402. operation='remove').splitlines()
  403. if len(files) > 10:
  404. files = files[:10]
  405. files.append('...')
  406. dlg = wx.MessageDialog(
  407. parent=self,
  408. message=_(
  409. "List of files to be removed:\n%(files)s\n\n"
  410. "Do you want really to remove <%(ext)s> extension?") %
  411. {'files': os.linesep.join(files),
  412. 'ext': ext},
  413. caption=_("Remove extension"),
  414. style=wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
  415. if dlg.ShowModal() == wx.ID_YES:
  416. RunCommand('g.extension', flags='f', parent=self, quiet=True,
  417. extension=ext, operation='remove')
  418. self.extList.LoadData()
  419. # update prompt
  420. globalvar.UpdateGRASSAddOnCommands(eList)
  421. toolboxesOutdated()
  422. def OnUpdate(self, event):
  423. """Update selected extensions"""
  424. eList = self._getSelectedExtensions()
  425. if not eList:
  426. return
  427. log = self.parent.GetLogWindow()
  428. for ext in eList:
  429. log.RunCmd(['g.extension', 'extension=%s' % ext,
  430. 'operation=add'])
  431. class CheckListExtension(GListCtrl):
  432. """List of mapset/owner/group"""
  433. def __init__(self, parent):
  434. GListCtrl.__init__(self, parent)
  435. # load extensions
  436. self.InsertColumn(0, _('Extension'))
  437. self.LoadData()
  438. def LoadData(self):
  439. """Load data into list"""
  440. self.DeleteAllItems()
  441. for ext in RunCommand('g.extension',
  442. quiet=True, parent=self, read=True,
  443. flags='a').splitlines():
  444. if ext:
  445. self.InsertItem(self.GetItemCount(), ext)
  446. def GetExtensions(self):
  447. """Get extensions to be un-installed
  448. """
  449. extList = list()
  450. for i in range(self.GetItemCount()):
  451. if self.IsChecked(i):
  452. name = self.GetItemText(i)
  453. if name:
  454. extList.append(name)
  455. return extList