vdigit.py 112 KB

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