model.py 78 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258
  1. """!
  2. @package gmodeler.model
  3. @brief wxGUI Graphical Modeler (base classes & read/write)
  4. Classes:
  5. - model::Model
  6. - model::ModelObject
  7. - model::ModelAction
  8. - model::ModelData
  9. - model::ModelRelation
  10. - model::ModelItem
  11. - model::ModelLoop
  12. - model::ModelCondition
  13. - model::ProcessModelFile
  14. - model::WriteModelFile
  15. - model::WritePythonFile
  16. - model::ModelParamDialog
  17. (C) 2010-2012 by the GRASS Development Team
  18. This program is free software under the GNU General Public License
  19. (>=v2). Read the file COPYING that comes with GRASS for details.
  20. @author Martin Landa <landa.martin gmail.com>
  21. """
  22. import os
  23. import getpass
  24. import copy
  25. import re
  26. import mimetypes
  27. import time
  28. import locale
  29. try:
  30. import xml.etree.ElementTree as etree
  31. except ImportError:
  32. import elementtree.ElementTree as etree # Python <= 2.4
  33. import wx
  34. from wx.lib import ogl
  35. from core import globalvar
  36. from core import utils
  37. from core.gcmd import GMessage, GException, GError, RunCommand, EncodeString, GWarning
  38. from core.settings import UserSettings
  39. from gui_core.forms import GUI, CmdPanel
  40. from gui_core.widgets import GNotebook
  41. from grass.script import core as grass
  42. from grass.script import task as gtask
  43. class Model(object):
  44. """!Class representing the model"""
  45. def __init__(self, canvas = None):
  46. self.items = list() # list of actions/loops/...
  47. # model properties
  48. self.properties = { 'name' : _("model"),
  49. 'description' : _("Script generated by wxGUI Graphical Modeler."),
  50. 'author' : getpass.getuser() }
  51. # model variables
  52. self.variables = dict()
  53. self.variablesParams = dict()
  54. self.canvas = canvas
  55. def GetCanvas(self):
  56. """!Get canvas or None"""
  57. return self.canvas
  58. def GetItems(self, objType = None):
  59. """!Get list of model items
  60. @param objType Object type to filter model objects
  61. """
  62. if not objType:
  63. return self.items
  64. result = list()
  65. for item in self.items:
  66. if isinstance(item, objType):
  67. result.append(item)
  68. return result
  69. def GetItem(self, aId):
  70. """!Get item of given id
  71. @param aId item id
  72. @return Model* instance
  73. @return None if no item found
  74. """
  75. ilist = self.GetItems()
  76. for item in ilist:
  77. if item.GetId() == aId:
  78. return item
  79. return None
  80. def GetNumItems(self, actionOnly = False):
  81. """!Get number of items"""
  82. if actionOnly:
  83. return len(self.GetItems(objType = ModelAction))
  84. return len(self.GetItems())
  85. def GetNextId(self):
  86. """!Get next id (data ignored)
  87. @return next id to be used (default: 1)
  88. """
  89. if len(self.items) < 1:
  90. return 1
  91. currId = self.items[-1].GetId()
  92. if currId > 0:
  93. return currId + 1
  94. return 1
  95. def GetProperties(self):
  96. """!Get model properties"""
  97. return self.properties
  98. def GetVariables(self, params = False):
  99. """!Get model variables"""
  100. if params:
  101. return self.variablesParams
  102. return self.variables
  103. def SetVariables(self, data):
  104. """!Set model variables"""
  105. self.variables = data
  106. def Reset(self):
  107. """!Reset model"""
  108. self.items = list()
  109. def RemoveItem(self, item):
  110. """!Remove item from model
  111. @return list of related items to remove/update
  112. """
  113. relList = list()
  114. upList = list()
  115. if not isinstance(item, ModelData):
  116. self.items.remove(item)
  117. if isinstance(item, ModelAction):
  118. for rel in item.GetRelations():
  119. relList.append(rel)
  120. data = rel.GetData()
  121. if len(data.GetRelations()) < 2:
  122. relList.append(data)
  123. else:
  124. upList.append(data)
  125. elif isinstance(item, ModelData):
  126. for rel in item.GetRelations():
  127. relList.append(rel)
  128. if rel.GetFrom() == self:
  129. relList.append(rel.GetTo())
  130. else:
  131. relList.append(rel.GetFrom())
  132. elif isinstance(item, ModelLoop):
  133. for rel in item.GetRelations():
  134. relList.append(rel)
  135. for action in self.GetItems():
  136. action.UnSetBlock(item)
  137. return relList, upList
  138. def FindAction(self, aId):
  139. """!Find action by id"""
  140. alist = self.GetItems(objType = ModelAction)
  141. for action in alist:
  142. if action.GetId() == aId:
  143. return action
  144. return None
  145. def GetData(self):
  146. """!Get list of data items"""
  147. result = list()
  148. dataItems = self.GetItems(objType = ModelData)
  149. for action in self.GetItems(objType = ModelAction):
  150. for rel in action.GetRelations():
  151. dataItem = rel.GetData()
  152. if dataItem not in result:
  153. result.append(dataItem)
  154. if dataItem in dataItems:
  155. dataItems.remove(dataItem)
  156. # standalone data
  157. if dataItems:
  158. result += dataItems
  159. return result
  160. def FindData(self, value, prompt):
  161. """!Find data item in the model
  162. @param value value
  163. @param prompt prompt
  164. @return ModelData instance
  165. @return None if not found
  166. """
  167. for data in self.GetData():
  168. if data.GetValue() == value and \
  169. data.GetPrompt() == prompt:
  170. return data
  171. return None
  172. def LoadModel(self, filename):
  173. """!Load model definition stored in GRASS Model XML file (gxm)
  174. @todo Validate against DTD
  175. Raise exception on error.
  176. """
  177. dtdFilename = os.path.join(globalvar.ETCWXDIR, "xml", "grass-gxm.dtd")
  178. # parse workspace file
  179. try:
  180. gxmXml = ProcessModelFile(etree.parse(filename))
  181. except StandardError, e:
  182. raise GException(e)
  183. if self.canvas:
  184. win = self.canvas.parent
  185. if gxmXml.pos:
  186. win.SetPosition(gxmXml.pos)
  187. if gxmXml.size:
  188. win.SetSize(gxmXml.size)
  189. # load properties
  190. self.properties = gxmXml.properties
  191. self.variables = gxmXml.variables
  192. # load model.GetActions()
  193. for action in gxmXml.actions:
  194. actionItem = ModelAction(parent = self,
  195. x = action['pos'][0],
  196. y = action['pos'][1],
  197. width = action['size'][0],
  198. height = action['size'][1],
  199. task = action['task'],
  200. id = action['id'])
  201. if action['disabled']:
  202. actionItem.Enable(False)
  203. self.AddItem(actionItem)
  204. actionItem.SetValid(actionItem.GetTask().get_options())
  205. actionItem.GetLog() # substitute variables (-> valid/invalid)
  206. # load data & relations
  207. for data in gxmXml.data:
  208. dataItem = ModelData(parent = self,
  209. x = data['pos'][0],
  210. y = data['pos'][1],
  211. width = data['size'][0],
  212. height = data['size'][1],
  213. prompt = data['prompt'],
  214. value = data['value'])
  215. dataItem.SetIntermediate(data['intermediate'])
  216. for rel in data['rels']:
  217. actionItem = self.FindAction(rel['id'])
  218. if rel['dir'] == 'from':
  219. relation = ModelRelation(parent = self, fromShape = dataItem,
  220. toShape = actionItem, param = rel['name'])
  221. else:
  222. relation = ModelRelation(parent = self, fromShape = actionItem,
  223. toShape = dataItem, param = rel['name'])
  224. relation.SetControlPoints(rel['points'])
  225. actionItem.AddRelation(relation)
  226. dataItem.AddRelation(relation)
  227. if self.canvas:
  228. dataItem.Update()
  229. # load loops
  230. for loop in gxmXml.loops:
  231. loopItem = ModelLoop(parent = self,
  232. x = loop['pos'][0],
  233. y = loop['pos'][1],
  234. width = loop['size'][0],
  235. height = loop['size'][1],
  236. text = loop['text'],
  237. id = loop['id'])
  238. self.AddItem(loopItem)
  239. # load conditions
  240. for condition in gxmXml.conditions:
  241. conditionItem = ModelCondition(parent = self,
  242. x = condition['pos'][0],
  243. y = condition['pos'][1],
  244. width = condition['size'][0],
  245. height = condition['size'][1],
  246. text = condition['text'],
  247. id = condition['id'])
  248. self.AddItem(conditionItem)
  249. # define loops & if/else items
  250. for loop in gxmXml.loops:
  251. alist = list()
  252. for aId in loop['items']:
  253. action = self.GetItem(aId)
  254. alist.append(action)
  255. loopItem = self.GetItem(loop['id'])
  256. loopItem.SetItems(alist)
  257. for action in loopItem.GetItems():
  258. action.SetBlock(loopItem)
  259. for condition in gxmXml.conditions:
  260. conditionItem = self.GetItem(condition['id'])
  261. for b in condition['items'].keys():
  262. alist = list()
  263. for aId in condition['items'][b]:
  264. action = self.GetItem(aId)
  265. alist.append(action)
  266. conditionItem.SetItems(alist, branch = b)
  267. items = conditionItem.GetItems()
  268. for b in items.keys():
  269. for action in items[b]:
  270. action.SetBlock(conditionItem)
  271. def AddItem(self, newItem):
  272. """!Add item to the list"""
  273. iId = newItem.GetId()
  274. i = 0
  275. for item in self.items:
  276. if item.GetId() > iId:
  277. self.items.insert(i, newItem)
  278. return
  279. i += 1
  280. self.items.append(newItem)
  281. def IsValid(self):
  282. """Return True if model is valid"""
  283. if self.Validate():
  284. return False
  285. return True
  286. def Validate(self):
  287. """!Validate model, return None if model is valid otherwise
  288. error string"""
  289. errList = list()
  290. variables = self.GetVariables().keys()
  291. pattern = re.compile(r'(.*)(%.+\s?)(.*)')
  292. for action in self.GetItems(objType = ModelAction):
  293. cmd = action.GetLog(string = False)
  294. task = GUI(show = None).ParseCommand(cmd = cmd)
  295. errList += map(lambda x: cmd[0] + ': ' + x, task.get_cmd_error())
  296. # check also variables
  297. for opt in cmd[1:]:
  298. if '=' not in opt:
  299. continue
  300. key, value = opt.split('=', 1)
  301. sval = pattern.search(value)
  302. if sval:
  303. var = sval.group(2).strip()[1:] # ignore '%'
  304. if var not in variables:
  305. report = True
  306. for item in filter(lambda x: isinstance(x, ModelLoop), action.GetBlock()):
  307. if var in item.GetText():
  308. report = False
  309. break
  310. if report:
  311. errList.append(cmd[0] + ": " + _("undefined variable '%s'") % var)
  312. ### TODO: check variables in file only optionally
  313. ### errList += self._substituteFile(action, checkOnly = True)
  314. return errList
  315. def _substituteFile(self, item, params = None, checkOnly = False):
  316. """!Subsitute variables in command file inputs
  317. @param checkOnly tuble - True to check variable, don't touch files
  318. @return list of undefined variables
  319. """
  320. errList = list()
  321. self.fileInput = dict()
  322. # collect ascii inputs
  323. for p in item.GetParams()['params']:
  324. if p.get('element', '') == 'file' and \
  325. p.get('prompt', '') == 'input' and \
  326. p.get('age', '') == 'old':
  327. filename = p.get('value', p.get('default', ''))
  328. if filename and \
  329. mimetypes.guess_type(filename)[0] == 'text/plain':
  330. self.fileInput[filename] = None
  331. for finput in self.fileInput:
  332. # read lines
  333. fd = open(finput, "r")
  334. try:
  335. data = self.fileInput[finput] = fd.read()
  336. finally:
  337. fd.close()
  338. # substitute variables
  339. write = False
  340. variables = self.GetVariables()
  341. for variable in variables:
  342. pattern = re.compile('%' + variable)
  343. value = ''
  344. if params and 'variables' in params:
  345. for p in params['variables']['params']:
  346. if variable == p.get('name', ''):
  347. if p.get('type', 'string') == 'string':
  348. value = p.get('value', '')
  349. else:
  350. value = str(p.get('value', ''))
  351. break
  352. if not value:
  353. value = variables[variable].get('value', '')
  354. data = pattern.sub(value, data)
  355. if not checkOnly:
  356. write = True
  357. pattern = re.compile(r'(.*)(%.+\s?)(.*)')
  358. sval = pattern.search(data)
  359. if sval:
  360. var = sval.group(2).strip()[1:] # ignore '%'
  361. cmd = item.GetLog(string = False)[0]
  362. errList.append(cmd + ": " + _("undefined variable '%s'") % var)
  363. if not checkOnly:
  364. if write:
  365. fd = open(finput, "w")
  366. try:
  367. fd.write(data)
  368. finally:
  369. fd.close()
  370. else:
  371. self.fileInput[finput] = None
  372. return errList
  373. def OnPrepare(self, item, params):
  374. self._substituteFile(item, params, checkOnly = False)
  375. def RunAction(self, item, params, log, onDone, onPrepare = None, statusbar = None):
  376. """!Run given action
  377. @param item action item
  378. @param params parameters dict
  379. @param log logging window
  380. @param onDone on-done method
  381. @param onPrepare on-prepare method
  382. @param statusbar wx.StatusBar instance or None
  383. """
  384. name = item.GetName()
  385. if name in params:
  386. paramsOrig = item.GetParams(dcopy = True)
  387. item.MergeParams(params[name])
  388. if statusbar:
  389. statusbar.SetStatusText(_('Running model...'), 0)
  390. data = { 'item' : item,
  391. 'params' : copy.deepcopy(params) }
  392. log.RunCmd(command = item.GetLog(string = False, substitute = params),
  393. onDone = onDone, onPrepare = self.OnPrepare, userData = data)
  394. if name in params:
  395. item.SetParams(paramsOrig)
  396. def Run(self, log, onDone, parent = None):
  397. """!Run model
  398. @param log logging window (see gconsole.GConsole)
  399. @param onDone on-done method
  400. @param parent window for messages or None
  401. """
  402. if self.GetNumItems() < 1:
  403. GMessage(parent = parent,
  404. message = _('Model is empty. Nothing to run.'))
  405. return
  406. statusbar = None
  407. if isinstance(parent, wx.Frame):
  408. statusbar = parent.GetStatusBar()
  409. # validation
  410. if statusbar:
  411. statusbar.SetStatusText(_('Validating model...'), 0)
  412. errList = self.Validate()
  413. if statusbar:
  414. statusbar.SetStatusText('', 0)
  415. if errList:
  416. dlg = wx.MessageDialog(parent = parent,
  417. message = _('Model is not valid. Do you want to '
  418. 'run the model anyway?\n\n%s') % '\n'.join(errList),
  419. caption = _("Run model?"),
  420. style = wx.YES_NO | wx.NO_DEFAULT |
  421. wx.ICON_QUESTION | wx.CENTRE)
  422. ret = dlg.ShowModal()
  423. dlg.Destroy()
  424. if ret != wx.ID_YES:
  425. return
  426. # parametrization
  427. params = self.Parameterize()
  428. delInterData = False
  429. if params:
  430. dlg = ModelParamDialog(parent = parent,
  431. params = params)
  432. dlg.CenterOnParent()
  433. ret = dlg.ShowModal()
  434. if ret != wx.ID_OK:
  435. dlg.Destroy()
  436. return
  437. err = dlg.GetErrors()
  438. delInterData = dlg.DeleteIntermediateData()
  439. dlg.Destroy()
  440. if err:
  441. GError(parent = parent, message = unicode('\n'.join(err)))
  442. return
  443. err = list()
  444. for key, item in params.iteritems():
  445. for p in item['params']:
  446. if p.get('value', '') == '':
  447. err.append((key, p.get('name', ''), p.get('description', '')))
  448. if err:
  449. GError(parent = parent,
  450. message = _("Variables below not defined:") + \
  451. "\n\n" + unicode('\n'.join(map(lambda x: "%s: %s (%s)" % (x[0], x[1], x[2]), err))))
  452. return
  453. log.cmdThread.SetId(-1)
  454. for item in self.GetItems():
  455. if not item.IsEnabled():
  456. continue
  457. if isinstance(item, ModelAction):
  458. if item.GetBlockId():
  459. continue
  460. self.RunAction(item, params, log, onDone)
  461. elif isinstance(item, ModelLoop):
  462. cond = item.GetText()
  463. # substitute variables in condition
  464. variables = self.GetVariables()
  465. for variable in variables:
  466. pattern = re.compile('%' + variable)
  467. if pattern.search(cond):
  468. value = ''
  469. if params and 'variables' in params:
  470. for p in params['variables']['params']:
  471. if variable == p.get('name', ''):
  472. value = p.get('value', '')
  473. break
  474. if not value:
  475. value = variables[variable].get('value', '')
  476. if not value:
  477. continue
  478. vtype = variables[variable].get('type', 'string')
  479. if vtype == 'string':
  480. value = '"' + value + '"'
  481. cond = pattern.sub(value, cond)
  482. # split condition
  483. condVar, condText = map(lambda x: x.strip(), re.split('\s*in\s*', cond))
  484. pattern = re.compile('%' + condVar)
  485. ### for vars()[condVar] in eval(condText): ?
  486. vlist = list()
  487. if condText[0] == '`' and condText[-1] == '`':
  488. # run command
  489. cmd, dcmd = utils.CmdToTuple(condText[1:-1].split(' '))
  490. ret = RunCommand(cmd,
  491. read = True,
  492. **dcmd)
  493. if ret:
  494. vlist = ret.splitlines()
  495. else:
  496. vlist = eval(condText)
  497. if 'variables' not in params:
  498. params['variables'] = { 'params' : [] }
  499. varDict = { 'name' : condVar, 'value' : '' }
  500. params['variables']['params'].append(varDict)
  501. for var in vlist:
  502. for action in item.GetItems():
  503. if not isinstance(action, ModelAction) or \
  504. not action.IsEnabled():
  505. continue
  506. varDict['value'] = var
  507. self.RunAction(item = action, params = params,
  508. log = log, onDone = onDone)
  509. params['variables']['params'].remove(varDict)
  510. if delInterData:
  511. self.DeleteIntermediateData(log)
  512. # discard values
  513. if params:
  514. for item in params.itervalues():
  515. for p in item['params']:
  516. p['value'] = ''
  517. def DeleteIntermediateData(self, log):
  518. """!Detele intermediate data"""
  519. rast, vect, rast3d, msg = self.GetIntermediateData()
  520. if rast:
  521. log.RunCmd(['g.remove', 'rast=%s' %','.join(rast)])
  522. if rast3d:
  523. log.RunCmd(['g.remove', 'rast3d=%s' %','.join(rast3d)])
  524. if vect:
  525. log.RunCmd(['g.remove', 'vect=%s' %','.join(vect)])
  526. def GetIntermediateData(self):
  527. """!Get info about intermediate data"""
  528. rast = list()
  529. rast3d = list()
  530. vect = list()
  531. for data in self.GetData():
  532. if not data.IsIntermediate():
  533. continue
  534. name = data.GetValue()
  535. prompt = data.GetPrompt()
  536. if prompt == 'raster':
  537. rast.append(name)
  538. elif prompt == 'vector':
  539. vect.append(name)
  540. elif prompt == 'rast3d':
  541. rast3d.append(name)
  542. msg = ''
  543. if rast:
  544. msg += '\n\n%s: ' % _('Raster maps')
  545. msg += ', '.join(rast)
  546. if rast3d:
  547. msg += '\n\n%s: ' % _('3D raster maps')
  548. msg += ', '.join(rast3d)
  549. if vect:
  550. msg += '\n\n%s: ' % _('Vector maps')
  551. msg += ', '.join(vect)
  552. return rast, vect, rast3d, msg
  553. def Update(self):
  554. """!Update model"""
  555. for item in self.items:
  556. item.Update()
  557. def IsParameterized(self):
  558. """!Return True if model is parameterized"""
  559. if self.Parameterize():
  560. return True
  561. return False
  562. def Parameterize(self):
  563. """!Return parameterized options"""
  564. result = dict()
  565. idx = 0
  566. if self.variables:
  567. params = list()
  568. result["variables"] = { 'flags' : list(),
  569. 'params' : params,
  570. 'idx' : idx }
  571. for name, values in self.variables.iteritems():
  572. gtype = values.get('type', 'string')
  573. if gtype in ('raster', 'vector', 'mapset', 'file'):
  574. gisprompt = True
  575. prompt = gtype
  576. if gtype == 'raster':
  577. element = 'cell'
  578. else:
  579. element = gtype
  580. ptype = 'string'
  581. else:
  582. gisprompt = False
  583. prompt = None
  584. element = None
  585. ptype = gtype
  586. params.append({ 'gisprompt' : gisprompt,
  587. 'multiple' : False,
  588. 'description' : values.get('description', ''),
  589. 'guidependency' : '',
  590. 'default' : '',
  591. 'age' : None,
  592. 'required' : True,
  593. 'value' : values.get('value', ''),
  594. 'label' : '',
  595. 'guisection' : '',
  596. 'key_desc' : '',
  597. 'values' : list(),
  598. 'parameterized' : False,
  599. 'values_desc' : list(),
  600. 'prompt' : prompt,
  601. 'element' : element,
  602. 'type' : ptype,
  603. 'name' : name })
  604. idx += 1
  605. for action in self.GetItems(objType = ModelAction):
  606. if not action.IsEnabled():
  607. continue
  608. name = action.GetName()
  609. params = action.GetParams()
  610. for f in params['flags']:
  611. if f.get('parameterized', False):
  612. if name not in result:
  613. result[name] = { 'flags' : list(),
  614. 'params': list(),
  615. 'idx' : idx }
  616. result[name]['flags'].append(f)
  617. for p in params['params']:
  618. if p.get('parameterized', False):
  619. if name not in result:
  620. result[name] = { 'flags' : list(),
  621. 'params': list(),
  622. 'idx' : idx }
  623. result[name]['params'].append(p)
  624. if name in result:
  625. idx += 1
  626. self.variablesParams = result # record parameters
  627. return result
  628. class ModelObject(object):
  629. def __init__(self, id = -1):
  630. self.id = id
  631. self.rels = list() # list of ModelRelations
  632. self.isEnabled = True
  633. self.inBlock = list() # list of related loops/conditions
  634. def __del__(self):
  635. pass
  636. def GetId(self):
  637. """!Get id"""
  638. return self.id
  639. def AddRelation(self, rel):
  640. """!Record new relation
  641. """
  642. self.rels.append(rel)
  643. def GetRelations(self, fdir = None):
  644. """!Get list of relations
  645. @param fdir True for 'from'
  646. """
  647. if fdir is None:
  648. return self.rels
  649. result = list()
  650. for rel in self.rels:
  651. if fdir == 'from':
  652. if rel.GetFrom() == self:
  653. result.append(rel)
  654. else:
  655. if rel.GetTo() == self:
  656. result.append(rel)
  657. return result
  658. def IsEnabled(self):
  659. """!Get True if action is enabled, otherwise False"""
  660. return self.isEnabled
  661. def Enable(self, enabled = True):
  662. """!Enable/disable action"""
  663. self.isEnabled = enabled
  664. self.Update()
  665. def Update(self):
  666. pass
  667. def SetBlock(self, item):
  668. """!Add object to the block (loop/condition)
  669. @param item reference to ModelLoop or ModelCondition which
  670. defines loops/condition
  671. """
  672. if item not in self.inBlock:
  673. self.inBlock.append(item)
  674. def UnSetBlock(self, item):
  675. """!Remove object from the block (loop/consition)
  676. @param item reference to ModelLoop or ModelCondition which
  677. defines loops/codition
  678. """
  679. if item in self.inBlock:
  680. self.inBlock.remove(item)
  681. def GetBlock(self):
  682. """!Get list of related ModelObject(s) which defines block
  683. (loop/condition)
  684. @return list of ModelObjects
  685. """
  686. return self.inBlock
  687. def GetBlockId(self):
  688. """!Get list of related ids which defines block
  689. @return list of ids
  690. """
  691. ret = list()
  692. for mo in self.inBlock:
  693. ret.append(mo.GetId())
  694. return ret
  695. class ModelAction(ModelObject, ogl.RectangleShape):
  696. """!Action class (GRASS module)"""
  697. def __init__(self, parent, x, y, id = -1, cmd = None, task = None, width = None, height = None):
  698. ModelObject.__init__(self, id)
  699. self.parent = parent
  700. self.task = task
  701. if not width:
  702. width = UserSettings.Get(group='modeler', key='action', subkey=('size', 'width'))
  703. if not height:
  704. height = UserSettings.Get(group='modeler', key='action', subkey=('size', 'height'))
  705. if cmd:
  706. self.task = GUI(show = None).ParseCommand(cmd = cmd)
  707. else:
  708. if task:
  709. self.task = task
  710. else:
  711. self.task = None
  712. self.propWin = None
  713. self.data = list() # list of connected data items
  714. self.isValid = False
  715. self.isParameterized = False
  716. if self.parent.GetCanvas():
  717. ogl.RectangleShape.__init__(self, width, height)
  718. self.SetCanvas(self.parent)
  719. self.SetX(x)
  720. self.SetY(y)
  721. self.SetPen(wx.BLACK_PEN)
  722. self._setPen()
  723. self._setBrush()
  724. self.SetId(id)
  725. if self.task:
  726. self.SetValid(self.task.get_options())
  727. def _setBrush(self, running = False):
  728. """!Set brush"""
  729. if running:
  730. color = UserSettings.Get(group='modeler', key='action',
  731. subkey=('color', 'running'))
  732. elif not self.isEnabled:
  733. color = UserSettings.Get(group='modeler', key='disabled',
  734. subkey='color')
  735. elif self.isValid:
  736. color = UserSettings.Get(group='modeler', key='action',
  737. subkey=('color', 'valid'))
  738. else:
  739. color = UserSettings.Get(group='modeler', key='action',
  740. subkey=('color', 'invalid'))
  741. wxColor = wx.Color(color[0], color[1], color[2])
  742. self.SetBrush(wx.Brush(wxColor))
  743. def _setPen(self):
  744. """!Set pen"""
  745. if self.isParameterized:
  746. width = int(UserSettings.Get(group='modeler', key='action',
  747. subkey=('width', 'parameterized')))
  748. else:
  749. width = int(UserSettings.Get(group='modeler', key='action',
  750. subkey=('width', 'default')))
  751. pen = self.GetPen()
  752. pen.SetWidth(width)
  753. self.SetPen(pen)
  754. def SetId(self, id):
  755. """!Set id"""
  756. self.id = id
  757. cmd = self.task.get_cmd(ignoreErrors = True)
  758. if cmd and len(cmd) > 0:
  759. self.ClearText()
  760. self.AddText('(%d) %s' % (self.id, cmd[0]))
  761. else:
  762. self.AddText('(%d) <<%s>>' % (self.id, _("unknown")))
  763. def SetProperties(self, params, propwin):
  764. """!Record properties dialog"""
  765. self.task.params = params['params']
  766. self.task.flags = params['flags']
  767. self.propWin = propwin
  768. def GetPropDialog(self):
  769. """!Get properties dialog"""
  770. return self.propWin
  771. def GetLog(self, string = True, substitute = None):
  772. """!Get logging info
  773. @param string True to get cmd as a string otherwise a list
  774. @param substitute dictionary of parameter to substitute or None
  775. """
  776. cmd = self.task.get_cmd(ignoreErrors = True, ignoreRequired = True,
  777. ignoreDefault = False)
  778. # substitute variables
  779. if substitute:
  780. variables = []
  781. if 'variables' in substitute:
  782. for p in substitute['variables']['params']:
  783. variables.append(p.get('name', ''))
  784. else:
  785. variables = self.parent.GetVariables()
  786. for variable in variables:
  787. pattern= re.compile('%' + variable)
  788. value = ''
  789. if substitute and 'variables' in substitute:
  790. for p in substitute['variables']['params']:
  791. if variable == p.get('name', ''):
  792. if p.get('type', 'string') == 'string':
  793. value = p.get('value', '')
  794. else:
  795. value = str(p.get('value', ''))
  796. break
  797. if not value:
  798. value = variables[variable].get('value', '')
  799. if not value:
  800. continue
  801. for idx in range(len(cmd)):
  802. if pattern.search(cmd[idx]):
  803. cmd[idx] = pattern.sub(value, cmd[idx])
  804. break
  805. idx += 1
  806. if string:
  807. if cmd is None:
  808. return ''
  809. else:
  810. return ' '.join(cmd)
  811. return cmd
  812. def GetName(self):
  813. """!Get name"""
  814. cmd = self.task.get_cmd(ignoreErrors = True)
  815. if cmd and len(cmd) > 0:
  816. return cmd[0]
  817. return _('unknown')
  818. def GetParams(self, dcopy = False):
  819. """!Get dictionary of parameters"""
  820. if dcopy:
  821. return copy.deepcopy(self.task.get_options())
  822. return self.task.get_options()
  823. def GetTask(self):
  824. """!Get grassTask instance"""
  825. return self.task
  826. def SetParams(self, params):
  827. """!Set dictionary of parameters"""
  828. self.task.params = params['params']
  829. self.task.flags = params['flags']
  830. def MergeParams(self, params):
  831. """!Merge dictionary of parameters"""
  832. if 'flags' in params:
  833. for f in params['flags']:
  834. self.task.set_flag(f['name'],
  835. f.get('value', False))
  836. if 'params' in params:
  837. for p in params['params']:
  838. self.task.set_param(p['name'],
  839. p.get('value', ''))
  840. def SetValid(self, options):
  841. """!Set validity for action
  842. @param options dictionary with flags and params (gtask)
  843. """
  844. self.isValid = True
  845. self.isParameterized = False
  846. for f in options['flags']:
  847. if f.get('parameterized', False):
  848. self.IsParameterized = True
  849. break
  850. for p in options['params']:
  851. if self.isValid and p.get('required', False) and \
  852. p.get('value', '') == '' and \
  853. p.get('default', '') == '':
  854. self.isValid = False
  855. if not self.isParameterized and p.get('parameterized', False):
  856. self.isParameterized = True
  857. if self.parent.GetCanvas():
  858. self._setBrush()
  859. self._setPen()
  860. def IsValid(self):
  861. """!Check validity (all required parameters set)"""
  862. return self.isValid
  863. def IsParameterized(self):
  864. """!Check if action is parameterized"""
  865. return self.isParameterized
  866. def FindData(self, name):
  867. """!Find data item by name"""
  868. for rel in self.GetRelations():
  869. data = rel.GetData()
  870. if name == rel.GetName() and name in data.GetName():
  871. return data
  872. return None
  873. def Update(self, running = False):
  874. """!Update action"""
  875. if running:
  876. self._setBrush(running = True)
  877. else:
  878. self._setBrush()
  879. self._setPen()
  880. def OnDraw(self, dc):
  881. """!Draw action in canvas"""
  882. self._setBrush()
  883. self._setPen()
  884. ogl.RectangleShape.Recentre(self, dc) # re-center text
  885. ogl.RectangleShape.OnDraw(self, dc)
  886. class ModelData(ModelObject, ogl.EllipseShape):
  887. def __init__(self, parent, x, y, value = '', prompt = '', width = None, height = None):
  888. """Data item class
  889. @param parent window parent
  890. @param x, y position of the shape
  891. @param fname, tname list of parameter names from / to
  892. @param value value
  893. @param prompt type of GIS element
  894. @param width,height dimension of the shape
  895. """
  896. ModelObject.__init__(self)
  897. self.parent = parent
  898. self.value = value
  899. self.prompt = prompt
  900. self.intermediate = False
  901. self.propWin = None
  902. if not width:
  903. width = UserSettings.Get(group='modeler', key='data', subkey=('size', 'width'))
  904. if not height:
  905. height = UserSettings.Get(group='modeler', key='data', subkey=('size', 'height'))
  906. if self.parent.GetCanvas():
  907. ogl.EllipseShape.__init__(self, width, height)
  908. self.SetCanvas(self.parent)
  909. self.SetX(x)
  910. self.SetY(y)
  911. self.SetPen(wx.BLACK_PEN)
  912. self._setBrush()
  913. self._setText()
  914. def IsIntermediate(self):
  915. """!Checks if data item is intermediate"""
  916. return self.intermediate
  917. def SetIntermediate(self, im):
  918. """!Set intermediate flag"""
  919. self.intermediate = im
  920. def OnDraw(self, dc):
  921. pen = self.GetPen()
  922. pen.SetWidth(1)
  923. if self.intermediate:
  924. pen.SetStyle(wx.SHORT_DASH)
  925. else:
  926. pen.SetStyle(wx.SOLID)
  927. self.SetPen(pen)
  928. ogl.EllipseShape.OnDraw(self, dc)
  929. def GetLog(self, string = True):
  930. """!Get logging info"""
  931. name = list()
  932. for rel in self.GetRelations():
  933. name.append(rel.GetName())
  934. if name:
  935. return '/'.join(name) + '=' + self.value + ' (' + self.prompt + ')'
  936. else:
  937. return self.value + ' (' + self.prompt + ')'
  938. def GetName(self):
  939. """!Get list of names"""
  940. name = list()
  941. for rel in self.GetRelations():
  942. name.append(rel.GetName())
  943. return name
  944. def GetPrompt(self):
  945. """!Get prompt"""
  946. return self.prompt
  947. def SetPrompt(self, prompt):
  948. """!Set prompt
  949. @param prompt
  950. """
  951. self.prompt = prompt
  952. def GetValue(self):
  953. """!Get value"""
  954. return self.value
  955. def SetValue(self, value):
  956. """!Set value
  957. @param value
  958. """
  959. self.value = value
  960. self._setText()
  961. for direction in ('from', 'to'):
  962. for rel in self.GetRelations(direction):
  963. if direction == 'from':
  964. action = rel.GetTo()
  965. else:
  966. action = rel.GetFrom()
  967. task = GUI(show = None).ParseCommand(cmd = action.GetLog(string = False))
  968. task.set_param(rel.GetName(), self.value)
  969. action.SetParams(params = task.get_options())
  970. def GetPropDialog(self):
  971. """!Get properties dialog"""
  972. return self.propWin
  973. def SetPropDialog(self, win):
  974. """!Get properties dialog"""
  975. self.propWin = win
  976. def _setBrush(self):
  977. """!Set brush"""
  978. if self.prompt == 'raster':
  979. color = UserSettings.Get(group = 'modeler', key = 'data',
  980. subkey = ('color', 'raster'))
  981. elif self.prompt == 'raster3d':
  982. color = UserSettings.Get(group = 'modeler', key = 'data',
  983. subkey = ('color', 'raster3d'))
  984. elif self.prompt == 'vector':
  985. color = UserSettings.Get(group = 'modeler', key = 'data',
  986. subkey = ('color', 'vector'))
  987. else:
  988. color = UserSettings.Get(group = 'modeler', key = 'action',
  989. subkey = ('color', 'invalid'))
  990. wxColor = wx.Color(color[0], color[1], color[2])
  991. self.SetBrush(wx.Brush(wxColor))
  992. def _setPen(self):
  993. """!Set pen"""
  994. isParameterized = False
  995. for rel in self.GetRelations('from'):
  996. if rel.GetTo().IsParameterized():
  997. isParameterized = True
  998. break
  999. if not isParameterized:
  1000. for rel in self.GetRelations('to'):
  1001. if rel.GetFrom().IsParameterized():
  1002. isParameterized = True
  1003. break
  1004. if isParameterized:
  1005. width = int(UserSettings.Get(group = 'modeler', key = 'action',
  1006. subkey = ('width', 'parameterized')))
  1007. else:
  1008. width = int(UserSettings.Get(group = 'modeler', key = 'action',
  1009. subkey = ('width', 'default')))
  1010. pen = self.GetPen()
  1011. pen.SetWidth(width)
  1012. self.SetPen(pen)
  1013. def _setText(self):
  1014. """!Update text"""
  1015. self.ClearText()
  1016. name = []
  1017. for rel in self.GetRelations():
  1018. name.append(rel.GetName())
  1019. self.AddText('/'.join(name))
  1020. if self.value:
  1021. self.AddText(self.value)
  1022. else:
  1023. self.AddText(_('<not defined>'))
  1024. def Update(self):
  1025. """!Update action"""
  1026. self._setBrush()
  1027. self._setPen()
  1028. self._setText()
  1029. class ModelRelation(ogl.LineShape):
  1030. """!Data - action relation"""
  1031. def __init__(self, parent, fromShape, toShape, param = ''):
  1032. self.fromShape = fromShape
  1033. self.toShape = toShape
  1034. self.param = param
  1035. self.parent = parent
  1036. self._points = None
  1037. if self.parent.GetCanvas():
  1038. ogl.LineShape.__init__(self)
  1039. def __del__(self):
  1040. if self in self.fromShape.rels:
  1041. self.fromShape.rels.remove(self)
  1042. if self in self.toShape.rels:
  1043. self.toShape.rels.remove(self)
  1044. def GetFrom(self):
  1045. """!Get id of 'from' shape"""
  1046. return self.fromShape
  1047. def GetTo(self):
  1048. """!Get id of 'to' shape"""
  1049. return self.toShape
  1050. def GetData(self):
  1051. """!Get related ModelData instance
  1052. @return ModelData instance
  1053. @return None if not found
  1054. """
  1055. if isinstance(self.fromShape, ModelData):
  1056. return self.fromShape
  1057. elif isinstance(self.toShape, ModelData):
  1058. return self.toShape
  1059. return None
  1060. def GetName(self):
  1061. """!Get parameter name"""
  1062. return self.param
  1063. def ResetShapes(self):
  1064. """!Reset related objects"""
  1065. self.fromShape.ResetControlPoints()
  1066. self.toShape.ResetControlPoints()
  1067. self.ResetControlPoints()
  1068. def SetControlPoints(self, points):
  1069. """!Set control points"""
  1070. self._points = points
  1071. def GetControlPoints(self):
  1072. """!Get list of control points"""
  1073. return self._points
  1074. def _setPen(self):
  1075. """!Set pen"""
  1076. pen = self.GetPen()
  1077. pen.SetWidth(1)
  1078. pen.SetStyle(wx.SOLID)
  1079. self.SetPen(pen)
  1080. def OnDraw(self, dc):
  1081. """!Draw relation"""
  1082. self._setPen()
  1083. ogl.LineShape.OnDraw(self, dc)
  1084. def SetName(self, param):
  1085. self.param = param
  1086. class ModelItem(ModelObject):
  1087. def __init__(self, parent, x, y, id = -1, width = None, height = None, text = '', items = []):
  1088. """!Abstract class for loops and conditions"""
  1089. ModelObject.__init__(self, id)
  1090. self.parent = parent
  1091. self.text = text
  1092. self.items = items # list of items in the loop
  1093. def GetText(self):
  1094. """!Get loop text"""
  1095. return self.text
  1096. def GetItems(self):
  1097. """!Get items (id)"""
  1098. return self.items
  1099. def SetId(self, id):
  1100. """!Set loop id"""
  1101. self.id = id
  1102. def SetText(self, cond):
  1103. """!Set loop text (condition)"""
  1104. self.text = cond
  1105. self.ClearText()
  1106. self.AddText('(' + str(self.id) + ') ' + self.text)
  1107. def GetLog(self):
  1108. """!Get log info"""
  1109. if self.text:
  1110. return _("Condition: ") + self.text
  1111. else:
  1112. return _("Condition: not defined")
  1113. def AddRelation(self, rel):
  1114. """!Record relation"""
  1115. self.rels.append(rel)
  1116. def Clear(self):
  1117. """!Clear object, remove rels"""
  1118. self.rels = list()
  1119. class ModelLoop(ModelItem, ogl.RectangleShape):
  1120. def __init__(self, parent, x, y, id = -1, width = None, height = None, text = '', items = []):
  1121. """!Defines a loop"""
  1122. ModelItem.__init__(self, parent, x, y, id, width, height, text, items)
  1123. if not width:
  1124. width = UserSettings.Get(group='modeler', key='loop', subkey=('size', 'width'))
  1125. if not height:
  1126. height = UserSettings.Get(group='modeler', key='loop', subkey=('size', 'height'))
  1127. if self.parent.GetCanvas():
  1128. ogl.RectangleShape.__init__(self, width, height)
  1129. self.SetCanvas(self.parent)
  1130. self.SetX(x)
  1131. self.SetY(y)
  1132. self.SetPen(wx.BLACK_PEN)
  1133. self.SetCornerRadius(100)
  1134. if text:
  1135. self.AddText('(' + str(self.id) + ') ' + text)
  1136. else:
  1137. self.AddText('(' + str(self.id) + ')')
  1138. self._setBrush()
  1139. def _setBrush(self):
  1140. """!Set brush"""
  1141. if not self.isEnabled:
  1142. color = UserSettings.Get(group='modeler', key='disabled',
  1143. subkey='color')
  1144. else:
  1145. color = UserSettings.Get(group='modeler', key='loop',
  1146. subkey=('color', 'valid'))
  1147. wxColor = wx.Color(color[0], color[1], color[2])
  1148. self.SetBrush(wx.Brush(wxColor))
  1149. def Enable(self, enabled = True):
  1150. """!Enable/disable action"""
  1151. for item in self.items:
  1152. if not isinstance(item, ModelAction):
  1153. continue
  1154. item.Enable(enabled)
  1155. ModelObject.Enable(self, enabled)
  1156. def Update(self):
  1157. self._setBrush()
  1158. def GetName(self):
  1159. """!Get name"""
  1160. return _("loop")
  1161. def SetItems(self, items):
  1162. """!Set items (id)"""
  1163. self.items = items
  1164. def OnDraw(self, dc):
  1165. """!Draw loop in canvas"""
  1166. self._setBrush()
  1167. ogl.RectangleShape.Recentre(self, dc) # re-center text
  1168. ogl.RectangleShape.OnDraw(self, dc)
  1169. class ModelCondition(ModelItem, ogl.PolygonShape):
  1170. def __init__(self, parent, x, y, id = -1, width = None, height = None, text = '',
  1171. items = { 'if' : [], 'else' : [] }):
  1172. """!Defines a if-else condition"""
  1173. ModelItem.__init__(self, parent, x, y, id, width, height, text, items)
  1174. if not width:
  1175. self.width = UserSettings.Get(group='modeler', key='if-else', subkey=('size', 'width'))
  1176. else:
  1177. self.width = width
  1178. if not height:
  1179. self.height = UserSettings.Get(group='modeler', key='if-else', subkey=('size', 'height'))
  1180. else:
  1181. self.height = height
  1182. if self.parent.GetCanvas():
  1183. ogl.PolygonShape.__init__(self)
  1184. points = [(0, - self.height / 2),
  1185. (self.width / 2, 0),
  1186. (0, self.height / 2),
  1187. (- self.width / 2, 0)]
  1188. self.Create(points)
  1189. self.SetCanvas(self.parent)
  1190. self.SetX(x)
  1191. self.SetY(y)
  1192. self.SetPen(wx.BLACK_PEN)
  1193. if text:
  1194. self.AddText('(' + str(self.id) + ') ' + text)
  1195. else:
  1196. self.AddText('(' + str(self.id) + ')')
  1197. def GetName(self):
  1198. """!Get name"""
  1199. return _("if-else")
  1200. def GetWidth(self):
  1201. """!Get object width"""
  1202. return self.width
  1203. def GetHeight(self):
  1204. """!Get object height"""
  1205. return self.height
  1206. def SetItems(self, items, branch = 'if'):
  1207. """!Set items (id)
  1208. @param items list of items
  1209. @param branch 'if' / 'else'
  1210. """
  1211. if branch in ['if', 'else']:
  1212. self.items[branch] = items
  1213. class ProcessModelFile:
  1214. """!Process GRASS model file (gxm)"""
  1215. def __init__(self, tree):
  1216. """!A ElementTree handler for the GXM XML file, as defined in
  1217. grass-gxm.dtd.
  1218. """
  1219. self.tree = tree
  1220. self.root = self.tree.getroot()
  1221. # list of actions, data
  1222. self.properties = dict()
  1223. self.variables = dict()
  1224. self.actions = list()
  1225. self.data = list()
  1226. self.loops = list()
  1227. self.conditions = list()
  1228. self._processWindow()
  1229. self._processProperties()
  1230. self._processVariables()
  1231. self._processItems()
  1232. self._processData()
  1233. def _filterValue(self, value):
  1234. """!Filter value
  1235. @param value
  1236. """
  1237. value = value.replace('&lt;', '<')
  1238. value = value.replace('&gt;', '>')
  1239. return value
  1240. def _getNodeText(self, node, tag, default = ''):
  1241. """!Get node text"""
  1242. p = node.find(tag)
  1243. if p is not None:
  1244. if p.text:
  1245. return utils.normalize_whitespace(p.text)
  1246. else:
  1247. return ''
  1248. return default
  1249. def _processWindow(self):
  1250. """!Process window properties"""
  1251. node = self.root.find('window')
  1252. if node is None:
  1253. self.pos = self.size = None
  1254. return
  1255. self.pos, self.size = self._getDim(node)
  1256. def _processProperties(self):
  1257. """!Process model properties"""
  1258. node = self.root.find('properties')
  1259. if node is None:
  1260. return
  1261. for key in ('name', 'description', 'author'):
  1262. self._processProperty(node, key)
  1263. for f in node.findall('flag'):
  1264. name = f.get('name', '')
  1265. if name == 'overwrite':
  1266. self.properties['overwrite'] = True
  1267. def _processProperty(self, pnode, name):
  1268. """!Process given property"""
  1269. node = pnode.find(name)
  1270. if node is not None:
  1271. self.properties[name] = node.text
  1272. else:
  1273. self.properties[name] = ''
  1274. def _processVariables(self):
  1275. """!Process model variables"""
  1276. vnode = self.root.find('variables')
  1277. if vnode is None:
  1278. return
  1279. for node in vnode.findall('variable'):
  1280. name = node.get('name', '')
  1281. if not name:
  1282. continue # should not happen
  1283. self.variables[name] = { 'type' : node.get('type', 'string') }
  1284. for key in ('description', 'value'):
  1285. self._processVariable(node, name, key)
  1286. def _processVariable(self, pnode, name, key):
  1287. """!Process given variable"""
  1288. node = pnode.find(key)
  1289. if node is not None:
  1290. if node.text:
  1291. self.variables[name][key] = node.text
  1292. def _processItems(self):
  1293. """!Process model items (actions, loops, conditions)"""
  1294. self._processActions()
  1295. self._processLoops()
  1296. self._processConditions()
  1297. def _processActions(self):
  1298. """!Process model file"""
  1299. for action in self.root.findall('action'):
  1300. pos, size = self._getDim(action)
  1301. disabled = False
  1302. task = action.find('task')
  1303. if task is not None:
  1304. if task.find('disabled') is not None:
  1305. disabled = True
  1306. task = self._processTask(task)
  1307. else:
  1308. task = None
  1309. aId = int(action.get('id', -1))
  1310. self.actions.append({ 'pos' : pos,
  1311. 'size' : size,
  1312. 'task' : task,
  1313. 'id' : aId,
  1314. 'disabled' : disabled })
  1315. def _getDim(self, node):
  1316. """!Get position and size of shape"""
  1317. pos = size = None
  1318. posAttr = node.get('pos', None)
  1319. if posAttr:
  1320. posVal = map(int, posAttr.split(','))
  1321. try:
  1322. pos = (posVal[0], posVal[1])
  1323. except:
  1324. pos = None
  1325. sizeAttr = node.get('size', None)
  1326. if sizeAttr:
  1327. sizeVal = map(int, sizeAttr.split(','))
  1328. try:
  1329. size = (sizeVal[0], sizeVal[1])
  1330. except:
  1331. size = None
  1332. return pos, size
  1333. def _processData(self):
  1334. """!Process model file"""
  1335. for data in self.root.findall('data'):
  1336. pos, size = self._getDim(data)
  1337. param = data.find('data-parameter')
  1338. prompt = value = None
  1339. if param is not None:
  1340. prompt = param.get('prompt', None)
  1341. value = self._filterValue(self._getNodeText(param, 'value'))
  1342. if data.find('intermediate') is None:
  1343. intermediate = False
  1344. else:
  1345. intermediate = True
  1346. rels = list()
  1347. for rel in data.findall('relation'):
  1348. defrel = { 'id' : int(rel.get('id', -1)),
  1349. 'dir' : rel.get('dir', 'to'),
  1350. 'name' : rel.get('name', '') }
  1351. points = list()
  1352. for point in rel.findall('point'):
  1353. x = self._filterValue(self._getNodeText(point, 'x'))
  1354. y = self._filterValue(self._getNodeText(point, 'y'))
  1355. points.append((float(x), float(y)))
  1356. defrel['points'] = points
  1357. rels.append(defrel)
  1358. self.data.append({ 'pos' : pos,
  1359. 'size': size,
  1360. 'prompt' : prompt,
  1361. 'value' : value,
  1362. 'intermediate' : intermediate,
  1363. 'rels' : rels })
  1364. def _processTask(self, node):
  1365. """!Process task
  1366. @return grassTask instance
  1367. @return None on error
  1368. """
  1369. cmd = list()
  1370. parameterized = list()
  1371. name = node.get('name', None)
  1372. if not name:
  1373. return None
  1374. cmd.append(name)
  1375. # flags
  1376. for f in node.findall('flag'):
  1377. flag = f.get('name', '')
  1378. if f.get('parameterized', '0') == '1':
  1379. parameterized.append(('flag', flag))
  1380. if f.get('value', '1') == '0':
  1381. continue
  1382. if len(flag) > 1:
  1383. cmd.append('--' + flag)
  1384. else:
  1385. cmd.append('-' + flag)
  1386. # parameters
  1387. for p in node.findall('parameter'):
  1388. name = p.get('name', '')
  1389. if p.find('parameterized') is not None:
  1390. parameterized.append(('param', name))
  1391. cmd.append('%s=%s' % (name,
  1392. self._filterValue(self._getNodeText(p, 'value'))))
  1393. task, err = GUI(show = None, checkError = True).ParseCommand(cmd = cmd)
  1394. if err:
  1395. GWarning(os.linesep.join(err))
  1396. for opt, name in parameterized:
  1397. if opt == 'flag':
  1398. task.set_flag(name, True, element = 'parameterized')
  1399. else:
  1400. task.set_param(name, True, element = 'parameterized')
  1401. return task
  1402. def _processLoops(self):
  1403. """!Process model loops"""
  1404. for node in self.root.findall('loop'):
  1405. pos, size = self._getDim(node)
  1406. text = self._filterValue(self._getNodeText(node, 'condition')).strip()
  1407. aid = list()
  1408. for anode in node.findall('item'):
  1409. try:
  1410. aid.append(int(anode.text))
  1411. except ValueError:
  1412. pass
  1413. self.loops.append({ 'pos' : pos,
  1414. 'size' : size,
  1415. 'text' : text,
  1416. 'id' : int(node.get('id', -1)),
  1417. 'items' : aid })
  1418. def _processConditions(self):
  1419. """!Process model conditions"""
  1420. for node in self.root.findall('if-else'):
  1421. pos, size = self._getDim(node)
  1422. text = self._filterValue(self._getNodeText(node, 'condition')).strip()
  1423. aid = { 'if' : list(),
  1424. 'else' : list() }
  1425. for b in aid.keys():
  1426. bnode = node.find(b)
  1427. if bnode is None:
  1428. continue
  1429. for anode in bnode.findall('item'):
  1430. try:
  1431. aid[b].append(int(anode.text))
  1432. except ValueError:
  1433. pass
  1434. self.conditions.append({ 'pos' : pos,
  1435. 'size' : size,
  1436. 'text' : text,
  1437. 'id' : int(node.get('id', -1)),
  1438. 'items' : aid })
  1439. class WriteModelFile:
  1440. """!Generic class for writing model file"""
  1441. def __init__(self, fd, model):
  1442. self.fd = fd
  1443. self.model = model
  1444. self.properties = model.GetProperties()
  1445. self.variables = model.GetVariables()
  1446. self.items = model.GetItems()
  1447. self.indent = 0
  1448. self._header()
  1449. self._window()
  1450. self._properties()
  1451. self._variables()
  1452. self._items()
  1453. dataList = list()
  1454. for action in model.GetItems(objType = ModelAction):
  1455. for rel in action.GetRelations():
  1456. dataItem = rel.GetData()
  1457. if dataItem not in dataList:
  1458. dataList.append(dataItem)
  1459. self._data(dataList)
  1460. self._footer()
  1461. def _filterValue(self, value):
  1462. """!Make value XML-valid"""
  1463. value = value.replace('<', '&lt;')
  1464. value = value.replace('>', '&gt;')
  1465. return value
  1466. def _header(self):
  1467. """!Write header"""
  1468. enc = locale.getdefaultlocale()[1]
  1469. self.fd.write('<?xml version="1.0" encoding="%s"?>\n' % enc)
  1470. self.fd.write('<!DOCTYPE gxm SYSTEM "grass-gxm.dtd">\n')
  1471. self.fd.write('%s<gxm>\n' % (' ' * self.indent))
  1472. self.indent += 4
  1473. def _footer(self):
  1474. """!Write footer"""
  1475. self.indent -= 4
  1476. self.fd.write('%s</gxm>\n' % (' ' * self.indent))
  1477. def _window(self):
  1478. """!Write window properties"""
  1479. canvas = self.model.GetCanvas()
  1480. if canvas is None:
  1481. return
  1482. win = canvas.parent
  1483. pos = win.GetPosition()
  1484. size = win.GetSize()
  1485. self.fd.write('%s<window pos="%d,%d" size="%d,%d" />\n' % \
  1486. (' ' * self.indent, pos[0], pos[1], size[0], size[1]))
  1487. def _properties(self):
  1488. """!Write model properties"""
  1489. self.fd.write('%s<properties>\n' % (' ' * self.indent))
  1490. self.indent += 4
  1491. if self.properties['name']:
  1492. self.fd.write('%s<name>%s</name>\n' % (' ' * self.indent, EncodeString(self.properties['name'])))
  1493. if self.properties['description']:
  1494. self.fd.write('%s<description>%s</description>\n' % (' ' * self.indent,
  1495. EncodeString(self.properties['description'])))
  1496. if self.properties['author']:
  1497. self.fd.write('%s<author>%s</author>\n' % (' ' * self.indent,
  1498. EncodeString(self.properties['author'])))
  1499. if 'overwrite' in self.properties and \
  1500. self.properties['overwrite']:
  1501. self.fd.write('%s<flag name="overwrite" />\n' % (' ' * self.indent))
  1502. self.indent -= 4
  1503. self.fd.write('%s</properties>\n' % (' ' * self.indent))
  1504. def _variables(self):
  1505. """!Write model variables"""
  1506. if not self.variables:
  1507. return
  1508. self.fd.write('%s<variables>\n' % (' ' * self.indent))
  1509. self.indent += 4
  1510. for name, values in self.variables.iteritems():
  1511. self.fd.write('%s<variable name="%s" type="%s">\n' % \
  1512. (' ' * self.indent, EncodeString(name), values['type']))
  1513. self.indent += 4
  1514. if 'value' in values:
  1515. self.fd.write('%s<value>%s</value>\n' % \
  1516. (' ' * self.indent, EncodeString(values['value'])))
  1517. if 'description' in values:
  1518. self.fd.write('%s<description>%s</description>\n' % \
  1519. (' ' * self.indent, EncodeString(values['description'])))
  1520. self.indent -= 4
  1521. self.fd.write('%s</variable>\n' % (' ' * self.indent))
  1522. self.indent -= 4
  1523. self.fd.write('%s</variables>\n' % (' ' * self.indent))
  1524. def _items(self):
  1525. """!Write actions/loops/conditions"""
  1526. for item in self.items:
  1527. if isinstance(item, ModelAction):
  1528. self._action(item)
  1529. elif isinstance(item, ModelLoop):
  1530. self._loop(item)
  1531. elif isinstance(item, ModelCondition):
  1532. self._condition(item)
  1533. def _action(self, action):
  1534. """!Write actions"""
  1535. self.fd.write('%s<action id="%d" name="%s" pos="%d,%d" size="%d,%d">\n' % \
  1536. (' ' * self.indent, action.GetId(), action.GetName(), action.GetX(), action.GetY(),
  1537. action.GetWidth(), action.GetHeight()))
  1538. self.indent += 4
  1539. self.fd.write('%s<task name="%s">\n' % (' ' * self.indent, action.GetLog(string = False)[0]))
  1540. self.indent += 4
  1541. if not action.IsEnabled():
  1542. self.fd.write('%s<disabled />\n' % (' ' * self.indent))
  1543. for key, val in action.GetParams().iteritems():
  1544. if key == 'flags':
  1545. for f in val:
  1546. if f.get('value', False) or f.get('parameterized', False):
  1547. if f.get('parameterized', False):
  1548. if f.get('value', False) == False:
  1549. self.fd.write('%s<flag name="%s" value="0" parameterized="1" />\n' %
  1550. (' ' * self.indent, f.get('name', '')))
  1551. else:
  1552. self.fd.write('%s<flag name="%s" parameterized="1" />\n' %
  1553. (' ' * self.indent, f.get('name', '')))
  1554. else:
  1555. self.fd.write('%s<flag name="%s" />\n' %
  1556. (' ' * self.indent, f.get('name', '')))
  1557. else: # parameter
  1558. for p in val:
  1559. if not p.get('value', '') and not p.get('parameterized', False):
  1560. continue
  1561. self.fd.write('%s<parameter name="%s">\n' %
  1562. (' ' * self.indent, p.get('name', '')))
  1563. self.indent += 4
  1564. if p.get('parameterized', False):
  1565. self.fd.write('%s<parameterized />\n' % (' ' * self.indent))
  1566. self.fd.write('%s<value>%s</value>\n' %
  1567. (' ' * self.indent, self._filterValue(p.get('value', ''))))
  1568. self.indent -= 4
  1569. self.fd.write('%s</parameter>\n' % (' ' * self.indent))
  1570. self.indent -= 4
  1571. self.fd.write('%s</task>\n' % (' ' * self.indent))
  1572. self.indent -= 4
  1573. self.fd.write('%s</action>\n' % (' ' * self.indent))
  1574. def _data(self, dataList):
  1575. """!Write data"""
  1576. for data in dataList:
  1577. self.fd.write('%s<data pos="%d,%d" size="%d,%d">\n' % \
  1578. (' ' * self.indent, data.GetX(), data.GetY(),
  1579. data.GetWidth(), data.GetHeight()))
  1580. self.indent += 4
  1581. self.fd.write('%s<data-parameter prompt="%s">\n' % \
  1582. (' ' * self.indent, data.GetPrompt()))
  1583. self.indent += 4
  1584. self.fd.write('%s<value>%s</value>\n' %
  1585. (' ' * self.indent, self._filterValue(data.GetValue())))
  1586. self.indent -= 4
  1587. self.fd.write('%s</data-parameter>\n' % (' ' * self.indent))
  1588. if data.IsIntermediate():
  1589. self.fd.write('%s<intermediate />\n' % (' ' * self.indent))
  1590. # relations
  1591. for ft in ('from', 'to'):
  1592. for rel in data.GetRelations(ft):
  1593. if ft == 'from':
  1594. aid = rel.GetTo().GetId()
  1595. else:
  1596. aid = rel.GetFrom().GetId()
  1597. self.fd.write('%s<relation dir="%s" id="%d" name="%s">\n' % \
  1598. (' ' * self.indent, ft, aid, rel.GetName()))
  1599. self.indent += 4
  1600. for point in rel.GetLineControlPoints()[1:-1]:
  1601. self.fd.write('%s<point>\n' % (' ' * self.indent))
  1602. self.indent += 4
  1603. x, y = point.Get()
  1604. self.fd.write('%s<x>%d</x>\n' % (' ' * self.indent, int(x)))
  1605. self.fd.write('%s<y>%d</y>\n' % (' ' * self.indent, int(y)))
  1606. self.indent -= 4
  1607. self.fd.write('%s</point>\n' % (' ' * self.indent))
  1608. self.indent -= 4
  1609. self.fd.write('%s</relation>\n' % (' ' * self.indent))
  1610. self.indent -= 4
  1611. self.fd.write('%s</data>\n' % (' ' * self.indent))
  1612. def _loop(self, loop):
  1613. """!Write loops"""
  1614. self.fd.write('%s<loop id="%d" pos="%d,%d" size="%d,%d">\n' % \
  1615. (' ' * self.indent, loop.GetId(), loop.GetX(), loop.GetY(),
  1616. loop.GetWidth(), loop.GetHeight()))
  1617. text = loop.GetText()
  1618. self.indent += 4
  1619. if text:
  1620. self.fd.write('%s<condition>%s</condition>\n' %
  1621. (' ' * self.indent, self._filterValue(text)))
  1622. for item in loop.GetItems():
  1623. self.fd.write('%s<item>%d</item>\n' %
  1624. (' ' * self.indent, item.GetId()))
  1625. self.indent -= 4
  1626. self.fd.write('%s</loop>\n' % (' ' * self.indent))
  1627. def _condition(self, condition):
  1628. """!Write conditions"""
  1629. bbox = condition.GetBoundingBoxMin()
  1630. self.fd.write('%s<if-else id="%d" pos="%d,%d" size="%d,%d">\n' % \
  1631. (' ' * self.indent, condition.GetId(), condition.GetX(), condition.GetY(),
  1632. bbox[0], bbox[1]))
  1633. text = condition.GetText()
  1634. self.indent += 4
  1635. if text:
  1636. self.fd.write('%s<condition>%s</condition>\n' %
  1637. (' ' * self.indent, self._filterValue(text)))
  1638. items = condition.GetItems()
  1639. for b in items.keys():
  1640. if len(items[b]) < 1:
  1641. continue
  1642. self.fd.write('%s<%s>\n' % (' ' * self.indent, b))
  1643. self.indent += 4
  1644. for item in items[b]:
  1645. self.fd.write('%s<item>%d</item>\n' %
  1646. (' ' * self.indent, item.GetId()))
  1647. self.indent -= 4
  1648. self.fd.write('%s</%s>\n' % (' ' * self.indent, b))
  1649. self.indent -= 4
  1650. self.fd.write('%s</if-else>\n' % (' ' * self.indent))
  1651. class WritePythonFile:
  1652. def __init__(self, fd, model):
  1653. """!Class for exporting model to Python script
  1654. @param fd file desciptor
  1655. """
  1656. self.fd = fd
  1657. self.model = model
  1658. self.indent = 4
  1659. self._writePython()
  1660. def _writePython(self):
  1661. """!Write model to file"""
  1662. properties = self.model.GetProperties()
  1663. self.fd.write(
  1664. r"""#!/usr/bin/env python
  1665. #
  1666. #%s
  1667. #
  1668. # MODULE: %s
  1669. #
  1670. # AUTHOR(S): %s
  1671. #
  1672. # PURPOSE: %s
  1673. #
  1674. # DATE: %s
  1675. #
  1676. #%s
  1677. """ % ('#' * 79,
  1678. EncodeString(properties['name']),
  1679. EncodeString(properties['author']),
  1680. EncodeString('\n# '.join(properties['description'].splitlines())),
  1681. time.asctime(),
  1682. '#' * 79))
  1683. self.fd.write(
  1684. r"""
  1685. import sys
  1686. import os
  1687. import atexit
  1688. import grass.script as grass
  1689. """)
  1690. # cleanup()
  1691. rast, vect, rast3d, msg = self.model.GetIntermediateData()
  1692. self.fd.write(
  1693. r"""
  1694. def cleanup():
  1695. """)
  1696. if rast:
  1697. self.fd.write(
  1698. r""" grass.run_command('g.remove',
  1699. rast=%s)
  1700. """ % ','.join(map(lambda x: "'" + x + "'", rast)))
  1701. if vect:
  1702. self.fd.write(
  1703. r""" grass.run_command('g.remove',
  1704. vect = %s)
  1705. """ % ','.join(map(lambda x: "'" + x + "'", vect)))
  1706. if rast3d:
  1707. self.fd.write(
  1708. r""" grass.run_command('g.remove',
  1709. rast3d = %s)
  1710. """ % ','.join(map(lambda x: "'" + x + "'", rast3d)))
  1711. if not rast and not vect and not rast3d:
  1712. self.fd.write(' pass\n')
  1713. self.fd.write("\ndef main():\n")
  1714. for item in self.model.GetItems():
  1715. self._writePythonItem(item)
  1716. self.fd.write("\n return 0\n")
  1717. self.fd.write(
  1718. r"""
  1719. if __name__ == "__main__":
  1720. options, flags = grass.parser()
  1721. atexit.register(cleanup)
  1722. sys.exit(main())
  1723. """)
  1724. def _writePythonItem(self, item, ignoreBlock = True, variables = []):
  1725. """!Write model object to Python file"""
  1726. if isinstance(item, ModelAction):
  1727. if ignoreBlock and item.GetBlockId(): # ignore items in loops of conditions
  1728. return
  1729. self._writePythonAction(item, variables = variables)
  1730. elif isinstance(item, ModelLoop) or isinstance(item, ModelCondition):
  1731. # substitute condition
  1732. variables = self.model.GetVariables()
  1733. cond = item.GetText()
  1734. for variable in variables:
  1735. pattern = re.compile('%' + variable)
  1736. if pattern.search(cond):
  1737. value = variables[variable].get('value', '')
  1738. if variables[variable].get('type', 'string') == 'string':
  1739. value = '"' + value + '"'
  1740. cond = pattern.sub(value, cond)
  1741. if isinstance(item, ModelLoop):
  1742. condVar, condText = map(lambda x: x.strip(), re.split('\s*in\s*', cond))
  1743. cond = "%sfor %s in " % (' ' * self.indent, condVar)
  1744. if condText[0] == '`' and condText[-1] == '`':
  1745. task = GUI(show = None).ParseCommand(cmd = utils.split(condText[1:-1]))
  1746. cond += "grass.read_command("
  1747. cond += self._getPythonActionCmd(task, len(cond), variables = [condVar]) + ".splitlines()"
  1748. else:
  1749. cond += condText
  1750. self.fd.write('%s:\n' % cond)
  1751. self.indent += 4
  1752. for action in item.GetItems():
  1753. self._writePythonItem(action, ignoreBlock = False, variables = [condVar])
  1754. self.indent -= 4
  1755. else: # ModelCondition
  1756. self.fd.write('%sif %s:\n' % (' ' * self.indent, cond))
  1757. self.indent += 4
  1758. condItems = item.GetItems()
  1759. for action in condItems['if']:
  1760. self._writePythonItem(action, ignoreBlock = False)
  1761. if condItems['else']:
  1762. self.indent -= 4
  1763. self.fd.write('%selse:\n' % (' ' * self.indent))
  1764. self.indent += 4
  1765. for action in condItems['else']:
  1766. self._writePythonItem(action, ignoreBlock = False)
  1767. self.indent += 4
  1768. def _writePythonAction(self, item, variables = []):
  1769. """!Write model action to Python file"""
  1770. task = GUI(show = None).ParseCommand(cmd = item.GetLog(string = False, substitute = self.model.GetVariables()))
  1771. strcmd = "%sgrass.run_command(" % (' ' * self.indent)
  1772. self.fd.write(strcmd + self._getPythonActionCmd(task, len(strcmd), variables) + '\n')
  1773. def _getPythonActionCmd(self, task, cmdIndent, variables = []):
  1774. opts = task.get_options()
  1775. ret = ''
  1776. flags = ''
  1777. params = list()
  1778. for f in opts['flags']:
  1779. if f.get('value', False):
  1780. name = f.get('name', '')
  1781. if len(name) > 1:
  1782. params.append('%s = True' % name)
  1783. else:
  1784. flags += name
  1785. for p in opts['params']:
  1786. name = p.get('name', None)
  1787. value = p.get('value', None)
  1788. if name and value:
  1789. ptype = p.get('type', 'string')
  1790. if value[0] == '%':
  1791. params.append("%s = %s" % (name, value[1:]))
  1792. elif ptype == 'string':
  1793. params.append('%s = "%s"' % (name, value))
  1794. else:
  1795. params.append("%s = %s" % (name, value))
  1796. ret += '"%s"' % task.get_name()
  1797. if flags:
  1798. ret += ",\n%sflags = '%s'" % (' ' * cmdIndent, flags)
  1799. if len(params) > 0:
  1800. ret += ",\n"
  1801. for opt in params[:-1]:
  1802. ret += "%s%s,\n" % (' ' * cmdIndent, opt)
  1803. ret += "%s%s)" % (' ' * cmdIndent, params[-1])
  1804. else:
  1805. ret += ")"
  1806. return ret
  1807. class ModelParamDialog(wx.Dialog):
  1808. def __init__(self, parent, params, id = wx.ID_ANY, title = _("Model parameters"),
  1809. style = wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER, **kwargs):
  1810. """!Model parameters dialog
  1811. """
  1812. self.parent = parent
  1813. self.params = params
  1814. self.tasks = list() # list of tasks/pages
  1815. wx.Dialog.__init__(self, parent = parent, id = id, title = title, style = style, **kwargs)
  1816. self.notebook = GNotebook(parent = self,
  1817. style = globalvar.FNPageDStyle)
  1818. panel = self._createPages()
  1819. wx.CallAfter(self.notebook.SetSelection, 0)
  1820. # intermediate data?
  1821. self.interData = wx.CheckBox(parent = self, label = _("Delete intermediate data when finish"))
  1822. self.interData.SetValue(True)
  1823. rast, vect, rast3d, msg = self.parent.GetModel().GetIntermediateData()
  1824. if not rast and not vect and not rast3d:
  1825. self.interData.Hide()
  1826. self.btnCancel = wx.Button(parent = self, id = wx.ID_CANCEL)
  1827. self.btnRun = wx.Button(parent = self, id = wx.ID_OK,
  1828. label = _("&Run"))
  1829. self.btnRun.SetDefault()
  1830. self._layout()
  1831. size = self.GetBestSize()
  1832. self.SetMinSize(size)
  1833. self.SetSize((size.width, size.height +
  1834. panel.constrained_size[1] -
  1835. panel.panelMinHeight))
  1836. def _layout(self):
  1837. btnSizer = wx.StdDialogButtonSizer()
  1838. btnSizer.AddButton(self.btnCancel)
  1839. btnSizer.AddButton(self.btnRun)
  1840. btnSizer.Realize()
  1841. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1842. mainSizer.Add(item = self.notebook, proportion = 1,
  1843. flag = wx.EXPAND)
  1844. if self.interData.IsShown():
  1845. mainSizer.Add(item = self.interData, proportion = 0,
  1846. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1847. mainSizer.Add(item = wx.StaticLine(parent = self, id = wx.ID_ANY,
  1848. style = wx.LI_HORIZONTAL),
  1849. proportion = 0,
  1850. flag = wx.EXPAND | wx.LEFT | wx.RIGHT, border = 5)
  1851. mainSizer.Add(item = btnSizer, proportion = 0,
  1852. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1853. self.SetSizer(mainSizer)
  1854. mainSizer.Fit(self)
  1855. def _createPages(self):
  1856. """!Create for each parameterized module its own page"""
  1857. nameOrdered = [''] * len(self.params.keys())
  1858. for name, params in self.params.iteritems():
  1859. nameOrdered[params['idx']] = name
  1860. for name in nameOrdered:
  1861. params = self.params[name]
  1862. panel = self._createPage(name, params)
  1863. if name == 'variables':
  1864. name = _('Variables')
  1865. self.notebook.AddPage(page = panel, text = name)
  1866. return panel
  1867. def _createPage(self, name, params):
  1868. """!Define notebook page"""
  1869. if name in globalvar.grassCmd:
  1870. task = gtask.grassTask(name)
  1871. else:
  1872. task = gtask.grassTask()
  1873. task.flags = params['flags']
  1874. task.params = params['params']
  1875. panel = CmdPanel(parent = self, id = wx.ID_ANY, task = task)
  1876. self.tasks.append(task)
  1877. return panel
  1878. def GetErrors(self):
  1879. """!Check for errors, get list of messages"""
  1880. errList = list()
  1881. for task in self.tasks:
  1882. errList += task.get_cmd_error()
  1883. return errList
  1884. def DeleteIntermediateData(self):
  1885. """!Check if to detele intermediate data"""
  1886. if self.interData.IsShown() and self.interData.IsChecked():
  1887. return True
  1888. return False