vdigit.py 90 KB

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