colorrules.py 72 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127
  1. """
  2. @package module.colorrules
  3. @brief Dialog for interactive management of raster/vector color tables
  4. and color rules.
  5. Classes:
  6. - colorrules::RulesPanel
  7. - colorrules::ColorTable
  8. - colorrules::RasterColorTable
  9. - colorrules::VectorColorTable
  10. - colorrules::ThematicVectorTable
  11. - colorrules::BufferedWindow
  12. (C) 2008-2015 by the GRASS Development Team
  13. This program is free software under the GNU General Public License
  14. (>=v2). Read the file COPYING that comes with GRASS for details.
  15. @author Michael Barton (Arizona State University)
  16. @author Martin Landa <landa.martin gmail.com> (various updates, pre-defined color table)
  17. @author Anna Kratochvilova <kratochanna gmail.com> (split to base and derived classes)
  18. """
  19. import os
  20. import shutil
  21. import copy
  22. import tempfile
  23. import six
  24. import wx
  25. import wx.lib.colourselect as csel
  26. import wx.lib.scrolledpanel as scrolled
  27. import wx.lib.filebrowsebutton as filebrowse
  28. import grass.script as grass
  29. from grass.script.task import cmdlist_to_tuple
  30. from core import globalvar
  31. from core import utils
  32. from core.gcmd import GMessage, RunCommand, GError
  33. from gui_core.gselect import Select, LayerSelect, ColumnSelect, VectorDBInfo
  34. from core.render import Map
  35. from gui_core.forms import GUI
  36. from core.debug import Debug as Debug
  37. from core.settings import UserSettings
  38. from gui_core.widgets import ColorTablesComboBox
  39. from gui_core.wrap import SpinCtrl, PseudoDC, TextCtrl, Button, StaticText, \
  40. StaticBox, EmptyBitmap, BitmapFromImage
  41. class RulesPanel:
  42. def __init__(self, parent, mapType, attributeType,
  43. properties, panelWidth=180):
  44. """Create rules panel
  45. :param mapType: raster/vector
  46. :param attributeType: color/size for choosing widget type
  47. :param properties: properties of classes derived from ColorTable
  48. :param panelWidth: width of scroll panel"""
  49. self.ruleslines = {}
  50. self.mapType = mapType
  51. self.attributeType = attributeType
  52. self.properties = properties
  53. self.parent = parent
  54. self.panelWidth = panelWidth
  55. self.mainSizer = wx.FlexGridSizer(cols=3, vgap=6, hgap=4)
  56. # put small border at the top of panel
  57. for i in range(3):
  58. self.mainSizer.Add(wx.Size(3, 3))
  59. self.mainPanel = scrolled.ScrolledPanel(
  60. parent, id=wx.ID_ANY, size=(self.panelWidth, 300),
  61. style=wx.TAB_TRAVERSAL | wx.SUNKEN_BORDER)
  62. # (un)check all
  63. self.checkAll = wx.CheckBox(parent, id=wx.ID_ANY, label=_("Check all"))
  64. self.checkAll.SetValue(True)
  65. # clear button
  66. self.clearAll = Button(parent, id=wx.ID_ANY, label=_("Clear all"))
  67. # determines how many rules should be added
  68. self.numRules = SpinCtrl(parent, id=wx.ID_ANY, min=1, max=1e6,
  69. initial=1, size=(150, -1))
  70. # add rules
  71. self.btnAdd = Button(parent, id=wx.ID_ADD)
  72. self.btnAdd.Bind(wx.EVT_BUTTON, self.OnAddRules)
  73. self.checkAll.Bind(wx.EVT_CHECKBOX, self.OnCheckAll)
  74. self.clearAll.Bind(wx.EVT_BUTTON, self.OnClearAll)
  75. self.mainPanel.SetSizer(self.mainSizer)
  76. self.mainPanel.SetAutoLayout(True)
  77. self.mainPanel.SetupScrolling()
  78. def Clear(self):
  79. """Clear and widgets and delete information"""
  80. self.ruleslines.clear()
  81. self.mainSizer.Clear(True)
  82. def OnCheckAll(self, event):
  83. """(Un)check all rules"""
  84. check = event.GetInt()
  85. for child in self.mainPanel.GetChildren():
  86. if child.GetName() == 'enable':
  87. child.SetValue(check)
  88. else:
  89. child.Enable(check)
  90. def OnClearAll(self, event):
  91. """Delete all widgets in panel"""
  92. self.Clear()
  93. def OnAddRules(self, event):
  94. """Add rules button pressed"""
  95. nrules = self.numRules.GetValue()
  96. self.AddRules(nrules)
  97. def AddRules(self, nrules, start=False):
  98. """Add rules
  99. :param start: set widgets (not append)
  100. """
  101. snum = len(self.ruleslines.keys())
  102. if start:
  103. snum = 0
  104. for num in range(snum, snum + nrules):
  105. # enable
  106. enable = wx.CheckBox(parent=self.mainPanel, id=num)
  107. enable.SetValue(True)
  108. enable.SetName('enable')
  109. enable.Bind(wx.EVT_CHECKBOX, self.OnRuleEnable)
  110. # value
  111. txt_ctrl = TextCtrl(parent=self.mainPanel, id=1000 + num,
  112. size=(80, -1),
  113. style=wx.TE_NOHIDESEL)
  114. if self.mapType == 'vector':
  115. txt_ctrl.SetToolTip(_("Enter vector attribute values"))
  116. txt_ctrl.Bind(wx.EVT_TEXT, self.OnRuleValue)
  117. txt_ctrl.SetName('source')
  118. if self.attributeType == 'color':
  119. # color
  120. columnCtrl = csel.ColourSelect(
  121. self.mainPanel, id=2000 + num,
  122. size=globalvar.DIALOG_COLOR_SIZE)
  123. columnCtrl.Bind(csel.EVT_COLOURSELECT, self.OnRuleColor)
  124. columnCtrl.SetName('target')
  125. if not start:
  126. self.ruleslines[enable.GetId()] = {'value': '',
  127. 'color': "0:0:0"}
  128. else:
  129. # size or width
  130. init = 2
  131. if self.attributeType == 'size':
  132. init = 100
  133. columnCtrl = SpinCtrl(self.mainPanel, id=2000 + num,
  134. size=(50, -1), min=1, max=1e4,
  135. initial=init)
  136. columnCtrl.Bind(wx.EVT_SPINCTRL, self.OnRuleSize)
  137. columnCtrl.Bind(wx.EVT_TEXT, self.OnRuleSize)
  138. columnCtrl.SetName('target')
  139. if not start:
  140. self.ruleslines[
  141. enable.GetId()] = {
  142. 'value': '',
  143. self.attributeType: init}
  144. self.mainSizer.Add(enable, proportion=0,
  145. flag=wx.ALIGN_CENTER_VERTICAL)
  146. self.mainSizer.Add(txt_ctrl, proportion=0,
  147. flag=wx.ALIGN_CENTER | wx.RIGHT, border=5)
  148. self.mainSizer.Add(columnCtrl, proportion=0,
  149. flag=wx.ALIGN_CENTER | wx.RIGHT, border=10)
  150. self.mainPanel.Layout()
  151. self.mainPanel.SetupScrolling(scroll_x=False)
  152. def OnRuleEnable(self, event):
  153. """Rule enabled/disabled"""
  154. id = event.GetId()
  155. if event.IsChecked():
  156. self.mainPanel.FindWindowById(id + 1000).Enable()
  157. self.mainPanel.FindWindowById(id + 2000).Enable()
  158. if self.mapType == 'vector' and not self.parent.GetParent().colorTable:
  159. vals = []
  160. vals.append(
  161. self.mainPanel.FindWindowById(
  162. id + 1000).GetValue())
  163. try:
  164. vals.append(
  165. self.mainPanel.FindWindowById(
  166. id + 1 + 1000).GetValue())
  167. except AttributeError:
  168. vals.append(None)
  169. value = self.SQLConvert(vals)
  170. else:
  171. value = self.mainPanel.FindWindowById(id + 1000).GetValue()
  172. color = self.mainPanel.FindWindowById(id + 2000).GetValue()
  173. if self.attributeType == 'color':
  174. # color
  175. color_str = str(color[0]) + ':' \
  176. + str(color[1]) + ':' \
  177. + str(color[2])
  178. self.ruleslines[id] = {'value': value,
  179. 'color': color_str}
  180. else:
  181. # size or width
  182. self.ruleslines[id] = {'value': value,
  183. self.attributeType: float(color)}
  184. else:
  185. self.mainPanel.FindWindowById(id + 1000).Disable()
  186. self.mainPanel.FindWindowById(id + 2000).Disable()
  187. del self.ruleslines[id]
  188. def OnRuleColor(self, event):
  189. """Rule color changed"""
  190. num = event.GetId()
  191. rgba_color = event.GetValue()
  192. rgb_string = str(rgba_color[0]) + ':' \
  193. + str(rgba_color[1]) + ':' \
  194. + str(rgba_color[2])
  195. self.ruleslines[num - 2000]['color'] = rgb_string
  196. def OnRuleSize(self, event):
  197. """Rule size changed"""
  198. num = event.GetId()
  199. size = event.GetInt()
  200. self.ruleslines[num - 2000][self.attributeType] = size
  201. def OnRuleValue(self, event):
  202. """Rule value changed"""
  203. num = event.GetId()
  204. val = event.GetString().strip()
  205. if val == '':
  206. return
  207. try:
  208. table = self.parent.colorTable
  209. except AttributeError:
  210. # due to panel/scrollpanel in vector dialog
  211. if isinstance(self.parent.GetParent(), RasterColorTable):
  212. table = self.parent.GetParent().colorTable
  213. else:
  214. table = self.parent.GetParent().GetParent().colorTable
  215. if table:
  216. self.SetRasterRule(num, val)
  217. else:
  218. self.SetVectorRule(num, val)
  219. def SetRasterRule(self, num, val):
  220. """Set raster rule"""
  221. self.ruleslines[num - 1000]['value'] = val
  222. def SetVectorRule(self, num, val):
  223. """Set vector rule"""
  224. vals = []
  225. vals.append(val)
  226. try:
  227. vals.append(self.mainPanel.FindWindowById(num + 1).GetValue())
  228. except AttributeError:
  229. vals.append(None)
  230. self.ruleslines[num - 1000]['value'] = self.SQLConvert(vals)
  231. def Enable(self, enable=True):
  232. """Enable/Disable all widgets"""
  233. for child in self.mainPanel.GetChildren():
  234. child.Enable(enable)
  235. sql = True
  236. self.LoadRulesline(sql) # todo
  237. self.btnAdd.Enable(enable)
  238. self.numRules.Enable(enable)
  239. self.checkAll.Enable(enable)
  240. self.clearAll.Enable(enable)
  241. def LoadRules(self):
  242. message = ""
  243. for item in range(len(self.ruleslines)):
  244. try:
  245. self.mainPanel.FindWindowById(
  246. item +
  247. 1000).SetValue(
  248. self.ruleslines[item]['value'])
  249. r, g, b = (0, 0, 0) # default
  250. if not self.ruleslines[item][self.attributeType]:
  251. if self.attributeType == 'color':
  252. self.ruleslines[item][
  253. self.attributeType] = '%d:%d:%d' % (
  254. r, g, b)
  255. elif self.attributeType == 'size':
  256. self.ruleslines[item][self.attributeType] = 100
  257. elif self.attributeType == 'width':
  258. self.ruleslines[item][self.attributeType] = 2
  259. if self.attributeType == 'color':
  260. try:
  261. r, g, b = map(
  262. int, self.ruleslines[item][
  263. self.attributeType].split(':'))
  264. except ValueError as e:
  265. message = _(
  266. "Bad color format. Use color format '0:0:0'")
  267. self.mainPanel.FindWindowById(
  268. item + 2000).SetValue((r, g, b))
  269. else:
  270. value = float(self.ruleslines[item][self.attributeType])
  271. self.mainPanel.FindWindowById(item + 2000).SetValue(value)
  272. except:
  273. continue
  274. if message:
  275. GMessage(parent=self.parent, message=message)
  276. return False
  277. return True
  278. def SQLConvert(self, vals):
  279. """Prepare value for SQL query"""
  280. if vals[0].isdigit():
  281. sqlrule = '%s=%s' % (self.properties['sourceColumn'], vals[0])
  282. if vals[1]:
  283. sqlrule += ' AND %s<%s' % (
  284. self.properties['sourceColumn'], vals[1])
  285. else:
  286. sqlrule = '%s=%s' % (self.properties['sourceColumn'], vals[0])
  287. return sqlrule
  288. class ColorTable(wx.Frame):
  289. def __init__(self, parent, title, layerTree=None, id=wx.ID_ANY,
  290. style=wx.DEFAULT_FRAME_STYLE | wx.RESIZE_BORDER,
  291. **kwargs):
  292. """Dialog for interactively entering rules for map management
  293. commands
  294. :param raster: True to raster otherwise vector
  295. :param nviz: True if ColorTable is called from nviz thematic mapping
  296. """
  297. self.parent = parent # GMFrame ?
  298. self.layerTree = layerTree # LayerTree or None
  299. wx.Frame.__init__(self, parent, id, title, style=style, **kwargs)
  300. self.SetIcon(
  301. wx.Icon(
  302. os.path.join(
  303. globalvar.ICONDIR,
  304. 'grass.ico'),
  305. wx.BITMAP_TYPE_ICO))
  306. self.panel = wx.Panel(parent=self, id=wx.ID_ANY)
  307. # instance of render.Map to be associated with display
  308. self.Map = Map()
  309. # input map to change
  310. self.inmap = ''
  311. # reference to layer with preview
  312. self.layer = None
  313. # layout
  314. self._doLayout()
  315. # bindings
  316. self.Bind(wx.EVT_BUTTON, self.OnHelp, self.btnHelp)
  317. self.selectionInput.Bind(wx.EVT_TEXT, self.OnSelectionInput)
  318. self.Bind(wx.EVT_BUTTON, self.OnCancel, self.btnCancel)
  319. self.Bind(wx.EVT_BUTTON, self.OnApply, self.btnApply)
  320. self.Bind(wx.EVT_BUTTON, self.OnOK, self.btnOK)
  321. self.Bind(wx.EVT_BUTTON, self.OnLoadDefaultTable, self.btnDefault)
  322. self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
  323. self.Bind(wx.EVT_BUTTON, self.OnPreview, self.btnPreview)
  324. def _initLayer(self):
  325. """Set initial layer when opening dialog"""
  326. # set map layer from layer tree, first selected,
  327. # if not the right type, than select another
  328. try:
  329. sel = self.layerTree.layer_selected
  330. if sel and self.layerTree.GetLayerInfo(
  331. sel, key='type') == self.mapType:
  332. layer = sel
  333. else:
  334. layer = self.layerTree.FindItemByData(
  335. key='type', value=self.mapType)
  336. except:
  337. layer = None
  338. if layer:
  339. mapLayer = self.layerTree.GetLayerInfo(layer, key='maplayer')
  340. name = mapLayer.GetName()
  341. type = mapLayer.GetType()
  342. self.selectionInput.SetValue(name)
  343. self.inmap = name
  344. def _createMapSelection(self, parent):
  345. """Create map selection part of dialog"""
  346. # top controls
  347. if self.mapType == 'raster':
  348. maplabel = _('Select raster map:')
  349. else:
  350. maplabel = _('Select vector map:')
  351. inputBox = StaticBox(parent, id=wx.ID_ANY,
  352. label=" %s " % maplabel)
  353. inputSizer = wx.StaticBoxSizer(inputBox, wx.VERTICAL)
  354. self.selectionInput = Select(parent=parent, id=wx.ID_ANY,
  355. size=globalvar.DIALOG_GSELECT_SIZE,
  356. type=self.mapType)
  357. # layout
  358. inputSizer.Add(
  359. self.selectionInput,
  360. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALL | wx.EXPAND,
  361. border=5)
  362. return inputSizer
  363. def _createFileSelection(self, parent):
  364. """Create file (open/save rules) selection part of dialog"""
  365. inputBox = StaticBox(
  366. parent, id=wx.ID_ANY, label=" %s " %
  367. _("Import or export color table:"))
  368. inputSizer = wx.StaticBoxSizer(inputBox, wx.HORIZONTAL)
  369. self.loadRules = filebrowse.FileBrowseButton(
  370. parent=parent, id=wx.ID_ANY, fileMask='*', labelText='',
  371. dialogTitle=_('Choose file to load color table'),
  372. buttonText=_('Load'),
  373. toolTip=_(
  374. "Type filename or click to choose "
  375. "file and load color table"),
  376. startDirectory=os.getcwd(),
  377. fileMode=wx.FD_OPEN, changeCallback=self.OnLoadRulesFile)
  378. self.saveRules = filebrowse.FileBrowseButton(
  379. parent=parent, id=wx.ID_ANY, fileMask='*', labelText='',
  380. dialogTitle=_('Choose file to save color table'),
  381. toolTip=_(
  382. "Type filename or click to choose "
  383. "file and save color table"),
  384. buttonText=_('Save'),
  385. startDirectory=os.getcwd(),
  386. fileMode=wx.FD_SAVE, changeCallback=self.OnSaveRulesFile)
  387. colorTable = ColorTablesComboBox(
  388. parent=parent,
  389. size=globalvar.DIALOG_COMBOBOX_SIZE,
  390. choices=utils.GetColorTables(),
  391. name="colorTableChoice")
  392. self.btnSet = Button(
  393. parent=parent,
  394. id=wx.ID_ANY,
  395. label=_("&Set"),
  396. name='btnSet')
  397. self.btnSet.Bind(wx.EVT_BUTTON, self.OnSetTable)
  398. self.btnSet.Enable(False)
  399. # layout
  400. gridSizer = wx.GridBagSizer(hgap=2, vgap=2)
  401. gridSizer.Add(StaticText(parent, label=_("Load color table:")),
  402. pos=(0, 0), flag=wx.ALIGN_CENTER_VERTICAL)
  403. gridSizer.Add(colorTable, pos=(0, 1))
  404. gridSizer.Add(self.btnSet, pos=(0, 2), flag=wx.ALIGN_RIGHT)
  405. gridSizer.Add(
  406. StaticText(
  407. parent, label=_('Load color table from file:')), pos=(
  408. 1, 0), flag=wx.ALIGN_CENTER_VERTICAL)
  409. gridSizer.Add(
  410. self.loadRules, pos=(
  411. 1, 1), span=(
  412. 1, 2), flag=wx.EXPAND)
  413. gridSizer.Add(
  414. StaticText(
  415. parent, label=_('Save color table to file:')), pos=(
  416. 2, 0), flag=wx.ALIGN_CENTER_VERTICAL)
  417. gridSizer.Add(
  418. self.saveRules, pos=(
  419. 2, 1), span=(
  420. 1, 2), flag=wx.EXPAND)
  421. gridSizer.AddGrowableCol(1)
  422. inputSizer.Add(gridSizer, proportion=1, flag=wx.EXPAND | wx.ALL,
  423. border=5)
  424. if self.mapType == 'vector':
  425. # parent is collapsible pane
  426. parent.SetSizer(inputSizer)
  427. return inputSizer
  428. def _createPreview(self, parent):
  429. """Create preview"""
  430. # initialize preview display
  431. self.InitDisplay()
  432. self.preview = BufferedWindow(parent, id=wx.ID_ANY, size=(400, 300),
  433. Map=self.Map)
  434. self.preview.EraseMap()
  435. def _createButtons(self, parent):
  436. """Create buttons for leaving dialog"""
  437. self.btnHelp = Button(parent, id=wx.ID_HELP)
  438. self.btnCancel = Button(parent, id=wx.ID_CANCEL)
  439. self.btnApply = Button(parent, id=wx.ID_APPLY)
  440. self.btnOK = Button(parent, id=wx.ID_OK)
  441. self.btnDefault = Button(parent, id=wx.ID_ANY,
  442. label=_("Reload default table"))
  443. self.btnOK.SetDefault()
  444. self.btnOK.Enable(False)
  445. self.btnApply.Enable(False)
  446. self.btnDefault.Enable(False)
  447. # layout
  448. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  449. btnSizer.Add(wx.Size(-1, -1), proportion=1)
  450. btnSizer.Add(self.btnDefault,
  451. flag=wx.LEFT | wx.RIGHT, border=5)
  452. btnSizer.Add(self.btnHelp,
  453. flag=wx.LEFT | wx.RIGHT, border=5)
  454. btnSizer.Add(self.btnCancel,
  455. flag=wx.LEFT | wx.RIGHT, border=5)
  456. btnSizer.Add(self.btnApply,
  457. flag=wx.LEFT | wx.RIGHT, border=5)
  458. btnSizer.Add(self.btnOK,
  459. flag=wx.LEFT | wx.RIGHT, border=5)
  460. return btnSizer
  461. def _createBody(self, parent):
  462. """Create dialog body consisting of rules and preview"""
  463. bodySizer = wx.GridBagSizer(hgap=5, vgap=5)
  464. row = 0
  465. # label with range
  466. self.cr_label = StaticText(parent, id=wx.ID_ANY)
  467. bodySizer.Add(self.cr_label, pos=(row, 0), span=(1, 3),
  468. flag=wx.ALL, border=5)
  469. row += 1
  470. # color table
  471. self.rulesPanel = RulesPanel(
  472. parent=parent,
  473. mapType=self.mapType,
  474. attributeType=self.attributeType,
  475. properties=self.properties)
  476. bodySizer.Add(self.rulesPanel.mainPanel, pos=(row, 0),
  477. span=(1, 2), flag=wx.EXPAND)
  478. # add two rules as default
  479. self.rulesPanel.AddRules(2)
  480. # preview window
  481. self._createPreview(parent=parent)
  482. bodySizer.Add(self.preview, pos=(row, 2),
  483. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER)
  484. row += 1
  485. # add ckeck all and clear all
  486. bodySizer.Add(
  487. self.rulesPanel.checkAll,
  488. flag=wx.ALIGN_CENTER_VERTICAL,
  489. pos=(
  490. row,
  491. 0))
  492. bodySizer.Add(self.rulesPanel.clearAll, pos=(row, 1))
  493. # preview button
  494. self.btnPreview = Button(parent, id=wx.ID_ANY,
  495. label=_("Preview"))
  496. bodySizer.Add(self.btnPreview, pos=(row, 2),
  497. flag=wx.ALIGN_RIGHT)
  498. self.btnPreview.Enable(False)
  499. self.btnPreview.SetToolTip(
  500. _("Show preview of map " "(current Map Display extent is used)."))
  501. row += 1
  502. # add rules button and spin to sizer
  503. bodySizer.Add(self.rulesPanel.numRules, pos=(row, 0),
  504. flag=wx.ALIGN_CENTER_VERTICAL)
  505. bodySizer.Add(self.rulesPanel.btnAdd, pos=(row, 1))
  506. bodySizer.AddGrowableRow(1)
  507. bodySizer.AddGrowableCol(2)
  508. return bodySizer
  509. def InitDisplay(self):
  510. """Initialize preview display, set dimensions and region
  511. """
  512. self.width = self.Map.width = 400
  513. self.height = self.Map.height = 300
  514. self.Map.geom = self.width, self.height
  515. def OnCloseWindow(self, event):
  516. """Window closed
  517. """
  518. self.OnCancel(event)
  519. def OnApply(self, event):
  520. return self._apply()
  521. def _apply(self, updatePreview=True):
  522. """Apply selected color table
  523. :return: True on success otherwise False
  524. """
  525. ret = self.CreateColorTable()
  526. if not ret:
  527. GMessage(parent=self, message=_("No valid color rules given."))
  528. else:
  529. # re-render preview and current map window
  530. if updatePreview:
  531. self.OnPreview(None)
  532. display = self.layerTree.GetMapDisplay()
  533. if display and display.IsAutoRendered():
  534. display.GetWindow().UpdateMap(render=True)
  535. return ret
  536. def OnOK(self, event):
  537. """Apply selected color table and close the dialog"""
  538. if self._apply(updatePreview=False):
  539. self.OnCancel(event)
  540. def OnCancel(self, event):
  541. """Do not apply any changes, remove associated
  542. rendered images and close the dialog"""
  543. self.Map.Clean()
  544. self.Destroy()
  545. def OnSetTable(self, event):
  546. """Load pre-defined color table"""
  547. ct = self.FindWindowByName("colorTableChoice").GetValue()
  548. # save original color table
  549. ctOriginal = RunCommand(
  550. 'r.colors.out',
  551. read=True,
  552. map=self.inmap,
  553. rules='-')
  554. # set new color table
  555. ret, err = RunCommand('r.colors', map=self.inmap,
  556. color=ct, getErrorMsg=True)
  557. if ret != 0:
  558. GError(err, parent=self)
  559. return
  560. ctNew = RunCommand(
  561. 'r.colors.out',
  562. read=True,
  563. map=self.inmap,
  564. rules='-')
  565. # restore original table
  566. RunCommand('r.colors', map=self.inmap, rules='-', stdin=ctOriginal)
  567. # load color table
  568. self.rulesPanel.Clear()
  569. self.ReadColorTable(ctable=ctNew)
  570. def OnSaveRulesFile(self, event):
  571. """Save color table to file"""
  572. path = event.GetString()
  573. if not path:
  574. return
  575. if os.path.exists(path):
  576. dlgOw = wx.MessageDialog(
  577. parent,
  578. message=_(
  579. "File <%s> already already exists. "
  580. "Do you want to overwrite it?") %
  581. path,
  582. caption=_("Overwrite?"),
  583. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  584. if dlgOw.ShowModal() != wx.ID_YES:
  585. return
  586. rulestxt = ''
  587. for rule in six.itervalues(self.rulesPanel.ruleslines):
  588. if 'value' not in rule:
  589. continue
  590. rulestxt += rule['value'] + ' ' + rule['color'] + '\n'
  591. if not rulestxt:
  592. GMessage(message=_("Nothing to save."),
  593. parent=self)
  594. return
  595. fd = open(path, 'w')
  596. fd.write(rulestxt)
  597. fd.close()
  598. def OnLoadRulesFile(self, event):
  599. """Load color table from file"""
  600. path = event.GetString()
  601. if not os.path.exists(path):
  602. return
  603. self.rulesPanel.Clear()
  604. fd = open(path, 'r')
  605. self.ReadColorTable(ctable=fd.read())
  606. fd.close()
  607. def ReadColorTable(self, ctable):
  608. """Read color table
  609. :param table: color table in format coming from r.colors.out"""
  610. rulesNumber = len(ctable.splitlines())
  611. self.rulesPanel.AddRules(rulesNumber)
  612. minim = maxim = count = 0
  613. for line in ctable.splitlines():
  614. try:
  615. value, color = map(lambda x: x.strip(), line.split(' '))
  616. except ValueError:
  617. GMessage(parent=self, message=_("Invalid color table format"))
  618. self.rulesPanel.Clear()
  619. return
  620. self.rulesPanel.ruleslines[count]['value'] = value
  621. self.rulesPanel.ruleslines[count]['color'] = color
  622. self.rulesPanel.mainPanel.FindWindowById(
  623. count + 1000).SetValue(value)
  624. rgb = list()
  625. for c in color.split(':'):
  626. rgb.append(int(c))
  627. self.rulesPanel.mainPanel.FindWindowById(
  628. count + 2000).SetColour(rgb)
  629. # range
  630. try:
  631. if float(value) < minim:
  632. minim = float(value)
  633. if float(value) > maxim:
  634. maxim = float(value)
  635. except ValueError: # nv, default
  636. pass
  637. count += 1
  638. if self.mapType == 'vector':
  639. # raster min, max is known from r.info
  640. self.properties['min'], self.properties['max'] = minim, maxim
  641. self.SetRangeLabel()
  642. self.OnPreview(tmp=True)
  643. def OnLoadDefaultTable(self, event):
  644. """Load internal color table"""
  645. self.LoadTable()
  646. def LoadTable(self, mapType='raster'):
  647. """Load current color table (using `r(v).colors.out`)
  648. :param mapType: map type (raster or vector)"""
  649. self.rulesPanel.Clear()
  650. if mapType == 'raster':
  651. cmd = ['r.colors.out',
  652. 'read=True',
  653. 'map=%s' % self.inmap,
  654. 'rules=-']
  655. else:
  656. cmd = ['v.colors.out',
  657. 'read=True',
  658. 'map=%s' % self.inmap,
  659. 'rules=-']
  660. if self.properties['sourceColumn'] and self.properties[
  661. 'sourceColumn'] != 'cat':
  662. cmd.append('column=%s' % self.properties['sourceColumn'])
  663. cmd = cmdlist_to_tuple(cmd)
  664. if self.inmap:
  665. ctable = RunCommand(cmd[0], **cmd[1])
  666. else:
  667. self.OnPreview()
  668. return
  669. self.ReadColorTable(ctable=ctable)
  670. def CreateColorTable(self, tmp=False):
  671. """Creates color table
  672. :return: True on success
  673. :return: False on failure
  674. """
  675. rulestxt = ''
  676. for rule in six.itervalues(self.rulesPanel.ruleslines):
  677. if 'value' not in rule: # skip empty rules
  678. continue
  679. if rule['value'] not in ('nv', 'default') and \
  680. rule['value'][-1] != '%' and \
  681. not self._IsNumber(rule['value']):
  682. GError(
  683. _("Invalid rule value '%s'. Unable to apply color table.") %
  684. rule['value'], parent=self)
  685. return False
  686. rulestxt += rule['value'] + ' ' + rule['color'] + '\n'
  687. if not rulestxt:
  688. return False
  689. gtemp = utils.GetTempfile()
  690. output = open(gtemp, "w")
  691. try:
  692. output.write(rulestxt)
  693. finally:
  694. output.close()
  695. cmd = ['%s.colors' % self.mapType[0], # r.colors/v.colors
  696. 'map=%s' % self.inmap,
  697. 'rules=%s' % gtemp]
  698. if self.mapType == 'vector' and self.properties['sourceColumn'] \
  699. and self.properties['sourceColumn'] != 'cat':
  700. cmd.append('column=%s' % self.properties['sourceColumn'])
  701. cmd = cmdlist_to_tuple(cmd)
  702. ret = RunCommand(cmd[0], **cmd[1])
  703. if ret != 0:
  704. return False
  705. return True
  706. def DoPreview(self, ltype, cmdlist):
  707. """Update preview (based on computational region)"""
  708. if not self.layer:
  709. self.layer = self.Map.AddLayer(
  710. ltype=ltype,
  711. name='preview',
  712. command=cmdlist,
  713. active=True,
  714. hidden=False,
  715. opacity=1.0,
  716. render=False)
  717. else:
  718. self.layer.SetCmd(cmdlist)
  719. # apply new color table and display preview
  720. self.CreateColorTable(tmp=True)
  721. self.preview.UpdatePreview()
  722. def RunHelp(self, cmd):
  723. """Show GRASS manual page"""
  724. RunCommand('g.manual',
  725. quiet=True,
  726. parent=self,
  727. entry=cmd)
  728. def SetMap(self, name):
  729. """Set map name and update dialog"""
  730. self.selectionInput.SetValue(name)
  731. def _IsNumber(self, s):
  732. """Check if 's' is a number"""
  733. try:
  734. float(s)
  735. return True
  736. except ValueError:
  737. return False
  738. class RasterColorTable(ColorTable):
  739. def __init__(self, parent, **kwargs):
  740. """Dialog for interactively entering color rules for raster maps"""
  741. self.mapType = 'raster'
  742. self.attributeType = 'color'
  743. self.colorTable = True
  744. # raster properties
  745. self.properties = {
  746. # min cat in raster map
  747. 'min': None,
  748. # max cat in raster map
  749. 'max': None,
  750. }
  751. ColorTable.__init__(self, parent, title=_(
  752. 'Create new color table for raster map'), **kwargs)
  753. self._initLayer()
  754. self.Map.GetRenderMgr().renderDone.connect(self._restoreColorTable)
  755. # self.SetMinSize(self.GetSize())
  756. self.SetMinSize((650, 700))
  757. def _doLayout(self):
  758. """Do main layout"""
  759. sizer = wx.BoxSizer(wx.VERTICAL)
  760. #
  761. # map selection
  762. #
  763. mapSelection = self._createMapSelection(parent=self.panel)
  764. sizer.Add(mapSelection, proportion=0,
  765. flag=wx.ALL | wx.EXPAND, border=5)
  766. #
  767. # manage extern tables
  768. #
  769. fileSelection = self._createFileSelection(parent=self.panel)
  770. sizer.Add(fileSelection, proportion=0,
  771. flag=wx.LEFT | wx.RIGHT | wx.EXPAND, border=5)
  772. #
  773. # body & preview
  774. #
  775. bodySizer = self._createBody(parent=self.panel)
  776. sizer.Add(bodySizer, proportion=1,
  777. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=5)
  778. #
  779. # buttons
  780. #
  781. btnSizer = self._createButtons(parent=self.panel)
  782. sizer.Add(wx.StaticLine(parent=self.panel, id=wx.ID_ANY,
  783. style=wx.LI_HORIZONTAL), proportion=0,
  784. flag=wx.EXPAND | wx.ALL, border=5)
  785. sizer.Add(btnSizer, proportion=0,
  786. flag=wx.ALL | wx.ALIGN_RIGHT, border=5)
  787. self.panel.SetSizer(sizer)
  788. sizer.Layout()
  789. sizer.Fit(self.panel)
  790. self.Layout()
  791. def OnSelectionInput(self, event):
  792. """Raster map selected"""
  793. if event:
  794. self.inmap = event.GetString()
  795. self.loadRules.SetValue('')
  796. self.saveRules.SetValue('')
  797. if self.inmap:
  798. if not grass.find_file(name=self.inmap, element='cell')['file']:
  799. self.inmap = None
  800. if not self.inmap:
  801. for btn in (self.btnPreview, self.btnOK,
  802. self.btnApply, self.btnDefault, self.btnSet):
  803. btn.Enable(False)
  804. self.LoadTable()
  805. return
  806. info = grass.raster_info(map=self.inmap)
  807. if info:
  808. self.properties['min'] = info['min']
  809. self.properties['max'] = info['max']
  810. self.LoadTable()
  811. else:
  812. self.inmap = ''
  813. self.properties['min'] = self.properties['max'] = None
  814. for btn in (self.btnPreview, self.btnOK,
  815. self.btnApply, self.btnDefault, self.btnSet):
  816. btn.Enable(False)
  817. self.preview.EraseMap()
  818. self.cr_label.SetLabel(
  819. _('Enter raster category values or percents'))
  820. return
  821. if info['datatype'] == 'CELL':
  822. mapRange = _('range')
  823. else:
  824. mapRange = _('fp range')
  825. self.cr_label.SetLabel(
  826. _('Enter raster category values or percents (%(range)s = %(min)d-%(max)d)') %
  827. {
  828. 'range': mapRange,
  829. 'min': self.properties['min'],
  830. 'max': self.properties['max']})
  831. for btn in (self.btnPreview, self.btnOK,
  832. self.btnApply, self.btnDefault, self.btnSet):
  833. btn.Enable()
  834. def OnPreview(self, tmp=True):
  835. """Update preview (based on computational region)"""
  836. if not self.inmap:
  837. self.preview.EraseMap()
  838. return
  839. cmdlist = ['d.rast',
  840. 'map=%s' % self.inmap]
  841. ltype = 'raster'
  842. # find existing color table and copy to temp file
  843. try:
  844. name, mapset = self.inmap.split('@')
  845. except ValueError:
  846. name = self.inmap
  847. mapset = grass.find_file(self.inmap, element='cell')['mapset']
  848. if not mapset:
  849. return
  850. self._tmp = tmp
  851. self._old_colrtable = None
  852. if mapset == grass.gisenv()['MAPSET']:
  853. self._old_colrtable = grass.find_file(
  854. name=name, element='colr')['file']
  855. else:
  856. self._old_colrtable = grass.find_file(
  857. name=name, element='colr2/' + mapset)['file']
  858. if self._old_colrtable:
  859. self._colrtemp = utils.GetTempfile()
  860. shutil.copyfile(self._old_colrtable, self._colrtemp)
  861. ColorTable.DoPreview(self, ltype, cmdlist)
  862. def _restoreColorTable(self):
  863. # restore previous color table
  864. if self._tmp:
  865. if self._old_colrtable:
  866. shutil.copyfile(self._colrtemp, self._old_colrtable)
  867. os.remove(self._colrtemp)
  868. del self._colrtemp, self._old_colrtable
  869. else:
  870. RunCommand('r.colors',
  871. parent=self,
  872. flags='r',
  873. map=self.inmap)
  874. del self._tmp
  875. def OnHelp(self, event):
  876. """Show GRASS manual page"""
  877. cmd = 'r.colors'
  878. ColorTable.RunHelp(self, cmd=cmd)
  879. class VectorColorTable(ColorTable):
  880. def __init__(self, parent, attributeType, **kwargs):
  881. """Dialog for interactively entering color rules for vector maps"""
  882. # dialog attributes
  883. self.mapType = 'vector'
  884. self.attributeType = attributeType # color, size, width
  885. # in version 7 v.colors used, otherwise color column only
  886. self.version7 = int(grass.version()['version'].split('.')[0]) >= 7
  887. self.colorTable = False
  888. self.updateColumn = True
  889. # vector properties
  890. self.properties = {
  891. # vector layer for attribute table to use for setting color
  892. 'layer': 1,
  893. # vector attribute table used for setting color
  894. 'table': '',
  895. # vector attribute column for assigning colors
  896. 'sourceColumn': '',
  897. # vector attribute column to use for loading colors
  898. 'loadColumn': '',
  899. # vector attribute column to use for storing colors
  900. 'storeColumn': '',
  901. # vector attribute column for temporary storing colors
  902. 'tmpColumn': 'tmp_0',
  903. # min value of attribute column/vector color table
  904. 'min': None,
  905. # max value of attribute column/vector color table
  906. 'max': None
  907. }
  908. self.columnsProp = {
  909. 'color': {
  910. 'name': 'GRASSRGB',
  911. 'type1': 'varchar(11)',
  912. 'type2': ['character']},
  913. 'size': {
  914. 'name': 'GRASSSIZE',
  915. 'type1': 'integer',
  916. 'type2': ['integer']},
  917. 'width': {
  918. 'name': 'GRASSWIDTH',
  919. 'type1': 'integer',
  920. 'type2': ['integer']}}
  921. ColorTable.__init__(self, parent=parent, title=_(
  922. 'Create new color rules for vector map'), **kwargs)
  923. # additional bindings for vector color management
  924. self.Bind(wx.EVT_COMBOBOX, self.OnLayerSelection, self.layerSelect)
  925. self._columnWidgetEvtHandler()
  926. self.Bind(wx.EVT_BUTTON, self.OnAddColumn, self.addColumn)
  927. self._initLayer()
  928. if self.colorTable:
  929. self.cr_label.SetLabel(
  930. _("Enter vector attribute values or percents:"))
  931. else:
  932. self.cr_label.SetLabel(_("Enter vector attribute values:"))
  933. # self.SetMinSize(self.GetSize())
  934. self.SetMinSize((650, 700))
  935. self.CentreOnScreen()
  936. self.Show()
  937. def _createVectorAttrb(self, parent):
  938. """Create part of dialog with layer/column selection"""
  939. inputBox = StaticBox(parent=parent, id=wx.ID_ANY,
  940. label=" %s " % _("Select vector columns"))
  941. cb_vl_label = StaticText(parent, id=wx.ID_ANY,
  942. label=_('Layer:'))
  943. cb_vc_label = StaticText(parent, id=wx.ID_ANY,
  944. label=_('Attribute column:'))
  945. if self.attributeType == 'color':
  946. labels = [_("Load color from column:"), _("Save color to column:")]
  947. elif self.attributeType == 'size':
  948. labels = [_("Load size from column:"), _("Save size to column:")]
  949. elif self.attributeType == 'width':
  950. labels = [_("Load width from column:"), _("Save width to column:")]
  951. if self.version7 and self.attributeType == 'color':
  952. self.useColumn = wx.CheckBox(
  953. parent, id=wx.ID_ANY,
  954. label=_("Use color column instead of color table:"))
  955. self.useColumn.Bind(wx.EVT_CHECKBOX, self.OnCheckColumn)
  956. fromColumnLabel = StaticText(parent, id=wx.ID_ANY,
  957. label=labels[0])
  958. toColumnLabel = StaticText(parent, id=wx.ID_ANY,
  959. label=labels[1])
  960. self.rgb_range_label = StaticText(parent, id=wx.ID_ANY)
  961. self.layerSelect = LayerSelect(parent)
  962. self.sourceColumn = ColumnSelect(parent)
  963. self.fromColumn = ColumnSelect(parent)
  964. self.toColumn = ColumnSelect(parent)
  965. self.addColumn = Button(parent, id=wx.ID_ANY,
  966. label=_('Add column'))
  967. self.addColumn.SetToolTip(
  968. _("Add GRASSRGB column to current attribute table."))
  969. # layout
  970. inputSizer = wx.StaticBoxSizer(inputBox, wx.VERTICAL)
  971. vSizer = wx.GridBagSizer(hgap=5, vgap=5)
  972. row = 0
  973. vSizer.Add(cb_vl_label, pos=(row, 0),
  974. flag=wx.ALIGN_CENTER_VERTICAL)
  975. vSizer.Add(self.layerSelect, pos=(row, 1),
  976. flag=wx.ALIGN_CENTER_VERTICAL)
  977. row += 1
  978. vSizer.Add(cb_vc_label, pos=(row, 0),
  979. flag=wx.ALIGN_CENTER_VERTICAL)
  980. vSizer.Add(self.sourceColumn, pos=(row, 1),
  981. flag=wx.ALIGN_CENTER_VERTICAL)
  982. vSizer.Add(self.rgb_range_label, pos=(row, 2),
  983. flag=wx.ALIGN_CENTER_VERTICAL)
  984. row += 1
  985. if self.version7 and self.attributeType == 'color':
  986. vSizer.Add(self.useColumn, pos=(row, 0), span=(1, 2),
  987. flag=wx.ALIGN_CENTER_VERTICAL)
  988. row += 1
  989. vSizer.Add(fromColumnLabel, pos=(row, 0),
  990. flag=wx.ALIGN_CENTER_VERTICAL)
  991. vSizer.Add(self.fromColumn, pos=(row, 1),
  992. flag=wx.ALIGN_CENTER_VERTICAL)
  993. row += 1
  994. vSizer.Add(toColumnLabel, pos=(row, 0),
  995. flag=wx.ALIGN_CENTER_VERTICAL)
  996. vSizer.Add(self.toColumn, pos=(row, 1),
  997. flag=wx.ALIGN_CENTER_VERTICAL)
  998. vSizer.Add(self.addColumn, pos=(row, 2),
  999. flag=wx.ALIGN_CENTER_VERTICAL)
  1000. inputSizer.Add(
  1001. vSizer,
  1002. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALL | wx.EXPAND,
  1003. border=5)
  1004. self.colorColumnSizer = vSizer
  1005. return inputSizer
  1006. def _doLayout(self):
  1007. """Do main layout"""
  1008. scrollPanel = scrolled.ScrolledPanel(parent=self.panel, id=wx.ID_ANY,
  1009. style=wx.TAB_TRAVERSAL)
  1010. scrollPanel.SetupScrolling()
  1011. sizer = wx.BoxSizer(wx.VERTICAL)
  1012. #
  1013. # map selection
  1014. #
  1015. mapSelection = self._createMapSelection(parent=scrollPanel)
  1016. sizer.Add(mapSelection, proportion=0,
  1017. flag=wx.ALL | wx.EXPAND, border=5)
  1018. #
  1019. # manage extern tables
  1020. #
  1021. if self.version7 and self.attributeType == 'color':
  1022. self.cp = wx.CollapsiblePane(
  1023. scrollPanel,
  1024. label=_("Import or export color table"),
  1025. id=wx.ID_ANY,
  1026. style=wx.CP_DEFAULT_STYLE | wx.CP_NO_TLW_RESIZE)
  1027. self.Bind(
  1028. wx.EVT_COLLAPSIBLEPANE_CHANGED,
  1029. self.OnPaneChanged,
  1030. self.cp)
  1031. self._createFileSelection(parent=self.cp.GetPane())
  1032. sizer.Add(self.cp, proportion=0,
  1033. flag=wx.ALL | wx.EXPAND, border=5)
  1034. #
  1035. # set vector attributes
  1036. #
  1037. vectorAttrb = self._createVectorAttrb(parent=scrollPanel)
  1038. sizer.Add(vectorAttrb, proportion=0,
  1039. flag=wx.ALL | wx.EXPAND, border=5)
  1040. #
  1041. # body & preview
  1042. #
  1043. bodySizer = self._createBody(parent=scrollPanel)
  1044. sizer.Add(bodySizer, proportion=1,
  1045. flag=wx.ALL | wx.EXPAND, border=5)
  1046. scrollPanel.SetSizer(sizer)
  1047. scrollPanel.Fit()
  1048. #
  1049. # buttons
  1050. #
  1051. btnSizer = self._createButtons(self.panel)
  1052. mainsizer = wx.BoxSizer(wx.VERTICAL)
  1053. mainsizer.Add(
  1054. scrollPanel,
  1055. proportion=1,
  1056. flag=wx.EXPAND | wx.ALL,
  1057. border=5)
  1058. mainsizer.Add(wx.StaticLine(parent=self.panel, id=wx.ID_ANY,
  1059. style=wx.LI_HORIZONTAL), proportion=0,
  1060. flag=wx.EXPAND | wx.ALL, border=5)
  1061. mainsizer.Add(btnSizer, proportion=0,
  1062. flag=wx.ALL | wx.ALIGN_RIGHT | wx.EXPAND, border=5)
  1063. self.panel.SetSizer(mainsizer)
  1064. mainsizer.Layout()
  1065. mainsizer.Fit(self.panel)
  1066. self.Layout()
  1067. def OnPaneChanged(self, event=None):
  1068. # redo the layout
  1069. self.panel.Layout()
  1070. # and also change the labels
  1071. if self.cp.IsExpanded():
  1072. self.cp.SetLabel('')
  1073. else:
  1074. self.cp.SetLabel(_("Import or export color table"))
  1075. def CheckMapset(self):
  1076. """Check if current vector is in current mapset"""
  1077. if grass.find_file(name=self.inmap, element='vector')[
  1078. 'mapset'] == grass.gisenv()['MAPSET']:
  1079. return True
  1080. else:
  1081. return False
  1082. def NoConnection(self, vectorName):
  1083. dlg = wx.MessageDialog(
  1084. parent=self,
  1085. message=_(
  1086. "Database connection for vector map <%s> "
  1087. "is not defined in DB file. Do you want to create and "
  1088. "connect new attribute table?") %
  1089. vectorName,
  1090. caption=_("No database connection defined"),
  1091. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION | wx.CENTRE)
  1092. if dlg.ShowModal() == wx.ID_YES:
  1093. dlg.Destroy()
  1094. GUI(parent=self).ParseCommand(['v.db.addtable', 'map=' + self.inmap],
  1095. completed=(self.CreateAttrTable, self.inmap, ''))
  1096. else:
  1097. dlg.Destroy()
  1098. def OnCheckColumn(self, event):
  1099. """Use color column instead of color table"""
  1100. if self.useColumn.GetValue():
  1101. self.properties['loadColumn'] = self.fromColumn.GetValue()
  1102. self.properties['storeColumn'] = self.toColumn.GetValue()
  1103. self.fromColumn.Enable(True)
  1104. self.toColumn.Enable(True)
  1105. self.colorTable = False
  1106. if self.properties['loadColumn']:
  1107. self.LoadTable()
  1108. else:
  1109. self.rulesPanel.Clear()
  1110. else:
  1111. self.properties['loadColumn'] = ''
  1112. self.properties['storeColumn'] = ''
  1113. self.fromColumn.Enable(False)
  1114. self.toColumn.Enable(False)
  1115. self.colorTable = True
  1116. self.LoadTable()
  1117. def EnableVectorAttributes(self, enable):
  1118. """Enable/disable part of dialog connected with db"""
  1119. for child in self.colorColumnSizer.GetChildren():
  1120. child.GetWindow().Enable(enable)
  1121. def DisableClearAll(self):
  1122. """Enable, disable the whole dialog"""
  1123. self.rulesPanel.Clear()
  1124. self.EnableVectorAttributes(False)
  1125. self.btnPreview.Enable(False)
  1126. self.btnOK.Enable(False)
  1127. self.btnApply.Enable(False)
  1128. self.preview.EraseMap()
  1129. def OnSelectionInput(self, event):
  1130. """Vector map selected"""
  1131. if event:
  1132. if self.inmap:
  1133. # switch to another map -> delete temporary column
  1134. self.DeleteTemporaryColumn()
  1135. self.inmap = event.GetString()
  1136. if self.version7 and self.attributeType == 'color':
  1137. self.loadRules.SetValue('')
  1138. self.saveRules.SetValue('')
  1139. if self.inmap:
  1140. if not grass.find_file(name=self.inmap, element='vector')['file']:
  1141. self.inmap = None
  1142. self.UpdateDialog()
  1143. def UpdateDialog(self):
  1144. """Update dialog after map selection"""
  1145. if not self.inmap:
  1146. self.DisableClearAll()
  1147. return
  1148. if not self.CheckMapset():
  1149. # v.colors doesn't need the map to be in current mapset
  1150. if not (self.version7 and self.attributeType == 'color'):
  1151. message = _(
  1152. "Selected map <%(map)s> is not in current mapset <%(mapset)s>. "
  1153. "Attribute table cannot be edited.") % {
  1154. 'map': self.inmap,
  1155. 'mapset': grass.gisenv()['MAPSET']}
  1156. wx.CallAfter(GMessage, parent=self, message=message)
  1157. self.DisableClearAll()
  1158. return
  1159. # check for db connection
  1160. self.dbInfo = VectorDBInfo(self.inmap)
  1161. enable = True
  1162. if not len(self.dbInfo.layers): # no connection
  1163. if not (self.version7 and self.attributeType ==
  1164. 'color'): # otherwise it doesn't matter
  1165. wx.CallAfter(self.NoConnection, self.inmap)
  1166. enable = False
  1167. for combo in (self.layerSelect, self.sourceColumn,
  1168. self.fromColumn, self.toColumn):
  1169. combo.SetValue("")
  1170. combo.Clear()
  1171. for prop in ('sourceColumn', 'loadColumn', 'storeColumn'):
  1172. self.properties[prop] = ''
  1173. self.EnableVectorAttributes(False)
  1174. else: # db connection exist
  1175. # initialize layer selection combobox
  1176. self.EnableVectorAttributes(True)
  1177. self.layerSelect.InsertLayers(self.inmap)
  1178. # initialize attribute table for layer=1
  1179. self.properties['layer'] = self.layerSelect.GetString(0)
  1180. self.layerSelect.SetStringSelection(self.properties['layer'])
  1181. layer = int(self.properties['layer'])
  1182. self.properties['table'] = self.dbInfo.layers[layer]['table']
  1183. if self.attributeType == 'color':
  1184. self.AddTemporaryColumn(type='varchar(11)')
  1185. else:
  1186. self.AddTemporaryColumn(type='integer')
  1187. # initialize column selection comboboxes
  1188. self.OnLayerSelection(event=None)
  1189. if self.version7 and self.attributeType == 'color':
  1190. self.useColumn.SetValue(False)
  1191. self.OnCheckColumn(event=None)
  1192. self.useColumn.Enable(self.CheckMapset())
  1193. else:
  1194. self.LoadTable()
  1195. self.btnPreview.Enable(enable)
  1196. self.btnOK.Enable(enable)
  1197. self.btnApply.Enable(enable)
  1198. def AddTemporaryColumn(self, type):
  1199. """Add temporary column to not overwrite the original values,
  1200. need to be deleted when closing dialog and unloading map
  1201. :param type: type of column (e.g. vachar(11))"""
  1202. if not self.CheckMapset():
  1203. return
  1204. # because more than one dialog with the same map can be opened we must test column name and
  1205. # create another one
  1206. while self.properties['tmpColumn'] in self.dbInfo.GetTableDesc(self.properties[
  1207. 'table']).keys():
  1208. name, idx = self.properties['tmpColumn'].split('_')
  1209. idx = int(idx)
  1210. idx += 1
  1211. self.properties['tmpColumn'] = name + '_' + str(idx)
  1212. if self.version7:
  1213. modul = 'v.db.addcolumn'
  1214. else:
  1215. modul = 'v.db.addcol'
  1216. ret = RunCommand(modul,
  1217. parent=self,
  1218. map=self.inmap,
  1219. layer=self.properties['layer'],
  1220. column='%s %s' % (self.properties['tmpColumn'], type))
  1221. def DeleteTemporaryColumn(self):
  1222. """Delete temporary column"""
  1223. if not self.CheckMapset():
  1224. return
  1225. if self.inmap:
  1226. if self.version7:
  1227. modul = 'v.db.dropcolumn'
  1228. else:
  1229. modul = 'v.db.dropcol'
  1230. ret = RunCommand(modul,
  1231. map=self.inmap,
  1232. layer=self.properties['layer'],
  1233. column=self.properties['tmpColumn'])
  1234. def OnLayerSelection(self, event):
  1235. # reset choices in column selection comboboxes if layer changes
  1236. vlayer = int(self.layerSelect.GetStringSelection())
  1237. self._columnWidgetEvtHandler(bind=False)
  1238. self.sourceColumn.InsertColumns(
  1239. vector=self.inmap, layer=vlayer,
  1240. type=['integer', 'double precision'],
  1241. dbInfo=self.dbInfo, excludeCols=['tmpColumn'])
  1242. self.sourceColumn.SetValue('cat')
  1243. self.properties['sourceColumn'] = self.sourceColumn.GetValue()
  1244. if self.attributeType == 'color':
  1245. type = ['character']
  1246. else:
  1247. type = ['integer']
  1248. self.fromColumn.InsertColumns(
  1249. vector=self.inmap,
  1250. layer=vlayer,
  1251. type=type,
  1252. dbInfo=self.dbInfo,
  1253. excludeCols=['tmpColumn'])
  1254. self.toColumn.InsertColumns(
  1255. vector=self.inmap,
  1256. layer=vlayer,
  1257. type=type,
  1258. dbInfo=self.dbInfo,
  1259. excludeCols=['tmpColumn'])
  1260. v = self.columnsProp[self.attributeType]['name']
  1261. found = False
  1262. if v in self.fromColumn.GetColumns():
  1263. found = True
  1264. if found != wx.NOT_FOUND:
  1265. self.fromColumn.SetValue(v)
  1266. self.toColumn.SetValue(v)
  1267. self.properties['loadColumn'] = v
  1268. self.properties['storeColumn'] = v
  1269. else:
  1270. self.properties['loadColumn'] = ''
  1271. self.properties['storeColumn'] = ''
  1272. self._columnWidgetEvtHandler()
  1273. if event:
  1274. self.LoadTable()
  1275. self.Update()
  1276. def OnSourceColumnSelection(self, event):
  1277. self.properties['sourceColumn'] = event.GetString()
  1278. self.LoadTable()
  1279. def OnAddColumn(self, event):
  1280. """Add GRASS(RGB,SIZE,WIDTH) column if it doesn't exist"""
  1281. if self.columnsProp[self.attributeType][
  1282. 'name'] not in self.fromColumn.GetColumns():
  1283. if self.version7:
  1284. modul = 'v.db.addcolumn'
  1285. else:
  1286. modul = 'v.db.addcol'
  1287. ret = RunCommand(
  1288. modul, map=self.inmap, layer=self.properties['layer'], columns='%s %s' %
  1289. (self.columnsProp[
  1290. self.attributeType]['name'], self.columnsProp[
  1291. self.attributeType]['type1']))
  1292. self.toColumn.InsertColumns(
  1293. self.inmap,
  1294. self.properties['layer'],
  1295. type=self.columnsProp[
  1296. self.attributeType]['type2'])
  1297. self.toColumn.SetValue(
  1298. self.columnsProp[
  1299. self.attributeType]['name'])
  1300. self.properties['storeColumn'] = self.toColumn.GetValue()
  1301. self.LoadTable()
  1302. else:
  1303. GMessage(parent=self,
  1304. message=_("%s column already exists.") %
  1305. self.columnsProp[self.attributeType]['name'])
  1306. def CreateAttrTable(self, dcmd, layer, params, propwin):
  1307. """Create attribute table"""
  1308. if dcmd:
  1309. cmd = cmdlist_to_tuple(dcmd)
  1310. ret = RunCommand(cmd[0], **cmd[1])
  1311. if ret == 0:
  1312. self.OnSelectionInput(None)
  1313. return True
  1314. for combo in (self.layerSelect, self.sourceColumn,
  1315. self.fromColumn, self.toColumn):
  1316. combo.SetValue("")
  1317. combo.Disable()
  1318. return False
  1319. def LoadTable(self):
  1320. """Load table"""
  1321. if self.colorTable:
  1322. ColorTable.LoadTable(self, mapType='vector')
  1323. else:
  1324. self.LoadRulesFromColumn()
  1325. def LoadRulesFromColumn(self):
  1326. """Load current column (GRASSRGB, size column)"""
  1327. self.rulesPanel.Clear()
  1328. if not self.properties['sourceColumn']:
  1329. self.preview.EraseMap()
  1330. return
  1331. busy = wx.BusyInfo(
  1332. _("Please wait, loading data from attribute table..."),
  1333. parent=self)
  1334. wx.GetApp().Yield()
  1335. columns = self.properties['sourceColumn']
  1336. if self.properties['loadColumn']:
  1337. columns += ',' + self.properties['loadColumn']
  1338. sep = ';'
  1339. if self.inmap:
  1340. outFile = tempfile.NamedTemporaryFile(mode='w+')
  1341. ret = RunCommand('v.db.select',
  1342. quiet=True,
  1343. flags='c',
  1344. map=self.inmap,
  1345. layer=self.properties['layer'],
  1346. columns=columns,
  1347. sep=sep,
  1348. stdout=outFile)
  1349. else:
  1350. self.preview.EraseMap()
  1351. del busy
  1352. return
  1353. outFile.seek(0)
  1354. i = 0
  1355. minim = maxim = 0.0
  1356. limit = 1000
  1357. colvallist = []
  1358. readvals = False
  1359. while True:
  1360. # os.linesep doesn't work here (MSYS)
  1361. record = outFile.readline().replace('\n', '')
  1362. if not record:
  1363. break
  1364. self.rulesPanel.ruleslines[i] = {}
  1365. if not self.properties['loadColumn']:
  1366. col1 = record
  1367. col2 = None
  1368. else:
  1369. col1, col2 = record.split(sep)
  1370. if float(col1) < minim:
  1371. minim = float(col1)
  1372. if float(col1) > maxim:
  1373. maxim = float(col1)
  1374. # color rules list should only have unique values of col1, not all
  1375. # records
  1376. if col1 not in colvallist:
  1377. self.rulesPanel.ruleslines[i]['value'] = col1
  1378. self.rulesPanel.ruleslines[i][self.attributeType] = col2
  1379. colvallist.append(col1)
  1380. i += 1
  1381. if i > limit and readvals is False:
  1382. dlg = wx.MessageDialog(parent=self, message=_(
  1383. "Number of loaded records reached %d, "
  1384. "displaying all the records will be time-consuming "
  1385. "and may lead to computer freezing, "
  1386. "do you still want to continue?") % i,
  1387. caption=_("Too many records"),
  1388. style=wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
  1389. if dlg.ShowModal() == wx.ID_YES:
  1390. readvals = True
  1391. dlg.Destroy()
  1392. else:
  1393. del busy
  1394. dlg.Destroy()
  1395. self.updateColumn = False
  1396. return
  1397. self.rulesPanel.AddRules(i, start=True)
  1398. ret = self.rulesPanel.LoadRules()
  1399. self.properties['min'], self.properties['max'] = minim, maxim
  1400. self.SetRangeLabel()
  1401. if ret:
  1402. self.OnPreview()
  1403. else:
  1404. self.rulesPanel.Clear()
  1405. del busy
  1406. def SetRangeLabel(self):
  1407. """Set labels with info about attribute column range"""
  1408. if self.properties['sourceColumn']:
  1409. ctype = self.dbInfo.GetTableDesc(
  1410. self.properties['table'])[
  1411. self.properties['sourceColumn']]['ctype']
  1412. else:
  1413. ctype = int
  1414. range = ''
  1415. if self.properties['min'] or self.properties['max']:
  1416. if ctype == float:
  1417. range = "%s: %.1f - %.1f)" % (_("range"),
  1418. self.properties['min'],
  1419. self.properties['max'])
  1420. elif ctype == int:
  1421. range = "%s: %d - %d)" % (_("range"),
  1422. self.properties['min'],
  1423. self.properties['max'])
  1424. if range:
  1425. if self.colorTable:
  1426. self.cr_label.SetLabel(
  1427. _("Enter vector attribute values or percents %s:") %
  1428. range)
  1429. else:
  1430. self.cr_label.SetLabel(
  1431. _("Enter vector attribute values %s:") %
  1432. range)
  1433. else:
  1434. if self.colorTable:
  1435. self.cr_label.SetLabel(
  1436. _("Enter vector attribute values or percents:"))
  1437. else:
  1438. self.cr_label.SetLabel(_("Enter vector attribute values:"))
  1439. def OnFromColSelection(self, event):
  1440. """Selection in combobox (for loading values) changed"""
  1441. self.properties['loadColumn'] = event.GetString()
  1442. self.LoadTable()
  1443. def OnToColSelection(self, event):
  1444. """Selection in combobox (for storing values) changed"""
  1445. self.properties['storeColumn'] = event.GetString()
  1446. def OnPreview(self, event=None, tmp=True):
  1447. """Update preview (based on computational region)"""
  1448. if self.colorTable:
  1449. self.OnTablePreview(tmp)
  1450. else:
  1451. self.OnColumnPreview()
  1452. def OnTablePreview(self, tmp):
  1453. """Update preview (based on computational region)"""
  1454. if not self.inmap:
  1455. self.preview.EraseMap()
  1456. return
  1457. ltype = 'vector'
  1458. cmdlist = ['d.vect',
  1459. 'map=%s' % self.inmap]
  1460. # find existing color table and copy to temp file
  1461. try:
  1462. name, mapset = self.inmap.split('@')
  1463. except ValueError:
  1464. name = self.inmap
  1465. mapset = grass.find_file(self.inmap, element='cell')['mapset']
  1466. if not mapset:
  1467. return
  1468. old_colrtable = None
  1469. if mapset == grass.gisenv()['MAPSET']:
  1470. old_colrtable = grass.find_file(
  1471. name='colr', element=os.path.join(
  1472. 'vector', name))['file']
  1473. else:
  1474. old_colrtable = grass.find_file(
  1475. name=name, element=os.path.join(
  1476. 'vcolr2', mapset))['file']
  1477. if old_colrtable:
  1478. colrtemp = utils.GetTempfile()
  1479. shutil.copyfile(old_colrtable, colrtemp)
  1480. ColorTable.DoPreview(self, ltype, cmdlist)
  1481. # restore previous color table
  1482. if tmp:
  1483. if old_colrtable:
  1484. shutil.copyfile(colrtemp, old_colrtable)
  1485. os.remove(colrtemp)
  1486. else:
  1487. RunCommand('v.colors',
  1488. parent=self,
  1489. flags='r',
  1490. map=self.inmap)
  1491. def OnColumnPreview(self):
  1492. """Update preview (based on computational region)"""
  1493. if not self.inmap or not self.properties['tmpColumn']:
  1494. self.preview.EraseMap()
  1495. return
  1496. cmdlist = ['d.vect',
  1497. 'map=%s' % self.inmap,
  1498. 'type=point,line,boundary,area']
  1499. if self.attributeType == 'color':
  1500. cmdlist.append('rgb_column=%s' % self.properties['tmpColumn'])
  1501. elif self.attributeType == 'size':
  1502. cmdlist.append('size_column=%s' % self.properties['tmpColumn'])
  1503. elif self.attributeType == 'width':
  1504. cmdlist.append('width_column=%s' % self.properties['tmpColumn'])
  1505. ltype = 'vector'
  1506. ColorTable.DoPreview(self, ltype, cmdlist)
  1507. def OnHelp(self, event):
  1508. """Show GRASS manual page"""
  1509. cmd = 'v.colors'
  1510. ColorTable.RunHelp(self, cmd=cmd)
  1511. def UseAttrColumn(self, useAttrColumn):
  1512. """Find layers and apply the changes in d.vect command"""
  1513. layers = self.layerTree.FindItemByData(key='name', value=self.inmap)
  1514. if not layers:
  1515. return
  1516. for layer in layers:
  1517. if self.layerTree.GetLayerInfo(layer, key='type') != 'vector':
  1518. continue
  1519. cmdlist = self.layerTree.GetLayerInfo(
  1520. layer, key='maplayer').GetCmd()
  1521. if self.attributeType == 'color':
  1522. if useAttrColumn:
  1523. cmdlist[1].update(
  1524. {'rgb_column': self.properties['storeColumn']})
  1525. else:
  1526. cmdlist[1].pop('rgb_column', None)
  1527. elif self.attributeType == 'size':
  1528. cmdlist[1].update(
  1529. {'size_column': self.properties['storeColumn']})
  1530. elif self.attributeType == 'width':
  1531. cmdlist[1].update(
  1532. {'width_column': self.properties['storeColumn']})
  1533. self.layerTree.SetLayerInfo(layer, key='cmd', value=cmdlist)
  1534. def CreateColorTable(self, tmp=False):
  1535. """Create color rules (color table or color column)"""
  1536. if self.colorTable:
  1537. ret = ColorTable.CreateColorTable(self)
  1538. else:
  1539. if self.updateColumn:
  1540. ret = self.UpdateColorColumn(tmp)
  1541. else:
  1542. ret = True
  1543. return ret
  1544. def UpdateColorColumn(self, tmp):
  1545. """Creates color table
  1546. :return: True on success
  1547. :return: False on failure
  1548. """
  1549. rulestxt = ''
  1550. for rule in six.itervalues(self.rulesPanel.ruleslines):
  1551. if 'value' not in rule: # skip empty rules
  1552. break
  1553. if tmp:
  1554. rgb_col = self.properties['tmpColumn']
  1555. else:
  1556. rgb_col = self.properties['storeColumn']
  1557. if not self.properties['storeColumn']:
  1558. GMessage(parent=self.parent, message=_(
  1559. "Please select column to save values to."))
  1560. rulestxt += "UPDATE %s SET %s='%s' WHERE %s ;\n" % (
  1561. self.properties['table'],
  1562. rgb_col, rule[self.attributeType],
  1563. rule['value'])
  1564. if not rulestxt:
  1565. return False
  1566. gtemp = utils.GetTempfile()
  1567. output = open(gtemp, "w")
  1568. try:
  1569. output.write(rulestxt)
  1570. finally:
  1571. output.close()
  1572. RunCommand('db.execute',
  1573. parent=self,
  1574. input=gtemp)
  1575. return True
  1576. def OnCancel(self, event):
  1577. """Do not apply any changes and close the dialog"""
  1578. self.DeleteTemporaryColumn()
  1579. self.Map.Clean()
  1580. self.Destroy()
  1581. def _apply(self, updatePreview=True):
  1582. """Apply selected color table
  1583. :return: True on success otherwise False
  1584. """
  1585. if self.colorTable:
  1586. self.UseAttrColumn(False)
  1587. else:
  1588. if not self.properties['storeColumn']:
  1589. GError(_("No color column defined. Operation canceled."),
  1590. parent=self)
  1591. return
  1592. self.UseAttrColumn(True)
  1593. return ColorTable._apply(self, updatePreview)
  1594. def _columnWidgetEvtHandler(self, bind=True):
  1595. """Bind/Unbind Column widgets handlers"""
  1596. widgets = [
  1597. {
  1598. 'widget': self.sourceColumn,
  1599. 'event': wx.EVT_TEXT,
  1600. 'handler': self.OnSourceColumnSelection,
  1601. },
  1602. {
  1603. 'widget': self.fromColumn,
  1604. 'event': wx.EVT_TEXT,
  1605. 'handler': self.OnFromColSelection,
  1606. },
  1607. {
  1608. 'widget': self.toColumn,
  1609. 'event': wx.EVT_TEXT,
  1610. 'handler': self.OnToColSelection,
  1611. },
  1612. ]
  1613. for widget in widgets:
  1614. if bind is True:
  1615. getattr(widget['widget'], 'Bind')(
  1616. widget['event'], widget['handler'],
  1617. )
  1618. else:
  1619. getattr(widget['widget'], 'Unbind')(widget['event'])
  1620. class ThematicVectorTable(VectorColorTable):
  1621. def __init__(self, parent, vectorType, **kwargs):
  1622. """Dialog for interactively entering color/size rules
  1623. for vector maps for thematic mapping in nviz"""
  1624. self.vectorType = vectorType
  1625. VectorColorTable.__init__(self, parent=parent, **kwargs)
  1626. self.SetTitle(_("Thematic mapping for vector map in 3D view"))
  1627. def _initLayer(self):
  1628. """Set initial layer when opening dialog"""
  1629. self.inmap = self.parent.GetLayerData(nvizType='vector', nameOnly=True)
  1630. self.selectionInput.SetValue(self.inmap)
  1631. self.selectionInput.Disable()
  1632. def _apply(self, updatePreview=True):
  1633. """Apply selected color table
  1634. :return: True on success otherwise False
  1635. """
  1636. ret = self.CreateColorTable()
  1637. if not ret:
  1638. GMessage(parent=self, message=_("No valid color rules given."))
  1639. data = self.parent.GetLayerData(nvizType='vector')
  1640. data['vector']['points']['thematic'][
  1641. 'layer'] = int(self.properties['layer'])
  1642. value = None
  1643. if self.properties['storeColumn']:
  1644. value = self.properties['storeColumn']
  1645. if not self.colorTable:
  1646. if self.attributeType == 'color':
  1647. data['vector'][self.vectorType][
  1648. 'thematic']['rgbcolumn'] = value
  1649. else:
  1650. data['vector'][self.vectorType][
  1651. 'thematic']['sizecolumn'] = value
  1652. else:
  1653. if self.attributeType == 'color':
  1654. data['vector'][self.vectorType]['thematic']['rgbcolumn'] = None
  1655. else:
  1656. data['vector'][self.vectorType][
  1657. 'thematic']['sizecolumn'] = None
  1658. data['vector'][self.vectorType]['thematic']['update'] = None
  1659. from nviz.main import haveNviz
  1660. if haveNviz:
  1661. from nviz.mapwindow import wxUpdateProperties
  1662. event = wxUpdateProperties(data=data)
  1663. wx.PostEvent(self.parent.mapWindow, event)
  1664. self.parent.mapWindow.Refresh(False)
  1665. return ret
  1666. class BufferedWindow(wx.Window):
  1667. """A Buffered window class"""
  1668. def __init__(self, parent, id,
  1669. style=wx.NO_FULL_REPAINT_ON_RESIZE,
  1670. Map=None, **kwargs):
  1671. wx.Window.__init__(self, parent, id, style=style, **kwargs)
  1672. self.parent = parent
  1673. self.Map = Map
  1674. # re-render the map from GRASS or just redraw image
  1675. self.render = True
  1676. # indicates whether or not a resize event has taken place
  1677. self.resize = False
  1678. #
  1679. # event bindings
  1680. #
  1681. self.Bind(wx.EVT_PAINT, self.OnPaint)
  1682. self.Bind(wx.EVT_IDLE, self.OnIdle)
  1683. self.Bind(wx.EVT_ERASE_BACKGROUND, lambda x: None)
  1684. #
  1685. # render output objects
  1686. #
  1687. # image file to be rendered
  1688. self.mapfile = None
  1689. # wx.Image object (self.mapfile)
  1690. self.img = None
  1691. self.pdc = PseudoDC()
  1692. # will store an off screen empty bitmap for saving to file
  1693. self._Buffer = None
  1694. # make sure that extents are updated at init
  1695. self.Map.region = self.Map.GetRegion()
  1696. self.Map.SetRegion()
  1697. self.Map.GetRenderMgr().renderDone.connect(self._updatePreviewFinished)
  1698. def Draw(self, pdc, img=None, pdctype='image'):
  1699. """Draws preview or clears window"""
  1700. pdc.BeginDrawing()
  1701. Debug.msg(3, "BufferedWindow.Draw(): pdctype=%s" % (pdctype))
  1702. if pdctype == 'clear': # erase the display
  1703. bg = wx.WHITE_BRUSH
  1704. pdc.SetBackground(bg)
  1705. pdc.Clear()
  1706. self.Refresh()
  1707. pdc.EndDrawing()
  1708. return
  1709. if pdctype == 'image' and img:
  1710. bg = wx.TRANSPARENT_BRUSH
  1711. pdc.SetBackground(bg)
  1712. bitmap = BitmapFromImage(img)
  1713. w, h = bitmap.GetSize()
  1714. pdc.DrawBitmap(bitmap, 0, 0, True) # draw the composite map
  1715. pdc.EndDrawing()
  1716. self.Refresh()
  1717. def OnPaint(self, event):
  1718. """Draw pseudo DC to buffer"""
  1719. self._Buffer = EmptyBitmap(self.Map.width, self.Map.height)
  1720. dc = wx.BufferedPaintDC(self, self._Buffer)
  1721. # use PrepareDC to set position correctly
  1722. # probably does nothing, removed from wxPython 2.9
  1723. # self.PrepareDC(dc)
  1724. # we need to clear the dc BEFORE calling PrepareDC
  1725. bg = wx.Brush(self.GetBackgroundColour())
  1726. dc.SetBackground(bg)
  1727. dc.Clear()
  1728. # create a clipping rect from our position and size
  1729. # and the Update Region
  1730. rgn = self.GetUpdateRegion()
  1731. r = rgn.GetBox()
  1732. # draw to the dc using the calculated clipping rect
  1733. self.pdc.DrawToDCClipped(dc, r)
  1734. def OnSize(self, event):
  1735. """Init image size to match window size"""
  1736. # set size of the input image
  1737. self.Map.width, self.Map.height = self.GetClientSize()
  1738. # Make new off screen bitmap: this bitmap will always have the
  1739. # current drawing in it, so it can be used to save the image to
  1740. # a file, or whatever.
  1741. self._Buffer = EmptyBitmap(self.Map.width, self.Map.height)
  1742. # get the image to be rendered
  1743. self.img = self.GetImage()
  1744. # update map display
  1745. if self.img and self.Map.width + self.Map.height > 0: # scale image during resize
  1746. self.img = self.img.Scale(self.Map.width, self.Map.height)
  1747. self.render = False
  1748. self.UpdatePreview()
  1749. # re-render image on idle
  1750. self.resize = True
  1751. def OnIdle(self, event):
  1752. """Only re-render a preview image from GRASS during
  1753. idle time instead of multiple times during resizing.
  1754. """
  1755. if self.resize:
  1756. self.render = True
  1757. self.UpdatePreview()
  1758. event.Skip()
  1759. def GetImage(self):
  1760. """Converts files to wx.Image"""
  1761. if self.Map.mapfile and os.path.isfile(self.Map.mapfile) and \
  1762. os.path.getsize(self.Map.mapfile):
  1763. img = wx.Image(self.Map.mapfile, wx.BITMAP_TYPE_ANY)
  1764. else:
  1765. img = None
  1766. return img
  1767. def UpdatePreview(self, img=None):
  1768. """Update canvas if window changes geometry"""
  1769. Debug.msg(
  1770. 2, "BufferedWindow.UpdatePreview(%s): render=%s" %
  1771. (img, self.render))
  1772. if not self.render:
  1773. return
  1774. # extent is taken from current map display
  1775. try:
  1776. self.Map.region = copy.deepcopy(
  1777. self.parent.parent.GetLayerTree().GetMap().GetCurrentRegion())
  1778. except AttributeError:
  1779. self.Map.region = self.Map.GetRegion()
  1780. # render new map images
  1781. self.mapfile = self.Map.Render(force=self.render)
  1782. def _updatePreviewFinished(self):
  1783. if not self.render:
  1784. return
  1785. self.img = self.GetImage()
  1786. self.resize = False
  1787. if not self.img:
  1788. return
  1789. # paint images to PseudoDC
  1790. self.pdc.Clear()
  1791. self.pdc.RemoveAll()
  1792. # draw map image background
  1793. self.Draw(self.pdc, self.img, pdctype='image')
  1794. self.resize = False
  1795. def EraseMap(self):
  1796. """Erase preview"""
  1797. self.Draw(self.pdc, pdctype='clear')