vdigit.py 64 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545
  1. """!
  2. @package vdigit
  3. @brief Dialogs for wxGUI vector digitizer
  4. Classes:
  5. - VDigit
  6. - VDigitSettingsDialog
  7. - VDigitCategoryDialog
  8. - CategoryListCtrl
  9. - VDigitZBulkDialog
  10. - VDigitDuplicatesDialog
  11. - CheckListFeature
  12. (C) 2007-2011 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 Martin Landa <landa.martin gmail.com>
  16. """
  17. import os
  18. import sys
  19. import string
  20. import copy
  21. import textwrap
  22. import traceback
  23. from threading import Thread
  24. import wx
  25. import wx.lib.colourselect as csel
  26. import wx.lib.mixins.listctrl as listmix
  27. import gcmd
  28. import dbm
  29. from debug import Debug as Debug
  30. import gselect
  31. import globalvar
  32. from units import Units
  33. from preferences import globalSettings as UserSettings
  34. try:
  35. from wxvdigit import IVDigit, GV_LINES
  36. haveVDigit = True
  37. errorMsg = ''
  38. except ImportError, err:
  39. haveVDigit = False
  40. errorMsg = err
  41. GV_LINES = -1
  42. class IVDigit:
  43. def __init__(self):
  44. pass
  45. class VDigit(IVDigit):
  46. def __init__(self, mapwindow):
  47. """!Base class of vector digitizer
  48. @param mapwindow reference to mapwindow (mapdisp_window.BufferedWindow) instance
  49. """
  50. IVDigit.__init__(self, mapwindow)
  51. class VDigitSettingsDialog(wx.Dialog):
  52. def __init__(self, parent, title, style = wx.DEFAULT_DIALOG_STYLE):
  53. """!Standard settings dialog for digitization purposes
  54. """
  55. wx.Dialog.__init__(self, parent = parent, id = wx.ID_ANY, title = title, style = style)
  56. self.parent = parent # mapdisplay.MapFrame class instance
  57. self.digit = self.parent.MapWindow.digit
  58. # notebook
  59. notebook = wx.Notebook(parent = self, id = wx.ID_ANY, style = wx.BK_DEFAULT)
  60. self._createSymbologyPage(notebook)
  61. self.digit.SetCategory()
  62. self._createGeneralPage(notebook)
  63. self._createAttributesPage(notebook)
  64. self._createQueryPage(notebook)
  65. # buttons
  66. btnApply = wx.Button(self, wx.ID_APPLY)
  67. btnCancel = wx.Button(self, wx.ID_CANCEL)
  68. btnSave = wx.Button(self, wx.ID_SAVE)
  69. btnSave.SetDefault()
  70. # bindigs
  71. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  72. btnApply.SetToolTipString(_("Apply changes for this session"))
  73. btnApply.SetDefault()
  74. btnSave.Bind(wx.EVT_BUTTON, self.OnSave)
  75. btnSave.SetToolTipString(_("Close dialog and save changes to user settings file"))
  76. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  77. btnCancel.SetToolTipString(_("Close dialog and ignore changes"))
  78. # sizers
  79. btnSizer = wx.StdDialogButtonSizer()
  80. btnSizer.AddButton(btnCancel)
  81. btnSizer.AddButton(btnApply)
  82. btnSizer.AddButton(btnSave)
  83. btnSizer.Realize()
  84. mainSizer = wx.BoxSizer(wx.VERTICAL)
  85. mainSizer.Add(item = notebook, proportion = 1, flag = wx.EXPAND | wx.ALL, border = 5)
  86. mainSizer.Add(item = btnSizer, proportion = 0,
  87. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  88. self.Bind(wx.EVT_CLOSE, self.OnCancel)
  89. self.SetSizer(mainSizer)
  90. mainSizer.Fit(self)
  91. def _createSymbologyPage(self, notebook):
  92. """!Create notebook page concerning symbology settings"""
  93. panel = wx.Panel(parent = notebook, id = wx.ID_ANY)
  94. notebook.AddPage(page = panel, text = _("Symbology"))
  95. sizer = wx.BoxSizer(wx.VERTICAL)
  96. flexSizer = wx.FlexGridSizer (cols = 3, hgap = 5, vgap = 5)
  97. flexSizer.AddGrowableCol(0)
  98. self.symbology = {}
  99. for label, key in self._symbologyData():
  100. textLabel = wx.StaticText(panel, wx.ID_ANY, label)
  101. color = csel.ColourSelect(panel, id = wx.ID_ANY,
  102. colour = UserSettings.Get(group = 'vdigit', key = 'symbol',
  103. subkey = [key, 'color']), size = globalvar.DIALOG_COLOR_SIZE)
  104. isEnabled = UserSettings.Get(group = 'vdigit', key = 'symbol',
  105. subkey = [key, 'enabled'])
  106. if isEnabled is not None:
  107. enabled = wx.CheckBox(panel, id = wx.ID_ANY, label = "")
  108. enabled.SetValue(isEnabled)
  109. self.symbology[key] = (enabled, color)
  110. else:
  111. enabled = (1, 1)
  112. self.symbology[key] = (None, color)
  113. flexSizer.Add(textLabel, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  114. flexSizer.Add(enabled, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  115. flexSizer.Add(color, proportion = 0, flag = wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  116. color.SetName("GetColour")
  117. sizer.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 10)
  118. panel.SetSizer(sizer)
  119. return panel
  120. def _createGeneralPage(self, notebook):
  121. """!Create notebook page concerning general settings"""
  122. panel = wx.Panel(parent = notebook, id = wx.ID_ANY)
  123. notebook.AddPage(page = panel, text = _("General"))
  124. border = wx.BoxSizer(wx.VERTICAL)
  125. #
  126. # display section
  127. #
  128. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Display"))
  129. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  130. flexSizer = wx.FlexGridSizer (cols = 3, hgap = 5, vgap = 5)
  131. flexSizer.AddGrowableCol(0)
  132. # line width
  133. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Line width"))
  134. self.lineWidthValue = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (75, -1),
  135. initial = UserSettings.Get(group = 'vdigit', key = "lineWidth", subkey = 'value'),
  136. min = 1, max = 1e6)
  137. units = wx.StaticText(parent = panel, id = wx.ID_ANY, size = (115, -1),
  138. label = UserSettings.Get(group = 'vdigit', key = "lineWidth", subkey = 'units'),
  139. style = wx.ALIGN_LEFT)
  140. flexSizer.Add(text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  141. flexSizer.Add(self.lineWidthValue, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  142. flexSizer.Add(units, proportion = 0, flag = wx.ALIGN_RIGHT | wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL | wx.LEFT,
  143. border = 10)
  144. sizer.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 1)
  145. border.Add(item = sizer, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 5)
  146. #
  147. # snapping section
  148. #
  149. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Snapping"))
  150. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  151. flexSizer = wx.FlexGridSizer(cols = 3, hgap = 5, vgap = 5)
  152. flexSizer.AddGrowableCol(0)
  153. # snapping
  154. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Snapping threshold"))
  155. self.snappingValue = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (75, -1),
  156. initial = UserSettings.Get(group = 'vdigit', key = "snapping", subkey = 'value'),
  157. min = -1, max = 1e6)
  158. self.snappingValue.Bind(wx.EVT_SPINCTRL, self.OnChangeSnappingValue)
  159. self.snappingValue.Bind(wx.EVT_TEXT, self.OnChangeSnappingValue)
  160. self.snappingUnit = wx.Choice(parent = panel, id = wx.ID_ANY, size = (125, -1),
  161. choices = ["screen pixels", "map units"])
  162. self.snappingUnit.SetStringSelection(UserSettings.Get(group = 'vdigit', key = "snapping", subkey = 'units'))
  163. self.snappingUnit.Bind(wx.EVT_CHOICE, self.OnChangeSnappingUnits)
  164. flexSizer.Add(text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  165. flexSizer.Add(self.snappingValue, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  166. flexSizer.Add(self.snappingUnit, proportion = 0, flag = wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  167. vertexSizer = wx.BoxSizer(wx.VERTICAL)
  168. self.snapVertex = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  169. label = _("Snap also to vertex"))
  170. self.snapVertex.SetValue(UserSettings.Get(group = 'vdigit', key = "snapToVertex", subkey = 'enabled'))
  171. vertexSizer.Add(item = self.snapVertex, proportion = 0, flag = wx.EXPAND)
  172. self.mapUnits = self.parent.MapWindow.Map.GetProjInfo()['units']
  173. self.snappingInfo = wx.StaticText(parent = panel, id = wx.ID_ANY,
  174. label = _("Snapping threshold is %(value).1f %(units)s") % \
  175. {'value' : self.digit.GetDisplay().GetThreshold(),
  176. 'units' : self.mapUnits})
  177. vertexSizer.Add(item = self.snappingInfo, proportion = 0,
  178. flag = wx.ALL | wx.EXPAND, border = 1)
  179. sizer.Add(item = flexSizer, proportion = 1, flag = wx.EXPAND)
  180. sizer.Add(item = vertexSizer, proportion = 1, flag = wx.EXPAND)
  181. border.Add(item = sizer, proportion = 0, flag = wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border = 5)
  182. #
  183. # select box
  184. #
  185. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Select vector features"))
  186. # feature type
  187. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  188. inSizer = wx.BoxSizer(wx.HORIZONTAL)
  189. self.selectFeature = {}
  190. for feature in ('point', 'line',
  191. 'centroid', 'boundary'):
  192. chkbox = wx.CheckBox(parent = panel, label = feature)
  193. self.selectFeature[feature] = chkbox.GetId()
  194. chkbox.SetValue(UserSettings.Get(group = 'vdigit', key = 'selectType',
  195. subkey = [feature, 'enabled']))
  196. inSizer.Add(item = chkbox, proportion = 0,
  197. flag = wx.EXPAND | wx.ALL, border = 5)
  198. sizer.Add(item = inSizer, proportion = 0, flag = wx.EXPAND)
  199. # threshold
  200. flexSizer = wx.FlexGridSizer (cols = 3, hgap = 5, vgap = 5)
  201. flexSizer.AddGrowableCol(0)
  202. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Select threshold"))
  203. self.selectThreshValue = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (75, -1),
  204. initial = UserSettings.Get(group = 'vdigit', key = "selectThresh", subkey = 'value'),
  205. min = 1, max = 1e6)
  206. units = wx.StaticText(parent = panel, id = wx.ID_ANY, size = (115, -1),
  207. label = UserSettings.Get(group = 'vdigit', key = "lineWidth", subkey = 'units'),
  208. style = wx.ALIGN_LEFT)
  209. flexSizer.Add(text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  210. flexSizer.Add(self.selectThreshValue, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  211. flexSizer.Add(units, proportion = 0, flag = wx.ALIGN_RIGHT | wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL | wx.LEFT,
  212. border = 10)
  213. self.selectIn = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  214. label = _("Select only features inside of selection bounding box"))
  215. self.selectIn.SetValue(UserSettings.Get(group = 'vdigit', key = "selectInside", subkey = 'enabled'))
  216. self.selectIn.SetToolTipString(_("By default are selected all features overlapping selection bounding box "))
  217. self.checkForDupl = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  218. label = _("Check for duplicates"))
  219. self.checkForDupl.SetValue(UserSettings.Get(group = 'vdigit', key = "checkForDupl", subkey = 'enabled'))
  220. sizer.Add(item = flexSizer, proportion = 0, flag = wx.EXPAND)
  221. sizer.Add(item = self.selectIn, proportion = 0, flag = wx.EXPAND | wx.ALL, border = 1)
  222. sizer.Add(item = self.checkForDupl, proportion = 0, flag = wx.EXPAND | wx.ALL, border = 1)
  223. border.Add(item = sizer, proportion = 0, flag = wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border = 5)
  224. #
  225. # digitize lines box
  226. #
  227. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Digitize line features"))
  228. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  229. self.intersect = wx.CheckBox(parent = panel, label = _("Break lines at intersection"))
  230. self.intersect.SetValue(UserSettings.Get(group = 'vdigit', key = 'breakLines', subkey = 'enabled'))
  231. sizer.Add(item = self.intersect, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  232. border.Add(item = sizer, proportion = 0, flag = wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border = 5)
  233. #
  234. # save-on-exit box
  235. #
  236. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Save changes"))
  237. # save changes on exit?
  238. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  239. self.save = wx.CheckBox(parent = panel, label = _("Save changes on exit"))
  240. self.save.SetValue(UserSettings.Get(group = 'vdigit', key = 'saveOnExit', subkey = 'enabled'))
  241. sizer.Add(item = self.save, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  242. border.Add(item = sizer, proportion = 0, flag = wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border = 5)
  243. panel.SetSizer(border)
  244. return panel
  245. def _createQueryPage(self, notebook):
  246. """!Create notebook page for query tool"""
  247. panel = wx.Panel(parent = notebook, id = wx.ID_ANY)
  248. notebook.AddPage(page = panel, text = _("Query tool"))
  249. border = wx.BoxSizer(wx.VERTICAL)
  250. #
  251. # query tool box
  252. #
  253. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Choose query tool"))
  254. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  255. LocUnits = self.parent.MapWindow.Map.GetProjInfo()['units']
  256. self.queryBox = wx.CheckBox(parent = panel, id = wx.ID_ANY, label = _("Select by box"))
  257. self.queryBox.SetValue(UserSettings.Get(group = 'vdigit', key = "query", subkey = 'box'))
  258. sizer.Add(item = self.queryBox, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  259. sizer.Add((0, 5))
  260. #
  261. # length
  262. #
  263. self.queryLength = wx.RadioButton(parent = panel, id = wx.ID_ANY, label = _("length"))
  264. self.queryLength.Bind(wx.EVT_RADIOBUTTON, self.OnChangeQuery)
  265. sizer.Add(item = self.queryLength, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  266. flexSizer = wx.FlexGridSizer (cols = 4, hgap = 5, vgap = 5)
  267. flexSizer.AddGrowableCol(0)
  268. txt = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Select lines"))
  269. self.queryLengthSL = wx.Choice (parent = panel, id = wx.ID_ANY,
  270. choices = [_("shorter than"), _("longer than")])
  271. self.queryLengthSL.SetSelection(UserSettings.Get(group = 'vdigit', key = "queryLength", subkey = 'than-selection'))
  272. self.queryLengthValue = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (100, -1),
  273. initial = 1,
  274. min = 0, max = 1e6)
  275. self.queryLengthValue.SetValue(UserSettings.Get(group = 'vdigit', key = "queryLength", subkey = 'thresh'))
  276. units = wx.StaticText(parent = panel, id = wx.ID_ANY, label = "%s" % LocUnits)
  277. flexSizer.Add(txt, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  278. flexSizer.Add(self.queryLengthSL, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  279. flexSizer.Add(self.queryLengthValue, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  280. flexSizer.Add(units, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  281. sizer.Add(item = flexSizer, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  282. #
  283. # dangle
  284. #
  285. self.queryDangle = wx.RadioButton(parent = panel, id = wx.ID_ANY, label = _("dangle"))
  286. self.queryDangle.Bind(wx.EVT_RADIOBUTTON, self.OnChangeQuery)
  287. sizer.Add(item = self.queryDangle, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  288. flexSizer = wx.FlexGridSizer (cols = 4, hgap = 5, vgap = 5)
  289. flexSizer.AddGrowableCol(0)
  290. txt = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Select dangles"))
  291. self.queryDangleSL = wx.Choice (parent = panel, id = wx.ID_ANY,
  292. choices = [_("shorter than"), _("longer than")])
  293. self.queryDangleSL.SetSelection(UserSettings.Get(group = 'vdigit', key = "queryDangle", subkey = 'than-selection'))
  294. self.queryDangleValue = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (100, -1),
  295. initial = 1,
  296. min = 0, max = 1e6)
  297. self.queryDangleValue.SetValue(UserSettings.Get(group = 'vdigit', key = "queryDangle", subkey = 'thresh'))
  298. units = wx.StaticText(parent = panel, id = wx.ID_ANY, label = "%s" % LocUnits)
  299. flexSizer.Add(txt, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  300. flexSizer.Add(self.queryDangleSL, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  301. flexSizer.Add(self.queryDangleValue, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  302. flexSizer.Add(units, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  303. sizer.Add(item = flexSizer, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  304. if UserSettings.Get(group = 'vdigit', key = "query", subkey = 'selection') == 0:
  305. self.queryLength.SetValue(True)
  306. else:
  307. self.queryDangle.SetValue(True)
  308. # enable & disable items
  309. self.OnChangeQuery(None)
  310. border.Add(item = sizer, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 5)
  311. panel.SetSizer(border)
  312. return panel
  313. def _createAttributesPage(self, notebook):
  314. """!Create notebook page for attributes"""
  315. panel = wx.Panel(parent = notebook, id = wx.ID_ANY)
  316. notebook.AddPage(page = panel, text = _("Attributes"))
  317. border = wx.BoxSizer(wx.VERTICAL)
  318. #
  319. # add new record
  320. #
  321. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Digitize new feature"))
  322. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  323. # checkbox
  324. self.addRecord = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  325. label = _("Add new record into table"))
  326. self.addRecord.SetValue(UserSettings.Get(group = 'vdigit', key = "addRecord", subkey = 'enabled'))
  327. sizer.Add(item = self.addRecord, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  328. # settings
  329. flexSizer = wx.FlexGridSizer(cols = 2, hgap = 3, vgap = 3)
  330. flexSizer.AddGrowableCol(0)
  331. settings = ((_("Layer"), 1), (_("Category"), 1), (_("Mode"), _("Next to use")))
  332. # layer
  333. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Layer"))
  334. self.layer = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (125, -1),
  335. min = 1, max = 1e3)
  336. self.layer.SetValue(int(UserSettings.Get(group = 'vdigit', key = "layer", subkey = 'value')))
  337. flexSizer.Add(item = text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  338. flexSizer.Add(item = self.layer, proportion = 0,
  339. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  340. # category number
  341. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Category number"))
  342. self.category = wx.SpinCtrl(parent = panel, id = wx.ID_ANY, size = (125, -1),
  343. initial = UserSettings.Get(group = 'vdigit', key = "category", subkey = 'value'),
  344. min = -1e9, max = 1e9)
  345. if UserSettings.Get(group = 'vdigit', key = "categoryMode", subkey = 'selection') != 1:
  346. self.category.Enable(False)
  347. flexSizer.Add(item = text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  348. flexSizer.Add(item = self.category, proportion = 0,
  349. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  350. # category mode
  351. text = wx.StaticText(parent = panel, id = wx.ID_ANY, label = _("Category mode"))
  352. self.categoryMode = wx.Choice(parent = panel, id = wx.ID_ANY, size = (125, -1),
  353. choices = [_("Next to use"), _("Manual entry"), _("No category")])
  354. self.categoryMode.SetSelection(UserSettings.Get(group = 'vdigit', key = "categoryMode", subkey = 'selection'))
  355. flexSizer.Add(item = text, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  356. flexSizer.Add(item = self.categoryMode, proportion = 0,
  357. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  358. sizer.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 1)
  359. border.Add(item = sizer, proportion = 0,
  360. flag = wx.ALL | wx.EXPAND, border = 5)
  361. #
  362. # delete existing record
  363. #
  364. box = wx.StaticBox (parent = panel, id = wx.ID_ANY, label = " %s " % _("Delete existing feature(s)"))
  365. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  366. # checkbox
  367. self.deleteRecord = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  368. label = _("Delete record from table"))
  369. self.deleteRecord.SetValue(UserSettings.Get(group = 'vdigit', key = "delRecord", subkey = 'enabled'))
  370. sizer.Add(item = self.deleteRecord, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 1)
  371. border.Add(item = sizer, proportion = 0,
  372. flag = wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border = 5)
  373. #
  374. # geometry attributes (currently only length and area are supported)
  375. #
  376. box = wx.StaticBox (parent = panel, id = wx.ID_ANY,
  377. label = " %s " % _("Geometry attributes"))
  378. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  379. gridSizer = wx.GridBagSizer(hgap = 3, vgap = 3)
  380. gridSizer.AddGrowableCol(0)
  381. self.geomAttrb = { 'length' : { 'label' : _('length') },
  382. 'area' : { 'label' : _('area') },
  383. 'perimeter' : { 'label' : _('perimeter') } }
  384. digitToolbar = self.parent.toolbars['vdigit']
  385. try:
  386. vectorName = digitToolbar.GetLayer().GetName()
  387. except AttributeError:
  388. vectorName = None # no vector selected for editing
  389. layer = UserSettings.Get(group = 'vdigit', key = "layer", subkey = 'value')
  390. mapLayer = self.parent.toolbars['vdigit'].GetLayer()
  391. tree = self.parent.tree
  392. item = tree.FindItemByData('maplayer', mapLayer)
  393. row = 0
  394. for attrb in ['length', 'area', 'perimeter']:
  395. # checkbox
  396. check = wx.CheckBox(parent = panel, id = wx.ID_ANY,
  397. label = self.geomAttrb[attrb]['label'])
  398. ### self.deleteRecord.SetValue(UserSettings.Get(group='vdigit', key="delRecord", subkey='enabled'))
  399. check.Bind(wx.EVT_CHECKBOX, self.OnGeomAttrb)
  400. # column (only numeric)
  401. column = gselect.ColumnSelect(parent = panel, size = (200, -1))
  402. column.InsertColumns(vector = vectorName,
  403. layer = layer, excludeKey = True,
  404. type = ['integer', 'double precision'])
  405. # units
  406. if attrb == 'area':
  407. choices = Units.GetUnitsList('area')
  408. else:
  409. choices = Units.GetUnitsList('length')
  410. win_units = wx.Choice(parent = panel, id = wx.ID_ANY,
  411. choices = choices, size = (120, -1))
  412. # default values
  413. check.SetValue(False)
  414. if item and tree.GetPyData(item)[0]['vdigit'] and \
  415. 'geomAttr' in tree.GetPyData(item)[0]['vdigit'] and \
  416. attrb in tree.GetPyData(item)[0]['vdigit']['geomAttr']:
  417. check.SetValue(True)
  418. column.SetStringSelection(tree.GetPyData(item)[0]['vdigit']['geomAttr'][attrb]['column'])
  419. if attrb == 'area':
  420. type = 'area'
  421. else:
  422. type = 'length'
  423. unitsIdx = Units.GetUnitsIndex(type,
  424. tree.GetPyData(item)[0]['vdigit']['geomAttr'][attrb]['units'])
  425. win_units.SetSelection(unitsIdx)
  426. if not vectorName:
  427. check.Enable(False)
  428. column.Enable(False)
  429. if not check.IsChecked():
  430. column.Enable(False)
  431. self.geomAttrb[attrb]['check'] = check.GetId()
  432. self.geomAttrb[attrb]['column'] = column.GetId()
  433. self.geomAttrb[attrb]['units'] = win_units.GetId()
  434. gridSizer.Add(item = check,
  435. flag = wx.ALIGN_CENTER_VERTICAL,
  436. pos = (row, 0))
  437. gridSizer.Add(item = column,
  438. pos = (row, 1))
  439. gridSizer.Add(item = win_units,
  440. pos = (row, 2))
  441. row += 1
  442. note = '\n'.join(textwrap.wrap(_("Note: These settings are stored "
  443. "in the workspace not in the vector digitizer "
  444. "preferences."), 55))
  445. gridSizer.Add(item = wx.StaticText(parent = panel, id = wx.ID_ANY,
  446. label = note),
  447. pos = (3, 0), span = (1, 3))
  448. sizer.Add(item = gridSizer, proportion = 1,
  449. flag = wx.ALL | wx.EXPAND, border = 1)
  450. border.Add(item = sizer, proportion = 0,
  451. flag = wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border = 5)
  452. # bindings
  453. self.Bind(wx.EVT_CHECKBOX, self.OnChangeAddRecord, self.addRecord)
  454. self.Bind(wx.EVT_CHOICE, self.OnChangeCategoryMode, self.categoryMode)
  455. self.Bind(wx.EVT_SPINCTRL, self.OnChangeLayer, self.layer)
  456. panel.SetSizer(border)
  457. return panel
  458. def _symbologyData(self):
  459. """!Data for _createSymbologyPage()
  460. label | checkbox | color
  461. """
  462. return (
  463. # ("Background", "symbolBackground"),
  464. (_("Highlight"), "highlight"),
  465. (_("Highlight (duplicates)"), "highlightDupl"),
  466. (_("Point"), "point"),
  467. (_("Line"), "line"),
  468. (_("Boundary (no area)"), "boundaryNo"),
  469. (_("Boundary (one area)"), "boundaryOne"),
  470. (_("Boundary (two areas)"), "boundaryTwo"),
  471. (_("Centroid (in area)"), "centroidIn"),
  472. (_("Centroid (outside area)"), "centroidOut"),
  473. (_("Centroid (duplicate in area)"), "centroidDup"),
  474. (_("Node (one line)"), "nodeOne"),
  475. (_("Node (two lines)"), "nodeTwo"),
  476. (_("Vertex"), "vertex"),
  477. (_("Area (closed boundary + centroid)"), "area"),
  478. (_("Direction"), "direction"),)
  479. def OnGeomAttrb(self, event):
  480. """!Register geometry attributes (enable/disable)
  481. """
  482. checked = event.IsChecked()
  483. id = event.GetId()
  484. key = None
  485. for attrb, val in self.geomAttrb.iteritems():
  486. if val['check'] == id:
  487. key = attrb
  488. break
  489. column = self.FindWindowById(self.geomAttrb[key]['column'])
  490. if checked:
  491. column.Enable()
  492. else:
  493. column.Enable(False)
  494. def OnChangeCategoryMode(self, event):
  495. """!Change category mode
  496. """
  497. mode = event.GetSelection()
  498. UserSettings.Set(group = 'vdigit', key = "categoryMode", subkey = 'selection', value = mode)
  499. if mode == 1: # manual entry
  500. self.category.Enable(True)
  501. elif self.category.IsEnabled(): # disable
  502. self.category.Enable(False)
  503. if mode == 2 and self.addRecord.IsChecked(): # no category
  504. self.addRecord.SetValue(False)
  505. self.digit.SetCategory()
  506. self.category.SetValue(UserSettings.Get(group = 'vdigit', key = 'category', subkey = 'value'))
  507. def OnChangeLayer(self, event):
  508. """!Layer changed
  509. """
  510. layer = event.GetInt()
  511. if layer > 0:
  512. UserSettings.Set(group = 'vdigit', key = 'layer', subkey = 'value', value = layer)
  513. self.digit.SetCategory()
  514. self.category.SetValue(UserSettings.Get(group = 'vdigit', key = 'category', subkey = 'value'))
  515. event.Skip()
  516. def OnChangeAddRecord(self, event):
  517. """!Checkbox 'Add new record' status changed
  518. """
  519. self.category.SetValue(self.digit.SetCategory())
  520. def OnChangeSnappingValue(self, event):
  521. """!Change snapping value - update static text
  522. """
  523. value = self.snappingValue.GetValue()
  524. if value < 0:
  525. region = self.parent.MapWindow.Map.GetRegion()
  526. res = (region['nsres'] + region['ewres']) / 2.
  527. threshold = self.digit.GetDisplay().GetThreshold(value = res)
  528. else:
  529. if self.snappingUnit.GetStringSelection() == "map units":
  530. threshold = value
  531. else:
  532. threshold = self.digit.GetDisplay().GetThreshold(value = value)
  533. if value == 0:
  534. self.snappingInfo.SetLabel(_("Snapping disabled"))
  535. elif value < 0:
  536. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s "
  537. "(based on comp. resolution)") %
  538. {'value' : threshold,
  539. 'units' : self.mapUnits.lower()})
  540. else:
  541. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  542. {'value' : threshold,
  543. 'units' : self.mapUnits.lower()})
  544. event.Skip()
  545. def OnChangeSnappingUnits(self, event):
  546. """!Snapping units change -> update static text
  547. """
  548. value = self.snappingValue.GetValue()
  549. units = self.snappingUnit.GetStringSelection()
  550. threshold = self.digit.GetDisplay().GetThreshold(value = value, units = units)
  551. if units == "map units":
  552. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  553. {'value' : value,
  554. 'units' : self.mapUnits})
  555. else:
  556. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  557. {'value' : threshold,
  558. 'units' : self.mapUnits})
  559. event.Skip()
  560. def OnChangeQuery(self, event):
  561. """!Change query
  562. """
  563. if self.queryLength.GetValue():
  564. # length
  565. self.queryLengthSL.Enable(True)
  566. self.queryLengthValue.Enable(True)
  567. self.queryDangleSL.Enable(False)
  568. self.queryDangleValue.Enable(False)
  569. else:
  570. # dangle
  571. self.queryLengthSL.Enable(False)
  572. self.queryLengthValue.Enable(False)
  573. self.queryDangleSL.Enable(True)
  574. self.queryDangleValue.Enable(True)
  575. def OnSave(self, event):
  576. """!Button 'Save' pressed
  577. """
  578. self.UpdateSettings()
  579. self.parent.toolbars['vdigit'].settingsDialog = None
  580. fileSettings = {}
  581. UserSettings.ReadSettingsFile(settings = fileSettings)
  582. fileSettings['vdigit'] = UserSettings.Get(group = 'vdigit')
  583. file = UserSettings.SaveToFile(fileSettings)
  584. self.parent.GetLayerManager().goutput.WriteLog(_('Vector digitizer settings saved to file <%s>.') % file)
  585. self.Destroy()
  586. event.Skip()
  587. def OnApply(self, event):
  588. """!Button 'Apply' pressed
  589. """
  590. self.UpdateSettings()
  591. def OnCancel(self, event):
  592. """!Button 'Cancel' pressed
  593. """
  594. self.parent.toolbars['vdigit'].settingsDialog = None
  595. self.Destroy()
  596. if event:
  597. event.Skip()
  598. def UpdateSettings(self):
  599. """!Update digitizer settings
  600. """
  601. self.parent.GetLayerManager().WorkspaceChanged() # geometry attributes
  602. # symbology
  603. for key, (enabled, color) in self.symbology.iteritems():
  604. if enabled:
  605. UserSettings.Set(group = 'vdigit', key = 'symbol',
  606. subkey = [key, 'enabled'],
  607. value = enabled.IsChecked())
  608. UserSettings.Set(group = 'vdigit', key = 'symbol',
  609. subkey = [key, 'color'],
  610. value = tuple(color.GetColour()))
  611. else:
  612. UserSettings.Set(group = 'vdigit', key = 'symbol',
  613. subkey = [key, 'color'],
  614. value = tuple(color.GetColour()))
  615. # display
  616. UserSettings.Set(group = 'vdigit', key = "lineWidth", subkey = 'value',
  617. value = int(self.lineWidthValue.GetValue()))
  618. # snapping
  619. UserSettings.Set(group = 'vdigit', key = "snapping", subkey = 'value',
  620. value = int(self.snappingValue.GetValue()))
  621. UserSettings.Set(group = 'vdigit', key = "snapping", subkey = 'units',
  622. value = self.snappingUnit.GetStringSelection())
  623. UserSettings.Set(group = 'vdigit', key = "snapToVertex", subkey = 'enabled',
  624. value = self.snapVertex.IsChecked())
  625. # digitize new feature
  626. UserSettings.Set(group = 'vdigit', key = "addRecord", subkey = 'enabled',
  627. value = self.addRecord.IsChecked())
  628. UserSettings.Set(group = 'vdigit', key = "layer", subkey = 'value',
  629. value = int(self.layer.GetValue()))
  630. UserSettings.Set(group = 'vdigit', key = "category", subkey = 'value',
  631. value = int(self.category.GetValue()))
  632. UserSettings.Set(group = 'vdigit', key = "categoryMode", subkey = 'selection',
  633. value = self.categoryMode.GetSelection())
  634. # delete existing feature
  635. UserSettings.Set(group = 'vdigit', key = "delRecord", subkey = 'enabled',
  636. value = self.deleteRecord.IsChecked())
  637. # geometry attributes (workspace)
  638. mapLayer = self.parent.toolbars['vdigit'].GetLayer()
  639. tree = self.parent.tree
  640. item = tree.FindItemByData('maplayer', mapLayer)
  641. for key, val in self.geomAttrb.iteritems():
  642. checked = self.FindWindowById(val['check']).IsChecked()
  643. column = self.FindWindowById(val['column']).GetValue()
  644. unitsIdx = self.FindWindowById(val['units']).GetSelection()
  645. if item and not tree.GetPyData(item)[0]['vdigit']:
  646. tree.GetPyData(item)[0]['vdigit'] = { 'geomAttr' : dict() }
  647. if checked: # enable
  648. if key == 'area':
  649. type = key
  650. else:
  651. type = 'length'
  652. unitsKey = Units.GetUnitsKey(type, unitsIdx)
  653. tree.GetPyData(item)[0]['vdigit']['geomAttr'][key] = { 'column' : column,
  654. 'units' : unitsKey }
  655. else:
  656. if item and tree.GetPyData(item)[0]['vdigit'] and \
  657. key in tree.GetPyData(item)[0]['vdigit']['geomAttr']:
  658. del tree.GetPyData(item)[0]['vdigit']['geomAttr'][key]
  659. # query tool
  660. if self.queryLength.GetValue():
  661. UserSettings.Set(group = 'vdigit', key = "query", subkey = 'selection',
  662. value = 0)
  663. else:
  664. UserSettings.Set(group = 'vdigit', key = "query", subkey = 'type',
  665. value = 1)
  666. UserSettings.Set(group = 'vdigit', key = "query", subkey = 'box',
  667. value = self.queryBox.IsChecked())
  668. UserSettings.Set(group = 'vdigit', key = "queryLength", subkey = 'than-selection',
  669. value = self.queryLengthSL.GetSelection())
  670. UserSettings.Set(group = 'vdigit', key = "queryLength", subkey = 'thresh',
  671. value = int(self.queryLengthValue.GetValue()))
  672. UserSettings.Set(group = 'vdigit', key = "queryDangle", subkey = 'than-selection',
  673. value = self.queryDangleSL.GetSelection())
  674. UserSettings.Set(group = 'vdigit', key = "queryDangle", subkey = 'thresh',
  675. value = int(self.queryDangleValue.GetValue()))
  676. # select features
  677. for feature in ('point', 'line',
  678. 'centroid', 'boundary'):
  679. UserSettings.Set(group = 'vdigit', key = 'selectType',
  680. subkey = [feature, 'enabled'],
  681. value = self.FindWindowById(self.selectFeature[feature]).IsChecked())
  682. UserSettings.Set(group = 'vdigit', key = "selectThresh", subkey = 'value',
  683. value = int(self.selectThreshValue.GetValue()))
  684. UserSettings.Set(group = 'vdigit', key = "checkForDupl", subkey = 'enabled',
  685. value = self.checkForDupl.IsChecked())
  686. UserSettings.Set(group = 'vdigit', key = "selectInside", subkey = 'enabled',
  687. value = self.selectIn.IsChecked())
  688. # on-exit
  689. UserSettings.Set(group = 'vdigit', key = "saveOnExit", subkey = 'enabled',
  690. value = self.save.IsChecked())
  691. # break lines
  692. UserSettings.Set(group = 'vdigit', key = "breakLines", subkey = 'enabled',
  693. value = self.intersect.IsChecked())
  694. self.digit.UpdateSettings()
  695. # redraw map if auto-rendering is enabled
  696. if self.parent.statusbarWin['render'].GetValue():
  697. self.parent.OnRender(None)
  698. class VDigitCategoryDialog(wx.Dialog, listmix.ColumnSorterMixin):
  699. def __init__(self, parent, title,
  700. map, query = None, cats = None,
  701. pos = wx.DefaultPosition,
  702. style = wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER):
  703. """!Dialog used to display/modify categories of vector objects
  704. @param parent
  705. @param title dialog title
  706. @param query {coordinates, qdist} - used by v.edit/v.what
  707. @param cats directory of lines (layer/categories) - used by vdigit
  708. @param pos
  709. @param style
  710. """
  711. # parent
  712. self.parent = parent # mapdisplay.BufferedWindow class instance
  713. # map name
  714. self.map = map
  715. # line : {layer: [categories]}
  716. self.cats = {}
  717. # do not display dialog if no line is found (-> self.cats)
  718. if cats is None:
  719. if self._getCategories(query[0], query[1]) == 0 or not self.line:
  720. Debug.msg(3, "VDigitCategoryDialog(): nothing found!")
  721. else:
  722. self.cats = cats
  723. for line in cats.keys():
  724. for layer in cats[line].keys():
  725. self.cats[line][layer] = list(cats[line][layer])
  726. layers = []
  727. for layer in self.parent.parent.digit.GetLayers():
  728. layers.append(str(layer))
  729. # make copy of cats (used for 'reload')
  730. self.cats_orig = copy.deepcopy(self.cats)
  731. wx.Dialog.__init__(self, parent = self.parent, id = wx.ID_ANY, title = title,
  732. style = style, pos = pos)
  733. # list of categories
  734. box = wx.StaticBox(parent = self, id = wx.ID_ANY,
  735. label = " %s " % _("List of categories - right-click to delete"))
  736. listSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  737. self.list = CategoryListCtrl(parent = self, id = wx.ID_ANY,
  738. style = wx.LC_REPORT |
  739. wx.BORDER_NONE |
  740. wx.LC_SORT_ASCENDING |
  741. wx.LC_HRULES |
  742. wx.LC_VRULES)
  743. # sorter
  744. self.fid = self.cats.keys()[0]
  745. self.itemDataMap = self.list.Populate(self.cats[self.fid])
  746. listmix.ColumnSorterMixin.__init__(self, 2)
  747. self.fidMulti = wx.Choice(parent = self, id = wx.ID_ANY,
  748. size = (150, -1))
  749. self.fidMulti.Bind(wx.EVT_CHOICE, self.OnFeature)
  750. self.fidText = wx.StaticText(parent = self, id = wx.ID_ANY)
  751. if len(self.cats.keys()) == 1:
  752. self.fidMulti.Show(False)
  753. self.fidText.SetLabel(str(self.fid))
  754. else:
  755. self.fidText.Show(False)
  756. choices = []
  757. for fid in self.cats.keys():
  758. choices.append(str(fid))
  759. self.fidMulti.SetItems(choices)
  760. self.fidMulti.SetSelection(0)
  761. listSizer.Add(item = self.list, proportion = 1, flag = wx.EXPAND)
  762. # add new category
  763. box = wx.StaticBox(parent = self, id = wx.ID_ANY,
  764. label = " %s " % _("Add new category"))
  765. addSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  766. flexSizer = wx.FlexGridSizer (cols = 5, hgap = 5, vgap = 5)
  767. flexSizer.AddGrowableCol(3)
  768. layerNewTxt = wx.StaticText(parent = self, id = wx.ID_ANY,
  769. label = "%s:" % _("Layer"))
  770. self.layerNew = wx.Choice(parent = self, id = wx.ID_ANY, size = (75, -1),
  771. choices = layers)
  772. if len(layers) > 0:
  773. self.layerNew.SetSelection(0)
  774. catNewTxt = wx.StaticText(parent = self, id = wx.ID_ANY,
  775. label = "%s:" % _("Category"))
  776. try:
  777. newCat = max(self.cats[self.fid][1]) + 1
  778. except KeyError:
  779. newCat = 1
  780. self.catNew = wx.SpinCtrl(parent = self, id = wx.ID_ANY, size = (75, -1),
  781. initial = newCat, min = 0, max = 1e9)
  782. btnAddCat = wx.Button(self, wx.ID_ADD)
  783. flexSizer.Add(item = layerNewTxt, proportion = 0,
  784. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  785. flexSizer.Add(item = self.layerNew, proportion = 0,
  786. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  787. flexSizer.Add(item = catNewTxt, proportion = 0,
  788. flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT | wx.LEFT,
  789. border = 10)
  790. flexSizer.Add(item = self.catNew, proportion = 0,
  791. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  792. flexSizer.Add(item = btnAddCat, proportion = 0,
  793. flag = wx.EXPAND | wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  794. addSizer.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 5)
  795. # buttons
  796. btnApply = wx.Button(self, wx.ID_APPLY)
  797. btnApply.SetToolTipString(_("Apply changes"))
  798. btnCancel = wx.Button(self, wx.ID_CANCEL)
  799. btnCancel.SetToolTipString(_("Ignore changes and close dialog"))
  800. btnOk = wx.Button(self, wx.ID_OK)
  801. btnOk.SetToolTipString(_("Apply changes and close dialog"))
  802. btnOk.SetDefault()
  803. # sizers
  804. btnSizer = wx.StdDialogButtonSizer()
  805. btnSizer.AddButton(btnCancel)
  806. #btnSizer.AddButton(btnReload)
  807. #btnSizer.SetNegativeButton(btnReload)
  808. btnSizer.AddButton(btnApply)
  809. btnSizer.AddButton(btnOk)
  810. btnSizer.Realize()
  811. mainSizer = wx.BoxSizer(wx.VERTICAL)
  812. mainSizer.Add(item = listSizer, proportion = 1,
  813. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  814. mainSizer.Add(item = addSizer, proportion = 0,
  815. flag = wx.EXPAND | wx.ALIGN_CENTER |
  816. wx.LEFT | wx.RIGHT | wx.BOTTOM, border = 5)
  817. fidSizer = wx.BoxSizer(wx.HORIZONTAL)
  818. fidSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  819. label = _("Feature id:")),
  820. proportion = 0, border = 5,
  821. flag = wx.ALIGN_CENTER_VERTICAL)
  822. fidSizer.Add(item = self.fidMulti, proportion = 0,
  823. flag = wx.EXPAND | wx.ALL, border = 5)
  824. fidSizer.Add(item = self.fidText, proportion = 0,
  825. flag = wx.EXPAND | wx.ALL, border = 5)
  826. mainSizer.Add(item = fidSizer, proportion = 0,
  827. flag = wx.EXPAND | wx.ALL, border = 5)
  828. mainSizer.Add(item = btnSizer, proportion = 0,
  829. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  830. self.SetSizer(mainSizer)
  831. mainSizer.Fit(self)
  832. self.SetAutoLayout(True)
  833. # set min size for dialog
  834. self.SetMinSize(self.GetBestSize())
  835. # bindings
  836. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  837. btnOk.Bind(wx.EVT_BUTTON, self.OnOK)
  838. btnAddCat.Bind(wx.EVT_BUTTON, self.OnAddCat)
  839. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  840. # list
  841. self.list.Bind(wx.EVT_COMMAND_RIGHT_CLICK, self.OnRightUp) #wxMSW
  842. self.list.Bind(wx.EVT_RIGHT_UP, self.OnRightUp) #wxGTK
  843. self.Bind(wx.EVT_LIST_BEGIN_LABEL_EDIT, self.OnBeginEdit, self.list)
  844. self.Bind(wx.EVT_LIST_END_LABEL_EDIT, self.OnEndEdit, self.list)
  845. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick, self.list)
  846. def GetListCtrl(self):
  847. """!Used by ColumnSorterMixin
  848. """
  849. return self.list
  850. def OnColClick(self, event):
  851. """!Click on column header (order by)
  852. """
  853. event.Skip()
  854. def OnBeginEdit(self, event):
  855. """!Editing of item started
  856. """
  857. event.Allow()
  858. def OnEndEdit(self, event):
  859. """!Finish editing of item
  860. """
  861. itemIndex = event.GetIndex()
  862. layerOld = int (self.list.GetItem(itemIndex, 0).GetText())
  863. catOld = int (self.list.GetItem(itemIndex, 1).GetText())
  864. if event.GetColumn() == 0:
  865. layerNew = int(event.GetLabel())
  866. catNew = catOld
  867. else:
  868. layerNew = layerOld
  869. catNew = int(event.GetLabel())
  870. try:
  871. if layerNew not in self.cats[self.fid].keys():
  872. self.cats[self.fid][layerNew] = []
  873. self.cats[self.fid][layerNew].append(catNew)
  874. self.cats[self.fid][layerOld].remove(catOld)
  875. except:
  876. event.Veto()
  877. self.list.SetStringItem(itemIndex, 0, str(layerNew))
  878. self.list.SetStringItem(itemIndex, 1, str(catNew))
  879. dlg = wx.MessageDialog(self, _("Unable to add new layer/category <%(layer)s/%(category)s>.\n"
  880. "Layer and category number must be integer.\n"
  881. "Layer number must be greater then zero.") %
  882. { 'layer': self.layerNew.GetStringSelection(),
  883. 'category' : str(self.catNew.GetValue()) },
  884. _("Error"), wx.OK | wx.ICON_ERROR)
  885. dlg.ShowModal()
  886. dlg.Destroy()
  887. return False
  888. def OnRightDown(self, event):
  889. """!Mouse right button down
  890. """
  891. x = event.GetX()
  892. y = event.GetY()
  893. item, flags = self.list.HitTest((x, y))
  894. if item != wx.NOT_FOUND and \
  895. flags & wx.LIST_HITTEST_ONITEM:
  896. self.list.Select(item)
  897. event.Skip()
  898. def OnRightUp(self, event):
  899. """!Mouse right button up
  900. """
  901. if not hasattr(self, "popupID1"):
  902. self.popupID1 = wx.NewId()
  903. self.popupID2 = wx.NewId()
  904. self.popupID3 = wx.NewId()
  905. self.Bind(wx.EVT_MENU, self.OnItemDelete, id = self.popupID1)
  906. self.Bind(wx.EVT_MENU, self.OnItemDeleteAll, id = self.popupID2)
  907. self.Bind(wx.EVT_MENU, self.OnReload, id = self.popupID3)
  908. # generate popup-menu
  909. menu = wx.Menu()
  910. menu.Append(self.popupID1, _("Delete selected"))
  911. if self.list.GetFirstSelected() == -1:
  912. menu.Enable(self.popupID1, False)
  913. menu.Append(self.popupID2, _("Delete all"))
  914. menu.AppendSeparator()
  915. menu.Append(self.popupID3, _("Reload"))
  916. self.PopupMenu(menu)
  917. menu.Destroy()
  918. def OnItemSelected(self, event):
  919. """!Item selected
  920. """
  921. event.Skip()
  922. def OnItemDelete(self, event):
  923. """!Delete selected item(s) from the list (layer/category pair)
  924. """
  925. item = self.list.GetFirstSelected()
  926. while item != -1:
  927. layer = int (self.list.GetItem(item, 0).GetText())
  928. cat = int (self.list.GetItem(item, 1).GetText())
  929. self.list.DeleteItem(item)
  930. self.cats[self.fid][layer].remove(cat)
  931. item = self.list.GetFirstSelected()
  932. event.Skip()
  933. def OnItemDeleteAll(self, event):
  934. """!Delete all items from the list
  935. """
  936. self.list.DeleteAllItems()
  937. self.cats[self.fid] = {}
  938. event.Skip()
  939. def OnFeature(self, event):
  940. """!Feature id changed (on duplicates)
  941. """
  942. self.fid = int(event.GetString())
  943. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  944. update = True)
  945. try:
  946. newCat = max(self.cats[self.fid][1]) + 1
  947. except KeyError:
  948. newCat = 1
  949. self.catNew.SetValue(newCat)
  950. event.Skip()
  951. def _getCategories(self, coords, qdist):
  952. """!Get layer/category pairs for all available
  953. layers
  954. Return True line found or False if not found
  955. """
  956. ret = gcmd.RunCommand('v.what',
  957. parent = self,
  958. quiet = True,
  959. map = self.map,
  960. east_north = '%f,%f' % \
  961. (float(coords[0]), float(coords[1])),
  962. distance = qdist)
  963. if not ret:
  964. return False
  965. for item in ret.splitlines():
  966. litem = item.lower()
  967. if "id:" in litem: # get line id
  968. self.line = int(item.split(':')[1].strip())
  969. elif "layer:" in litem: # add layer
  970. layer = int(item.split(':')[1].strip())
  971. if layer not in self.cats.keys():
  972. self.cats[layer] = []
  973. elif "category:" in litem: # add category
  974. self.cats[layer].append(int(item.split(':')[1].strip()))
  975. return True
  976. def OnReload(self, event):
  977. """!Reload button pressed
  978. """
  979. # restore original list
  980. self.cats = copy.deepcopy(self.cats_orig)
  981. # polulate list
  982. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  983. update = True)
  984. event.Skip()
  985. def OnCancel(self, event):
  986. """!Cancel button pressed
  987. """
  988. self.parent.parent.dialogs['category'] = None
  989. if self.parent.parent.digit:
  990. self.parent.parent.digit.GetDisplay().SetSelected([])
  991. self.parent.UpdateMap(render = False)
  992. else:
  993. self.parent.parent.OnRender(None)
  994. self.Close()
  995. def OnApply(self, event):
  996. """!Apply button pressed
  997. """
  998. for fid in self.cats.keys():
  999. newfid = self.ApplyChanges(fid)
  1000. if fid == self.fid:
  1001. self.fid = newfid
  1002. def ApplyChanges(self, fid):
  1003. """!Apply changes
  1004. @param fid feature id
  1005. """
  1006. cats = self.cats[fid]
  1007. cats_orig = self.cats_orig[fid]
  1008. # action : (catsFrom, catsTo)
  1009. check = {'catadd': (cats, cats_orig),
  1010. 'catdel': (cats_orig, cats)}
  1011. newfid = -1
  1012. # add/delete new category
  1013. for action, catsCurr in check.iteritems():
  1014. for layer in catsCurr[0].keys():
  1015. catList = []
  1016. for cat in catsCurr[0][layer]:
  1017. if layer not in catsCurr[1].keys() or \
  1018. cat not in catsCurr[1][layer]:
  1019. catList.append(cat)
  1020. if catList != []:
  1021. if action == 'catadd':
  1022. add = True
  1023. else:
  1024. add = False
  1025. newfid = self.parent.parent.digit.SetLineCats(fid, layer,
  1026. catList, add)
  1027. if len(self.cats.keys()) == 1:
  1028. self.fidText.SetLabel("%d" % newfid)
  1029. else:
  1030. choices = self.fidMulti.GetItems()
  1031. choices[choices.index(str(fid))] = str(newfid)
  1032. self.fidMulti.SetItems(choices)
  1033. self.fidMulti.SetStringSelection(str(newfid))
  1034. self.cats[newfid] = self.cats[fid]
  1035. del self.cats[fid]
  1036. fid = newfid
  1037. if self.fid < 0:
  1038. wx.MessageBox(parent = self, message = _("Unable to update vector map."),
  1039. caption = _("Error"), style = wx.OK | wx.ICON_ERROR)
  1040. self.cats_orig[fid] = copy.deepcopy(cats)
  1041. return newfid
  1042. def OnOK(self, event):
  1043. """!OK button pressed
  1044. """
  1045. self.OnApply(event)
  1046. self.OnCancel(event)
  1047. def OnAddCat(self, event):
  1048. """!Button 'Add' new category pressed
  1049. """
  1050. try:
  1051. layer = int(self.layerNew.GetStringSelection())
  1052. cat = int(self.catNew.GetValue())
  1053. if layer <= 0:
  1054. raise ValueError
  1055. except ValueError:
  1056. dlg = wx.MessageDialog(self, _("Unable to add new layer/category <%(layer)s/%(category)s>.\n"
  1057. "Layer and category number must be integer.\n"
  1058. "Layer number must be greater then zero.") %
  1059. {'layer' : str(self.layerNew.GetValue()),
  1060. 'category' : str(self.catNew.GetValue())},
  1061. _("Error"), wx.OK | wx.ICON_ERROR)
  1062. dlg.ShowModal()
  1063. dlg.Destroy()
  1064. return False
  1065. if layer not in self.cats[self.fid].keys():
  1066. self.cats[self.fid][layer] = []
  1067. self.cats[self.fid][layer].append(cat)
  1068. # reload list
  1069. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1070. update = True)
  1071. # update category number for add
  1072. self.catNew.SetValue(cat + 1)
  1073. event.Skip()
  1074. return True
  1075. def GetLine(self):
  1076. """!Get id of selected line of 'None' if no line is selected
  1077. """
  1078. return self.cats.keys()
  1079. def UpdateDialog(self, query = None, cats = None):
  1080. """!Update dialog
  1081. @param query {coordinates, distance} - v.what
  1082. @param cats directory layer/cats - vdigit
  1083. Return True if updated otherwise False
  1084. """
  1085. # line: {layer: [categories]}
  1086. self.cats = {}
  1087. # do not display dialog if no line is found (-> self.cats)
  1088. if cats is None:
  1089. ret = self._getCategories(query[0], query[1])
  1090. else:
  1091. self.cats = cats
  1092. for line in cats.keys():
  1093. for layer in cats[line].keys():
  1094. self.cats[line][layer] = list(cats[line][layer])
  1095. ret = 1
  1096. if ret == 0 or len(self.cats.keys()) < 1:
  1097. Debug.msg(3, "VDigitCategoryDialog(): nothing found!")
  1098. return False
  1099. # make copy of cats (used for 'reload')
  1100. self.cats_orig = copy.deepcopy(self.cats)
  1101. # polulate list
  1102. self.fid = self.cats.keys()[0]
  1103. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1104. update = True)
  1105. try:
  1106. newCat = max(self.cats[self.fid][1]) + 1
  1107. except KeyError:
  1108. newCat = 1
  1109. self.catNew.SetValue(newCat)
  1110. if len(self.cats.keys()) == 1:
  1111. self.fidText.Show(True)
  1112. self.fidMulti.Show(False)
  1113. self.fidText.SetLabel("%d" % self.fid)
  1114. else:
  1115. self.fidText.Show(False)
  1116. self.fidMulti.Show(True)
  1117. choices = []
  1118. for fid in self.cats.keys():
  1119. choices.append(str(fid))
  1120. self.fidMulti.SetItems(choices)
  1121. self.fidMulti.SetSelection(0)
  1122. self.Layout()
  1123. return True
  1124. class CategoryListCtrl(wx.ListCtrl,
  1125. listmix.ListCtrlAutoWidthMixin,
  1126. listmix.TextEditMixin):
  1127. def __init__(self, parent, id, pos = wx.DefaultPosition,
  1128. size = wx.DefaultSize, style = 0):
  1129. """!List of layers/categories"""
  1130. self.parent = parent
  1131. wx.ListCtrl.__init__(self, parent, id, pos, size, style)
  1132. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1133. listmix.TextEditMixin.__init__(self)
  1134. def Populate(self, cats, update = False):
  1135. """!Populate the list
  1136. """
  1137. itemData = {} # requested by sorter
  1138. if not update:
  1139. self.InsertColumn(0, _("Layer"))
  1140. self.InsertColumn(1, _("Category"))
  1141. else:
  1142. self.DeleteAllItems()
  1143. i = 1
  1144. for layer in cats.keys():
  1145. catsList = cats[layer]
  1146. for cat in catsList:
  1147. index = self.InsertStringItem(sys.maxint, str(catsList[0]))
  1148. self.SetStringItem(index, 0, str(layer))
  1149. self.SetStringItem(index, 1, str(cat))
  1150. self.SetItemData(index, i)
  1151. itemData[i] = (str(layer), str(cat))
  1152. i = i + 1
  1153. if not update:
  1154. self.SetColumnWidth(0, 100)
  1155. self.SetColumnWidth(1, wx.LIST_AUTOSIZE)
  1156. self.currentItem = 0
  1157. return itemData
  1158. class VDigitZBulkDialog(wx.Dialog):
  1159. def __init__(self, parent, title, nselected, style = wx.DEFAULT_DIALOG_STYLE):
  1160. """!Dialog used for Z bulk-labeling tool
  1161. """
  1162. wx.Dialog.__init__(self, parent = parent, id = wx.ID_ANY, title = title, style = style)
  1163. self.parent = parent # mapdisplay.BufferedWindow class instance
  1164. # panel = wx.Panel(parent=self, id=wx.ID_ANY)
  1165. border = wx.BoxSizer(wx.VERTICAL)
  1166. txt = wx.StaticText(parent = self,
  1167. label = _("%d lines selected for z bulk-labeling") % nselected);
  1168. border.Add(item = txt, proportion = 0, flag = wx.ALL | wx.EXPAND, border = 5)
  1169. box = wx.StaticBox (parent = self, id = wx.ID_ANY, label = " %s " % _("Set value"))
  1170. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1171. flexSizer = wx.FlexGridSizer (cols = 2, hgap = 5, vgap = 5)
  1172. flexSizer.AddGrowableCol(0)
  1173. # starting value
  1174. txt = wx.StaticText(parent = self,
  1175. label = _("Starting value"));
  1176. self.value = wx.SpinCtrl(parent = self, id = wx.ID_ANY, size = (150, -1),
  1177. initial = 0,
  1178. min = -1e6, max = 1e6)
  1179. flexSizer.Add(txt, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  1180. flexSizer.Add(self.value, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1181. # step
  1182. txt = wx.StaticText(parent = self,
  1183. label = _("Step"))
  1184. self.step = wx.SpinCtrl(parent = self, id = wx.ID_ANY, size = (150, -1),
  1185. initial = 0,
  1186. min = 0, max = 1e6)
  1187. flexSizer.Add(txt, proportion = 0, flag = wx.ALIGN_CENTER_VERTICAL)
  1188. flexSizer.Add(self.step, proportion = 0, flag = wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1189. sizer.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 1)
  1190. border.Add(item = sizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 0)
  1191. # buttons
  1192. btnCancel = wx.Button(self, wx.ID_CANCEL)
  1193. btnOk = wx.Button(self, wx.ID_OK)
  1194. btnOk.SetDefault()
  1195. # sizers
  1196. btnSizer = wx.StdDialogButtonSizer()
  1197. btnSizer.AddButton(btnCancel)
  1198. btnSizer.AddButton(btnOk)
  1199. btnSizer.Realize()
  1200. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1201. mainSizer.Add(item = border, proportion = 1, flag = wx.EXPAND | wx.ALL, border = 5)
  1202. mainSizer.Add(item = btnSizer, proportion = 0,
  1203. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1204. self.SetSizer(mainSizer)
  1205. mainSizer.Fit(self)
  1206. class VDigitDuplicatesDialog(wx.Dialog):
  1207. def __init__(self, parent, data, title = _("List of duplicates"),
  1208. style = wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  1209. pos = wx.DefaultPosition):
  1210. """!Show duplicated feature ids
  1211. """
  1212. wx.Dialog.__init__(self, parent = parent, id = wx.ID_ANY, title = title, style = style,
  1213. pos = pos)
  1214. self.parent = parent # BufferedWindow
  1215. self.data = data
  1216. self.winList = []
  1217. # panel = wx.Panel(parent=self, id=wx.ID_ANY)
  1218. # notebook
  1219. self.notebook = wx.Notebook(parent = self, id = wx.ID_ANY, style = wx.BK_DEFAULT)
  1220. id = 1
  1221. for key in self.data.keys():
  1222. panel = wx.Panel(parent = self.notebook, id = wx.ID_ANY)
  1223. self.notebook.AddPage(page = panel, text = " %d " % (id))
  1224. # notebook body
  1225. border = wx.BoxSizer(wx.VERTICAL)
  1226. win = CheckListFeature(parent = panel, data = list(self.data[key]))
  1227. self.winList.append(win.GetId())
  1228. border.Add(item = win, proportion = 1,
  1229. flag = wx.ALL | wx.EXPAND, border = 5)
  1230. panel.SetSizer(border)
  1231. id += 1
  1232. # buttons
  1233. btnCancel = wx.Button(self, wx.ID_CANCEL)
  1234. btnOk = wx.Button(self, wx.ID_OK)
  1235. btnOk.SetDefault()
  1236. # sizers
  1237. btnSizer = wx.StdDialogButtonSizer()
  1238. btnSizer.AddButton(btnCancel)
  1239. btnSizer.AddButton(btnOk)
  1240. btnSizer.Realize()
  1241. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1242. mainSizer.Add(item = self.notebook, proportion = 1, flag = wx.EXPAND | wx.ALL, border = 5)
  1243. mainSizer.Add(item = btnSizer, proportion = 0,
  1244. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  1245. self.SetSizer(mainSizer)
  1246. mainSizer.Fit(self)
  1247. self.SetAutoLayout(True)
  1248. # set min size for dialog
  1249. self.SetMinSize((250, 180))
  1250. def GetUnSelected(self):
  1251. """!Get unselected items (feature id)
  1252. @return list of ids
  1253. """
  1254. ids = []
  1255. for id in self.winList:
  1256. wlist = self.FindWindowById(id)
  1257. for item in range(wlist.GetItemCount()):
  1258. if not wlist.IsChecked(item):
  1259. ids.append(int(wlist.GetItem(item, 0).GetText()))
  1260. return ids
  1261. class CheckListFeature(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin, listmix.CheckListCtrlMixin):
  1262. def __init__(self, parent, data,
  1263. pos = wx.DefaultPosition, log = None):
  1264. """!List of mapset/owner/group
  1265. """
  1266. self.parent = parent
  1267. self.data = data
  1268. wx.ListCtrl.__init__(self, parent, wx.ID_ANY,
  1269. style = wx.LC_REPORT)
  1270. listmix.CheckListCtrlMixin.__init__(self)
  1271. self.log = log
  1272. # setup mixins
  1273. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1274. self.LoadData(self.data)
  1275. def LoadData(self, data):
  1276. """!Load data into list
  1277. """
  1278. self.InsertColumn(0, _('Feature id'))
  1279. self.InsertColumn(1, _('Layer (Categories)'))
  1280. for item in data:
  1281. index = self.InsertStringItem(sys.maxint, str(item[0]))
  1282. self.SetStringItem(index, 1, str(item[1]))
  1283. # enable all items by default
  1284. for item in range(self.GetItemCount()):
  1285. self.CheckItem(item, True)
  1286. self.SetColumnWidth(col = 0, width = wx.LIST_AUTOSIZE_USEHEADER)
  1287. self.SetColumnWidth(col = 1, width = wx.LIST_AUTOSIZE_USEHEADER)
  1288. def OnCheckItem(self, index, flag):
  1289. """!Mapset checked/unchecked
  1290. """
  1291. pass