frame.py 71 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933
  1. """!
  2. @package gmodeler.frame
  3. @brief wxGUI Graphical Modeler for creating, editing, and managing models
  4. Classes:
  5. - frame::ModelToolbar
  6. - frame::ModelFrame
  7. - frame::ModelCanvas
  8. - frame::ModelEvtHandler
  9. - frame::ModelListCtrl
  10. - frame::VariablePanel
  11. - frame::ValiableListCtrl
  12. - frame::ItemPanel
  13. - frame::ItemListCtrl
  14. - frame::ItemCheckListCtrl
  15. (C) 2010-2011 by the GRASS Development Team
  16. This program is free software under the GNU General Public License
  17. (>=v2). Read the file COPYING that comes with GRASS for details.
  18. @author Martin Landa <landa.martin gmail.com>
  19. """
  20. import os
  21. import sys
  22. import time
  23. import stat
  24. import textwrap
  25. import tempfile
  26. import copy
  27. import re
  28. import wx
  29. from wx.lib import ogl
  30. import wx.lib.flatnotebook as FN
  31. import wx.lib.mixins.listctrl as listmix
  32. from core import globalvar
  33. from gui_core.widgets import GNotebook
  34. from gui_core.goutput import GMConsole
  35. from core.debug import Debug
  36. from core.gcmd import GMessage, GException, GWarning, GError, RunCommand
  37. from gui_core.dialogs import GetImageHandlers
  38. from gui_core.preferences import PreferencesBaseDialog
  39. from core.settings import UserSettings
  40. from core.menudata import MenuData
  41. from gui_core.toolbars import BaseToolbar
  42. from gui_core.menu import Menu
  43. from gmodeler.menudata import ModelerData
  44. from icons.icon import Icons
  45. from gui_core.forms import GUI
  46. from gmodeler.model import *
  47. from gmodeler.dialogs import *
  48. from grass.script import core as grass
  49. class ModelToolbar(BaseToolbar):
  50. """!Graphical modeler toolbaro (see gmodeler.py)
  51. """
  52. def __init__(self, parent):
  53. BaseToolbar.__init__(self, parent)
  54. self.InitToolbar(self._toolbarData())
  55. # realize the toolbar
  56. self.Realize()
  57. def _toolbarData(self):
  58. """!Toolbar data"""
  59. icons = Icons['modeler']
  60. return self._getToolbarData((('new', icons['new'],
  61. self.parent.OnModelNew),
  62. ('open', icons['open'],
  63. self.parent.OnModelOpen),
  64. ('save', icons['save'],
  65. self.parent.OnModelSave),
  66. ('image', icons['toImage'],
  67. self.parent.OnExportImage),
  68. ('python', icons['toPython'],
  69. self.parent.OnExportPython),
  70. (None, ),
  71. ('action', icons['actionAdd'],
  72. self.parent.OnAddAction),
  73. ('data', icons['dataAdd'],
  74. self.parent.OnAddData),
  75. ('relation', icons['relation'],
  76. self.parent.OnDefineRelation),
  77. ('loop', icons['loop'],
  78. self.parent.OnDefineLoop),
  79. (None, ),
  80. ('redraw', icons['redraw'],
  81. self.parent.OnCanvasRefresh),
  82. ('validate', icons['validate'],
  83. self.parent.OnValidateModel),
  84. ('run', icons['run'],
  85. self.parent.OnRunModel),
  86. (None, ),
  87. ("variables", icons['variables'],
  88. self.parent.OnVariables),
  89. ("settings", icons['settings'],
  90. self.parent.OnPreferences),
  91. ("help", Icons['misc']['help'],
  92. self.parent.OnHelp),
  93. (None, ),
  94. ('quit', icons['quit'],
  95. self.parent.OnCloseWindow))
  96. )
  97. class ModelFrame(wx.Frame):
  98. def __init__(self, parent, id = wx.ID_ANY,
  99. title = _("GRASS GIS Graphical Modeler"), **kwargs):
  100. """!Graphical modeler main window
  101. @param parent parent window
  102. @param id window id
  103. @param title window title
  104. @param kwargs wx.Frames' arguments
  105. """
  106. self.parent = parent
  107. self.searchDialog = None # module search dialog
  108. self.baseTitle = title
  109. self.modelFile = None # loaded model
  110. self.modelChanged = False
  111. self.cursors = {
  112. "default" : wx.StockCursor(wx.CURSOR_ARROW),
  113. "cross" : wx.StockCursor(wx.CURSOR_CROSS),
  114. }
  115. wx.Frame.__init__(self, parent = parent, id = id, title = title, **kwargs)
  116. self.SetName("Modeler")
  117. self.SetIcon(wx.Icon(os.path.join(globalvar.ETCICONDIR, 'grass.ico'), wx.BITMAP_TYPE_ICO))
  118. self.menubar = Menu(parent = self, data = ModelerData())
  119. self.SetMenuBar(self.menubar)
  120. self.toolbar = ModelToolbar(parent = self)
  121. self.SetToolBar(self.toolbar)
  122. self.statusbar = self.CreateStatusBar(number = 1)
  123. self.notebook = GNotebook(parent = self,
  124. style = FN.FNB_FANCY_TABS | FN.FNB_BOTTOM |
  125. FN.FNB_NO_NAV_BUTTONS | FN.FNB_NO_X_BUTTON)
  126. self.canvas = ModelCanvas(self)
  127. self.canvas.SetBackgroundColour(wx.WHITE)
  128. self.canvas.SetCursor(self.cursors["default"])
  129. self.model = Model(self.canvas)
  130. self.variablePanel = VariablePanel(parent = self)
  131. self.itemPanel = ItemPanel(parent = self)
  132. self.goutput = GMConsole(parent = self, notebook = self.notebook)
  133. self.notebook.AddPage(page = self.canvas, text=_('Model'), name = 'model')
  134. self.notebook.AddPage(page = self.itemPanel, text=_('Items'), name = 'items')
  135. self.notebook.AddPage(page = self.variablePanel, text=_('Variables'), name = 'variables')
  136. self.notebook.AddPage(page = self.goutput, text=_('Command output'), name = 'output')
  137. wx.CallAfter(self.notebook.SetSelectionByName, 'model')
  138. wx.CallAfter(self.ModelChanged, False)
  139. self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
  140. self.Bind(wx.EVT_SIZE, self.OnSize)
  141. self._layout()
  142. self.SetMinSize((475, 300))
  143. self.SetSize((640, 480))
  144. # fix goutput's pane size
  145. if self.goutput:
  146. self.goutput.SetSashPosition(int(self.GetSize()[1] * .75))
  147. def _layout(self):
  148. """!Do layout"""
  149. sizer = wx.BoxSizer(wx.VERTICAL)
  150. sizer.Add(item = self.notebook, proportion = 1,
  151. flag = wx.EXPAND)
  152. self.SetAutoLayout(True)
  153. self.SetSizer(sizer)
  154. sizer.Fit(self)
  155. self.Layout()
  156. def _addEvent(self, item):
  157. """!Add event to item"""
  158. evthandler = ModelEvtHandler(self.statusbar,
  159. self)
  160. evthandler.SetShape(item)
  161. evthandler.SetPreviousHandler(item.GetEventHandler())
  162. item.SetEventHandler(evthandler)
  163. def GetCanvas(self):
  164. """!Get canvas"""
  165. return self.canvas
  166. def GetModel(self):
  167. """!Get model"""
  168. return self.model
  169. def ModelChanged(self, changed = True):
  170. """!Update window title"""
  171. self.modelChanged = changed
  172. if self.modelFile:
  173. if self.modelChanged:
  174. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile) + '*')
  175. else:
  176. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile))
  177. else:
  178. self.SetTitle(self.baseTitle)
  179. def OnVariables(self, event):
  180. """!Switch to variables page"""
  181. self.notebook.SetSelectionByName('variables')
  182. def OnRemoveItem(self, event):
  183. """!Remove shape
  184. """
  185. self.GetCanvas().RemoveSelected()
  186. def OnCanvasRefresh(self, event):
  187. """!Refresh canvas"""
  188. self.SetStatusText(_("Redrawing model..."), 0)
  189. self.GetCanvas().Refresh()
  190. self.SetStatusText("", 0)
  191. def OnCmdRun(self, event):
  192. """!Run command"""
  193. try:
  194. action = self.GetModel().GetItems()[event.pid]
  195. if hasattr(action, "task"):
  196. action.Update(running = True)
  197. except IndexError:
  198. pass
  199. def OnCmdPrepare(self, event):
  200. """!Prepare for running command"""
  201. event.onPrepare(item = event.userData['item'],
  202. params = event.userData['params'])
  203. def OnCmdDone(self, event):
  204. """!Command done (or aborted)"""
  205. try:
  206. action = self.GetModel().GetItems()[event.pid]
  207. if hasattr(action, "task"):
  208. action.Update(running = True)
  209. except IndexError:
  210. pass
  211. def OnCloseWindow(self, event):
  212. """!Close window"""
  213. if self.modelChanged and \
  214. UserSettings.Get(group='manager', key='askOnQuit', subkey='enabled'):
  215. if self.modelFile:
  216. message = _("Do you want to save changes in the model?")
  217. else:
  218. message = _("Do you want to store current model settings "
  219. "to model file?")
  220. # ask user to save current settings
  221. dlg = wx.MessageDialog(self,
  222. message = message,
  223. caption=_("Quit Graphical Modeler"),
  224. style = wx.YES_NO | wx.YES_DEFAULT |
  225. wx.CANCEL | wx.ICON_QUESTION | wx.CENTRE)
  226. ret = dlg.ShowModal()
  227. if ret == wx.ID_YES:
  228. if not self.modelFile:
  229. self.OnWorkspaceSaveAs()
  230. else:
  231. self.WriteModelFile(self.modelFile)
  232. elif ret == wx.ID_CANCEL:
  233. dlg.Destroy()
  234. return
  235. dlg.Destroy()
  236. self.Destroy()
  237. def OnSize(self, event):
  238. """Window resized, save to the model"""
  239. self.ModelChanged()
  240. event.Skip()
  241. def OnPreferences(self, event):
  242. """!Open preferences dialog"""
  243. dlg = PreferencesDialog(parent = self)
  244. dlg.CenterOnParent()
  245. dlg.ShowModal()
  246. self.canvas.Refresh()
  247. def OnHelp(self, event):
  248. """!Show help"""
  249. if self.parent and self.parent.GetName() == 'LayerManager':
  250. log = self.parent.GetLogWindow()
  251. log.RunCmd(['g.manual',
  252. 'entry=wxGUI.Modeler'])
  253. else:
  254. RunCommand('g.manual',
  255. quiet = True,
  256. entry = 'wxGUI.Modeler')
  257. def OnModelProperties(self, event):
  258. """!Model properties dialog"""
  259. dlg = PropertiesDialog(parent = self)
  260. dlg.CentreOnParent()
  261. properties = self.model.GetProperties()
  262. dlg.Init(properties)
  263. if dlg.ShowModal() == wx.ID_OK:
  264. self.ModelChanged()
  265. for key, value in dlg.GetValues().iteritems():
  266. properties[key] = value
  267. for action in self.model.GetItems(objType = ModelAction):
  268. action.GetTask().set_flag('overwrite', properties['overwrite'])
  269. dlg.Destroy()
  270. def OnDeleteData(self, event):
  271. """!Delete intermediate data"""
  272. rast, vect, rast3d, msg = self.model.GetIntermediateData()
  273. if not rast and not vect and not rast3d:
  274. GMessage(parent = self,
  275. message = _('Nothing to delete.'))
  276. return
  277. dlg = wx.MessageDialog(parent = self,
  278. message= _("Do you want to permanently delete data?%s" % msg),
  279. caption=_("Delete intermediate data?"),
  280. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  281. ret = dlg.ShowModal()
  282. if ret == wx.ID_YES:
  283. dlg.Destroy()
  284. if rast:
  285. self.goutput.RunCmd(['g.remove', 'rast=%s' %','.join(rast)])
  286. if rast3d:
  287. self.goutput.RunCmd(['g.remove', 'rast3d=%s' %','.join(rast3d)])
  288. if vect:
  289. self.goutput.RunCmd(['g.remove', 'vect=%s' %','.join(vect)])
  290. self.SetStatusText(_("%d maps deleted from current mapset") % \
  291. int(len(rast) + len(rast3d) + len(vect)))
  292. return
  293. dlg.Destroy()
  294. def OnModelNew(self, event):
  295. """!Create new model"""
  296. Debug.msg(4, "ModelFrame.OnModelNew():")
  297. # ask user to save current model
  298. if self.modelFile and self.modelChanged:
  299. self.OnModelSave()
  300. elif self.modelFile is None and \
  301. (self.model.GetNumItems() > 0 or len(self.model.GetData()) > 0):
  302. dlg = wx.MessageDialog(self, message=_("Current model is not empty. "
  303. "Do you want to store current settings "
  304. "to model file?"),
  305. caption=_("Create new model?"),
  306. style=wx.YES_NO | wx.YES_DEFAULT |
  307. wx.CANCEL | wx.ICON_QUESTION)
  308. ret = dlg.ShowModal()
  309. if ret == wx.ID_YES:
  310. self.OnModelSaveAs()
  311. elif ret == wx.ID_CANCEL:
  312. dlg.Destroy()
  313. return
  314. dlg.Destroy()
  315. # delete all items
  316. self.canvas.GetDiagram().DeleteAllShapes()
  317. self.model.Reset()
  318. self.canvas.Refresh()
  319. self.itemPanel.Update()
  320. self.variablePanel.Reset()
  321. # no model file loaded
  322. self.modelFile = None
  323. self.modelChanged = False
  324. self.SetTitle(self.baseTitle)
  325. def OnModelOpen(self, event):
  326. """!Load model from file"""
  327. filename = ''
  328. dlg = wx.FileDialog(parent = self, message=_("Choose model file"),
  329. defaultDir = os.getcwd(),
  330. wildcard=_("GRASS Model File (*.gxm)|*.gxm"))
  331. if dlg.ShowModal() == wx.ID_OK:
  332. filename = dlg.GetPath()
  333. if not filename:
  334. return
  335. Debug.msg(4, "ModelFrame.OnModelOpen(): filename=%s" % filename)
  336. # close current model
  337. self.OnModelClose()
  338. self.LoadModelFile(filename)
  339. self.modelFile = filename
  340. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile))
  341. self.SetStatusText(_('%(items)d items (%(actions)d actions) loaded into model') % \
  342. { 'items' : self.model.GetNumItems(),
  343. 'actions' : self.model.GetNumItems(actionOnly = True) }, 0)
  344. def OnModelSave(self, event = None):
  345. """!Save model to file"""
  346. if self.modelFile and self.modelChanged:
  347. dlg = wx.MessageDialog(self, message=_("Model file <%s> already exists. "
  348. "Do you want to overwrite this file?") % \
  349. self.modelFile,
  350. caption=_("Save model"),
  351. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  352. if dlg.ShowModal() == wx.ID_NO:
  353. dlg.Destroy()
  354. else:
  355. Debug.msg(4, "ModelFrame.OnModelSave(): filename=%s" % self.modelFile)
  356. self.WriteModelFile(self.modelFile)
  357. self.SetStatusText(_('File <%s> saved') % self.modelFile, 0)
  358. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile))
  359. elif not self.modelFile:
  360. self.OnModelSaveAs(None)
  361. def OnModelSaveAs(self, event):
  362. """!Create model to file as"""
  363. filename = ''
  364. dlg = wx.FileDialog(parent = self,
  365. message = _("Choose file to save current model"),
  366. defaultDir = os.getcwd(),
  367. wildcard=_("GRASS Model File (*.gxm)|*.gxm"),
  368. style=wx.FD_SAVE)
  369. if dlg.ShowModal() == wx.ID_OK:
  370. filename = dlg.GetPath()
  371. if not filename:
  372. return
  373. # check for extension
  374. if filename[-4:] != ".gxm":
  375. filename += ".gxm"
  376. if os.path.exists(filename):
  377. dlg = wx.MessageDialog(parent = self,
  378. message=_("Model file <%s> already exists. "
  379. "Do you want to overwrite this file?") % filename,
  380. caption=_("File already exists"),
  381. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  382. if dlg.ShowModal() != wx.ID_YES:
  383. dlg.Destroy()
  384. return
  385. Debug.msg(4, "GMFrame.OnModelSaveAs(): filename=%s" % filename)
  386. self.WriteModelFile(filename)
  387. self.modelFile = filename
  388. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile))
  389. self.SetStatusText(_('File <%s> saved') % self.modelFile, 0)
  390. def OnModelClose(self, event = None):
  391. """!Close model file"""
  392. Debug.msg(4, "ModelFrame.OnModelClose(): file=%s" % self.modelFile)
  393. # ask user to save current model
  394. if self.modelFile and self.modelChanged:
  395. self.OnModelSave()
  396. elif self.modelFile is None and \
  397. (self.model.GetNumItems() > 0 or len(self.model.GetData()) > 0):
  398. dlg = wx.MessageDialog(self, message=_("Current model is not empty. "
  399. "Do you want to store current settings "
  400. "to model file?"),
  401. caption=_("Create new model?"),
  402. style=wx.YES_NO | wx.YES_DEFAULT |
  403. wx.CANCEL | wx.ICON_QUESTION)
  404. ret = dlg.ShowModal()
  405. if ret == wx.ID_YES:
  406. self.OnModelSaveAs()
  407. elif ret == wx.ID_CANCEL:
  408. dlg.Destroy()
  409. return
  410. dlg.Destroy()
  411. self.modelFile = None
  412. self.SetTitle(self.baseTitle)
  413. self.canvas.GetDiagram().DeleteAllShapes()
  414. self.model.Reset()
  415. self.canvas.Refresh()
  416. def OnRunModel(self, event):
  417. """!Run entire model"""
  418. self.model.Run(self.goutput, self.OnDone, parent = self)
  419. def OnDone(self, cmd, returncode):
  420. """!Computation finished"""
  421. self.SetStatusText('', 0)
  422. # restore original files
  423. if hasattr(self.model, "fileInput"):
  424. for finput in self.model.fileInput:
  425. data = self.model.fileInput[finput]
  426. if not data:
  427. continue
  428. fd = open(finput, "w")
  429. try:
  430. fd.write(data)
  431. finally:
  432. fd.close()
  433. del self.model.fileInput
  434. def OnValidateModel(self, event, showMsg = True):
  435. """!Validate entire model"""
  436. if self.model.GetNumItems() < 1:
  437. GMessage(parent = self,
  438. message = _('Model is empty. Nothing to validate.'))
  439. return
  440. self.SetStatusText(_('Validating model...'), 0)
  441. errList = self.model.Validate()
  442. self.SetStatusText('', 0)
  443. if errList:
  444. GWarning(parent = self,
  445. message = _('Model is not valid.\n\n%s') % '\n'.join(errList))
  446. else:
  447. GMessage(parent = self,
  448. message = _('Model is valid.'))
  449. def OnExportImage(self, event):
  450. """!Export model to image (default image)
  451. """
  452. xminImg = 0
  453. xmaxImg = 0
  454. yminImg = 0
  455. ymaxImg = 0
  456. # get current size of canvas
  457. for shape in self.canvas.GetDiagram().GetShapeList():
  458. w, h = shape.GetBoundingBoxMax()
  459. x = shape.GetX()
  460. y = shape.GetY()
  461. xmin = x - w / 2
  462. xmax = x + w / 2
  463. ymin = y - h / 2
  464. ymax = y + h / 2
  465. if xmin < xminImg:
  466. xminImg = xmin
  467. if xmax > xmaxImg:
  468. xmaxImg = xmax
  469. if ymin < yminImg:
  470. yminImg = ymin
  471. if ymax > ymaxImg:
  472. ymaxImg = ymax
  473. size = wx.Size(int(xmaxImg - xminImg) + 50,
  474. int(ymaxImg - yminImg) + 50)
  475. bitmap = wx.EmptyBitmap(width = size.width, height = size.height)
  476. filetype, ltype = GetImageHandlers(wx.ImageFromBitmap(bitmap))
  477. dlg = wx.FileDialog(parent = self,
  478. message = _("Choose a file name to save the image (no need to add extension)"),
  479. defaultDir = "",
  480. defaultFile = "",
  481. wildcard = filetype,
  482. style=wx.SAVE | wx.FD_OVERWRITE_PROMPT)
  483. if dlg.ShowModal() == wx.ID_OK:
  484. path = dlg.GetPath()
  485. if not path:
  486. dlg.Destroy()
  487. return
  488. base, ext = os.path.splitext(path)
  489. fileType = ltype[dlg.GetFilterIndex()]['type']
  490. extType = ltype[dlg.GetFilterIndex()]['ext']
  491. if ext != extType:
  492. path = base + '.' + extType
  493. dc = wx.MemoryDC(bitmap)
  494. dc.SetBackground(wx.WHITE_BRUSH)
  495. dc.SetBackgroundMode(wx.SOLID)
  496. dc.BeginDrawing()
  497. self.canvas.GetDiagram().Clear(dc)
  498. self.canvas.GetDiagram().Redraw(dc)
  499. dc.EndDrawing()
  500. bitmap.SaveFile(path, fileType)
  501. self.SetStatusText(_("Model exported to <%s>") % path)
  502. dlg.Destroy()
  503. def OnExportPython(self, event):
  504. """!Export model to Python script"""
  505. filename = ''
  506. dlg = wx.FileDialog(parent = self,
  507. message = _("Choose file to save"),
  508. defaultDir = os.getcwd(),
  509. wildcard=_("Python script (*.py)|*.py"),
  510. style=wx.FD_SAVE)
  511. if dlg.ShowModal() == wx.ID_OK:
  512. filename = dlg.GetPath()
  513. if not filename:
  514. return
  515. # check for extension
  516. if filename[-3:] != ".py":
  517. filename += ".py"
  518. if os.path.exists(filename):
  519. dlg = wx.MessageDialog(self, message=_("File <%s> already exists. "
  520. "Do you want to overwrite this file?") % filename,
  521. caption=_("Save file"),
  522. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  523. if dlg.ShowModal() == wx.ID_NO:
  524. dlg.Destroy()
  525. return
  526. dlg.Destroy()
  527. fd = open(filename, "w")
  528. try:
  529. WritePythonFile(fd, self.model)
  530. finally:
  531. fd.close()
  532. # executable file
  533. os.chmod(filename, stat.S_IRWXU | stat.S_IWUSR)
  534. self.SetStatusText(_("Model exported to <%s>") % filename)
  535. def OnDefineRelation(self, event):
  536. """!Define relation between data and action items"""
  537. self.canvas.SetCursor(self.cursors["cross"])
  538. self.defineRelation = { 'from' : None,
  539. 'to' : None }
  540. def OnDefineLoop(self, event):
  541. """!Define new loop in the model"""
  542. self.ModelChanged()
  543. width, height = self.canvas.GetSize()
  544. loop = ModelLoop(self, x = width/2, y = height/2,
  545. id = self.model.GetNumItems() + 1)
  546. self.canvas.diagram.AddShape(loop)
  547. loop.Show(True)
  548. self._addEvent(loop)
  549. self.model.AddItem(loop)
  550. self.canvas.Refresh()
  551. def OnDefineCondition(self, event):
  552. """!Define new condition in the model"""
  553. self.ModelChanged()
  554. width, height = self.canvas.GetSize()
  555. cond = ModelCondition(self, x = width/2, y = height/2,
  556. id = self.model.GetNumItems() + 1)
  557. self.canvas.diagram.AddShape(cond)
  558. cond.Show(True)
  559. self._addEvent(cond)
  560. self.model.AddItem(cond)
  561. self.canvas.Refresh()
  562. def OnAddAction(self, event):
  563. """!Add action to model"""
  564. if self.searchDialog is None:
  565. self.searchDialog = ModelSearchDialog(self)
  566. self.searchDialog.CentreOnParent()
  567. else:
  568. self.searchDialog.Reset()
  569. if self.searchDialog.ShowModal() == wx.ID_CANCEL:
  570. self.searchDialog.Hide()
  571. return
  572. cmd = self.searchDialog.GetCmd()
  573. self.searchDialog.Hide()
  574. self.ModelChanged()
  575. # add action to canvas
  576. width, height = self.canvas.GetSize()
  577. action = ModelAction(self.model, cmd = cmd, x = width/2, y = height/2,
  578. id = self.model.GetNextId())
  579. overwrite = self.model.GetProperties().get('overwrite', None)
  580. if overwrite is not None:
  581. action.GetTask().set_flag('overwrite', overwrite)
  582. self.canvas.diagram.AddShape(action)
  583. action.Show(True)
  584. self._addEvent(action)
  585. self.model.AddItem(action)
  586. self.itemPanel.Update()
  587. self.canvas.Refresh()
  588. time.sleep(.1)
  589. # show properties dialog
  590. win = action.GetPropDialog()
  591. if not win:
  592. if action.IsValid():
  593. self.GetOptData(dcmd = action.GetLog(string = False), layer = action,
  594. params = action.GetParams(), propwin = None)
  595. else:
  596. GUI(parent = self, show = True).ParseCommand(action.GetLog(string = False),
  597. completed = (self.GetOptData, action, action.GetParams()))
  598. elif win and not win.IsShown():
  599. win.Show()
  600. if win:
  601. win.Raise()
  602. def OnAddData(self, event):
  603. """!Add data item to model
  604. """
  605. # add action to canvas
  606. width, height = self.canvas.GetSize()
  607. data = ModelData(self, x = width/2, y = height/2)
  608. dlg = ModelDataDialog(parent = self, shape = data)
  609. data.SetPropDialog(dlg)
  610. dlg.CentreOnParent()
  611. ret = dlg.ShowModal()
  612. dlg.Destroy()
  613. if ret != wx.ID_OK:
  614. return
  615. data.Update()
  616. self.canvas.diagram.AddShape(data)
  617. data.Show(True)
  618. self.ModelChanged()
  619. self._addEvent(data)
  620. self.model.AddItem(data)
  621. self.canvas.Refresh()
  622. def OnHelp(self, event):
  623. """!Display manual page"""
  624. grass.run_command('g.manual',
  625. entry = 'wxGUI.Modeler')
  626. def OnAbout(self, event):
  627. """!Display About window"""
  628. info = wx.AboutDialogInfo()
  629. info.SetIcon(wx.Icon(os.path.join(globalvar.ETCICONDIR, 'grass.ico'), wx.BITMAP_TYPE_ICO))
  630. info.SetName(_('wxGUI Graphical Modeler'))
  631. info.SetWebSite('http://grass.osgeo.org')
  632. year = grass.version()['date']
  633. info.SetDescription(_('(C) 2010-%s by the GRASS Development Team\n\n') % year +
  634. '\n'.join(textwrap.wrap(_('This program is free software under the GNU General Public License'
  635. '(>=v2). Read the file COPYING that comes with GRASS for details.'), 75)))
  636. wx.AboutBox(info)
  637. def GetOptData(self, dcmd, layer, params, propwin):
  638. """!Process action data"""
  639. if params: # add data items
  640. width, height = self.canvas.GetSize()
  641. x = [width/2 + 200, width/2 - 200]
  642. for p in params['params']:
  643. if p.get('prompt', '') in ('raster', 'vector', 'raster3d') and \
  644. (p.get('value', None) or \
  645. (p.get('age', 'old') != 'old' and p.get('required', 'no') == 'yes')):
  646. data = layer.FindData(p.get('name', ''))
  647. if data:
  648. data.SetValue(p.get('value', ''))
  649. data.Update()
  650. continue
  651. data = self.model.FindData(p.get('value', ''),
  652. p.get('prompt', ''))
  653. if data:
  654. if p.get('age', 'old') == 'old':
  655. rel = ModelRelation(parent = self, fromShape = data,
  656. toShape = layer, param = p.get('name', ''))
  657. else:
  658. rel = ModelRelation(parent = self, fromShape = layer,
  659. toShape = data, param = p.get('name', ''))
  660. layer.AddRelation(rel)
  661. data.AddRelation(rel)
  662. self.AddLine(rel)
  663. data.Update()
  664. continue
  665. data = ModelData(self, value = p.get('value', ''),
  666. prompt = p.get('prompt', ''),
  667. x = x.pop(), y = height/2)
  668. self._addEvent(data)
  669. self.canvas.diagram.AddShape(data)
  670. data.Show(True)
  671. if p.get('age', 'old') == 'old':
  672. rel = ModelRelation(parent = self, fromShape = data,
  673. toShape = layer, param = p.get('name', ''))
  674. else:
  675. rel = ModelRelation(parent = self, fromShape = layer,
  676. toShape = data, param = p.get('name', ''))
  677. layer.AddRelation(rel)
  678. data.AddRelation(rel)
  679. self.AddLine(rel)
  680. data.Update()
  681. # valid / parameterized ?
  682. layer.SetValid(params)
  683. self.canvas.Refresh()
  684. if dcmd:
  685. layer.SetProperties(params, propwin)
  686. self.SetStatusText(layer.GetLog(), 0)
  687. def AddLine(self, rel):
  688. """!Add connection between model objects
  689. @param rel relation
  690. """
  691. fromShape = rel.GetFrom()
  692. toShape = rel.GetTo()
  693. rel.SetCanvas(self)
  694. rel.SetPen(wx.BLACK_PEN)
  695. rel.SetBrush(wx.BLACK_BRUSH)
  696. rel.AddArrow(ogl.ARROW_ARROW)
  697. points = rel.GetControlPoints()
  698. rel.MakeLineControlPoints(2)
  699. if points:
  700. for x, y in points:
  701. rel.InsertLineControlPoint(point = wx.RealPoint(x, y))
  702. self._addEvent(rel)
  703. try:
  704. fromShape.AddLine(rel, toShape)
  705. except TypeError:
  706. pass # bug when connecting ModelCondition and ModelLoop - to be fixed
  707. self.canvas.diagram.AddShape(rel)
  708. rel.Show(True)
  709. def LoadModelFile(self, filename):
  710. """!Load model definition stored in GRASS Model XML file (gxm)
  711. """
  712. try:
  713. self.model.LoadModel(filename)
  714. except GException, e:
  715. GError(parent = self,
  716. message = _("Reading model file <%s> failed.\n"
  717. "Invalid file, unable to parse XML document.") % filename)
  718. self.modelFile = filename
  719. self.SetTitle(self.baseTitle + " - " + os.path.basename(self.modelFile))
  720. self.SetStatusText(_("Please wait, loading model..."), 0)
  721. # load actions
  722. for item in self.model.GetItems(objType = ModelAction):
  723. self._addEvent(item)
  724. self.canvas.diagram.AddShape(item)
  725. item.Show(True)
  726. # relations/data
  727. for rel in item.GetRelations():
  728. if rel.GetFrom() == item:
  729. dataItem = rel.GetTo()
  730. else:
  731. dataItem = rel.GetFrom()
  732. self._addEvent(dataItem)
  733. self.canvas.diagram.AddShape(dataItem)
  734. self.AddLine(rel)
  735. dataItem.Show(True)
  736. # load loops
  737. for item in self.model.GetItems(objType = ModelLoop):
  738. self._addEvent(item)
  739. self.canvas.diagram.AddShape(item)
  740. item.Show(True)
  741. # connect items in the loop
  742. self.DefineLoop(item)
  743. # load conditions
  744. for item in self.model.GetItems(objType = ModelCondition):
  745. self._addEvent(item)
  746. self.canvas.diagram.AddShape(item)
  747. item.Show(True)
  748. # connect items in the condition
  749. self.DefineCondition(item)
  750. # load variables
  751. self.variablePanel.Update()
  752. self.itemPanel.Update()
  753. self.SetStatusText('', 0)
  754. # final updates
  755. for action in self.model.GetItems(objType = ModelAction):
  756. action.SetValid(action.GetParams())
  757. action.Update()
  758. self.canvas.Refresh(True)
  759. def WriteModelFile(self, filename):
  760. """!Save model to model file, recover original file on error.
  761. @return True on success
  762. @return False on failure
  763. """
  764. self.ModelChanged(False)
  765. tmpfile = tempfile.TemporaryFile(mode='w+b')
  766. try:
  767. WriteModelFile(fd = tmpfile, model = self.model)
  768. except StandardError:
  769. GError(parent = self,
  770. message = _("Writing current settings to model file failed."))
  771. return False
  772. try:
  773. mfile = open(filename, "w")
  774. tmpfile.seek(0)
  775. for line in tmpfile.readlines():
  776. mfile.write(line)
  777. except IOError:
  778. wx.MessageBox(parent = self,
  779. message = _("Unable to open file <%s> for writing.") % filename,
  780. caption = _("Error"),
  781. style = wx.OK | wx.ICON_ERROR | wx.CENTRE)
  782. return False
  783. mfile.close()
  784. return True
  785. def DefineLoop(self, loop):
  786. """!Define loop with given list of items"""
  787. parent = loop
  788. items = loop.GetItems()
  789. if not items:
  790. return
  791. # remove defined relations first
  792. for rel in loop.GetRelations():
  793. self.canvas.GetDiagram().RemoveShape(rel)
  794. loop.Clear()
  795. for item in items:
  796. rel = ModelRelation(parent = self, fromShape = parent, toShape = item)
  797. dx = item.GetX() - parent.GetX()
  798. dy = item.GetY() - parent.GetY()
  799. loop.AddRelation(rel)
  800. if dx != 0:
  801. rel.SetControlPoints(((parent.GetX(), parent.GetY() + dy / 2),
  802. (parent.GetX() + dx, parent.GetY() + dy / 2)))
  803. self.AddLine(rel)
  804. parent = item
  805. # close loop
  806. item = loop.GetItems()[-1]
  807. rel = ModelRelation(parent = self, fromShape = item, toShape = loop)
  808. loop.AddRelation(rel)
  809. self.AddLine(rel)
  810. dx = (item.GetX() - loop.GetX()) + loop.GetWidth() / 2 + 50
  811. dy = item.GetHeight() / 2 + 50
  812. rel.MakeLineControlPoints(0)
  813. rel.InsertLineControlPoint(point = wx.RealPoint(loop.GetX() - loop.GetWidth() / 2 ,
  814. loop.GetY()))
  815. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX(),
  816. item.GetY() + item.GetHeight() / 2))
  817. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX(),
  818. item.GetY() + dy))
  819. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX() - dx,
  820. item.GetY() + dy))
  821. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX() - dx,
  822. loop.GetY()))
  823. self.canvas.Refresh()
  824. def DefineCondition(self, condition):
  825. """!Define if-else statement with given list of items"""
  826. parent = condition
  827. items = condition.GetItems()
  828. if not items['if'] and not items['else']:
  829. return
  830. # remove defined relations first
  831. for rel in condition.GetRelations():
  832. self.canvas.GetDiagram().RemoveShape(rel)
  833. condition.Clear()
  834. dxIf = condition.GetX() + condition.GetWidth() / 2
  835. dxElse = condition.GetX() - condition.GetWidth() / 2
  836. dy = condition.GetY()
  837. for branch in items.keys():
  838. for item in items[branch]:
  839. rel = ModelRelation(parent = self, fromShape = parent,
  840. toShape = item)
  841. condition.AddRelation(rel)
  842. self.AddLine(rel)
  843. rel.MakeLineControlPoints(0)
  844. if branch == 'if':
  845. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX() - item.GetWidth() / 2, item.GetY()))
  846. rel.InsertLineControlPoint(point = wx.RealPoint(dxIf, dy))
  847. else:
  848. rel.InsertLineControlPoint(point = wx.RealPoint(dxElse, dy))
  849. rel.InsertLineControlPoint(point = wx.RealPoint(item.GetX() - item.GetWidth() / 2, item.GetY()))
  850. parent = item
  851. self.canvas.Refresh()
  852. class ModelCanvas(ogl.ShapeCanvas):
  853. """!Canvas where model is drawn"""
  854. def __init__(self, parent):
  855. self.parent = parent
  856. ogl.OGLInitialize()
  857. ogl.ShapeCanvas.__init__(self, parent)
  858. self.diagram = ogl.Diagram()
  859. self.SetDiagram(self.diagram)
  860. self.diagram.SetCanvas(self)
  861. self.SetScrollbars(20, 20, 1000/20, 1000/20)
  862. self.Bind(wx.EVT_CHAR, self.OnChar)
  863. def OnChar(self, event):
  864. """!Key pressed"""
  865. kc = event.GetKeyCode()
  866. diagram = self.GetDiagram()
  867. if kc == wx.WXK_DELETE:
  868. self.RemoveSelected()
  869. def RemoveSelected(self):
  870. """!Remove selected shapes"""
  871. self.parent.ModelChanged()
  872. diagram = self.GetDiagram()
  873. for shape in diagram.GetShapeList():
  874. if not shape.Selected():
  875. continue
  876. remList, upList = self.parent.GetModel().RemoveItem(shape)
  877. shape.Select(False)
  878. diagram.RemoveShape(shape)
  879. shape.__del__()
  880. for item in remList:
  881. diagram.RemoveShape(item)
  882. item.__del__()
  883. for item in upList:
  884. item.Update()
  885. self.Refresh()
  886. class ModelEvtHandler(ogl.ShapeEvtHandler):
  887. """!Model event handler class"""
  888. def __init__(self, log, frame):
  889. ogl.ShapeEvtHandler.__init__(self)
  890. self.log = log
  891. self.frame = frame
  892. self.x = self.y = None
  893. def OnLeftClick(self, x, y, keys = 0, attachment = 0):
  894. """!Left mouse button pressed -> select item & update statusbar"""
  895. shape = self.GetShape()
  896. canvas = shape.GetCanvas()
  897. dc = wx.ClientDC(canvas)
  898. canvas.PrepareDC(dc)
  899. if hasattr(self.frame, 'defineRelation'):
  900. drel = self.frame.defineRelation
  901. if drel['from'] is None:
  902. drel['from'] = shape
  903. elif drel['to'] is None:
  904. drel['to'] = shape
  905. rel = ModelRelation(parent = self.frame, fromShape = drel['from'],
  906. toShape = drel['to'])
  907. dlg = ModelRelationDialog(parent = self.frame,
  908. shape = rel)
  909. if dlg.IsValid():
  910. ret = dlg.ShowModal()
  911. if ret == wx.ID_OK:
  912. option = dlg.GetOption()
  913. rel.SetName(option)
  914. drel['from'].AddRelation(rel)
  915. drel['to'].AddRelation(rel)
  916. drel['from'].Update()
  917. params = { 'params' : [{ 'name' : option,
  918. 'value' : drel['from'].GetValue()}] }
  919. drel['to'].MergeParams(params)
  920. self.frame.AddLine(rel)
  921. dlg.Destroy()
  922. del self.frame.defineRelation
  923. if shape.Selected():
  924. shape.Select(False, dc)
  925. else:
  926. redraw = False
  927. shapeList = canvas.GetDiagram().GetShapeList()
  928. toUnselect = list()
  929. for s in shapeList:
  930. if s.Selected():
  931. toUnselect.append(s)
  932. shape.Select(True, dc)
  933. for s in toUnselect:
  934. s.Select(False, dc)
  935. canvas.Refresh(False)
  936. if hasattr(shape, "GetLog"):
  937. self.log.SetStatusText(shape.GetLog(), 0)
  938. else:
  939. self.log.SetStatusText('', 0)
  940. def OnLeftDoubleClick(self, x, y, keys = 0, attachment = 0):
  941. """!Left mouse button pressed (double-click) -> show properties"""
  942. self.OnProperties()
  943. def OnProperties(self, event = None):
  944. """!Show properties dialog"""
  945. self.frame.ModelChanged()
  946. shape = self.GetShape()
  947. if isinstance(shape, ModelAction):
  948. module = GUI(parent = self.frame, show = True).ParseCommand(shape.GetLog(string = False),
  949. completed = (self.frame.GetOptData, shape, shape.GetParams()))
  950. elif isinstance(shape, ModelData):
  951. dlg = ModelDataDialog(parent = self.frame, shape = shape)
  952. shape.SetPropDialog(dlg)
  953. dlg.CentreOnParent()
  954. dlg.Show()
  955. elif isinstance(shape, ModelLoop):
  956. dlg = ModelLoopDialog(parent = self.frame, shape = shape)
  957. dlg.CentreOnParent()
  958. if dlg.ShowModal() == wx.ID_OK:
  959. shape.SetText(dlg.GetCondition())
  960. alist = list()
  961. ids = dlg.GetItems()
  962. for aId in ids['unchecked']:
  963. action = self.frame.GetModel().GetItem(aId)
  964. action.UnSetBlock(shape)
  965. for aId in ids['checked']:
  966. action = self.frame.GetModel().GetItem(aId)
  967. action.SetBlock(shape)
  968. if action:
  969. alist.append(action)
  970. shape.SetItems(alist)
  971. self.frame.DefineLoop(shape)
  972. self.frame.SetStatusText(shape.GetLog(), 0)
  973. self.frame.GetCanvas().Refresh()
  974. dlg.Destroy()
  975. elif isinstance(shape, ModelCondition):
  976. dlg = ModelConditionDialog(parent = self.frame, shape = shape)
  977. dlg.CentreOnParent()
  978. if dlg.ShowModal() == wx.ID_OK:
  979. shape.SetText(dlg.GetCondition())
  980. ids = dlg.GetItems()
  981. for b in ids.keys():
  982. alist = list()
  983. for aId in ids[b]['unchecked']:
  984. action = self.frame.GetModel().GetItem(aId)
  985. action.UnSetBlock(shape)
  986. for aId in ids[b]['checked']:
  987. action = self.frame.GetModel().GetItem(aId)
  988. action.SetBlock(shape)
  989. if action:
  990. alist.append(action)
  991. shape.SetItems(alist, branch = b)
  992. self.frame.DefineCondition(shape)
  993. self.frame.GetCanvas().Refresh()
  994. dlg.Destroy()
  995. def OnBeginDragLeft(self, x, y, keys = 0, attachment = 0):
  996. """!Drag shape (begining)"""
  997. self.frame.ModelChanged()
  998. if self._previousHandler:
  999. self._previousHandler.OnBeginDragLeft(x, y, keys, attachment)
  1000. def OnEndDragLeft(self, x, y, keys = 0, attachment = 0):
  1001. """!Drag shape (end)"""
  1002. if self._previousHandler:
  1003. self._previousHandler.OnEndDragLeft(x, y, keys, attachment)
  1004. shape = self.GetShape()
  1005. if isinstance(shape, ModelLoop):
  1006. self.frame.DefineLoop(shape)
  1007. elif isinstance(shape, ModelCondition):
  1008. self.frame.DefineCondition(shape)
  1009. for mo in shape.GetBlock():
  1010. if isinstance(mo, ModelLoop):
  1011. self.frame.DefineLoop(mo)
  1012. elif isinstance(mo, ModelCondition):
  1013. self.frame.DefineCondition(mo)
  1014. def OnEndSize(self, x, y):
  1015. """!Resize shape"""
  1016. self.frame.ModelChanged()
  1017. if self._previousHandler:
  1018. self._previousHandler.OnEndSize(x, y)
  1019. def OnRightClick(self, x, y, keys = 0, attachment = 0):
  1020. """!Right click -> pop-up menu"""
  1021. if not hasattr (self, "popupID"):
  1022. self.popupID = dict()
  1023. for key in ('remove', 'enable', 'addPoint',
  1024. 'delPoint', 'intermediate', 'props', 'id'):
  1025. self.popupID[key] = wx.NewId()
  1026. # record coordinates
  1027. self.x = x
  1028. self.y = y
  1029. shape = self.GetShape()
  1030. popupMenu = wx.Menu()
  1031. popupMenu.Append(self.popupID['remove'], text=_('Remove'))
  1032. self.frame.Bind(wx.EVT_MENU, self.OnRemove, id = self.popupID['remove'])
  1033. if isinstance(shape, ModelAction) or isinstance(shape, ModelLoop):
  1034. if shape.IsEnabled():
  1035. popupMenu.Append(self.popupID['enable'], text=_('Disable'))
  1036. self.frame.Bind(wx.EVT_MENU, self.OnDisable, id = self.popupID['enable'])
  1037. else:
  1038. popupMenu.Append(self.popupID['enable'], text=_('Enable'))
  1039. self.frame.Bind(wx.EVT_MENU, self.OnEnable, id = self.popupID['enable'])
  1040. if isinstance(shape, ModelRelation):
  1041. popupMenu.AppendSeparator()
  1042. popupMenu.Append(self.popupID['addPoint'], text=_('Add control point'))
  1043. self.frame.Bind(wx.EVT_MENU, self.OnAddPoint, id = self.popupID['addPoint'])
  1044. popupMenu.Append(self.popupID['delPoint'], text=_('Remove control point'))
  1045. self.frame.Bind(wx.EVT_MENU, self.OnRemovePoint, id = self.popupID['delPoint'])
  1046. if len(shape.GetLineControlPoints()) == 2:
  1047. popupMenu.Enable(self.popupID['delPoint'], False)
  1048. if isinstance(shape, ModelData) and '@' not in shape.GetValue():
  1049. popupMenu.AppendSeparator()
  1050. popupMenu.Append(self.popupID['intermediate'], text=_('Intermediate'),
  1051. kind = wx.ITEM_CHECK)
  1052. if self.GetShape().IsIntermediate():
  1053. popupMenu.Check(self.popupID['intermediate'], True)
  1054. self.frame.Bind(wx.EVT_MENU, self.OnIntermediate, id = self.popupID['intermediate'])
  1055. if isinstance(shape, ModelData) or \
  1056. isinstance(shape, ModelAction) or \
  1057. isinstance(shape, ModelLoop):
  1058. popupMenu.AppendSeparator()
  1059. popupMenu.Append(self.popupID['props'], text=_('Properties'))
  1060. self.frame.Bind(wx.EVT_MENU, self.OnProperties, id = self.popupID['props'])
  1061. self.frame.PopupMenu(popupMenu)
  1062. popupMenu.Destroy()
  1063. def OnDisable(self, event):
  1064. """!Disable action"""
  1065. self._onEnable(False)
  1066. def OnEnable(self, event):
  1067. """!Disable action"""
  1068. self._onEnable(True)
  1069. def _onEnable(self, enable):
  1070. shape = self.GetShape()
  1071. shape.Enable(enable)
  1072. self.frame.ModelChanged()
  1073. self.frame.canvas.Refresh()
  1074. def OnAddPoint(self, event):
  1075. """!Add control point"""
  1076. shape = self.GetShape()
  1077. shape.InsertLineControlPoint(point = wx.RealPoint(self.x, self.y))
  1078. shape.ResetShapes()
  1079. shape.Select(True)
  1080. self.frame.ModelChanged()
  1081. self.frame.canvas.Refresh()
  1082. def OnRemovePoint(self, event):
  1083. """!Remove control point"""
  1084. shape = self.GetShape()
  1085. shape.DeleteLineControlPoint()
  1086. shape.Select(False)
  1087. shape.Select(True)
  1088. self.frame.ModelChanged()
  1089. self.frame.canvas.Refresh()
  1090. def OnIntermediate(self, event):
  1091. """!Mark data as intermediate"""
  1092. self.frame.ModelChanged()
  1093. shape = self.GetShape()
  1094. shape.SetIntermediate(event.IsChecked())
  1095. self.frame.canvas.Refresh()
  1096. def OnRemove(self, event):
  1097. """!Remove shape
  1098. """
  1099. self.frame.GetCanvas().RemoveSelected()
  1100. self.frame.itemPanel.Update()
  1101. class ModelListCtrl(wx.ListCtrl,
  1102. listmix.ListCtrlAutoWidthMixin,
  1103. listmix.TextEditMixin,
  1104. listmix.ColumnSorterMixin):
  1105. def __init__(self, parent, columns, id = wx.ID_ANY,
  1106. style = wx.LC_REPORT | wx.BORDER_NONE |
  1107. wx.LC_SORT_ASCENDING |wx.LC_HRULES |
  1108. wx.LC_VRULES, **kwargs):
  1109. """!List of model variables"""
  1110. self.parent = parent
  1111. self.columns = columns
  1112. self.shape = None
  1113. try:
  1114. self.frame = parent.parent
  1115. except AttributeError:
  1116. self.frame = None
  1117. wx.ListCtrl.__init__(self, parent, id = id, style = style, **kwargs)
  1118. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1119. listmix.TextEditMixin.__init__(self)
  1120. listmix.ColumnSorterMixin.__init__(self, 4)
  1121. i = 0
  1122. for col in columns:
  1123. self.InsertColumn(i, col)
  1124. self.SetColumnWidth(i, wx.LIST_AUTOSIZE_USEHEADER)
  1125. i += 1
  1126. self.itemDataMap = {} # requested by sorter
  1127. self.itemCount = 0
  1128. self.Bind(wx.EVT_LIST_BEGIN_LABEL_EDIT, self.OnBeginEdit)
  1129. self.Bind(wx.EVT_LIST_END_LABEL_EDIT, self.OnEndEdit)
  1130. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick)
  1131. self.Bind(wx.EVT_COMMAND_RIGHT_CLICK, self.OnRightUp) #wxMSW
  1132. self.Bind(wx.EVT_RIGHT_UP, self.OnRightUp) #wxGTK
  1133. def OnBeginEdit(self, event):
  1134. """!Editing of item started"""
  1135. event.Allow()
  1136. def OnEndEdit(self, event):
  1137. """!Finish editing of item"""
  1138. pass
  1139. def OnColClick(self, event):
  1140. """!Click on column header (order by)"""
  1141. event.Skip()
  1142. class VariablePanel(wx.Panel):
  1143. def __init__(self, parent, id = wx.ID_ANY,
  1144. **kwargs):
  1145. """!Manage model variables panel
  1146. """
  1147. self.parent = parent
  1148. wx.Panel.__init__(self, parent = parent, id = id, **kwargs)
  1149. self.listBox = wx.StaticBox(parent = self, id = wx.ID_ANY,
  1150. label=" %s " % _("List of variables - right-click to delete"))
  1151. self.list = VariableListCtrl(parent = self,
  1152. columns = [_("Name"), _("Data type"),
  1153. _("Default value"), _("Description")])
  1154. # add new category
  1155. self.addBox = wx.StaticBox(parent = self, id = wx.ID_ANY,
  1156. label = " %s " % _("Add new variable"))
  1157. self.name = wx.TextCtrl(parent = self, id = wx.ID_ANY)
  1158. wx.CallAfter(self.name.SetFocus)
  1159. self.type = wx.Choice(parent = self, id = wx.ID_ANY,
  1160. choices = [_("integer"),
  1161. _("float"),
  1162. _("string"),
  1163. _("raster"),
  1164. _("vector"),
  1165. _("mapset"),
  1166. _("file")])
  1167. self.type.SetSelection(2) # string
  1168. self.value = wx.TextCtrl(parent = self, id = wx.ID_ANY)
  1169. self.desc = wx.TextCtrl(parent = self, id = wx.ID_ANY)
  1170. # buttons
  1171. self.btnAdd = wx.Button(parent = self, id = wx.ID_ADD)
  1172. self.btnAdd.SetToolTipString(_("Add new variable to the model"))
  1173. self.btnAdd.Enable(False)
  1174. # bindings
  1175. self.name.Bind(wx.EVT_TEXT, self.OnText)
  1176. self.value.Bind(wx.EVT_TEXT, self.OnText)
  1177. self.desc.Bind(wx.EVT_TEXT, self.OnText)
  1178. self.btnAdd.Bind(wx.EVT_BUTTON, self.OnAdd)
  1179. self._layout()
  1180. def _layout(self):
  1181. """!Layout dialog"""
  1182. listSizer = wx.StaticBoxSizer(self.listBox, wx.VERTICAL)
  1183. listSizer.Add(item = self.list, proportion = 1,
  1184. flag = wx.EXPAND)
  1185. addSizer = wx.StaticBoxSizer(self.addBox, wx.VERTICAL)
  1186. gridSizer = wx.GridBagSizer(hgap = 5, vgap = 5)
  1187. gridSizer.AddGrowableCol(1)
  1188. gridSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  1189. label = "%s:" % _("Name")),
  1190. flag = wx.ALIGN_CENTER_VERTICAL,
  1191. pos = (0, 0))
  1192. gridSizer.Add(item = self.name,
  1193. pos = (0, 1),
  1194. flag = wx.EXPAND)
  1195. gridSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  1196. label = "%s:" % _("Data type")),
  1197. flag = wx.ALIGN_CENTER_VERTICAL,
  1198. pos = (0, 2))
  1199. gridSizer.Add(item = self.type,
  1200. pos = (0, 3))
  1201. gridSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  1202. label = "%s:" % _("Default value")),
  1203. flag = wx.ALIGN_CENTER_VERTICAL,
  1204. pos = (1, 0))
  1205. gridSizer.Add(item = self.value,
  1206. pos = (1, 1), span = (1, 3),
  1207. flag = wx.EXPAND)
  1208. gridSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  1209. label = "%s:" % _("Description")),
  1210. flag = wx.ALIGN_CENTER_VERTICAL,
  1211. pos = (2, 0))
  1212. gridSizer.Add(item = self.desc,
  1213. pos = (2, 1), span = (1, 3),
  1214. flag = wx.EXPAND)
  1215. addSizer.Add(item = gridSizer,
  1216. flag = wx.EXPAND)
  1217. addSizer.Add(item = self.btnAdd, proportion = 0,
  1218. flag = wx.TOP | wx.ALIGN_RIGHT, border = 5)
  1219. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1220. mainSizer.Add(item = listSizer, proportion = 1,
  1221. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1222. mainSizer.Add(item = addSizer, proportion = 0,
  1223. flag = wx.EXPAND | wx.ALIGN_CENTER |
  1224. wx.LEFT | wx.RIGHT | wx.BOTTOM, border = 5)
  1225. self.SetSizer(mainSizer)
  1226. mainSizer.Fit(self)
  1227. def OnText(self, event):
  1228. """!Text entered"""
  1229. if self.name.GetValue():
  1230. self.btnAdd.Enable()
  1231. else:
  1232. self.btnAdd.Enable(False)
  1233. def OnAdd(self, event):
  1234. """!Add new variable to the list"""
  1235. msg = self.list.Append(self.name.GetValue(),
  1236. self.type.GetStringSelection(),
  1237. self.value.GetValue(),
  1238. self.desc.GetValue())
  1239. self.name.SetValue('')
  1240. self.name.SetFocus()
  1241. if msg:
  1242. GError(parent = self,
  1243. message = msg)
  1244. else:
  1245. self.type.SetSelection(2) # string
  1246. self.value.SetValue('')
  1247. self.desc.SetValue('')
  1248. self.UpdateModelVariables()
  1249. def UpdateModelVariables(self):
  1250. """!Update model variables"""
  1251. variables = dict()
  1252. for values in self.list.GetData().itervalues():
  1253. name = values[0]
  1254. variables[name] = { 'type' : str(values[1]) }
  1255. if values[2]:
  1256. variables[name]['value'] = values[2]
  1257. if values[3]:
  1258. variables[name]['description'] = values[3]
  1259. self.parent.GetModel().SetVariables(variables)
  1260. self.parent.ModelChanged()
  1261. def Update(self):
  1262. """!Reload list of variables"""
  1263. self.list.OnReload(None)
  1264. def Reset(self):
  1265. """!Remove all variables"""
  1266. self.list.DeleteAllItems()
  1267. self.parent.GetModel().SetVariables([])
  1268. class VariableListCtrl(ModelListCtrl):
  1269. def __init__(self, parent, columns, **kwargs):
  1270. """!List of model variables"""
  1271. ModelListCtrl.__init__(self, parent, columns, **kwargs)
  1272. self.SetColumnWidth(2, 200) # default value
  1273. def GetListCtrl(self):
  1274. """!Used by ColumnSorterMixin"""
  1275. return self
  1276. def GetData(self):
  1277. """!Get list data"""
  1278. return self.itemDataMap
  1279. def Populate(self, data):
  1280. """!Populate the list"""
  1281. self.itemDataMap = dict()
  1282. i = 0
  1283. for name, values in data.iteritems():
  1284. self.itemDataMap[i] = [name, values['type'],
  1285. values.get('value', ''),
  1286. values.get('description', '')]
  1287. i += 1
  1288. self.itemCount = len(self.itemDataMap.keys())
  1289. self.DeleteAllItems()
  1290. i = 0
  1291. for name, vtype, value, desc in self.itemDataMap.itervalues():
  1292. index = self.InsertStringItem(sys.maxint, name)
  1293. self.SetStringItem(index, 0, name)
  1294. self.SetStringItem(index, 1, vtype)
  1295. self.SetStringItem(index, 2, value)
  1296. self.SetStringItem(index, 3, desc)
  1297. self.SetItemData(index, i)
  1298. i += 1
  1299. def Append(self, name, vtype, value, desc):
  1300. """!Append new item to the list
  1301. @return None on success
  1302. @return error string
  1303. """
  1304. for iname, ivtype, ivalue, idesc in self.itemDataMap.itervalues():
  1305. if iname == name:
  1306. return _("Variable <%s> already exists in the model. "
  1307. "Adding variable failed.") % name
  1308. index = self.InsertStringItem(sys.maxint, name)
  1309. self.SetStringItem(index, 0, name)
  1310. self.SetStringItem(index, 1, vtype)
  1311. self.SetStringItem(index, 2, value)
  1312. self.SetStringItem(index, 3, desc)
  1313. self.SetItemData(index, self.itemCount)
  1314. self.itemDataMap[self.itemCount] = [name, vtype, value, desc]
  1315. self.itemCount += 1
  1316. return None
  1317. def OnRemove(self, event):
  1318. """!Remove selected variable(s) from the model"""
  1319. item = self.GetFirstSelected()
  1320. while item != -1:
  1321. self.DeleteItem(item)
  1322. del self.itemDataMap[item]
  1323. item = self.GetFirstSelected()
  1324. self.parent.UpdateModelVariables()
  1325. event.Skip()
  1326. def OnRemoveAll(self, event):
  1327. """!Remove all variable(s) from the model"""
  1328. dlg = wx.MessageBox(parent=self,
  1329. message=_("Do you want to delete all variables from "
  1330. "the model?"),
  1331. caption=_("Delete variables"),
  1332. style=wx.YES_NO | wx.CENTRE)
  1333. if dlg != wx.YES:
  1334. return
  1335. self.DeleteAllItems()
  1336. self.itemDataMap = dict()
  1337. self.parent.UpdateModelVariables()
  1338. def OnEndEdit(self, event):
  1339. """!Finish editing of item"""
  1340. itemIndex = event.GetIndex()
  1341. columnIndex = event.GetColumn()
  1342. nameOld = self.GetItem(itemIndex, 0).GetText()
  1343. if columnIndex == 0: # TODO
  1344. event.Veto()
  1345. self.itemDataMap[itemIndex][columnIndex] = event.GetText()
  1346. self.parent.UpdateModelVariables()
  1347. def OnReload(self, event):
  1348. """!Reload list of variables"""
  1349. self.Populate(self.parent.parent.GetModel().GetVariables())
  1350. def OnRightUp(self, event):
  1351. """!Mouse right button up"""
  1352. if not hasattr(self, "popupID1"):
  1353. self.popupID1 = wx.NewId()
  1354. self.popupID2 = wx.NewId()
  1355. self.popupID3 = wx.NewId()
  1356. self.Bind(wx.EVT_MENU, self.OnRemove, id = self.popupID1)
  1357. self.Bind(wx.EVT_MENU, self.OnRemoveAll, id = self.popupID2)
  1358. self.Bind(wx.EVT_MENU, self.OnReload, id = self.popupID3)
  1359. # generate popup-menu
  1360. menu = wx.Menu()
  1361. menu.Append(self.popupID1, _("Delete selected"))
  1362. menu.Append(self.popupID2, _("Delete all"))
  1363. if self.GetFirstSelected() == -1:
  1364. menu.Enable(self.popupID1, False)
  1365. menu.Enable(self.popupID2, False)
  1366. menu.AppendSeparator()
  1367. menu.Append(self.popupID3, _("Reload"))
  1368. self.PopupMenu(menu)
  1369. menu.Destroy()
  1370. class ItemPanel(wx.Panel):
  1371. def __init__(self, parent, id = wx.ID_ANY,
  1372. **kwargs):
  1373. """!Manage model items
  1374. """
  1375. self.parent = parent
  1376. wx.Panel.__init__(self, parent = parent, id = id, **kwargs)
  1377. self.listBox = wx.StaticBox(parent = self, id = wx.ID_ANY,
  1378. label=" %s " % _("List of items - right-click to delete"))
  1379. self.list = ItemListCtrl(parent = self,
  1380. columns = [_("ID"), _("Name"), _("In block"),
  1381. _("Command / Condition")])
  1382. self._layout()
  1383. def _layout(self):
  1384. """!Layout dialog"""
  1385. listSizer = wx.StaticBoxSizer(self.listBox, wx.VERTICAL)
  1386. listSizer.Add(item = self.list, proportion = 1,
  1387. flag = wx.EXPAND)
  1388. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1389. mainSizer.Add(item = listSizer, proportion = 1,
  1390. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1391. self.SetSizer(mainSizer)
  1392. mainSizer.Fit(self)
  1393. def Update(self):
  1394. """!Reload list of variables"""
  1395. self.list.OnReload(None)
  1396. class ItemListCtrl(ModelListCtrl):
  1397. def __init__(self, parent, columns, disablePopup = False, **kwargs):
  1398. """!List of model actions"""
  1399. self.disablePopup = disablePopup
  1400. ModelListCtrl.__init__(self, parent, columns, **kwargs)
  1401. self.SetColumnWidth(1, 100)
  1402. self.SetColumnWidth(2, 65)
  1403. def GetListCtrl(self):
  1404. """!Used by ColumnSorterMixin"""
  1405. return self
  1406. def GetData(self):
  1407. """!Get list data"""
  1408. return self.itemDataMap
  1409. def Populate(self, data):
  1410. """!Populate the list"""
  1411. self.itemDataMap = dict()
  1412. if self.shape:
  1413. if isinstance(self.shape, ModelCondition):
  1414. if self.GetName() == 'ElseBlockList':
  1415. shapeItems = map(lambda x: x.GetId(), self.shape.GetItems()['else'])
  1416. else:
  1417. shapeItems = map(lambda x: x.GetId(), self.shape.GetItems()['if'])
  1418. else:
  1419. shapeItems = map(lambda x: x.GetId(), self.shape.GetItems())
  1420. else:
  1421. shapeItems = list()
  1422. i = 0
  1423. if len(self.columns) == 3: # ItemCheckList
  1424. checked = list()
  1425. for action in data:
  1426. if isinstance(action, ModelData) or \
  1427. action == self.shape:
  1428. continue
  1429. if len(self.columns) == 3:
  1430. self.itemDataMap[i] = [str(action.GetId()),
  1431. action.GetName(),
  1432. action.GetLog()]
  1433. aId = action.GetBlockId()
  1434. if action.GetId() in shapeItems:
  1435. checked.append(aId)
  1436. else:
  1437. checked.append(None)
  1438. else:
  1439. bId = action.GetBlockId()
  1440. if not bId:
  1441. bId = ''
  1442. self.itemDataMap[i] = [str(action.GetId()),
  1443. action.GetName(),
  1444. ','.join(map(str, bId)),
  1445. action.GetLog()]
  1446. i += 1
  1447. self.itemCount = len(self.itemDataMap.keys())
  1448. self.DeleteAllItems()
  1449. i = 0
  1450. if len(self.columns) == 3:
  1451. for aid, name, desc in self.itemDataMap.itervalues():
  1452. index = self.InsertStringItem(sys.maxint, aid)
  1453. self.SetStringItem(index, 0, aid)
  1454. self.SetStringItem(index, 1, name)
  1455. self.SetStringItem(index, 2, desc)
  1456. self.SetItemData(index, i)
  1457. if checked[i]:
  1458. self.CheckItem(index, True)
  1459. i += 1
  1460. else:
  1461. for aid, name, inloop, desc in self.itemDataMap.itervalues():
  1462. index = self.InsertStringItem(sys.maxint, aid)
  1463. self.SetStringItem(index, 0, aid)
  1464. self.SetStringItem(index, 1, name)
  1465. self.SetStringItem(index, 2, inloop)
  1466. self.SetStringItem(index, 3, desc)
  1467. self.SetItemData(index, i)
  1468. i += 1
  1469. def OnRemove(self, event):
  1470. """!Remove selected action(s) from the model"""
  1471. model = self.frame.GetModel()
  1472. canvas = self.frame.GetCanvas()
  1473. item = self.GetFirstSelected()
  1474. while item != -1:
  1475. self.DeleteItem(item)
  1476. del self.itemDataMap[item]
  1477. aId = self.GetItem(item, 0).GetText()
  1478. action = model.GetItem(int(aId))
  1479. if not action:
  1480. item = self.GetFirstSelected()
  1481. continue
  1482. model.RemoveItem(action)
  1483. canvas.GetDiagram().RemoveShape(action)
  1484. self.frame.ModelChanged()
  1485. item = self.GetFirstSelected()
  1486. canvas.Refresh()
  1487. event.Skip()
  1488. def OnRemoveAll(self, event):
  1489. """!Remove all variable(s) from the model"""
  1490. deleteDialog = wx.MessageBox(parent=self,
  1491. message=_("Selected data records (%d) will permanently deleted "
  1492. "from table. Do you want to delete them?") % \
  1493. (len(self.listOfSQLStatements)),
  1494. caption=_("Delete records"),
  1495. style=wx.YES_NO | wx.CENTRE)
  1496. if deleteDialog != wx.YES:
  1497. return False
  1498. self.DeleteAllItems()
  1499. self.itemDataMap = dict()
  1500. self.parent.UpdateModelVariables()
  1501. def OnEndEdit(self, event):
  1502. """!Finish editing of item"""
  1503. itemIndex = event.GetIndex()
  1504. columnIndex = event.GetColumn()
  1505. self.itemDataMap[itemIndex][columnIndex] = event.GetText()
  1506. aId = int(self.GetItem(itemIndex, 0).GetText())
  1507. action = self.frame.GetModel().GetItem(aId)
  1508. if not action:
  1509. event.Veto()
  1510. if columnIndex == 0:
  1511. action.SetId(int(event.GetText()))
  1512. self.frame.ModelChanged()
  1513. def OnReload(self, event = None):
  1514. """!Reload list of actions"""
  1515. self.Populate(self.frame.GetModel().GetItems())
  1516. def OnRightUp(self, event):
  1517. """!Mouse right button up"""
  1518. if self.disablePopup:
  1519. return
  1520. if not hasattr(self, "popupID1"):
  1521. self.popupID1 = wx.NewId()
  1522. self.popupID2 = wx.NewId()
  1523. self.popupID3 = wx.NewId()
  1524. self.popupID4 = wx.NewId()
  1525. self.Bind(wx.EVT_MENU, self.OnRemove, id = self.popupID1)
  1526. self.Bind(wx.EVT_MENU, self.OnRemoveAll, id = self.popupID2)
  1527. self.Bind(wx.EVT_MENU, self.OnReload, id = self.popupID3)
  1528. self.Bind(wx.EVT_MENU, self.OnNormalize, id = self.popupID4)
  1529. # generate popup-menu
  1530. menu = wx.Menu()
  1531. menu.Append(self.popupID1, _("Delete selected"))
  1532. menu.Append(self.popupID2, _("Delete all"))
  1533. if self.GetFirstSelected() == -1:
  1534. menu.Enable(self.popupID1, False)
  1535. menu.Enable(self.popupID2, False)
  1536. menu.AppendSeparator()
  1537. menu.Append(self.popupID4, _("Normalize"))
  1538. menu.Append(self.popupID3, _("Reload"))
  1539. self.PopupMenu(menu)
  1540. menu.Destroy()
  1541. def OnNormalize(self, event):
  1542. """!Update id of actions"""
  1543. model = self.frame.GetModel()
  1544. aId = 1
  1545. for item in model.GetItems():
  1546. item.SetId(aId)
  1547. aId += 1
  1548. self.OnReload(None)
  1549. self.frame.GetCanvas().Refresh()
  1550. self.frame.ModelChanged()
  1551. class ItemCheckListCtrl(ItemListCtrl, listmix.CheckListCtrlMixin):
  1552. def __init__(self, parent, shape, columns, window = None, **kwargs):
  1553. self.parent = parent
  1554. self.window = window
  1555. ItemListCtrl.__init__(self, parent, columns, disablePopup = True, **kwargs)
  1556. listmix.CheckListCtrlMixin.__init__(self)
  1557. self.SetColumnWidth(0, 50)
  1558. self.shape = shape
  1559. def OnBeginEdit(self, event):
  1560. """!Disable editing"""
  1561. event.Veto()
  1562. def OnCheckItem(self, index, flag):
  1563. """!Item checked/unchecked"""
  1564. name = self.GetName()
  1565. if name == 'IfBlockList' and self.window:
  1566. self.window.OnCheckItemIf(index, flag)
  1567. elif name == 'ElseBlockList' and self.window:
  1568. self.window.OnCheckItemElse(index, flag)
  1569. def GetItems(self):
  1570. """!Get list of selected actions"""
  1571. ids = { 'checked' : list(),
  1572. 'unchecked' : list() }
  1573. for i in range(self.GetItemCount()):
  1574. iId = int(self.GetItem(i, 0).GetText())
  1575. if self.IsChecked(i):
  1576. ids['checked'].append(iId)
  1577. else:
  1578. ids['unchecked'].append(iId)
  1579. return ids
  1580. def CheckItemById(self, aId, flag):
  1581. """!Check/uncheck given item by id"""
  1582. for i in range(self.GetItemCount()):
  1583. iId = int(self.GetItem(i, 0).GetText())
  1584. if iId == aId:
  1585. self.CheckItem(i, flag)
  1586. break
  1587. def main():
  1588. import gettext
  1589. gettext.install('grasswxpy', os.path.join(os.getenv("GISBASE"), 'locale'), unicode = True)
  1590. app = wx.PySimpleApp()
  1591. wx.InitAllImageHandlers()
  1592. frame = ModelFrame(parent = None)
  1593. if len(sys.argv) > 1:
  1594. frame.LoadModelFile(sys.argv[1])
  1595. frame.Show()
  1596. app.MainLoop()
  1597. if __name__ == "__main__":
  1598. main()