frame.py 57 KB

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