dialogs.py 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177
  1. """
  2. @package gmodeler.dialogs
  3. @brief wxGUI Graphical Modeler - dialogs
  4. Classes:
  5. - dialogs::ModelDataDialog
  6. - dialogs::ModelSearchDialog
  7. - dialogs::ModelRelationDialog
  8. - dialogs::ModelItemDialog
  9. - dialogs::ModelLoopDialog
  10. - dialogs::ModelConditionDialog
  11. - dialogs::ModelListCtrl
  12. - dialogs::ValiableListCtrl
  13. - dialogs::ItemListCtrl
  14. - dialogs::ItemCheckListCtrl
  15. (C) 2010-2016 by the GRASS Development Team
  16. This program is free software under the GNU General Public License
  17. (>=v2). Read the file COPYING that comes with GRASS for details.
  18. @author Martin Landa <landa.martin gmail.com>
  19. """
  20. import os
  21. import six
  22. import wx
  23. import wx.lib.mixins.listctrl as listmix
  24. from core import globalvar
  25. from core import utils
  26. from gui_core.widgets import SearchModuleWidget, SimpleValidator
  27. from core.gcmd import GError
  28. from gui_core.dialogs import SimpleDialog, MapLayersDialogForModeler
  29. from gui_core.prompt import GPromptSTC
  30. from gui_core.gselect import Select, ElementSelect
  31. from gmodeler.model import *
  32. from lmgr.menudata import LayerManagerMenuData
  33. from gui_core.wrap import (
  34. Button,
  35. StaticText,
  36. StaticBox,
  37. TextCtrl,
  38. Menu,
  39. ListCtrl,
  40. NewId,
  41. CheckListCtrlMixin,
  42. )
  43. class ModelDataDialog(SimpleDialog):
  44. """Data item properties dialog"""
  45. def __init__(self, parent, shape, title=_("Data properties")):
  46. self.parent = parent
  47. self.shape = shape
  48. label, etype = self._getLabel()
  49. self.etype = etype
  50. SimpleDialog.__init__(self, parent, title)
  51. self.element = Select(
  52. parent=self.panel,
  53. type=self.shape.GetPrompt(),
  54. validator=SimpleValidator(callback=self.ValidatorCallback),
  55. )
  56. if shape.GetValue():
  57. self.element.SetValue(shape.GetValue())
  58. self.Bind(wx.EVT_BUTTON, self.OnOK, self.btnOK)
  59. self.Bind(wx.EVT_BUTTON, self.OnCancel, self.btnCancel)
  60. if self.etype:
  61. self.typeSelect = ElementSelect(
  62. parent=self.panel,
  63. elements=["raster", "raster_3d", "vector"],
  64. size=globalvar.DIALOG_GSELECT_SIZE,
  65. )
  66. self.typeSelect.Bind(wx.EVT_CHOICE, self.OnType)
  67. self.typeSelect.SetSelection(0)
  68. self.element.SetType("raster")
  69. if shape.GetValue():
  70. self.btnOK.Enable()
  71. self._layout()
  72. self.SetMinSize(self.GetSize())
  73. def _getLabel(self):
  74. etype = False
  75. prompt = self.shape.GetPrompt()
  76. if prompt == "raster":
  77. label = _("Name of raster map:")
  78. elif prompt == "vector":
  79. label = _("Name of vector map:")
  80. else:
  81. etype = True
  82. label = _("Name of element:")
  83. return label, etype
  84. def _layout(self):
  85. """Do layout"""
  86. if self.etype:
  87. self.dataSizer.Add(
  88. StaticText(
  89. parent=self.panel, id=wx.ID_ANY, label=_("Type of element:")
  90. ),
  91. proportion=0,
  92. flag=wx.ALL,
  93. border=1,
  94. )
  95. self.dataSizer.Add(self.typeSelect, proportion=0, flag=wx.ALL, border=1)
  96. self.dataSizer.Add(
  97. StaticText(parent=self.panel, id=wx.ID_ANY, label=_("Name of element:")),
  98. proportion=0,
  99. flag=wx.ALL,
  100. border=1,
  101. )
  102. self.dataSizer.Add(
  103. self.element, proportion=0, flag=wx.EXPAND | wx.ALL, border=1
  104. )
  105. self.panel.SetSizer(self.sizer)
  106. self.sizer.Fit(self)
  107. def GetType(self):
  108. """Get element type"""
  109. if not self.etype:
  110. return
  111. return self.element.tcp.GetType()
  112. def OnType(self, event):
  113. """Select element type"""
  114. evalue = self.typeSelect.GetValue(event.GetString())
  115. self.element.SetType(evalue)
  116. def OnOK(self, event):
  117. """Ok pressed"""
  118. self.shape.SetValue(self.element.GetValue())
  119. if self.etype:
  120. elem = self.GetType()
  121. if elem == "raster":
  122. self.shape.SetPrompt("raster")
  123. elif elem == "vector":
  124. self.shape.SetPrompt("vector")
  125. elif elem == "raster_3d":
  126. self.shape.SetPrompt("raster_3d")
  127. self.parent.canvas.Refresh()
  128. self.parent.SetStatusText("", 0)
  129. self.shape.SetPropDialog(None)
  130. if self.IsModal():
  131. event.Skip()
  132. else:
  133. self.Destroy()
  134. def OnCancel(self, event):
  135. """Cancel pressed"""
  136. self.shape.SetPropDialog(None)
  137. if self.IsModal():
  138. event.Skip()
  139. else:
  140. self.Destroy()
  141. class ModelSearchDialog(wx.Dialog):
  142. def __init__(
  143. self,
  144. parent,
  145. giface,
  146. title=_("Add GRASS command to the model"),
  147. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  148. **kwargs,
  149. ):
  150. """Graphical modeler module search window
  151. :param parent: parent window
  152. :param id: window id
  153. :param title: window title
  154. :param kwargs: wx.Dialogs' arguments
  155. """
  156. self.parent = parent
  157. wx.Dialog.__init__(self, parent=parent, id=wx.ID_ANY, title=title, **kwargs)
  158. self.SetName("ModelerDialog")
  159. self.SetIcon(
  160. wx.Icon(os.path.join(globalvar.ICONDIR, "grass.ico"), wx.BITMAP_TYPE_ICO)
  161. )
  162. self._command = None
  163. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  164. self.cmdBox = StaticBox(
  165. parent=self.panel, id=wx.ID_ANY, label=" %s " % _("Command")
  166. )
  167. self.labelBox = StaticBox(
  168. parent=self.panel, id=wx.ID_ANY, label=" %s " % _("Label and comment")
  169. )
  170. # menu data for search widget and prompt
  171. menuModel = LayerManagerMenuData()
  172. self.cmd_prompt = GPromptSTC(
  173. parent=self, giface=giface, menuModel=menuModel.GetModel()
  174. )
  175. self.cmd_prompt.promptRunCmd.connect(self.OnCommand)
  176. self.cmd_prompt.commandSelected.connect(
  177. lambda command: self.label.SetValue(command)
  178. )
  179. self.search = SearchModuleWidget(
  180. parent=self.panel, model=menuModel.GetModel(), showTip=True
  181. )
  182. self.search.moduleSelected.connect(
  183. lambda name: self.cmd_prompt.SetTextAndFocus(name + " ")
  184. )
  185. wx.CallAfter(self.cmd_prompt.SetFocus)
  186. self.label = TextCtrl(parent=self.panel, id=wx.ID_ANY)
  187. self.comment = TextCtrl(parent=self.panel, id=wx.ID_ANY, style=wx.TE_MULTILINE)
  188. self.btnCancel = Button(self.panel, wx.ID_CANCEL)
  189. self.btnOk = Button(self.panel, wx.ID_OK)
  190. self.btnOk.SetDefault()
  191. self.Bind(wx.EVT_BUTTON, self.OnOk, self.btnOk)
  192. self.Bind(wx.EVT_BUTTON, self.OnCancel, self.btnCancel)
  193. self._layout()
  194. self.SetSize((500, -1))
  195. def _layout(self):
  196. cmdSizer = wx.StaticBoxSizer(self.cmdBox, wx.VERTICAL)
  197. cmdSizer.Add(self.cmd_prompt, proportion=1, flag=wx.EXPAND)
  198. labelSizer = wx.StaticBoxSizer(self.labelBox, wx.VERTICAL)
  199. gridSizer = wx.GridBagSizer(hgap=5, vgap=5)
  200. gridSizer.Add(
  201. StaticText(parent=self.panel, id=wx.ID_ANY, label=_("Label:")),
  202. flag=wx.ALIGN_CENTER_VERTICAL,
  203. pos=(0, 0),
  204. )
  205. gridSizer.Add(self.label, pos=(0, 1), flag=wx.EXPAND)
  206. gridSizer.Add(
  207. StaticText(parent=self.panel, id=wx.ID_ANY, label=_("Comment:")),
  208. flag=wx.ALIGN_CENTER_VERTICAL,
  209. pos=(1, 0),
  210. )
  211. gridSizer.Add(self.comment, pos=(1, 1), flag=wx.EXPAND)
  212. gridSizer.AddGrowableRow(1)
  213. gridSizer.AddGrowableCol(1)
  214. labelSizer.Add(gridSizer, proportion=1, flag=wx.EXPAND)
  215. btnSizer = wx.StdDialogButtonSizer()
  216. btnSizer.AddButton(self.btnCancel)
  217. btnSizer.AddButton(self.btnOk)
  218. btnSizer.Realize()
  219. mainSizer = wx.BoxSizer(wx.VERTICAL)
  220. mainSizer.Add(self.search, proportion=0, flag=wx.EXPAND | wx.ALL, border=3)
  221. mainSizer.Add(
  222. cmdSizer,
  223. proportion=1,
  224. flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.TOP,
  225. border=3,
  226. )
  227. mainSizer.Add(
  228. labelSizer,
  229. proportion=1,
  230. flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.TOP,
  231. border=3,
  232. )
  233. mainSizer.Add(btnSizer, proportion=0, flag=wx.EXPAND | wx.ALL, border=5)
  234. self.panel.SetSizer(mainSizer)
  235. mainSizer.Fit(self)
  236. self.Layout()
  237. def GetPanel(self):
  238. """Get dialog panel"""
  239. return self.panel
  240. def _getCmd(self):
  241. line = self.cmd_prompt.GetCurLine()[0].strip()
  242. if len(line) == 0:
  243. cmd = list()
  244. else:
  245. cmd = utils.split(str(line))
  246. return cmd
  247. def GetCmd(self):
  248. """Get command"""
  249. return self._command
  250. def GetLabel(self):
  251. """Get label and comment"""
  252. return self.label.GetValue(), self.comment.GetValue()
  253. def ValidateCmd(self, cmd):
  254. if len(cmd) < 1:
  255. GError(
  256. parent=self,
  257. message=_(
  258. "Command not defined.\n\n" "Unable to add new action to the model."
  259. ),
  260. )
  261. return False
  262. if cmd[0] not in globalvar.grassCmd:
  263. GError(
  264. parent=self,
  265. message=_(
  266. "'%s' is not a GRASS module.\n\n"
  267. "Unable to add new action to the model."
  268. )
  269. % cmd[0],
  270. )
  271. return False
  272. return True
  273. def OnCommand(self, cmd):
  274. """Command in prompt confirmed"""
  275. if self.ValidateCmd(cmd):
  276. self._command = cmd
  277. self.EndModal(wx.ID_OK)
  278. def OnOk(self, event):
  279. """Button 'OK' pressed"""
  280. cmd = self._getCmd()
  281. if self.ValidateCmd(cmd):
  282. self._command = cmd
  283. self.EndModal(wx.ID_OK)
  284. def OnCancel(self, event):
  285. """Cancel pressed, close window"""
  286. self.Hide()
  287. def Reset(self):
  288. """Reset dialog"""
  289. self.search.Reset()
  290. self.label.SetValue("")
  291. self.comment.SetValue("")
  292. self.cmd_prompt.OnCmdErase(None)
  293. self.cmd_prompt.SetFocus()
  294. class ModelRelationDialog(wx.Dialog):
  295. """Relation properties dialog"""
  296. def __init__(
  297. self,
  298. parent,
  299. shape,
  300. id=wx.ID_ANY,
  301. title=_("Relation properties"),
  302. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  303. **kwargs,
  304. ):
  305. self.parent = parent
  306. self.shape = shape
  307. options = self._getOptions()
  308. if not options:
  309. self.valid = False
  310. return
  311. self.valid = True
  312. wx.Dialog.__init__(self, parent, id, title, style=style, **kwargs)
  313. self.SetIcon(
  314. wx.Icon(os.path.join(globalvar.ICONDIR, "grass.ico"), wx.BITMAP_TYPE_ICO)
  315. )
  316. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  317. self.fromBox = StaticBox(
  318. parent=self.panel, id=wx.ID_ANY, label=" %s " % _("From")
  319. )
  320. self.toBox = StaticBox(parent=self.panel, id=wx.ID_ANY, label=" %s " % _("To"))
  321. self.option = wx.ComboBox(
  322. parent=self.panel, id=wx.ID_ANY, style=wx.CB_READONLY, choices=options
  323. )
  324. self.option.Bind(wx.EVT_COMBOBOX, self.OnOption)
  325. self.btnCancel = Button(self.panel, wx.ID_CANCEL)
  326. self.btnOk = Button(self.panel, wx.ID_OK)
  327. self.btnOk.Enable(False)
  328. self._layout()
  329. def _layout(self):
  330. mainSizer = wx.BoxSizer(wx.VERTICAL)
  331. fromSizer = wx.StaticBoxSizer(self.fromBox, wx.VERTICAL)
  332. self._layoutShape(shape=self.shape.GetFrom(), sizer=fromSizer)
  333. toSizer = wx.StaticBoxSizer(self.toBox, wx.VERTICAL)
  334. self._layoutShape(shape=self.shape.GetTo(), sizer=toSizer)
  335. btnSizer = wx.StdDialogButtonSizer()
  336. btnSizer.AddButton(self.btnCancel)
  337. btnSizer.AddButton(self.btnOk)
  338. btnSizer.Realize()
  339. mainSizer.Add(fromSizer, proportion=0, flag=wx.EXPAND | wx.ALL, border=5)
  340. mainSizer.Add(
  341. toSizer,
  342. proportion=0,
  343. flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM,
  344. border=5,
  345. )
  346. mainSizer.Add(
  347. btnSizer, proportion=0, flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5
  348. )
  349. self.panel.SetSizer(mainSizer)
  350. mainSizer.Fit(self.panel)
  351. self.Layout()
  352. self.SetSize(self.GetBestSize())
  353. def _layoutShape(self, shape, sizer):
  354. if isinstance(shape, ModelData):
  355. sizer.Add(
  356. StaticText(
  357. parent=self.panel,
  358. id=wx.ID_ANY,
  359. label=_("Data: %s") % shape.GetLog(),
  360. ),
  361. proportion=1,
  362. flag=wx.EXPAND | wx.ALL,
  363. border=5,
  364. )
  365. elif isinstance(shape, ModelAction):
  366. gridSizer = wx.GridBagSizer(hgap=5, vgap=5)
  367. gridSizer.Add(
  368. StaticText(parent=self.panel, id=wx.ID_ANY, label=_("Command:")),
  369. pos=(0, 0),
  370. )
  371. gridSizer.Add(
  372. StaticText(parent=self.panel, id=wx.ID_ANY, label=shape.GetLabel()),
  373. pos=(0, 1),
  374. )
  375. gridSizer.Add(
  376. StaticText(parent=self.panel, id=wx.ID_ANY, label=_("Option:")),
  377. flag=wx.ALIGN_CENTER_VERTICAL,
  378. pos=(1, 0),
  379. )
  380. gridSizer.Add(self.option, pos=(1, 1))
  381. sizer.Add(gridSizer, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  382. def _getOptions(self):
  383. """Get relevant options"""
  384. items = []
  385. fromShape = self.shape.GetFrom()
  386. if not isinstance(fromShape, ModelData):
  387. GError(
  388. parent=self.parent,
  389. message=_(
  390. "Relation doesn't start with data item.\n" "Unable to add relation."
  391. ),
  392. )
  393. return items
  394. toShape = self.shape.GetTo()
  395. if not isinstance(toShape, ModelAction):
  396. GError(
  397. parent=self.parent,
  398. message=_(
  399. "Relation doesn't point to GRASS command.\n"
  400. "Unable to add relation."
  401. ),
  402. )
  403. return items
  404. prompt = fromShape.GetPrompt()
  405. task = toShape.GetTask()
  406. for p in task.get_options()["params"]:
  407. if p.get("prompt", "") == prompt and "name" in p:
  408. items.append(p["name"])
  409. if not items:
  410. GError(
  411. parent=self.parent,
  412. message=_("No relevant option found.\n" "Unable to add relation."),
  413. )
  414. return items
  415. def GetOption(self):
  416. """Get selected option"""
  417. return self.option.GetStringSelection()
  418. def IsValid(self):
  419. """Check if relation is valid"""
  420. return self.valid
  421. def OnOption(self, event):
  422. """Set option"""
  423. if event.GetString():
  424. self.btnOk.Enable()
  425. else:
  426. self.btnOk.Enable(False)
  427. class ModelItemDialog(wx.Dialog):
  428. """Abstract item properties dialog"""
  429. def __init__(
  430. self,
  431. parent,
  432. shape,
  433. title,
  434. id=wx.ID_ANY,
  435. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  436. **kwargs,
  437. ):
  438. self.parent = parent
  439. self.shape = shape
  440. wx.Dialog.__init__(self, parent, id, title=title, style=style, **kwargs)
  441. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  442. self.condBox = StaticBox(
  443. parent=self.panel, id=wx.ID_ANY, label=" %s " % _("Condition")
  444. )
  445. self.condText = TextCtrl(
  446. parent=self.panel, id=wx.ID_ANY, value=shape.GetLabel()
  447. )
  448. self.itemList = ItemCheckListCtrl(
  449. parent=self.panel,
  450. columns=[_("Label"), _("Command")],
  451. shape=shape,
  452. frame=parent,
  453. )
  454. self.itemList.Populate(self.parent.GetModel().GetItems())
  455. self.btnCancel = Button(parent=self.panel, id=wx.ID_CANCEL)
  456. self.btnOk = Button(parent=self.panel, id=wx.ID_OK)
  457. self.btnOk.SetDefault()
  458. def _layout(self):
  459. """Do layout (virtual method)"""
  460. pass
  461. def GetCondition(self):
  462. """Get loop condition"""
  463. return self.condText.GetValue()
  464. class ModelLoopDialog(ModelItemDialog):
  465. """Loop properties dialog"""
  466. def __init__(
  467. self,
  468. parent,
  469. shape,
  470. id=wx.ID_ANY,
  471. title=_("Loop properties"),
  472. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  473. **kwargs,
  474. ):
  475. ModelItemDialog.__init__(self, parent, shape, title, style=style, **kwargs)
  476. self.listBox = StaticBox(
  477. parent=self.panel, id=wx.ID_ANY, label=" %s " % _("List of items in loop")
  478. )
  479. self.btnSeries = Button(parent=self.panel, id=wx.ID_ANY, label=_("Series"))
  480. self.btnSeries.SetToolTip(_("Define map series as condition for the loop"))
  481. self.btnSeries.Bind(wx.EVT_BUTTON, self.OnSeries)
  482. self._layout()
  483. self.SetMinSize(self.GetSize())
  484. self.SetSize((500, 400))
  485. def _layout(self):
  486. """Do layout"""
  487. sizer = wx.BoxSizer(wx.VERTICAL)
  488. condSizer = wx.StaticBoxSizer(self.condBox, wx.HORIZONTAL)
  489. condSizer.Add(self.condText, proportion=1, flag=wx.ALL, border=3)
  490. condSizer.Add(self.btnSeries, proportion=0, flag=wx.EXPAND)
  491. listSizer = wx.StaticBoxSizer(self.listBox, wx.VERTICAL)
  492. listSizer.Add(self.itemList, proportion=1, flag=wx.EXPAND | wx.ALL, border=3)
  493. btnSizer = wx.StdDialogButtonSizer()
  494. btnSizer.AddButton(self.btnCancel)
  495. btnSizer.AddButton(self.btnOk)
  496. btnSizer.Realize()
  497. sizer.Add(condSizer, proportion=0, flag=wx.EXPAND | wx.ALL, border=3)
  498. sizer.Add(
  499. listSizer, proportion=1, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=3
  500. )
  501. sizer.Add(
  502. btnSizer, proportion=0, flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5
  503. )
  504. self.panel.SetSizer(sizer)
  505. sizer.Fit(self.panel)
  506. self.Layout()
  507. def GetItems(self):
  508. """Get list of selected actions"""
  509. return self.itemList.GetItems()
  510. def OnSeries(self, event):
  511. """Define map series as condition"""
  512. dialog = MapLayersDialogForModeler(
  513. parent=self, title=_("Define series of maps")
  514. )
  515. if dialog.ShowModal() != wx.ID_OK:
  516. dialog.Destroy()
  517. return
  518. cond = dialog.GetDSeries()
  519. if not cond:
  520. cond = "map in {}".format(list(map(str, dialog.GetMapLayers())))
  521. self.condText.SetValue(cond)
  522. dialog.Destroy()
  523. class ModelConditionDialog(ModelItemDialog):
  524. """Condition properties dialog"""
  525. def __init__(
  526. self,
  527. parent,
  528. shape,
  529. id=wx.ID_ANY,
  530. title=_("If-else properties"),
  531. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  532. **kwargs,
  533. ):
  534. ModelItemDialog.__init__(self, parent, shape, title, style=style, **kwargs)
  535. self.listBoxIf = StaticBox(
  536. parent=self.panel,
  537. id=wx.ID_ANY,
  538. label=" %s " % _("List of items in 'if' block"),
  539. )
  540. self.itemListIf = self.itemList
  541. self.itemListIf.SetName("IfBlockList")
  542. self.listBoxElse = StaticBox(
  543. parent=self.panel,
  544. id=wx.ID_ANY,
  545. label=" %s " % _("List of items in 'else' block"),
  546. )
  547. self.itemListElse = ItemCheckListCtrl(
  548. parent=self.panel,
  549. columns=[_("Label"), _("Command")],
  550. shape=shape,
  551. frame=parent,
  552. )
  553. self.itemListElse.SetName("ElseBlockList")
  554. self.itemListElse.Populate(self.parent.GetModel().GetItems())
  555. self._layout()
  556. self.SetMinSize(self.GetSize())
  557. self.SetSize((500, 400))
  558. def _layout(self):
  559. """Do layout"""
  560. sizer = wx.BoxSizer(wx.VERTICAL)
  561. condSizer = wx.StaticBoxSizer(self.condBox, wx.VERTICAL)
  562. condSizer.Add(self.condText, proportion=1, flag=wx.EXPAND)
  563. listIfSizer = wx.StaticBoxSizer(self.listBoxIf, wx.VERTICAL)
  564. listIfSizer.Add(self.itemListIf, proportion=1, flag=wx.EXPAND)
  565. listElseSizer = wx.StaticBoxSizer(self.listBoxElse, wx.VERTICAL)
  566. listElseSizer.Add(self.itemListElse, proportion=1, flag=wx.EXPAND)
  567. btnSizer = wx.StdDialogButtonSizer()
  568. btnSizer.AddButton(self.btnCancel)
  569. btnSizer.AddButton(self.btnOk)
  570. btnSizer.Realize()
  571. sizer.Add(condSizer, proportion=0, flag=wx.EXPAND | wx.ALL, border=3)
  572. sizer.Add(
  573. listIfSizer, proportion=1, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=3
  574. )
  575. sizer.Add(
  576. listElseSizer, proportion=1, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=3
  577. )
  578. sizer.Add(
  579. btnSizer, proportion=0, flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5
  580. )
  581. self.panel.SetSizer(sizer)
  582. sizer.Fit(self.panel)
  583. self.Layout()
  584. def OnCheckItemIf(self, index, flag):
  585. """Item in if-block checked/unchecked"""
  586. if flag is False:
  587. return
  588. aId = int(self.itemListIf.GetItem(index, 0).GetText())
  589. if aId in self.itemListElse.GetItems()["checked"]:
  590. self.itemListElse.CheckItemById(aId, False)
  591. def OnCheckItemElse(self, index, flag):
  592. """Item in else-block checked/unchecked"""
  593. if flag is False:
  594. return
  595. aId = int(self.itemListElse.GetItem(index, 0).GetText())
  596. if aId in self.itemListIf.GetItems()["checked"]:
  597. self.itemListIf.CheckItemById(aId, False)
  598. def GetItems(self):
  599. """Get items"""
  600. return {"if": self.itemListIf.GetItems(), "else": self.itemListElse.GetItems()}
  601. class ModelListCtrl(ListCtrl, listmix.ListCtrlAutoWidthMixin, listmix.TextEditMixin):
  602. def __init__(
  603. self,
  604. parent,
  605. columns,
  606. frame,
  607. id=wx.ID_ANY,
  608. columnsNotEditable=[],
  609. style=wx.LC_REPORT | wx.BORDER_NONE | wx.LC_HRULES | wx.LC_VRULES,
  610. **kwargs,
  611. ):
  612. """List of model variables"""
  613. self.parent = parent
  614. self.columns = columns
  615. self.shape = None
  616. self.frame = frame
  617. self.columnNotEditable = columnsNotEditable
  618. ListCtrl.__init__(self, parent, id=id, style=style, **kwargs)
  619. listmix.ListCtrlAutoWidthMixin.__init__(self)
  620. listmix.TextEditMixin.__init__(self)
  621. i = 0
  622. for col in columns:
  623. self.InsertColumn(i, col)
  624. self.SetColumnWidth(i, wx.LIST_AUTOSIZE_USEHEADER)
  625. i += 1
  626. self.itemDataMap = {} # requested by sorter
  627. self.itemCount = 0
  628. self.Bind(wx.EVT_LIST_BEGIN_LABEL_EDIT, self.OnBeginEdit)
  629. self.Bind(wx.EVT_LIST_END_LABEL_EDIT, self.OnEndEdit)
  630. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick)
  631. self.Bind(wx.EVT_COMMAND_RIGHT_CLICK, self.OnRightUp) # wxMSW
  632. self.Bind(wx.EVT_RIGHT_UP, self.OnRightUp) # wxGTK
  633. def OnBeginEdit(self, event):
  634. """Editing of item started"""
  635. column = event.GetColumn()
  636. if self.columnNotEditable and column in self.columnNotEditable:
  637. event.Veto()
  638. self.SetItemState(
  639. event.GetIndex(),
  640. wx.LIST_STATE_SELECTED,
  641. wx.LIST_STATE_SELECTED | wx.LIST_STATE_FOCUSED,
  642. )
  643. else:
  644. event.Allow()
  645. def OnEndEdit(self, event):
  646. """Finish editing of item"""
  647. pass
  648. def GetListCtrl(self):
  649. """Used by ColumnSorterMixin"""
  650. return self
  651. def OnColClick(self, event):
  652. """Click on column header (order by)"""
  653. event.Skip()
  654. class VariableListCtrl(ModelListCtrl):
  655. def __init__(self, parent, columns, **kwargs):
  656. """List of model variables"""
  657. ModelListCtrl.__init__(self, parent, columns, **kwargs)
  658. self.SetColumnWidth(2, 200) # default value
  659. def GetData(self):
  660. """Get list data"""
  661. return self.itemDataMap
  662. def Populate(self, data):
  663. """Populate the list"""
  664. self.itemDataMap = dict()
  665. i = 0
  666. for name, values in six.iteritems(data):
  667. self.itemDataMap[i] = [
  668. name,
  669. values["type"],
  670. values.get("value", ""),
  671. values.get("description", ""),
  672. ]
  673. i += 1
  674. self.itemCount = len(self.itemDataMap.keys())
  675. self.DeleteAllItems()
  676. i = 0
  677. for name, vtype, value, desc in six.itervalues(self.itemDataMap):
  678. index = self.InsertItem(i, name)
  679. self.SetItem(index, 0, name)
  680. self.SetItem(index, 1, vtype)
  681. self.SetItem(index, 2, value)
  682. self.SetItem(index, 3, desc)
  683. self.SetItemData(index, i)
  684. i += 1
  685. def Append(self, name, vtype, value, desc):
  686. """Append new item to the list
  687. :return: None on success
  688. :return: error string
  689. """
  690. for iname, ivtype, ivalue, idesc in six.itervalues(self.itemDataMap):
  691. if iname == name:
  692. return (
  693. _(
  694. "Variable <%s> already exists in the model. "
  695. "Adding variable failed."
  696. )
  697. % name
  698. )
  699. index = self.InsertItem(self.GetItemCount(), name)
  700. self.SetItem(index, 0, name)
  701. self.SetItem(index, 1, vtype)
  702. self.SetItem(index, 2, value)
  703. self.SetItem(index, 3, desc)
  704. self.SetItemData(index, self.itemCount)
  705. self.itemDataMap[self.itemCount] = [name, vtype, value, desc]
  706. self.itemCount += 1
  707. return None
  708. def OnRemove(self, event):
  709. """Remove selected variable(s) from the model"""
  710. item = self.GetFirstSelected()
  711. while item != -1:
  712. self.DeleteItem(item)
  713. del self.itemDataMap[item]
  714. item = self.GetFirstSelected()
  715. self.parent.UpdateModelVariables()
  716. event.Skip()
  717. def OnRemoveAll(self, event):
  718. """Remove all variable(s) from the model"""
  719. dlg = wx.MessageBox(
  720. parent=self,
  721. message=_("Do you want to delete all variables from " "the model?"),
  722. caption=_("Delete variables"),
  723. style=wx.YES_NO | wx.CENTRE,
  724. )
  725. if dlg != wx.YES:
  726. return
  727. self.DeleteAllItems()
  728. self.itemDataMap = dict()
  729. self.parent.UpdateModelVariables()
  730. def OnEndEdit(self, event):
  731. """Finish editing of item"""
  732. itemIndex = event.GetIndex()
  733. columnIndex = event.GetColumn()
  734. nameOld = self.GetItem(itemIndex, 0).GetText()
  735. if columnIndex == 0: # TODO
  736. event.Veto()
  737. self.itemDataMap[itemIndex][columnIndex] = event.GetText()
  738. self.parent.UpdateModelVariables()
  739. def OnReload(self, event):
  740. """Reload list of variables"""
  741. self.Populate(self.parent.parent.GetModel().GetVariables())
  742. def OnRightUp(self, event):
  743. """Mouse right button up"""
  744. if not hasattr(self, "popupID1"):
  745. self.popupID1 = NewId()
  746. self.popupID2 = NewId()
  747. self.popupID3 = NewId()
  748. self.Bind(wx.EVT_MENU, self.OnRemove, id=self.popupID1)
  749. self.Bind(wx.EVT_MENU, self.OnRemoveAll, id=self.popupID2)
  750. self.Bind(wx.EVT_MENU, self.OnReload, id=self.popupID3)
  751. # generate popup-menu
  752. menu = Menu()
  753. menu.Append(self.popupID1, _("Delete selected"))
  754. menu.Append(self.popupID2, _("Delete all"))
  755. if self.GetFirstSelected() == -1:
  756. menu.Enable(self.popupID1, False)
  757. menu.Enable(self.popupID2, False)
  758. menu.AppendSeparator()
  759. menu.Append(self.popupID3, _("Reload"))
  760. self.PopupMenu(menu)
  761. menu.Destroy()
  762. class ItemListCtrl(ModelListCtrl):
  763. def __init__(self, parent, columns, frame, disablePopup=False, **kwargs):
  764. """List of model actions"""
  765. self.disablePopup = disablePopup
  766. ModelListCtrl.__init__(self, parent, columns, frame, **kwargs)
  767. self.itemIdMap = list()
  768. self.SetColumnWidth(0, 100)
  769. self.SetColumnWidth(1, 75)
  770. if len(self.columns) >= 3:
  771. self.SetColumnWidth(2, 100)
  772. def GetData(self):
  773. """Get list data"""
  774. return self.itemDataMap
  775. def Populate(self, data):
  776. """Populate the list"""
  777. self.itemDataMap = dict()
  778. self.itemIdMap = list()
  779. if self.shape:
  780. items = self.frame.GetModel().GetItems(objType=ModelAction)
  781. if isinstance(self.shape, ModelCondition):
  782. if self.GetLabel() == "ElseBlockList":
  783. shapeItems = map(
  784. lambda x: x.GetId(), self.shape.GetItems(items)["else"]
  785. )
  786. else:
  787. shapeItems = map(
  788. lambda x: x.GetId(), self.shape.GetItems(items)["if"]
  789. )
  790. else:
  791. shapeItems = map(lambda x: x.GetId(), self.shape.GetItems(items))
  792. else:
  793. shapeItems = list()
  794. i = 0
  795. if len(self.columns) == 2: # ItemCheckList
  796. checked = list()
  797. for action in data:
  798. if isinstance(action, ModelData) or action == self.shape:
  799. continue
  800. self.itemIdMap.append(action.GetId())
  801. if len(self.columns) == 2:
  802. self.itemDataMap[i] = [action.GetLabel(), action.GetLog()]
  803. aId = action.GetBlockId()
  804. if action.GetId() in shapeItems:
  805. checked.append(aId)
  806. else:
  807. checked.append(None)
  808. else:
  809. bId = action.GetBlockId()
  810. if not bId:
  811. bId = _("No")
  812. else:
  813. bId = _("Yes")
  814. options = action.GetParameterizedParams()
  815. params = []
  816. for f in options["flags"]:
  817. params.append("-{0}".format(f["name"]))
  818. for p in options["params"]:
  819. params.append(p["name"])
  820. self.itemDataMap[i] = [
  821. action.GetLabel(),
  822. bId,
  823. ",".join(params),
  824. action.GetLog(),
  825. ]
  826. i += 1
  827. self.itemCount = len(self.itemDataMap.keys())
  828. self.DeleteAllItems()
  829. i = 0
  830. if len(self.columns) == 2:
  831. for name, desc in six.itervalues(self.itemDataMap):
  832. index = self.InsertItem(i, str(i))
  833. self.SetItem(index, 0, name)
  834. self.SetItem(index, 1, desc)
  835. self.SetItemData(index, i)
  836. if checked[i]:
  837. self.CheckItem(index, True)
  838. i += 1
  839. else:
  840. for name, inloop, param, desc in six.itervalues(self.itemDataMap):
  841. index = self.InsertItem(i, str(i))
  842. self.SetItem(index, 0, name)
  843. self.SetItem(index, 1, inloop)
  844. self.SetItem(index, 2, param)
  845. self.SetItem(index, 3, desc)
  846. self.SetItemData(index, i)
  847. i += 1
  848. def OnRemove(self, event):
  849. """Remove selected action(s) from the model"""
  850. model = self.frame.GetModel()
  851. canvas = self.frame.GetCanvas()
  852. item = self.GetFirstSelected()
  853. while item != -1:
  854. self.DeleteItem(item)
  855. del self.itemDataMap[item]
  856. action = model.GetItem(item + 1) # action indices starts at 1
  857. if not action:
  858. item = self.GetFirstSelected()
  859. continue
  860. canvas.RemoveShapes([action])
  861. self.frame.ModelChanged()
  862. item = self.GetFirstSelected()
  863. canvas.Refresh()
  864. event.Skip()
  865. def OnEndEdit(self, event):
  866. """Finish editing of item"""
  867. itemIndex = event.GetIndex()
  868. columnIndex = event.GetColumn()
  869. self.itemDataMap[itemIndex][columnIndex] = event.GetText()
  870. action = self.frame.GetModel().GetItem(itemIndex + 1)
  871. if not action:
  872. event.Veto()
  873. return
  874. action.SetLabel(label=event.GetText())
  875. self.frame.ModelChanged()
  876. def OnReload(self, event=None):
  877. """Reload list of actions"""
  878. self.Populate(self.frame.GetModel().GetItems(objType=ModelAction))
  879. def OnRightUp(self, event):
  880. """Mouse right button up"""
  881. if self.disablePopup:
  882. return
  883. if not hasattr(self, "popupId"):
  884. self.popupID = dict()
  885. self.popupID["remove"] = NewId()
  886. self.popupID["reload"] = NewId()
  887. self.Bind(wx.EVT_MENU, self.OnRemove, id=self.popupID["remove"])
  888. self.Bind(wx.EVT_MENU, self.OnReload, id=self.popupID["reload"])
  889. # generate popup-menu
  890. menu = Menu()
  891. menu.Append(self.popupID["remove"], _("Delete selected"))
  892. if self.GetFirstSelected() == -1:
  893. menu.Enable(self.popupID["remove"], False)
  894. menu.AppendSeparator()
  895. menu.Append(self.popupID["reload"], _("Reload"))
  896. self.PopupMenu(menu)
  897. menu.Destroy()
  898. def MoveItems(self, items, up):
  899. """Move items in the list
  900. :param items: list of items to move
  901. :param up: True to move up otherwise down
  902. """
  903. if len(items) < 1:
  904. return
  905. if items[0] == 0 and up:
  906. del items[0]
  907. if len(items) < 1:
  908. return
  909. if items[-1] == len(self.itemDataMap.keys()) - 1 and not up:
  910. del items[-1]
  911. if len(items) < 1:
  912. return
  913. model = self.frame.GetModel()
  914. modelActions = model.GetItems(objType=ModelAction)
  915. idxList = dict()
  916. itemsToSelect = list()
  917. for i in items:
  918. if up:
  919. idx = i - 1
  920. else:
  921. idx = i + 1
  922. itemsToSelect.append(idx)
  923. idxList[model.GetItemIndex(modelActions[i])] = model.GetItemIndex(
  924. modelActions[idx]
  925. )
  926. # reorganize model items
  927. model.ReorderItems(idxList)
  928. model.Normalize()
  929. self.Populate(model.GetItems(objType=ModelAction))
  930. # re-selected originally selected item
  931. for item in itemsToSelect:
  932. self.SetItemState(
  933. item,
  934. wx.LIST_STATE_SELECTED,
  935. wx.LIST_STATE_SELECTED | wx.LIST_STATE_FOCUSED,
  936. )
  937. class ItemCheckListCtrl(ItemListCtrl, CheckListCtrlMixin):
  938. def __init__(self, parent, shape, columns, frame, **kwargs):
  939. self.parent = parent
  940. self.frame = frame
  941. ItemListCtrl.__init__(self, parent, columns, frame, disablePopup=True, **kwargs)
  942. CheckListCtrlMixin.__init__(self)
  943. self.SetColumnWidth(0, 100)
  944. self.shape = shape
  945. def OnBeginEdit(self, event):
  946. """Disable editing"""
  947. event.Veto()
  948. def OnCheckItem(self, index, flag):
  949. """Item checked/unchecked"""
  950. name = self.GetLabel()
  951. if name == "IfBlockList" and self.window:
  952. self.window.OnCheckItemIf(index, flag)
  953. elif name == "ElseBlockList" and self.window:
  954. self.window.OnCheckItemElse(index, flag)
  955. def GetItems(self):
  956. """Get list of selected actions"""
  957. ids = {"checked": list(), "unchecked": list()}
  958. # action ids start at 1
  959. for i in range(self.GetItemCount()):
  960. if self.IsItemChecked(i):
  961. ids["checked"].append(self.itemIdMap[i])
  962. else:
  963. ids["unchecked"].append(self.itemIdMap[i])
  964. return ids
  965. def CheckItemById(self, aId, flag):
  966. """Check/uncheck given item by id"""
  967. for i in range(self.GetItemCount()):
  968. iId = int(self.GetItem(i, 0).GetText())
  969. if iId == aId:
  970. self.CheckItem(i, flag)
  971. break