vdigit.py 92 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444
  1. """
  2. @package vdigit
  3. @brief Vector digitizer extension
  4. Progress:
  5. (1) v.edit called on the background (class VEdit) (removed in r?)
  6. (2) Reimplentation of v.digit (VDigit)
  7. Import:
  8. from vdigit import VDigit as VDigit
  9. Classes:
  10. - AbstractDigit
  11. - VDigit
  12. - AbstractDisplayDriver
  13. - CDisplayDriver
  14. - VDigitSettingsDialog
  15. - VDigitCategoryDialog
  16. - VDigitZBulkDialog
  17. - VDigitDuplicatesDialog
  18. - VDigitVBuildDialog
  19. (C) 2007-2009 by the GRASS Development Team
  20. This program is free software under the GNU General Public
  21. License (>=v2). Read the file COPYING that comes with GRASS
  22. for details.
  23. @author Martin Landa <landa.martin gmail.com>
  24. """
  25. import os
  26. import sys
  27. import string
  28. import copy
  29. from threading import Thread
  30. import wx
  31. import wx.lib.colourselect as csel
  32. import wx.lib.mixins.listctrl as listmix
  33. import gcmd
  34. import dbm
  35. from debug import Debug as Debug
  36. import gselect
  37. import globalvar
  38. from preferences import globalSettings as UserSettings
  39. try:
  40. digitPath = os.path.join(globalvar.ETCWXDIR, "vdigit")
  41. sys.path.append(digitPath)
  42. import grass7_wxvdigit as wxvdigit
  43. GV_LINES = wxvdigit.GV_LINES
  44. PseudoDC = wxvdigit.PseudoDC
  45. digitErr = ''
  46. except ImportError, err:
  47. GV_LINES = None
  48. PseudoDC = wx.PseudoDC
  49. digitErr = err
  50. print >> sys.stderr, "%sWARNING: Digitization tool is disabled (%s). " \
  51. "Detailed information in README file." % \
  52. (os.linesep, err)
  53. class AbstractDigit:
  54. """
  55. Abstract digitization class
  56. """
  57. def __init__(self, mapwindow):
  58. """Initialization
  59. @param mapwindow reference to mapwindow (MapFrame) instance
  60. @param settings initial settings of digitization tool
  61. """
  62. self.map = None
  63. self.mapWindow = mapwindow
  64. Debug.msg (3, "AbstractDigit.__init__(): map=%s" % \
  65. self.map)
  66. #self.SetCategory()
  67. self.driver = CDisplayDriver(self, mapwindow)
  68. def __del__(self):
  69. pass
  70. def SetCategoryNextToUse(self):
  71. """Find maximum category number in the map layer
  72. and update Digit.settings['category']
  73. @return 'True' on success, 'False' on failure
  74. """
  75. # vector map layer without categories, reset to '1'
  76. UserSettings.Set(group='vdigit', key='category', subkey='value', value=1)
  77. if self.map:
  78. cat = self.digit.GetCategory(UserSettings.Get(group='vdigit', key='layer', subkey='value'))
  79. cat += 1
  80. UserSettings.Set(group='vdigit', key='category', subkey='value',
  81. value=cat)
  82. def SetCategory(self):
  83. """Return category number to use (according Settings)"""
  84. if UserSettings.Get(group='vdigit', key="categoryMode", subkey='selection') == 0:
  85. self.SetCategoryNextToUse()
  86. return UserSettings.Get(group='vdigit', key="category", subkey='value')
  87. def SetMapName(self, map):
  88. """Set map name
  89. @param map map name to be set up or None (will close currently edited map)
  90. """
  91. Debug.msg (3, "AbstractDigit.SetMapName map=%s" % map)
  92. self.map = map
  93. try:
  94. ret = self.driver.Reset(self.map)
  95. except StandardError, e:
  96. raise gcmd.DigitError(parent=self.mapWindow.parent,
  97. message="%s %s (%s)" % (_('Unable to initialize display driver, '
  98. 'see README file for more information.\n\n'
  99. 'Details:'), e, digitErr))
  100. if map and ret == -1:
  101. raise gcmd.DigitError(parent=self.mapWindow.parent,
  102. message=_('Unable to open vector map <%s> for editing.\n\n'
  103. 'Data are probably corrupted, '
  104. 'try to run v.build to rebuild '
  105. 'the topology (Vector->Develop vector map->'
  106. 'Create/rebuild topology).') % map)
  107. if not map and ret != 0:
  108. raise gcmd.DigitError(parent=self.mapWindow.parent,
  109. message=_('Unable to open vector map <%s> for editing.\n\n'
  110. 'Data are probably corrupted, '
  111. 'try to run v.build to rebuild '
  112. 'the topology (Vector->Develop vector map->'
  113. 'Create/rebuild topology).') % map)
  114. self.digit.InitCats()
  115. def SelectLinesByQueryThresh(self):
  116. """Generic method used for SelectLinesByQuery()
  117. -- to get threshold value"""
  118. thresh = 0.0
  119. if UserSettings.Get(group='vdigit', key='query', subkey='selection') == 0:
  120. thresh = UserSettings.Get(group='vdigit', key='queryLength', subkey='thresh')
  121. if UserSettings.Get(group='vdigit', key="queryLength", subkey='than-selection') == 0:
  122. thresh = -1 * thresh
  123. else:
  124. thresh = UserSettings.Get(group='vdigit', key='queryDangle', subkey='thresh')
  125. if UserSettings.Get(group='vdigit', key="queryDangle", subkey='than-selection') == 0:
  126. thresh = -1 * thresh
  127. return thresh
  128. def GetSelectType(self):
  129. """Get type(s) to be selected
  130. Used by SelectLinesByBox() and SelectLinesByPoint()"""
  131. type = 0
  132. for feature in (('point', wxvdigit.GV_POINT),
  133. ('line', wxvdigit.GV_LINE),
  134. ('centroid', wxvdigit.GV_CENTROID),
  135. ('boundary', wxvdigit.GV_BOUNDARY)):
  136. if UserSettings.Get(group='vdigit', key='selectType',
  137. subkey=[feature[0], 'enabled']) is True:
  138. type |= feature[1]
  139. return type
  140. def SelectLinesFromBackgroundMap(self, pos1, pos2):
  141. """Select features from background map
  142. @param pos1,pos2 bounding box
  143. """
  144. bgmap = str(UserSettings.Get(group='vdigit', key='bgmap', subkey='value',
  145. internal=True))
  146. if bgmap == '':
  147. Debug.msg(4, "VEdit.SelectLinesFromBackgroundMap(): []")
  148. return []
  149. x1, y1 = pos1
  150. x2, y2 = pos2
  151. ret = gcmd.RunCommand('v.edit',
  152. parent = self,
  153. quiet = True,
  154. read = True,
  155. map = bgmap,
  156. tool = 'select',
  157. bbox = '%f,%f,%f,%f' % (x1, y1, x2, y2))
  158. if not ret:
  159. x, y = pos1
  160. ret = gcmd.RunCommand('v.edit',
  161. parent = self,
  162. quiet = True,
  163. read = True,
  164. map = bgmap,
  165. tool = 'select',
  166. coords = '%f,%f' % (x, y),
  167. thresh = self.driver.GetThreshold(type='selectThresh'))
  168. if not ret:
  169. return []
  170. output = ret.splitlines()[0] # first line
  171. ids = output.split(',')
  172. ids = map(int, ids) # str -> int
  173. Debug.msg(4, "VEdit.SelectLinesFromBackgroundMap(): %s" % \
  174. ",".join(["%d" % v for v in ids]))
  175. return ids
  176. class VDigit(AbstractDigit):
  177. """
  178. Prototype of digitization class based on v.digit reimplementation
  179. Under development (wxWidgets C/C++ background)
  180. """
  181. def __init__(self, mapwindow):
  182. """Initialization
  183. @param mapwindow reference to mapwindow (MapFrame) instance
  184. @param settings initial settings of digitization tool
  185. """
  186. AbstractDigit.__init__(self, mapwindow)
  187. try:
  188. self.digit = wxvdigit.Digit(self.driver.GetDevice(),
  189. mapwindow)
  190. except (ImportError, NameError):
  191. self.digit = None
  192. self.toolbar = mapwindow.parent.toolbars['vdigit']
  193. self.UpdateSettings()
  194. def __del__(self):
  195. del self.digit
  196. def AddPoint (self, map, point, x, y, z=None):
  197. """Add new point/centroid
  198. @param map map name (unused, for compatability with VEdit)
  199. @param point feature type (if true point otherwise centroid)
  200. @param x,y,z coordinates
  201. """
  202. if UserSettings.Get(group='vdigit', key="categoryMode", subkey='selection') == 2:
  203. layer = -1 # -> no category
  204. cat = -1
  205. else:
  206. layer = UserSettings.Get(group='vdigit', key="layer", subkey='value')
  207. cat = self.SetCategory()
  208. if point:
  209. type = wxvdigit.GV_POINT
  210. else:
  211. type = wxvdigit.GV_CENTROID
  212. snap, thresh = self.__getSnapThreshold()
  213. bgmap = str(UserSettings.Get(group='vdigit', key="bgmap",
  214. subkey='value', internal=True))
  215. if z:
  216. ret = self.digit.AddLine(type, [x, y, z], layer, cat,
  217. bgmap, snap, thresh)
  218. else:
  219. ret = self.digit.AddLine(type, [x, y], layer, cat,
  220. bgmap, snap, thresh)
  221. self.toolbar.EnableUndo()
  222. return ret
  223. def AddLine (self, map, line, coords):
  224. """Add line/boundary
  225. @param map map name (unused, for compatability with VEdit)
  226. @param line feature type (if True line, otherwise boundary)
  227. @param coords list of coordinates
  228. """
  229. if len(coords) < 2:
  230. return
  231. if UserSettings.Get(group='vdigit', key="categoryMode", subkey='selection') == 2:
  232. layer = -1 # -> no category
  233. cat = -1
  234. else:
  235. layer = UserSettings.Get(group='vdigit', key="layer", subkey='value')
  236. cat = self.SetCategory()
  237. if line:
  238. type = wxvdigit.GV_LINE
  239. else:
  240. type = wxvdigit.GV_BOUNDARY
  241. listCoords = []
  242. for c in coords:
  243. for x in c:
  244. listCoords.append(x)
  245. snap, thresh = self.__getSnapThreshold()
  246. bgmap = str(UserSettings.Get(group='vdigit', key="bgmap",
  247. subkey='value', internal=True))
  248. ret = self.digit.AddLine(type, listCoords, layer, cat,
  249. bgmap, snap, thresh)
  250. self.toolbar.EnableUndo()
  251. return ret
  252. def DeleteSelectedLines(self):
  253. """Delete selected features
  254. @return number of deleted lines
  255. """
  256. nlines = self.digit.DeleteLines(UserSettings.Get(group='vdigit', key='delRecord', subkey='enabled'))
  257. if nlines > 0:
  258. self.toolbar.EnableUndo()
  259. return nlines
  260. def MoveSelectedLines(self, move):
  261. """Move selected features
  262. @param move direction (x, y)
  263. """
  264. snap, thresh = self.__getSnapThreshold()
  265. bgmap = str(UserSettings.Get(group='vdigit', key="bgmap",
  266. subkey='value', internal=True))
  267. try:
  268. nlines = self.digit.MoveLines(move[0], move[1], 0.0, # TODO 3D
  269. bgmap, snap, thresh)
  270. except SystemExit:
  271. pass
  272. if nlines > 0:
  273. self.toolbar.EnableUndo()
  274. return nlines
  275. def MoveSelectedVertex(self, coords, move):
  276. """Move selected vertex of the line
  277. @param coords click coordinates
  278. @param move X,Y direction
  279. @return 1 vertex moved
  280. @return 0 vertex not moved (not found, line is not selected)
  281. """
  282. snap, thresh = self.__getSnapThreshold()
  283. bgmap = str(UserSettings.Get(group='vdigit', key="bgmap",
  284. subkey='value', internal=True))
  285. moved = self.digit.MoveVertex(coords[0], coords[1], 0.0, # TODO 3D
  286. move[0], move[1], 0.0,
  287. bgmap, snap,
  288. self.driver.GetThreshold(type='selectThresh'), thresh)
  289. if moved:
  290. self.toolbar.EnableUndo()
  291. return moved
  292. def AddVertex(self, coords):
  293. """Add new vertex to the selected line/boundary on position 'coords'
  294. @param coords coordinates to add vertex
  295. @return 1 vertex added
  296. @return 0 nothing changed
  297. @return -1 on failure
  298. """
  299. added = self.digit.ModifyLineVertex(1, coords[0], coords[1], 0.0, # TODO 3D
  300. self.driver.GetThreshold(type='selectThresh'))
  301. if added > 0:
  302. self.toolbar.EnableUndo()
  303. return added
  304. def RemoveVertex(self, coords):
  305. """Remove vertex from the selected line/boundary on position 'coords'
  306. @param coords coordinates to remove vertex
  307. @return 1 vertex removed
  308. @return 0 nothing changed
  309. @return -1 on failure
  310. """
  311. deleted = self.digit.ModifyLineVertex(0, coords[0], coords[1], 0.0, # TODO 3D
  312. self.driver.GetThreshold(type='selectThresh'))
  313. if deleted > 0:
  314. self.toolbar.EnableUndo()
  315. return deleted
  316. def SplitLine(self, coords):
  317. """Split selected line/boundary on position 'coords'
  318. @param coords coordinates to split line
  319. @return 1 line modified
  320. @return 0 nothing changed
  321. @return -1 error
  322. """
  323. ret = self.digit.SplitLine(coords[0], coords[1], 0.0, # TODO 3D
  324. self.driver.GetThreshold('selectThresh'))
  325. if ret > 0:
  326. self.toolbar.EnableUndo()
  327. return ret
  328. def EditLine(self, line, coords):
  329. """Edit existing line/boundary
  330. @param line id of line to be modified
  331. @param coords list of coordinates of modified line
  332. @return feature id of new line
  333. @return -1 on error
  334. """
  335. try:
  336. lineid = line[0]
  337. except:
  338. lineid = -1
  339. if len(coords) < 2:
  340. self.DeleteSelectedLines()
  341. return 0
  342. listCoords = []
  343. for c in coords:
  344. for x in c:
  345. listCoords.append(x)
  346. snap, thresh = self.__getSnapThreshold()
  347. bgmap = str(UserSettings.Get(group='vdigit', key="bgmap",
  348. subkey='value', internal=True))
  349. try:
  350. ret = self.digit.RewriteLine(lineid, listCoords,
  351. bgmap, snap, thresh)
  352. except SystemExit:
  353. pass
  354. if ret > 0:
  355. self.toolbar.EnableUndo()
  356. return ret
  357. def FlipLine(self):
  358. """Flip selected lines/boundaries
  359. @return number of modified lines
  360. @return -1 on error
  361. """
  362. ret = self.digit.FlipLines()
  363. if ret > 0:
  364. self.toolbar.EnableUndo()
  365. return ret
  366. def MergeLine(self):
  367. """Merge selected lines/boundaries
  368. @return number of modified lines
  369. @return -1 on error
  370. """
  371. ret = self.digit.MergeLines()
  372. if ret > 0:
  373. self.toolbar.EnableUndo()
  374. return ret
  375. def BreakLine(self):
  376. """Break selected lines/boundaries
  377. @return number of modified lines
  378. @return -1 on error
  379. """
  380. ret = self.digit.BreakLines()
  381. if ret > 0:
  382. self.toolbar.EnableUndo()
  383. return ret
  384. def SnapLine(self):
  385. """Snap selected lines/boundaries
  386. @return on success
  387. @return -1 on error
  388. """
  389. snap, thresh = self.__getSnapThreshold()
  390. ret = self.digit.SnapLines(thresh)
  391. if ret == 0:
  392. self.toolbar.EnableUndo()
  393. return ret
  394. def ConnectLine(self):
  395. """Connect selected lines/boundaries
  396. @return 1 lines connected
  397. @return 0 lines not connected
  398. @return -1 on error
  399. """
  400. snap, thresh = self.__getSnapThreshold()
  401. ret = self.digit.ConnectLines(thresh)
  402. if ret > 0:
  403. self.toolbar.EnableUndo()
  404. return ret
  405. def CopyLine(self, ids=[]):
  406. """Copy features from (background) vector map
  407. @param ids list of line ids to be copied
  408. @return number of copied features
  409. @return -1 on error
  410. """
  411. bgmap = str(UserSettings.Get(group='vdigit', key='bgmap',
  412. subkey='value', internal=True))
  413. if len(bgmap) > 0:
  414. ret = self.digit.CopyLines(ids, bgmap)
  415. else:
  416. ret = self.digit.CopyLines(ids, None)
  417. if ret > 0:
  418. self.toolbar.EnableUndo()
  419. return ret
  420. def CopyCats(self, fromId, toId, copyAttrb=False):
  421. """Copy given categories to objects with id listed in ids
  422. @param cats ids of 'from' feature
  423. @param ids ids of 'to' feature(s)
  424. @return number of modified features
  425. @return -1 on error
  426. """
  427. if len(fromId) == 0 or len(toId) == 0:
  428. return 0
  429. ret = self.digit.CopyCats(fromId, toId, copyAttrb)
  430. if ret > 0:
  431. self.toolbar.EnableUndo()
  432. return ret
  433. def SelectLinesByQuery(self, pos1, pos2):
  434. """Select features by query
  435. @param pos1, pos2 bounding box definition
  436. """
  437. thresh = self.SelectLinesByQueryThresh()
  438. w, n = pos1
  439. e, s = pos2
  440. query = wxvdigit.QUERY_UNKNOWN
  441. if UserSettings.Get(group='vdigit', key='query', subkey='selection') == 0:
  442. query = wxvdigit.QUERY_LENGTH
  443. else:
  444. query = wxvdigit.QUERY_DANGLE
  445. type = wxvdigit.GV_POINTS | wxvdigit.GV_LINES # TODO: 3D
  446. ids = self.digit.SelectLinesByQuery(w, n, 0.0, e, s, 1000.0,
  447. UserSettings.Get(group='vdigit', key='query', subkey='box'),
  448. query, type, thresh)
  449. Debug.msg(4, "VDigit.SelectLinesByQuery(): %s" % \
  450. ",".join(["%d" % v for v in ids]))
  451. return ids
  452. def GetLineCats(self, line=-1):
  453. """Get layer/category pairs from given (selected) line
  454. @param line feature id (-1 for first selected line)
  455. """
  456. return dict(self.digit.GetLineCats(line))
  457. def SetLineCats(self, line, layer, cats, add=True):
  458. """Set categories for given line and layer
  459. @param line feature id
  460. @param layer layer number (-1 for first selected line)
  461. @param cats list of categories
  462. @param add if True to add, otherwise do delete categories
  463. @return new feature id (feature need to be rewritten)
  464. @return -1 on error
  465. """
  466. ret = self.digit.SetLineCats(line, layer, cats, add)
  467. if ret > 0:
  468. self.toolbar.EnableUndo()
  469. return ret
  470. def GetLayers(self):
  471. """Get list of layers"""
  472. return self.digit.GetLayers()
  473. def TypeConvForSelectedLines(self):
  474. """Feature type conversion for selected objects.
  475. Supported conversions:
  476. - point <-> centroid
  477. - line <-> boundary
  478. @return number of modified features
  479. @return -1 on error
  480. """
  481. ret = self.digit.TypeConvLines()
  482. if ret > 0:
  483. self.toolbar.EnableUndo()
  484. return ret
  485. def Undo(self, level=-1):
  486. """Undo action
  487. @param level levels to undo (0 to revert all)
  488. @return id of current changeset
  489. """
  490. try:
  491. ret = self.digit.Undo(level)
  492. except SystemExit:
  493. ret = -2
  494. if ret == -2:
  495. raise gcmd.DigitError, _("Undo failed, data corrupted.")
  496. self.mapWindow.UpdateMap(render=False)
  497. if ret < 0: # disable undo tool
  498. self.toolbar.EnableUndo(False)
  499. def GetUndoLevel(self):
  500. """Get undo level (number of active changesets)"""
  501. return self.digit.GetUndoLevel()
  502. def UpdateSettings(self):
  503. """Update digit settigs"""
  504. if not self.digit:
  505. return
  506. self.digit.UpdateSettings(UserSettings.Get(group='vdigit', key='breakLines',
  507. subkey='enabled'),
  508. UserSettings.Get(group='vdigit', key='addCentroid',
  509. subkey='enabled'),
  510. UserSettings.Get(group='vdigit', key='catBoundary',
  511. subkey='enabled'))
  512. def __getSnapThreshold(self):
  513. """Get snap mode and threshold value
  514. @return (snap, thresh)
  515. """
  516. thresh = self.driver.GetThreshold()
  517. if thresh > 0.0:
  518. if UserSettings.Get(group='vdigit', key='snapToVertex', subkey='enabled') is True:
  519. snap = wxvdigit.SNAPVERTEX
  520. else:
  521. snap = wxvdigit.SNAP
  522. else:
  523. snap = wxvdigit.NO_SNAP
  524. return (snap, thresh)
  525. class Digit(VDigit):
  526. """Default digit class"""
  527. def __init__(self, mapwindow):
  528. VDigit.__init__(self, mapwindow)
  529. self.type = 'vdigit'
  530. def __del__(self):
  531. VDigit.__del__(self)
  532. class AbstractDisplayDriver:
  533. """Abstract classs for display driver"""
  534. def __init__(self, parent, mapwindow):
  535. """Initialization
  536. @param parent
  537. @param mapwindow reference to mapwindow (MFrame)
  538. """
  539. self.parent = parent
  540. self.mapwindow = mapwindow
  541. self.ids = {} # dict[g6id] = [pdcId]
  542. self.selected = [] # list of selected objects (grassId!)
  543. def GetThreshold(self, type='snapping', value=None, units=None):
  544. """Return threshold in map units
  545. @param value threshold to be set up
  546. @param units units (map, screen)
  547. """
  548. if value is None:
  549. value = UserSettings.Get(group='vdigit', key=type, subkey='value')
  550. if units is None:
  551. units = UserSettings.Get(group='vdigit', key=type, subkey='units')
  552. reg = self.mapwindow.Map.region
  553. if value < 0:
  554. value = (reg['nsres'] + reg['ewres']) / 2.
  555. if units == "screen pixels":
  556. # pixel -> cell
  557. if reg['nsres'] > reg['ewres']:
  558. res = reg['nsres']
  559. else:
  560. res = reg['ewres']
  561. threshold = value * res
  562. else:
  563. threshold = value
  564. Debug.msg(4, "AbstractDisplayDriver.GetThreshold(): type=%s, thresh=%f" % (type, threshold))
  565. return threshold
  566. class CDisplayDriver(AbstractDisplayDriver):
  567. """
  568. Display driver using grass7_wxdriver module
  569. """
  570. def __init__(self, parent, mapwindow):
  571. """Initialization
  572. @param parent
  573. @param mapwindow reference to mapwindow (MFrame)
  574. """
  575. AbstractDisplayDriver.__init__(self, parent, mapwindow)
  576. self.mapWindow = mapwindow
  577. # initialize wx display driver
  578. try:
  579. self.__display = wxvdigit.DisplayDriver(mapwindow.pdcVector,
  580. mapwindow.pdcTmp)
  581. except:
  582. self.__display = None
  583. self.UpdateSettings()
  584. def GetDevice(self):
  585. """Get device"""
  586. return self.__display
  587. def SetDevice(self, pdc):
  588. """Set device for driver
  589. @param pdc wx.PseudoDC instance
  590. """
  591. self.__display.SetDevice(pdc)
  592. def Reset(self, map):
  593. """Reset map
  594. Open or close the vector map by driver.
  595. @param map map name or None to close the map
  596. @return 0 on success (close map)
  597. @return topo level on success (open map)
  598. @return non-zero (close map)
  599. @return -1 on error (open map)
  600. """
  601. if map:
  602. name, mapset = map.split('@')
  603. try:
  604. ret = self.__display.OpenMap(str(name), str(mapset), True)
  605. except SystemExit:
  606. ret = -1
  607. else:
  608. ret = self.__display.CloseMap()
  609. return ret
  610. def ReloadMap(self):
  611. """Reload map (close and re-open).
  612. Needed for v.edit, TODO: get rid of that..."""
  613. Debug.msg(4, "CDisplayDriver.ReloadMap():")
  614. self.__display.ReloadMap()
  615. def DrawMap(self):
  616. """Draw vector map layer content
  617. @return wx.Image instance
  618. """
  619. nlines = self.__display.DrawMap(True) # force
  620. Debug.msg(3, "CDisplayDriver.DrawMap(): nlines=%d" % nlines)
  621. return nlines
  622. def SelectLinesByBox(self, begin, end, type=0, drawSeg=False):
  623. """Select vector features by given bounding box.
  624. If type is given, only vector features of given type are selected.
  625. @param begin,end bounding box definition
  626. @param type select only objects of given type
  627. """
  628. x1, y1 = begin
  629. x2, y2 = end
  630. inBox = UserSettings.Get(group='vdigit', key='selectInside', subkey='enabled')
  631. nselected = self.__display.SelectLinesByBox(x1, y1, -1.0 * wxvdigit.PORT_DOUBLE_MAX,
  632. x2, y2, wxvdigit.PORT_DOUBLE_MAX,
  633. type, inBox, drawSeg)
  634. Debug.msg(4, "CDisplayDriver.SelectLinesByBox(): selected=%d" % \
  635. nselected)
  636. return nselected
  637. def SelectLineByPoint(self, point, type=0):
  638. """Select vector feature by coordinates of click point (in given threshold).
  639. If type is given, only vector features of given type are selected.
  640. @param point click coordinates (bounding box given by threshold)
  641. @param type select only objects of given type
  642. """
  643. pointOnLine = self.__display.SelectLineByPoint(point[0], point[1], 0.0,
  644. self.GetThreshold(type='selectThresh'),
  645. type, 0); # without_z
  646. if len(pointOnLine) > 0:
  647. Debug.msg(4, "CDisplayDriver.SelectLineByPoint(): pointOnLine=%f,%f" % \
  648. (pointOnLine[0], pointOnLine[1]))
  649. return pointOnLine
  650. else:
  651. Debug.msg(4, "CDisplayDriver.SelectLineByPoint(): no line found")
  652. return None
  653. def GetSelected(self, grassId=True):
  654. """Return ids of selected vector features
  655. @param grassId if grassId is True returns GRASS ids, otherwise
  656. internal ids of objects drawn in PseudoDC"""
  657. if grassId:
  658. selected = self.__display.GetSelected(True)
  659. else:
  660. selected = self.__display.GetSelected(False)
  661. Debug.msg(4, "CDisplayDriver.GetSelected(): grassId=%d, ids=%s" % \
  662. (grassId, (",".join(["%d" % v for v in selected]))))
  663. return selected
  664. def GetSelectedCoord(self):
  665. """Return ids of selected vector features and their coordinates"""
  666. return dict(self.__display.GetSelectedCoord())
  667. def GetRegionSelected(self):
  668. """Get minimal region extent of selected features (ids/cats)"""
  669. return self.__display.GetRegionSelected()
  670. def GetDuplicates(self):
  671. """Return ids of (selected) duplicated vector features
  672. """
  673. # -> id : (list of ids)
  674. dupl = dict(self.__display.GetDuplicates())
  675. # -> id : ((id, cat), ...)
  676. dupl_full = {}
  677. for key in dupl.keys():
  678. dupl_full[key] = []
  679. for id in dupl[key]:
  680. catStr = ''
  681. cats = self.parent.GetLineCats(line=id)
  682. for layer in cats.keys():
  683. if len(cats[layer]) > 0:
  684. catStr = "%d: (" % layer
  685. for cat in cats[layer]:
  686. catStr += "%d," % cat
  687. catStr = catStr.rstrip(',')
  688. catStr += ')'
  689. dupl_full[key].append([id, catStr])
  690. return dupl_full
  691. def GetSelectedVertex(self, coords):
  692. """Get PseudoDC id(s) of vertex (of selected line)
  693. on position 'coords'
  694. @param coords click position
  695. """
  696. x, y = coords
  697. id = self.__display.GetSelectedVertex(x, y, self.GetThreshold(type='selectThresh'))
  698. Debug.msg(4, "CDisplayDriver.GetSelectedVertex(): id=%s" % \
  699. (",".join(["%d" % v for v in id])))
  700. return id
  701. def SetSelected(self, id, field=-1):
  702. """Set selected vector features
  703. @param id list of feature ids/categories to be selected
  704. @param field field(layer) number, -1 for ids instead of cats
  705. """
  706. Debug.msg(4, "CDisplayDriver.SetSelected(): id=%s" % \
  707. id)
  708. self.__display.SetSelected(id, field)
  709. def UnSelect(self, id):
  710. """Unselect vector features
  711. @param id list of feature id(s)
  712. """
  713. Debug.msg(4, "CDisplayDriver.UnSelect(): id=%s" % \
  714. ",".join(["%d" % v for v in id]))
  715. self.__display.UnSelect(id)
  716. def UpdateRegion(self):
  717. """Set geographical region
  718. Needed for 'cell2pixel' conversion"""
  719. map = self.mapwindow.Map
  720. reg = map.region
  721. self.__display.SetRegion(reg['n'],
  722. reg['s'],
  723. reg['e'],
  724. reg['w'],
  725. reg['nsres'],
  726. reg['ewres'],
  727. reg['center_easting'],
  728. reg['center_northing'],
  729. map.width, map.height)
  730. def GetMapBoundingBox(self):
  731. """Return bounding box of given vector map layer
  732. @return (w,s,b,e,n,t)
  733. """
  734. return self.__display.GetMapBoundingBox()
  735. def DrawSelected(self, draw=True):
  736. """Show/hide selected features"""
  737. self.__display.DrawSelected(draw)
  738. def UpdateSettings(self, alpha=255):
  739. """Update display driver settings"""
  740. # TODO map units
  741. if not self.__display:
  742. return
  743. color = {}
  744. for symbol in ("highlight",
  745. "highlightDupl",
  746. "point",
  747. "line",
  748. "boundaryNo",
  749. "boundaryOne",
  750. "boundaryTwo",
  751. "centroidIn",
  752. "centroidOut",
  753. "centroidDup",
  754. "nodeOne",
  755. "nodeTwo",
  756. "vertex",
  757. "area",
  758. "direction"):
  759. color[symbol] = wx.Color(UserSettings.Get(group='vdigit', key='symbol',
  760. subkey=[symbol, 'color'])[0],
  761. UserSettings.Get(group='vdigit', key='symbol',
  762. subkey=[symbol, 'color'])[1],
  763. UserSettings.Get(group='vdigit', key='symbol',
  764. subkey=[symbol, 'color'])[2]).GetRGB()
  765. self.__display.UpdateSettings (color['highlight'],
  766. UserSettings.Get(group='vdigit', key='checkForDupl',
  767. subkey='enabled'),
  768. color['highlightDupl'],
  769. UserSettings.Get(group='vdigit', key='symbol',
  770. subkey=['point', 'enabled']),
  771. color['point'],
  772. UserSettings.Get(group='vdigit', key='symbol',
  773. subkey=['line', 'enabled']),
  774. color['line'],
  775. UserSettings.Get(group='vdigit', key='symbol',
  776. subkey=['boundaryNo', 'enabled']),
  777. color['boundaryNo'],
  778. UserSettings.Get(group='vdigit', key='symbol',
  779. subkey=['boundaryOne', 'enabled']),
  780. color['boundaryOne'],
  781. UserSettings.Get(group='vdigit', key='symbol',
  782. subkey=['boundaryTwo', 'enabled']),
  783. color['boundaryTwo'],
  784. UserSettings.Get(group='vdigit', key='symbol',
  785. subkey=['centroidIn', 'enabled']),
  786. color['centroidIn'],
  787. UserSettings.Get(group='vdigit', key='symbol',
  788. subkey=['centroidOut', 'enabled']),
  789. color['centroidOut'],
  790. UserSettings.Get(group='vdigit', key='symbol',
  791. subkey=['centroidDup', 'enabled']),
  792. color['centroidDup'],
  793. UserSettings.Get(group='vdigit', key='symbol',
  794. subkey=['nodeOne', 'enabled']),
  795. color['nodeOne'],
  796. UserSettings.Get(group='vdigit', key='symbol',
  797. subkey=['nodeTwo', 'enabled']),
  798. color['nodeTwo'],
  799. UserSettings.Get(group='vdigit', key='symbol',
  800. subkey=['vertex', 'enabled']),
  801. color['vertex'],
  802. UserSettings.Get(group='vdigit', key='symbol',
  803. subkey=['area', 'enabled']),
  804. color['area'],
  805. UserSettings.Get(group='vdigit', key='symbol',
  806. subkey=['direction', 'enabled']),
  807. color['direction'],
  808. UserSettings.Get(group='vdigit', key='lineWidth',
  809. subkey='value'),
  810. alpha)
  811. class VDigitSettingsDialog(wx.Dialog):
  812. """
  813. Standard settings dialog for digitization purposes
  814. """
  815. def __init__(self, parent, title, style=wx.DEFAULT_DIALOG_STYLE):
  816. wx.Dialog.__init__(self, parent=parent, id=wx.ID_ANY, title=title, style=style)
  817. self.parent = parent # mapdisplay.BufferedWindow class instance
  818. # notebook
  819. notebook = wx.Notebook(parent=self, id=wx.ID_ANY, style=wx.BK_DEFAULT)
  820. self.__CreateSymbologyPage(notebook)
  821. parent.digit.SetCategory() # update category number (next to use)
  822. self.__CreateGeneralPage(notebook)
  823. self.__CreateAttributesPage(notebook)
  824. self.__CreateQueryPage(notebook)
  825. # buttons
  826. btnApply = wx.Button(self, wx.ID_APPLY)
  827. btnCancel = wx.Button(self, wx.ID_CANCEL)
  828. btnSave = wx.Button(self, wx.ID_SAVE)
  829. btnSave.SetDefault()
  830. # bindigs
  831. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  832. btnApply.SetToolTipString(_("Apply changes for this session"))
  833. btnApply.SetDefault()
  834. btnSave.Bind(wx.EVT_BUTTON, self.OnSave)
  835. btnSave.SetToolTipString(_("Close dialog and save changes to user settings file"))
  836. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  837. btnCancel.SetToolTipString(_("Close dialog and ignore changes"))
  838. # sizers
  839. btnSizer = wx.StdDialogButtonSizer()
  840. btnSizer.AddButton(btnCancel)
  841. btnSizer.AddButton(btnApply)
  842. btnSizer.AddButton(btnSave)
  843. btnSizer.Realize()
  844. mainSizer = wx.BoxSizer(wx.VERTICAL)
  845. mainSizer.Add(item=notebook, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  846. mainSizer.Add(item=btnSizer, proportion=0,
  847. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  848. self.Bind(wx.EVT_CLOSE, self.OnCancel)
  849. self.SetSizer(mainSizer)
  850. mainSizer.Fit(self)
  851. def __CreateSymbologyPage(self, notebook):
  852. """Create notebook page concerning with symbology settings"""
  853. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  854. notebook.AddPage(page=panel, text=_("Symbology"))
  855. sizer = wx.BoxSizer(wx.VERTICAL)
  856. flexSizer = wx.FlexGridSizer (cols=3, hgap=5, vgap=5)
  857. flexSizer.AddGrowableCol(0)
  858. self.symbology = {}
  859. for label, key in self.__SymbologyData():
  860. textLabel = wx.StaticText(panel, wx.ID_ANY, label)
  861. color = csel.ColourSelect(panel, id=wx.ID_ANY,
  862. colour=UserSettings.Get(group='vdigit', key='symbol',
  863. subkey=[key, 'color']), size=(25, 25))
  864. isEnabled = UserSettings.Get(group='vdigit', key='symbol',
  865. subkey=[key, 'enabled'])
  866. if isEnabled is not None:
  867. enabled = wx.CheckBox(panel, id=wx.ID_ANY, label="")
  868. enabled.SetValue(isEnabled)
  869. self.symbology[key] = (enabled, color)
  870. else:
  871. enabled = (1, 1)
  872. self.symbology[key] = (None, color)
  873. flexSizer.Add(textLabel, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  874. flexSizer.Add(enabled, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  875. flexSizer.Add(color, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  876. color.SetName("GetColour")
  877. sizer.Add(item=flexSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=10)
  878. panel.SetSizer(sizer)
  879. return panel
  880. def __CreateGeneralPage(self, notebook):
  881. """Create notebook page concerning with symbology settings"""
  882. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  883. notebook.AddPage(page=panel, text=_("General"))
  884. border = wx.BoxSizer(wx.VERTICAL)
  885. #
  886. # display section
  887. #
  888. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Display"))
  889. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  890. flexSizer = wx.FlexGridSizer (cols=3, hgap=5, vgap=5)
  891. flexSizer.AddGrowableCol(0)
  892. # line width
  893. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Line width"))
  894. self.lineWidthValue = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(75, -1),
  895. initial=UserSettings.Get(group='vdigit', key="lineWidth", subkey='value'),
  896. min=1, max=1e6)
  897. units = wx.StaticText(parent=panel, id=wx.ID_ANY, size=(115, -1),
  898. label=UserSettings.Get(group='vdigit', key="lineWidth", subkey='units'),
  899. style=wx.ALIGN_LEFT)
  900. flexSizer.Add(text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  901. flexSizer.Add(self.lineWidthValue, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  902. flexSizer.Add(units, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL | wx.LEFT,
  903. border=10)
  904. sizer.Add(item=flexSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=1)
  905. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=5)
  906. #
  907. # snapping section
  908. #
  909. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Snapping"))
  910. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  911. flexSizer = wx.FlexGridSizer (cols=3, hgap=5, vgap=5)
  912. flexSizer.AddGrowableCol(0)
  913. # snapping
  914. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Snapping threshold"))
  915. self.snappingValue = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(75, -1),
  916. initial=UserSettings.Get(group='vdigit', key="snapping", subkey='value'),
  917. min=-1, max=1e6)
  918. self.snappingValue.Bind(wx.EVT_SPINCTRL, self.OnChangeSnappingValue)
  919. self.snappingValue.Bind(wx.EVT_TEXT, self.OnChangeSnappingValue)
  920. self.snappingUnit = wx.Choice(parent=panel, id=wx.ID_ANY, size=(125, -1),
  921. choices=["screen pixels", "map units"])
  922. self.snappingUnit.SetStringSelection(UserSettings.Get(group='vdigit', key="snapping", subkey='units'))
  923. self.snappingUnit.Bind(wx.EVT_CHOICE, self.OnChangeSnappingUnits)
  924. flexSizer.Add(text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  925. flexSizer.Add(self.snappingValue, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  926. flexSizer.Add(self.snappingUnit, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  927. vertexSizer = wx.BoxSizer(wx.VERTICAL)
  928. self.snapVertex = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  929. label=_("Snap also to vertex"))
  930. self.snapVertex.SetValue(UserSettings.Get(group='vdigit', key="snapToVertex", subkey='enabled'))
  931. vertexSizer.Add(item=self.snapVertex, proportion=0, flag=wx.EXPAND)
  932. self.mapUnits = self.parent.MapWindow.Map.ProjInfo()['units']
  933. self.snappingInfo = wx.StaticText(parent=panel, id=wx.ID_ANY,
  934. label=_("Snapping threshold is %(value).1f %(units)s") % \
  935. {'value' : self.parent.digit.driver.GetThreshold(),
  936. 'units' : self.mapUnits})
  937. vertexSizer.Add(item=self.snappingInfo, proportion=0,
  938. flag=wx.ALL | wx.EXPAND, border=1)
  939. sizer.Add(item=flexSizer, proportion=1, flag=wx.EXPAND)
  940. sizer.Add(item=vertexSizer, proportion=1, flag=wx.EXPAND)
  941. border.Add(item=sizer, proportion=0, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=5)
  942. #
  943. # select box
  944. #
  945. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Select vector features"))
  946. # feature type
  947. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  948. inSizer = wx.BoxSizer(wx.HORIZONTAL)
  949. self.selectFeature = {}
  950. for feature in ('point', 'line',
  951. 'centroid', 'boundary'):
  952. chkbox = wx.CheckBox(parent=panel, label=feature)
  953. self.selectFeature[feature] = chkbox.GetId()
  954. chkbox.SetValue(UserSettings.Get(group='vdigit', key='selectType',
  955. subkey=[feature, 'enabled']))
  956. inSizer.Add(item=chkbox, proportion=0,
  957. flag=wx.EXPAND | wx.ALL, border=5)
  958. sizer.Add(item=inSizer, proportion=0, flag=wx.EXPAND)
  959. # threshold
  960. flexSizer = wx.FlexGridSizer (cols=3, hgap=5, vgap=5)
  961. flexSizer.AddGrowableCol(0)
  962. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Select threshold"))
  963. self.selectThreshValue = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(75, -1),
  964. initial=UserSettings.Get(group='vdigit', key="selectThresh", subkey='value'),
  965. min=1, max=1e6)
  966. units = wx.StaticText(parent=panel, id=wx.ID_ANY, size=(115, -1),
  967. label=UserSettings.Get(group='vdigit', key="lineWidth", subkey='units'),
  968. style=wx.ALIGN_LEFT)
  969. flexSizer.Add(text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  970. flexSizer.Add(self.selectThreshValue, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  971. flexSizer.Add(units, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL | wx.LEFT,
  972. border=10)
  973. self.selectIn = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  974. label=_("Select only features inside of selection bounding box"))
  975. self.selectIn.SetValue(UserSettings.Get(group='vdigit', key="selectInside", subkey='enabled'))
  976. self.selectIn.SetToolTipString(_("By default are selected all features overlapping selection bounding box "))
  977. self.checkForDupl = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  978. label=_("Check for duplicates"))
  979. self.checkForDupl.SetValue(UserSettings.Get(group='vdigit', key="checkForDupl", subkey='enabled'))
  980. sizer.Add(item=flexSizer, proportion=0, flag=wx.EXPAND)
  981. sizer.Add(item=self.selectIn, proportion=0, flag=wx.EXPAND | wx.ALL, border=1)
  982. sizer.Add(item=self.checkForDupl, proportion=0, flag=wx.EXPAND | wx.ALL, border=1)
  983. border.Add(item=sizer, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border=5)
  984. #
  985. # digitize lines box
  986. #
  987. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Digitize line features"))
  988. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  989. self.intersect = wx.CheckBox(parent=panel, label=_("Break lines at intersection"))
  990. self.intersect.SetValue(UserSettings.Get(group='vdigit', key='breakLines', subkey='enabled'))
  991. sizer.Add(item=self.intersect, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  992. border.Add(item=sizer, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border=5)
  993. #
  994. # save-on-exit box
  995. #
  996. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Save changes"))
  997. # save changes on exit?
  998. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  999. self.save = wx.CheckBox(parent=panel, label=_("Save changes on exit"))
  1000. self.save.SetValue(UserSettings.Get(group='vdigit', key='saveOnExit', subkey='enabled'))
  1001. sizer.Add(item=self.save, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1002. border.Add(item=sizer, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM, border=5)
  1003. panel.SetSizer(border)
  1004. return panel
  1005. def __CreateQueryPage(self, notebook):
  1006. """Create notebook page for query tool"""
  1007. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1008. notebook.AddPage(page=panel, text=_("Query tool"))
  1009. border = wx.BoxSizer(wx.VERTICAL)
  1010. #
  1011. # query tool box
  1012. #
  1013. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Choose query tool"))
  1014. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1015. LocUnits = self.parent.MapWindow.Map.ProjInfo()['units']
  1016. self.queryBox = wx.CheckBox(parent=panel, id=wx.ID_ANY, label=_("Select by box"))
  1017. self.queryBox.SetValue(UserSettings.Get(group='vdigit', key="query", subkey='box'))
  1018. sizer.Add(item=self.queryBox, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1019. sizer.Add((0, 5))
  1020. #
  1021. # length
  1022. #
  1023. self.queryLength = wx.RadioButton(parent=panel, id=wx.ID_ANY, label=_("length"))
  1024. self.queryLength.Bind(wx.EVT_RADIOBUTTON, self.OnChangeQuery)
  1025. sizer.Add(item=self.queryLength, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1026. flexSizer = wx.FlexGridSizer (cols=4, hgap=5, vgap=5)
  1027. flexSizer.AddGrowableCol(0)
  1028. txt = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Select lines"))
  1029. self.queryLengthSL = wx.Choice (parent=panel, id=wx.ID_ANY,
  1030. choices = [_("shorter than"), _("longer than")])
  1031. self.queryLengthSL.SetSelection(UserSettings.Get(group='vdigit', key="queryLength", subkey='than-selection'))
  1032. self.queryLengthValue = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(100, -1),
  1033. initial=1,
  1034. min=0, max=1e6)
  1035. self.queryLengthValue.SetValue(UserSettings.Get(group='vdigit', key="queryLength", subkey='thresh'))
  1036. units = wx.StaticText(parent=panel, id=wx.ID_ANY, label="%s" % LocUnits)
  1037. flexSizer.Add(txt, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1038. flexSizer.Add(self.queryLengthSL, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1039. flexSizer.Add(self.queryLengthValue, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1040. flexSizer.Add(units, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1041. sizer.Add(item=flexSizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1042. #
  1043. # dangle
  1044. #
  1045. self.queryDangle = wx.RadioButton(parent=panel, id=wx.ID_ANY, label=_("dangle"))
  1046. self.queryDangle.Bind(wx.EVT_RADIOBUTTON, self.OnChangeQuery)
  1047. sizer.Add(item=self.queryDangle, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1048. flexSizer = wx.FlexGridSizer (cols=4, hgap=5, vgap=5)
  1049. flexSizer.AddGrowableCol(0)
  1050. txt = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Select dangles"))
  1051. self.queryDangleSL = wx.Choice (parent=panel, id=wx.ID_ANY,
  1052. choices = [_("shorter than"), _("longer than")])
  1053. self.queryDangleSL.SetSelection(UserSettings.Get(group='vdigit', key="queryDangle", subkey='than-selection'))
  1054. self.queryDangleValue = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(100, -1),
  1055. initial=1,
  1056. min=0, max=1e6)
  1057. self.queryDangleValue.SetValue(UserSettings.Get(group='vdigit', key="queryDangle", subkey='thresh'))
  1058. units = wx.StaticText(parent=panel, id=wx.ID_ANY, label="%s" % LocUnits)
  1059. flexSizer.Add(txt, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1060. flexSizer.Add(self.queryDangleSL, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1061. flexSizer.Add(self.queryDangleValue, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1062. flexSizer.Add(units, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1063. sizer.Add(item=flexSizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1064. if UserSettings.Get(group='vdigit', key="query", subkey='selection') == 0:
  1065. self.queryLength.SetValue(True)
  1066. else:
  1067. self.queryDangle.SetValue(True)
  1068. # enable & disable items
  1069. self.OnChangeQuery(None)
  1070. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=5)
  1071. panel.SetSizer(border)
  1072. return panel
  1073. def __CreateAttributesPage(self, notebook):
  1074. """Create notebook page for query tool"""
  1075. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1076. notebook.AddPage(page=panel, text=_("Attributes"))
  1077. border = wx.BoxSizer(wx.VERTICAL)
  1078. #
  1079. # add new record
  1080. #
  1081. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Digitize new feature"))
  1082. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1083. # checkbox
  1084. self.addRecord = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1085. label=_("Add new record into table"))
  1086. self.addRecord.SetValue(UserSettings.Get(group='vdigit', key="addRecord", subkey='enabled'))
  1087. sizer.Add(item=self.addRecord, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1088. # settings
  1089. flexSizer = wx.FlexGridSizer(cols=2, hgap=3, vgap=3)
  1090. flexSizer.AddGrowableCol(0)
  1091. settings = ((_("Layer"), 1), (_("Category"), 1), (_("Mode"), _("Next to use")))
  1092. # layer
  1093. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Layer"))
  1094. if self.parent.digit.map:
  1095. layers = map(str, self.parent.digit.GetLayers())
  1096. if len(layers) == 0:
  1097. layers = [str(UserSettings.Get(group='vdigit', key="layer", subkey='value')), ]
  1098. else:
  1099. layers = [str(UserSettings.Get(group='vdigit', key="layer", subkey='value')), ]
  1100. self.layer = wx.Choice(parent=panel, id=wx.ID_ANY, size=(125, -1),
  1101. choices=layers)
  1102. self.layer.SetStringSelection(str(UserSettings.Get(group='vdigit', key="layer", subkey='value')))
  1103. flexSizer.Add(item=text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1104. flexSizer.Add(item=self.layer, proportion=0,
  1105. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1106. # category number
  1107. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Category number"))
  1108. self.category = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(125, -1),
  1109. initial=UserSettings.Get(group='vdigit', key="category", subkey='value'),
  1110. min=-1e9, max=1e9)
  1111. if UserSettings.Get(group='vdigit', key="categoryMode", subkey='selection') != 1:
  1112. self.category.Enable(False)
  1113. flexSizer.Add(item=text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1114. flexSizer.Add(item=self.category, proportion=0,
  1115. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1116. # category mode
  1117. text = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Category mode"))
  1118. self.categoryMode = wx.Choice(parent=panel, id=wx.ID_ANY, size=(125, -1),
  1119. choices=[_("Next to use"), _("Manual entry"), _("No category")])
  1120. self.categoryMode.SetSelection(UserSettings.Get(group='vdigit', key="categoryMode", subkey='selection'))
  1121. flexSizer.Add(item=text, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1122. flexSizer.Add(item=self.categoryMode, proportion=0,
  1123. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1124. sizer.Add(item=flexSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=1)
  1125. border.Add(item=sizer, proportion=0,
  1126. flag=wx.ALL | wx.EXPAND, border=5)
  1127. #
  1128. # digitize new area
  1129. #
  1130. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Digitize new area"))
  1131. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1132. # add centroid
  1133. self.addCentroid = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1134. label=_("Add centroid to left/right area"))
  1135. self.addCentroid.SetValue(UserSettings.Get(group='vdigit', key="addCentroid", subkey='enabled'))
  1136. sizer.Add(item=self.addCentroid, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1137. # attach category to boundary
  1138. self.catBoundary = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1139. label=_("Do not attach category to boundaries"))
  1140. self.catBoundary.SetValue(UserSettings.Get(group='vdigit', key="catBoundary", subkey='enabled'))
  1141. sizer.Add(item=self.catBoundary, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1142. border.Add(item=sizer, proportion=0,
  1143. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=5)
  1144. #
  1145. # delete existing record
  1146. #
  1147. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Delete existing feature(s)"))
  1148. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1149. # checkbox
  1150. self.deleteRecord = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1151. label=_("Delete record from table"))
  1152. self.deleteRecord.SetValue(UserSettings.Get(group='vdigit', key="delRecord", subkey='enabled'))
  1153. sizer.Add(item=self.deleteRecord, proportion=0, flag=wx.ALL | wx.EXPAND, border=1)
  1154. border.Add(item=sizer, proportion=0,
  1155. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=5)
  1156. # bindings
  1157. self.Bind(wx.EVT_CHECKBOX, self.OnChangeAddRecord, self.addRecord)
  1158. self.Bind(wx.EVT_CHOICE, self.OnChangeCategoryMode, self.categoryMode)
  1159. self.Bind(wx.EVT_CHOICE, self.OnChangeLayer, self.layer)
  1160. panel.SetSizer(border)
  1161. return panel
  1162. def __SymbologyData(self):
  1163. """
  1164. Data for __CreateSymbologyPage()
  1165. label | checkbox | color
  1166. """
  1167. return (
  1168. # ("Background", "symbolBackground"),
  1169. (_("Highlight"), "highlight"),
  1170. (_("Highlight (duplicates)"), "highlightDupl"),
  1171. (_("Point"), "point"),
  1172. (_("Line"), "line"),
  1173. (_("Boundary (no area)"), "boundaryNo"),
  1174. (_("Boundary (one area)"), "boundaryOne"),
  1175. (_("Boundary (two areas)"), "boundaryTwo"),
  1176. (_("Centroid (in area)"), "centroidIn"),
  1177. (_("Centroid (outside area)"), "centroidOut"),
  1178. (_("Centroid (duplicate in area)"), "centroidDup"),
  1179. (_("Node (one line)"), "nodeOne"),
  1180. (_("Node (two lines)"), "nodeTwo"),
  1181. (_("Vertex"), "vertex"),
  1182. (_("Area (closed boundary + centroid)"), "area"),
  1183. (_("Direction"), "direction"),)
  1184. def OnChangeCategoryMode(self, event):
  1185. """Change category mode"""
  1186. mode = event.GetSelection()
  1187. UserSettings.Set(group='vdigit', key="categoryMode", subkey='selection', value=mode)
  1188. if mode == 1: # manual entry
  1189. self.category.Enable(True)
  1190. elif self.category.IsEnabled(): # disable
  1191. self.category.Enable(False)
  1192. if mode == 2 and self.addRecord.IsChecked(): # no category
  1193. self.addRecord.SetValue(False)
  1194. self.parent.digit.SetCategory()
  1195. self.category.SetValue(UserSettings.Get(group='vdigit', key='category', subkey='value'))
  1196. def OnChangeLayer(self, event):
  1197. """Layer changed"""
  1198. layer = int(event.GetString())
  1199. if layer > 0:
  1200. UserSettings.Set(group='vdigit', key='layer', subkey='value', value=layer)
  1201. self.parent.digit.SetCategory()
  1202. self.category.SetValue(UserSettings.Get(group='vdigit', key='category', subkey='value'))
  1203. event.Skip()
  1204. def OnChangeAddRecord(self, event):
  1205. """Checkbox 'Add new record' status changed"""
  1206. self.category.SetValue(self.parent.digit.SetCategory())
  1207. def OnChangeSnappingValue(self, event):
  1208. """Change snapping value - update static text"""
  1209. value = self.snappingValue.GetValue()
  1210. if value < 0:
  1211. region = self.parent.MapWindow.Map.GetRegion()
  1212. res = (region['nsres'] + region['ewres']) / 2.
  1213. threshold = self.parent.digit.driver.GetThreshold(value=res)
  1214. else:
  1215. if self.snappingUnit.GetStringSelection() == "map units":
  1216. threshold = value
  1217. else:
  1218. threshold = self.parent.digit.driver.GetThreshold(value=value)
  1219. if value == 0:
  1220. self.snappingInfo.SetLabel(_("Snapping disabled"))
  1221. elif value < 0:
  1222. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s "
  1223. "(based on comp. resolution)") %
  1224. {'value' : threshold,
  1225. 'units' : self.mapUnits.lower()})
  1226. else:
  1227. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  1228. {'value' : threshold,
  1229. 'units' : self.mapUnits.lower()})
  1230. event.Skip()
  1231. def OnChangeSnappingUnits(self, event):
  1232. """Snapping units change -> update static text"""
  1233. value = self.snappingValue.GetValue()
  1234. units = self.snappingUnit.GetStringSelection()
  1235. threshold = self.parent.digit.driver.GetThreshold(value=value, units=units)
  1236. if units == "map units":
  1237. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  1238. {'value' : value,
  1239. 'units' : self.mapUnits})
  1240. else:
  1241. self.snappingInfo.SetLabel(_("Snapping threshold is %(value).1f %(units)s") %
  1242. {'value' : threshold,
  1243. 'units' : self.mapUnits})
  1244. event.Skip()
  1245. def OnChangeQuery(self, event):
  1246. """Change query"""
  1247. if self.queryLength.GetValue():
  1248. # length
  1249. self.queryLengthSL.Enable(True)
  1250. self.queryLengthValue.Enable(True)
  1251. self.queryDangleSL.Enable(False)
  1252. self.queryDangleValue.Enable(False)
  1253. else:
  1254. # dangle
  1255. self.queryLengthSL.Enable(False)
  1256. self.queryLengthValue.Enable(False)
  1257. self.queryDangleSL.Enable(True)
  1258. self.queryDangleValue.Enable(True)
  1259. def OnSave(self, event):
  1260. """Button 'Save' clicked"""
  1261. self.UpdateSettings()
  1262. self.parent.toolbars['vdigit'].settingsDialog = None
  1263. fileSettings = {}
  1264. UserSettings.ReadSettingsFile(settings=fileSettings)
  1265. fileSettings['vdigit'] = UserSettings.Get(group='vdigit')
  1266. file = UserSettings.SaveToFile(fileSettings)
  1267. self.parent.gismanager.goutput.WriteLog(_('Vector digitizer settings saved to file <%s>.') % file)
  1268. self.Destroy()
  1269. event.Skip()
  1270. def OnApply(self, event):
  1271. """Button 'Apply' clicked"""
  1272. self.UpdateSettings()
  1273. def OnCancel(self, event):
  1274. """Button 'Cancel' clicked"""
  1275. self.parent.toolbars['vdigit'].settingsDialog = None
  1276. self.Destroy()
  1277. if event:
  1278. event.Skip()
  1279. def UpdateSettings(self):
  1280. """Update UserSettings"""
  1281. # symbology
  1282. for key, (enabled, color) in self.symbology.iteritems():
  1283. if enabled:
  1284. UserSettings.Set(group='vdigit', key='symbol',
  1285. subkey=[key, 'enabled'],
  1286. value=enabled.IsChecked())
  1287. UserSettings.Set(group='vdigit', key='symbol',
  1288. subkey=[key, 'color'],
  1289. value=tuple(color.GetColour()))
  1290. else:
  1291. UserSettings.Set(group='vdigit', key='symbol',
  1292. subkey=[key, 'color'],
  1293. value=tuple(color.GetColour()))
  1294. # display
  1295. UserSettings.Set(group='vdigit', key="lineWidth", subkey='value',
  1296. value=int(self.lineWidthValue.GetValue()))
  1297. # snapping
  1298. UserSettings.Set(group='vdigit', key="snapping", subkey='value',
  1299. value=int(self.snappingValue.GetValue()))
  1300. UserSettings.Set(group='vdigit', key="snapping", subkey='units',
  1301. value=self.snappingUnit.GetStringSelection())
  1302. UserSettings.Set(group='vdigit', key="snapToVertex", subkey='enabled',
  1303. value=self.snapVertex.IsChecked())
  1304. # digitize new feature
  1305. UserSettings.Set(group='vdigit', key="addRecord", subkey='enabled',
  1306. value=self.addRecord.IsChecked())
  1307. UserSettings.Set(group='vdigit', key="layer", subkey='value',
  1308. value=int(self.layer.GetStringSelection()))
  1309. UserSettings.Set(group='vdigit', key="category", subkey='value',
  1310. value=int(self.category.GetValue()))
  1311. UserSettings.Set(group='vdigit', key="categoryMode", subkey='selection',
  1312. value=self.categoryMode.GetSelection())
  1313. # digitize new area
  1314. UserSettings.Set(group='vdigit', key="addCentroid", subkey='enabled',
  1315. value=self.addCentroid.IsChecked())
  1316. UserSettings.Set(group='vdigit', key="catBoundary", subkey='enabled',
  1317. value=self.catBoundary.IsChecked())
  1318. # delete existing feature
  1319. UserSettings.Set(group='vdigit', key="delRecord", subkey='enabled',
  1320. value=self.deleteRecord.IsChecked())
  1321. # snapping threshold
  1322. self.parent.digit.threshold = self.parent.digit.driver.GetThreshold()
  1323. # query tool
  1324. if self.queryLength.GetValue():
  1325. UserSettings.Set(group='vdigit', key="query", subkey='selection',
  1326. value=0)
  1327. else:
  1328. UserSettings.Set(group='vdigit', key="query", subkey='type',
  1329. value=1)
  1330. UserSettings.Set(group='vdigit', key="query", subkey='box',
  1331. value=self.queryBox.IsChecked())
  1332. UserSettings.Set(group='vdigit', key="queryLength", subkey='than-selection',
  1333. value=self.queryLengthSL.GetSelection())
  1334. UserSettings.Set(group='vdigit', key="queryLength", subkey='thresh',
  1335. value=int(self.queryLengthValue.GetValue()))
  1336. UserSettings.Set(group='vdigit', key="queryDangle", subkey='than-selection',
  1337. value=self.queryDangleSL.GetSelection())
  1338. UserSettings.Set(group='vdigit', key="queryDangle", subkey='thresh',
  1339. value=int(self.queryDangleValue.GetValue()))
  1340. # select features
  1341. for feature in ('point', 'line',
  1342. 'centroid', 'boundary'):
  1343. UserSettings.Set(group='vdigit', key='selectType',
  1344. subkey=[feature, 'enabled'],
  1345. value=self.FindWindowById(self.selectFeature[feature]).IsChecked())
  1346. UserSettings.Set(group='vdigit', key="selectThresh", subkey='value',
  1347. value=int(self.selectThreshValue.GetValue()))
  1348. UserSettings.Set(group='vdigit', key="checkForDupl", subkey='enabled',
  1349. value=self.checkForDupl.IsChecked())
  1350. UserSettings.Set(group='vdigit', key="selectInside", subkey='enabled',
  1351. value=self.selectIn.IsChecked())
  1352. # on-exit
  1353. UserSettings.Set(group='vdigit', key="saveOnExit", subkey='enabled',
  1354. value=self.save.IsChecked())
  1355. # break lines
  1356. UserSettings.Set(group='vdigit', key="breakLines", subkey='enabled',
  1357. value=self.intersect.IsChecked())
  1358. # update driver settings
  1359. self.parent.digit.driver.UpdateSettings()
  1360. # update digit settings
  1361. self.parent.digit.UpdateSettings()
  1362. # redraw map if auto-rendering is enabled
  1363. if self.parent.autoRender.GetValue():
  1364. self.parent.OnRender(None)
  1365. class VDigitCategoryDialog(wx.Dialog, listmix.ColumnSorterMixin):
  1366. """
  1367. Dialog used to display/modify categories of vector objects
  1368. @param parent
  1369. @param title dialog title
  1370. @param query {coordinates, qdist} - used by v.edit/v.what
  1371. @param cats directory of lines (layer/categories) - used by vdigit
  1372. @param pos
  1373. @param style
  1374. """
  1375. def __init__(self, parent, title,
  1376. map, query=None, cats=None,
  1377. pos=wx.DefaultPosition,
  1378. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER):
  1379. # parent
  1380. self.parent = parent # mapdisplay.BufferedWindow class instance
  1381. # map name
  1382. self.map = map
  1383. # line : {layer: [categories]}
  1384. self.cats = {}
  1385. # do not display dialog if no line is found (-> self.cats)
  1386. if cats is None:
  1387. if self.__GetCategories(query[0], query[1]) == 0 or not self.line:
  1388. Debug.msg(3, "VDigitCategoryDialog(): nothing found!")
  1389. else:
  1390. self.cats = cats
  1391. for line in cats.keys():
  1392. for layer in cats[line].keys():
  1393. self.cats[line][layer] = list(cats[line][layer])
  1394. layers = []
  1395. for layer in self.parent.parent.digit.GetLayers():
  1396. layers.append(str(layer))
  1397. # make copy of cats (used for 'reload')
  1398. self.cats_orig = copy.deepcopy(self.cats)
  1399. wx.Dialog.__init__(self, parent=self.parent, id=wx.ID_ANY, title=title,
  1400. style=style, pos=pos)
  1401. # list of categories
  1402. box = wx.StaticBox(parent=self, id=wx.ID_ANY,
  1403. label=" %s " % _("List of categories - right-click to delete"))
  1404. listSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1405. self.list = CategoryListCtrl(parent=self, id=wx.ID_ANY,
  1406. style=wx.LC_REPORT |
  1407. wx.BORDER_NONE |
  1408. wx.LC_SORT_ASCENDING |
  1409. wx.LC_HRULES |
  1410. wx.LC_VRULES)
  1411. # sorter
  1412. self.fid = self.cats.keys()[0]
  1413. self.itemDataMap = self.list.Populate(self.cats[self.fid])
  1414. listmix.ColumnSorterMixin.__init__(self, 2)
  1415. self.fidMulti = wx.Choice(parent=self, id=wx.ID_ANY,
  1416. size=(150, -1))
  1417. self.fidMulti.Bind(wx.EVT_CHOICE, self.OnFeature)
  1418. self.fidText = wx.StaticText(parent=self, id=wx.ID_ANY)
  1419. if len(self.cats.keys()) == 1:
  1420. self.fidMulti.Show(False)
  1421. self.fidText.SetLabel(str(self.fid))
  1422. else:
  1423. self.fidText.Show(False)
  1424. choices = []
  1425. for fid in self.cats.keys():
  1426. choices.append(str(fid))
  1427. self.fidMulti.SetItems(choices)
  1428. self.fidMulti.SetSelection(0)
  1429. listSizer.Add(item=self.list, proportion=1, flag=wx.EXPAND)
  1430. # add new category
  1431. box = wx.StaticBox(parent=self, id=wx.ID_ANY,
  1432. label=" %s " % _("Add new category"))
  1433. addSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1434. flexSizer = wx.FlexGridSizer (cols=5, hgap=5, vgap=5)
  1435. flexSizer.AddGrowableCol(3)
  1436. layerNewTxt = wx.StaticText(parent=self, id=wx.ID_ANY,
  1437. label="%s:" % _("Layer"))
  1438. self.layerNew = wx.Choice(parent=self, id=wx.ID_ANY, size=(75, -1),
  1439. choices=layers)
  1440. if len(layers) > 0:
  1441. self.layerNew.SetSelection(0)
  1442. catNewTxt = wx.StaticText(parent=self, id=wx.ID_ANY,
  1443. label="%s:" % _("Category"))
  1444. try:
  1445. newCat = max(self.cats[self.fid][1]) + 1
  1446. except KeyError:
  1447. newCat = 1
  1448. self.catNew = wx.SpinCtrl(parent=self, id=wx.ID_ANY, size=(75, -1),
  1449. initial=newCat, min=0, max=1e9)
  1450. btnAddCat = wx.Button(self, wx.ID_ADD)
  1451. flexSizer.Add(item=layerNewTxt, proportion=0,
  1452. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1453. flexSizer.Add(item=self.layerNew, proportion=0,
  1454. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1455. flexSizer.Add(item=catNewTxt, proportion=0,
  1456. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT | wx.LEFT,
  1457. border=10)
  1458. flexSizer.Add(item=self.catNew, proportion=0,
  1459. flag=wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  1460. flexSizer.Add(item=btnAddCat, proportion=0,
  1461. flag=wx.EXPAND | wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1462. addSizer.Add(item=flexSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1463. # buttons
  1464. btnApply = wx.Button(self, wx.ID_APPLY)
  1465. btnApply.SetToolTipString(_("Apply changes"))
  1466. btnCancel = wx.Button(self, wx.ID_CANCEL)
  1467. btnCancel.SetToolTipString(_("Ignore changes and close dialog"))
  1468. btnOk = wx.Button(self, wx.ID_OK)
  1469. btnOk.SetToolTipString(_("Apply changes and close dialog"))
  1470. btnOk.SetDefault()
  1471. # sizers
  1472. btnSizer = wx.StdDialogButtonSizer()
  1473. btnSizer.AddButton(btnCancel)
  1474. #btnSizer.AddButton(btnReload)
  1475. #btnSizer.SetNegativeButton(btnReload)
  1476. btnSizer.AddButton(btnApply)
  1477. btnSizer.AddButton(btnOk)
  1478. btnSizer.Realize()
  1479. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1480. mainSizer.Add(item=listSizer, proportion=1,
  1481. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  1482. mainSizer.Add(item=addSizer, proportion=0,
  1483. flag=wx.EXPAND | wx.ALIGN_CENTER |
  1484. wx.LEFT | wx.RIGHT | wx.BOTTOM, border=5)
  1485. fidSizer = wx.BoxSizer(wx.HORIZONTAL)
  1486. fidSizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY,
  1487. label=_("Feature id:")),
  1488. proportion=0, border=5,
  1489. flag=wx.ALIGN_CENTER_VERTICAL)
  1490. fidSizer.Add(item=self.fidMulti, proportion=0,
  1491. flag=wx.EXPAND | wx.ALL, border=5)
  1492. fidSizer.Add(item=self.fidText, proportion=0,
  1493. flag=wx.EXPAND | wx.ALL, border=5)
  1494. mainSizer.Add(item=fidSizer, proportion=0,
  1495. flag=wx.EXPAND | wx.ALL, border=5)
  1496. mainSizer.Add(item=btnSizer, proportion=0,
  1497. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  1498. self.SetSizer(mainSizer)
  1499. mainSizer.Fit(self)
  1500. self.SetAutoLayout(True)
  1501. # set min size for dialog
  1502. self.SetMinSize(self.GetBestSize())
  1503. # bindings
  1504. # buttons
  1505. #btnReload.Bind(wx.EVT_BUTTON, self.OnReload)
  1506. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  1507. btnOk.Bind(wx.EVT_BUTTON, self.OnOK)
  1508. btnAddCat.Bind(wx.EVT_BUTTON, self.OnAddCat)
  1509. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  1510. # list
  1511. # self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected, self.list)
  1512. # self.list.Bind(wx.EVT_RIGHT_DOWN, self.OnRightDown)
  1513. self.list.Bind(wx.EVT_COMMAND_RIGHT_CLICK, self.OnRightUp) #wxMSW
  1514. self.list.Bind(wx.EVT_RIGHT_UP, self.OnRightUp) #wxGTK
  1515. self.Bind(wx.EVT_LIST_BEGIN_LABEL_EDIT, self.OnBeginEdit, self.list)
  1516. self.Bind(wx.EVT_LIST_END_LABEL_EDIT, self.OnEndEdit, self.list)
  1517. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick, self.list)
  1518. def GetListCtrl(self):
  1519. """Used by ColumnSorterMixin"""
  1520. return self.list
  1521. def OnColClick(self, event):
  1522. """Click on column header (order by)"""
  1523. event.Skip()
  1524. def OnBeginEdit(self, event):
  1525. """Editing of item started"""
  1526. event.Allow()
  1527. def OnEndEdit(self, event):
  1528. """Finish editing of item"""
  1529. itemIndex = event.GetIndex()
  1530. layerOld = int (self.list.GetItem(itemIndex, 0).GetText())
  1531. catOld = int (self.list.GetItem(itemIndex, 1).GetText())
  1532. if event.GetColumn() == 0:
  1533. layerNew = int(event.GetLabel())
  1534. catNew = catOld
  1535. else:
  1536. layerNew = layerOld
  1537. catNew = int(event.GetLabel())
  1538. try:
  1539. if layerNew not in self.cats[self.fid].keys():
  1540. self.cats[self.fid][layerNew] = []
  1541. self.cats[self.fid][layerNew].append(catNew)
  1542. self.cats[self.fid][layerOld].remove(catOld)
  1543. except:
  1544. event.Veto()
  1545. self.list.SetStringItem(itemIndex, 0, str(layerNew))
  1546. self.list.SetStringItem(itemIndex, 1, str(catNew))
  1547. dlg = wx.MessageDialog(self, _("Unable to add new layer/category <%(layer)s/%(category)s>.\n"
  1548. "Layer and category number must be integer.\n"
  1549. "Layer number must be greater then zero.") %
  1550. { 'layer': self.layerNew.GetStringSelection(),
  1551. 'category' : str(self.catNew.GetValue()) },
  1552. _("Error"), wx.OK | wx.ICON_ERROR)
  1553. dlg.ShowModal()
  1554. dlg.Destroy()
  1555. return False
  1556. def OnRightDown(self, event):
  1557. """Mouse right button down"""
  1558. x = event.GetX()
  1559. y = event.GetY()
  1560. item, flags = self.list.HitTest((x, y))
  1561. if item != wx.NOT_FOUND and \
  1562. flags & wx.LIST_HITTEST_ONITEM:
  1563. self.list.Select(item)
  1564. event.Skip()
  1565. def OnRightUp(self, event):
  1566. """Mouse right button up"""
  1567. if not hasattr(self, "popupID1"):
  1568. self.popupID1 = wx.NewId()
  1569. self.popupID2 = wx.NewId()
  1570. self.popupID3 = wx.NewId()
  1571. self.Bind(wx.EVT_MENU, self.OnItemDelete, id=self.popupID1)
  1572. self.Bind(wx.EVT_MENU, self.OnItemDeleteAll, id=self.popupID2)
  1573. self.Bind(wx.EVT_MENU, self.OnReload, id=self.popupID3)
  1574. # generate popup-menu
  1575. menu = wx.Menu()
  1576. menu.Append(self.popupID1, _("Delete selected"))
  1577. if self.list.GetFirstSelected() == -1:
  1578. menu.Enable(self.popupID1, False)
  1579. menu.Append(self.popupID2, _("Delete all"))
  1580. menu.AppendSeparator()
  1581. menu.Append(self.popupID3, _("Reload"))
  1582. self.PopupMenu(menu)
  1583. menu.Destroy()
  1584. def OnItemSelected(self, event):
  1585. """Item selected"""
  1586. event.Skip()
  1587. def OnItemDelete(self, event):
  1588. """Delete selected item(s) from the list (layer/category pair)"""
  1589. item = self.list.GetFirstSelected()
  1590. while item != -1:
  1591. layer = int (self.list.GetItem(item, 0).GetText())
  1592. cat = int (self.list.GetItem(item, 1).GetText())
  1593. self.list.DeleteItem(item)
  1594. self.cats[self.fid][layer].remove(cat)
  1595. item = self.list.GetFirstSelected()
  1596. event.Skip()
  1597. def OnItemDeleteAll(self, event):
  1598. """Delete all items from the list"""
  1599. self.list.DeleteAllItems()
  1600. self.cats[self.fid] = {}
  1601. event.Skip()
  1602. def OnFeature(self, event):
  1603. """Feature id changed (on duplicates)"""
  1604. self.fid = int(event.GetString())
  1605. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1606. update=True)
  1607. try:
  1608. newCat = max(self.cats[self.fid][1]) + 1
  1609. except KeyError:
  1610. newCat = 1
  1611. self.catNew.SetValue(newCat)
  1612. event.Skip()
  1613. def __GetCategories(self, coords, qdist):
  1614. """Get layer/category pairs for all available
  1615. layers
  1616. Return True line found or False if not found"""
  1617. ret = gcmd.RunCommand('v.what',
  1618. parent = self,
  1619. quiet = True,
  1620. map = self.map,
  1621. east_north = '%f,%f' % \
  1622. (float(coords[0]), float(coords[1])),
  1623. distance = qdist)
  1624. if not ret:
  1625. return False
  1626. for item in ret.splitlines():
  1627. litem = item.lower()
  1628. if "id:" in litem: # get line id
  1629. self.line = int(item.split(':')[1].strip())
  1630. elif "layer:" in litem: # add layer
  1631. layer = int(item.split(':')[1].strip())
  1632. if layer not in self.cats.keys():
  1633. self.cats[layer] = []
  1634. elif "category:" in litem: # add category
  1635. self.cats[layer].append(int(item.split(':')[1].strip()))
  1636. return True
  1637. def OnReload(self, event):
  1638. """Reload button pressed"""
  1639. # restore original list
  1640. self.cats = copy.deepcopy(self.cats_orig)
  1641. # polulate list
  1642. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1643. update=True)
  1644. event.Skip()
  1645. def OnCancel(self, event):
  1646. """Cancel button pressed"""
  1647. self.parent.parent.dialogs['category'] = None
  1648. if self.parent.parent.digit:
  1649. self.parent.parent.digit.driver.SetSelected([])
  1650. self.parent.UpdateMap(render=False)
  1651. else:
  1652. self.parent.parent.OnRender(None)
  1653. self.Close()
  1654. def OnApply(self, event):
  1655. """Apply button pressed"""
  1656. for fid in self.cats.keys():
  1657. newfid = self.ApplyChanges(fid)
  1658. if fid == self.fid:
  1659. self.fid = newfid
  1660. def ApplyChanges(self, fid):
  1661. cats = self.cats[fid]
  1662. cats_orig = self.cats_orig[fid]
  1663. # action : (catsFrom, catsTo)
  1664. check = {'catadd': (cats, cats_orig),
  1665. 'catdel': (cats_orig, cats)}
  1666. newfid = -1
  1667. # add/delete new category
  1668. for action, catsCurr in check.iteritems():
  1669. for layer in catsCurr[0].keys():
  1670. catList = []
  1671. for cat in catsCurr[0][layer]:
  1672. if layer not in catsCurr[1].keys() or \
  1673. cat not in catsCurr[1][layer]:
  1674. catList.append(cat)
  1675. if catList != []:
  1676. if action == 'catadd':
  1677. add = True
  1678. else:
  1679. add = False
  1680. newfid = self.parent.parent.digit.SetLineCats(fid, layer,
  1681. catList, add)
  1682. if len(self.cats.keys()) == 1:
  1683. self.fidText.SetLabel("%d" % newfid)
  1684. else:
  1685. choices = self.fidMulti.GetItems()
  1686. choices[choices.index(str(fid))] = str(newfid)
  1687. self.fidMulti.SetItems(choices)
  1688. self.fidMulti.SetStringSelection(str(newfid))
  1689. self.cats[newfid] = self.cats[fid]
  1690. del self.cats[fid]
  1691. fid = newfid
  1692. if self.fid < 0:
  1693. wx.MessageBox(parent=self, message=_("Unable to update vector map."),
  1694. caption=_("Error"), style=wx.OK | wx.ICON_ERROR)
  1695. self.cats_orig[fid] = copy.deepcopy(cats)
  1696. return newfid
  1697. def OnOK(self, event):
  1698. """OK button pressed"""
  1699. self.OnApply(event)
  1700. self.OnCancel(event)
  1701. def OnAddCat(self, event):
  1702. """Button 'Add' new category pressed"""
  1703. try:
  1704. layer = int(self.layerNew.GetStringSelection())
  1705. cat = int(self.catNew.GetValue())
  1706. if layer <= 0:
  1707. raise ValueError
  1708. except ValueError:
  1709. dlg = wx.MessageDialog(self, _("Unable to add new layer/category <%(layer)s/%(category)s>.\n"
  1710. "Layer and category number must be integer.\n"
  1711. "Layer number must be greater then zero.") %
  1712. {'layer' : str(self.layerNew.GetValue()),
  1713. 'category' : str(self.catNew.GetValue())},
  1714. _("Error"), wx.OK | wx.ICON_ERROR)
  1715. dlg.ShowModal()
  1716. dlg.Destroy()
  1717. return False
  1718. if layer not in self.cats[self.fid].keys():
  1719. self.cats[self.fid][layer] = []
  1720. self.cats[self.fid][layer].append(cat)
  1721. # reload list
  1722. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1723. update=True)
  1724. # update category number for add
  1725. self.catNew.SetValue(cat + 1)
  1726. event.Skip()
  1727. return True
  1728. def GetLine(self):
  1729. """Get id of selected line of 'None' if no line is selected"""
  1730. return self.cats.keys()
  1731. def UpdateDialog(self, query=None, cats=None):
  1732. """Update dialog
  1733. @param query {coordinates, distance} - v.edit/v.what
  1734. @param cats directory layer/cats - vdigit
  1735. Return True if updated otherwise False
  1736. """
  1737. # line: {layer: [categories]}
  1738. self.cats = {}
  1739. # do not display dialog if no line is found (-> self.cats)
  1740. if cats is None:
  1741. ret = self.__GetCategories(query[0], query[1])
  1742. else:
  1743. self.cats = cats
  1744. for line in cats.keys():
  1745. for layer in cats[line].keys():
  1746. self.cats[line][layer] = list(cats[line][layer])
  1747. ret = 1
  1748. if ret == 0 or len(self.cats.keys()) < 1:
  1749. Debug.msg(3, "VDigitCategoryDialog(): nothing found!")
  1750. return False
  1751. # make copy of cats (used for 'reload')
  1752. self.cats_orig = copy.deepcopy(self.cats)
  1753. # polulate list
  1754. self.fid = self.cats.keys()[0]
  1755. self.itemDataMap = self.list.Populate(self.cats[self.fid],
  1756. update=True)
  1757. try:
  1758. newCat = max(self.cats[self.fid][1]) + 1
  1759. except KeyError:
  1760. newCat = 1
  1761. self.catNew.SetValue(newCat)
  1762. if len(self.cats.keys()) == 1:
  1763. self.fidText.Show(True)
  1764. self.fidMulti.Show(False)
  1765. self.fidText.SetLabel("%d" % self.fid)
  1766. else:
  1767. self.fidText.Show(False)
  1768. self.fidMulti.Show(True)
  1769. choices = []
  1770. for fid in self.cats.keys():
  1771. choices.append(str(fid))
  1772. self.fidMulti.SetItems(choices)
  1773. self.fidMulti.SetSelection(0)
  1774. self.Layout()
  1775. return True
  1776. class CategoryListCtrl(wx.ListCtrl,
  1777. listmix.ListCtrlAutoWidthMixin,
  1778. listmix.TextEditMixin):
  1779. """List of layers/categories"""
  1780. def __init__(self, parent, id, pos=wx.DefaultPosition,
  1781. size=wx.DefaultSize, style=0):
  1782. self.parent = parent
  1783. wx.ListCtrl.__init__(self, parent, id, pos, size, style)
  1784. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1785. listmix.TextEditMixin.__init__(self)
  1786. def Populate(self, cats, update=False):
  1787. """Populate the list"""
  1788. itemData = {} # requested by sorter
  1789. if not update:
  1790. self.InsertColumn(0, _("Layer"))
  1791. self.InsertColumn(1, _("Category"))
  1792. else:
  1793. self.DeleteAllItems()
  1794. i = 1
  1795. for layer in cats.keys():
  1796. catsList = cats[layer]
  1797. for cat in catsList:
  1798. index = self.InsertStringItem(sys.maxint, str(catsList[0]))
  1799. self.SetStringItem(index, 0, str(layer))
  1800. self.SetStringItem(index, 1, str(cat))
  1801. self.SetItemData(index, i)
  1802. itemData[i] = (str(layer), str(cat))
  1803. i = i + 1
  1804. if not update:
  1805. self.SetColumnWidth(0, 100)
  1806. self.SetColumnWidth(1, wx.LIST_AUTOSIZE)
  1807. self.currentItem = 0
  1808. return itemData
  1809. class VDigitZBulkDialog(wx.Dialog):
  1810. """
  1811. Dialog used for Z bulk-labeling tool
  1812. """
  1813. def __init__(self, parent, title, nselected, style=wx.DEFAULT_DIALOG_STYLE):
  1814. wx.Dialog.__init__(self, parent=parent, id=wx.ID_ANY, title=title, style=style)
  1815. self.parent = parent # mapdisplay.BufferedWindow class instance
  1816. # panel = wx.Panel(parent=self, id=wx.ID_ANY)
  1817. border = wx.BoxSizer(wx.VERTICAL)
  1818. txt = wx.StaticText(parent=self,
  1819. label=_("%d lines selected for z bulk-labeling") % nselected);
  1820. border.Add(item=txt, proportion=0, flag=wx.ALL | wx.EXPAND, border=5)
  1821. box = wx.StaticBox (parent=self, id=wx.ID_ANY, label=" %s " % _("Set value"))
  1822. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1823. flexSizer = wx.FlexGridSizer (cols=2, hgap=5, vgap=5)
  1824. flexSizer.AddGrowableCol(0)
  1825. # starting value
  1826. txt = wx.StaticText(parent=self,
  1827. label=_("Starting value"));
  1828. self.value = wx.SpinCtrl(parent=self, id=wx.ID_ANY, size=(150, -1),
  1829. initial=0,
  1830. min=-1e6, max=1e6)
  1831. flexSizer.Add(txt, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1832. flexSizer.Add(self.value, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1833. # step
  1834. txt = wx.StaticText(parent=self,
  1835. label=_("Step"))
  1836. self.step = wx.SpinCtrl(parent=self, id=wx.ID_ANY, size=(150, -1),
  1837. initial=0,
  1838. min=0, max=1e6)
  1839. flexSizer.Add(txt, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1840. flexSizer.Add(self.step, proportion=0, flag=wx.ALIGN_CENTER | wx.FIXED_MINSIZE)
  1841. sizer.Add(item=flexSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=1)
  1842. border.Add(item=sizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1843. # buttons
  1844. btnCancel = wx.Button(self, wx.ID_CANCEL)
  1845. btnOk = wx.Button(self, wx.ID_OK)
  1846. btnOk.SetDefault()
  1847. # sizers
  1848. btnSizer = wx.StdDialogButtonSizer()
  1849. btnSizer.AddButton(btnCancel)
  1850. btnSizer.AddButton(btnOk)
  1851. btnSizer.Realize()
  1852. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1853. mainSizer.Add(item=border, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  1854. mainSizer.Add(item=btnSizer, proportion=0,
  1855. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  1856. self.SetSizer(mainSizer)
  1857. mainSizer.Fit(self)
  1858. class VDigitDuplicatesDialog(wx.Dialog):
  1859. """
  1860. Show duplicated feature ids
  1861. """
  1862. def __init__(self, parent, data, title=_("List of duplicates"),
  1863. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  1864. pos=wx.DefaultPosition):
  1865. wx.Dialog.__init__(self, parent=parent, id=wx.ID_ANY, title=title, style=style,
  1866. pos=pos)
  1867. self.parent = parent # BufferedWindow
  1868. self.data = data
  1869. self.winList = []
  1870. # panel = wx.Panel(parent=self, id=wx.ID_ANY)
  1871. # notebook
  1872. self.notebook = wx.Notebook(parent=self, id=wx.ID_ANY, style=wx.BK_DEFAULT)
  1873. id = 1
  1874. for key in self.data.keys():
  1875. panel = wx.Panel(parent=self.notebook, id=wx.ID_ANY)
  1876. self.notebook.AddPage(page=panel, text=" %d " % (id))
  1877. # notebook body
  1878. border = wx.BoxSizer(wx.VERTICAL)
  1879. win = CheckListFeature(parent=panel, data=list(self.data[key]))
  1880. self.winList.append(win.GetId())
  1881. border.Add(item=win, proportion=1,
  1882. flag=wx.ALL | wx.EXPAND, border=5)
  1883. panel.SetSizer(border)
  1884. id += 1
  1885. # buttons
  1886. btnCancel = wx.Button(self, wx.ID_CANCEL)
  1887. btnOk = wx.Button(self, wx.ID_OK)
  1888. btnOk.SetDefault()
  1889. # sizers
  1890. btnSizer = wx.StdDialogButtonSizer()
  1891. btnSizer.AddButton(btnCancel)
  1892. btnSizer.AddButton(btnOk)
  1893. btnSizer.Realize()
  1894. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1895. mainSizer.Add(item=self.notebook, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  1896. mainSizer.Add(item=btnSizer, proportion=0,
  1897. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  1898. self.SetSizer(mainSizer)
  1899. mainSizer.Fit(self)
  1900. self.SetAutoLayout(True)
  1901. # set min size for dialog
  1902. self.SetMinSize((250, 180))
  1903. def GetUnSelected(self):
  1904. """Get unselected items (feature id)
  1905. @return list of ids
  1906. """
  1907. ids = []
  1908. for id in self.winList:
  1909. wlist = self.FindWindowById(id)
  1910. for item in range(wlist.GetItemCount()):
  1911. if not wlist.IsChecked(item):
  1912. ids.append(int(wlist.GetItem(item, 0).GetText()))
  1913. return ids
  1914. class CheckListFeature(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin, listmix.CheckListCtrlMixin):
  1915. """List of mapset/owner/group"""
  1916. def __init__(self, parent, data,
  1917. pos=wx.DefaultPosition, log=None):
  1918. self.parent = parent
  1919. self.data = data
  1920. wx.ListCtrl.__init__(self, parent, wx.ID_ANY,
  1921. style=wx.LC_REPORT)
  1922. listmix.CheckListCtrlMixin.__init__(self)
  1923. self.log = log
  1924. # setup mixins
  1925. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1926. self.LoadData(self.data)
  1927. def LoadData(self, data):
  1928. """Load data into list"""
  1929. self.InsertColumn(0, _('Feature id'))
  1930. self.InsertColumn(1, _('Layer (Categories)'))
  1931. for item in data:
  1932. index = self.InsertStringItem(sys.maxint, str(item[0]))
  1933. self.SetStringItem(index, 1, str(item[1]))
  1934. # enable all items by default
  1935. for item in range(self.GetItemCount()):
  1936. self.CheckItem(item, True)
  1937. self.SetColumnWidth(col=0, width=wx.LIST_AUTOSIZE_USEHEADER)
  1938. self.SetColumnWidth(col=1, width=wx.LIST_AUTOSIZE_USEHEADER)
  1939. def OnCheckItem(self, index, flag):
  1940. """Mapset checked/unchecked"""
  1941. pass