manager.py 105 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840
  1. """!
  2. @package gcp.manager
  3. @brief Georectification module for GRASS GIS. Includes ground control
  4. point management and interactive point and click GCP creation
  5. Classes:
  6. - manager::GCPWizard
  7. - manager::LocationPage
  8. - manager::GroupPage
  9. - manager::DispMapPage
  10. - manager::GCP
  11. - manager::GCPList
  12. - manager::VectGroup
  13. - manager::EditGCP
  14. - manager::GrSettingsDialog
  15. (C) 2006-2012 by the GRASS Development Team
  16. This program is free software under the GNU General Public License
  17. (>=v2). Read the file COPYING that comes with GRASS for details.
  18. @author Original author Michael Barton
  19. @author Original version improved by Martin Landa <landa.martin gmail.com>
  20. @author Rewritten by Markus Metz redesign georectfier -> GCP Manage
  21. @author Support for GraphicsSet added by Stepan Turek <stepan.turek seznam.cz> (2012)
  22. """
  23. import os
  24. import sys
  25. import shutil
  26. import time
  27. from copy import copy
  28. import wx
  29. from wx.lib.mixins.listctrl import CheckListCtrlMixin, ColumnSorterMixin, ListCtrlAutoWidthMixin
  30. import wx.lib.colourselect as csel
  31. import wx.wizard as wiz
  32. import grass.script as grass
  33. from core import globalvar
  34. from core import utils
  35. from core.render import Map
  36. from gui_core.gselect import Select, LocationSelect, MapsetSelect
  37. from gui_core.dialogs import GroupDialog
  38. from core.gcmd import RunCommand, GMessage, GError, GWarning
  39. from core.settings import UserSettings
  40. from gcp.mapdisplay import MapFrame
  41. from location_wizard.wizard import TitledPage as TitledPage
  42. #
  43. # global variables
  44. #
  45. global src_map
  46. global tgt_map
  47. global maptype
  48. src_map = ''
  49. tgt_map = { 'raster' : '',
  50. 'vector' : '' }
  51. maptype = 'raster'
  52. def getSmallUpArrowImage():
  53. stream = open(os.path.join(globalvar.ETCIMGDIR, 'small_up_arrow.png'), 'rb')
  54. try:
  55. img = wx.ImageFromStream(stream)
  56. finally:
  57. stream.close()
  58. return img
  59. def getSmallDnArrowImage():
  60. stream = open(os.path.join(globalvar.ETCIMGDIR, 'small_down_arrow.png'), 'rb')
  61. try:
  62. img = wx.ImageFromStream(stream)
  63. finally:
  64. stream.close()
  65. stream.close()
  66. return img
  67. class GCPWizard(object):
  68. """
  69. Start wizard here and finish wizard here
  70. """
  71. def __init__(self, parent):
  72. self.parent = parent # GMFrame
  73. #
  74. # get environmental variables
  75. #
  76. self.grassdatabase = grass.gisenv()['GISDBASE']
  77. #
  78. # read original environment settings
  79. #
  80. self.target_gisrc = os.environ['GISRC']
  81. self.gisrc_dict = {}
  82. try:
  83. f = open(self.target_gisrc, 'r')
  84. for line in f.readlines():
  85. line = line.replace('\n', '').strip()
  86. if len(line) < 1:
  87. continue
  88. key, value = line.split(':', 1)
  89. self.gisrc_dict[key.strip()] = value.strip()
  90. finally:
  91. f.close()
  92. self.currentlocation = self.gisrc_dict['LOCATION_NAME']
  93. self.currentmapset = self.gisrc_dict['MAPSET']
  94. # location for xy map to georectify
  95. self.newlocation = ''
  96. # mapset for xy map to georectify
  97. self.newmapset = ''
  98. global maptype
  99. global src_map
  100. global tgt_map
  101. #src_map = ''
  102. #tgt_map = ''
  103. maptype = 'raster'
  104. # GISRC file for source location/mapset of map(s) to georectify
  105. self.source_gisrc = ''
  106. self.src_maps = []
  107. #
  108. # define wizard pages
  109. #
  110. self.wizard = wiz.Wizard(parent=parent, id=wx.ID_ANY, title=_("Setup for georectification"))
  111. self.startpage = LocationPage(self.wizard, self)
  112. self.grouppage = GroupPage(self.wizard, self)
  113. self.mappage = DispMapPage(self.wizard, self)
  114. #
  115. # set the initial order of the pages
  116. #
  117. self.startpage.SetNext(self.grouppage)
  118. self.grouppage.SetPrev(self.startpage)
  119. self.grouppage.SetNext(self.mappage)
  120. self.mappage.SetPrev(self.grouppage)
  121. #
  122. # do pages layout
  123. #
  124. self.startpage.DoLayout()
  125. self.grouppage.DoLayout()
  126. self.mappage.DoLayout()
  127. self.wizard.FitToPage(self.startpage)
  128. # self.Bind(wx.EVT_CLOSE, self.Cleanup)
  129. # self.parent.Bind(wx.EVT_ACTIVATE, self.OnGLMFocus)
  130. success = False
  131. #
  132. # run wizard
  133. #
  134. if self.wizard.RunWizard(self.startpage):
  135. success = self.OnWizFinished()
  136. if success == False:
  137. GMessage(parent = self.parent,
  138. message = _("Georectifying setup canceled."))
  139. self.Cleanup()
  140. else:
  141. GMessage(parent = self.parent,
  142. message = _("Georectifying setup canceled."))
  143. self.Cleanup()
  144. #
  145. # start GCP display
  146. #
  147. if success != False:
  148. # instance of render.Map to be associated with display
  149. self.SwitchEnv('source')
  150. self.SrcMap = Map(gisrc=self.source_gisrc)
  151. self.SwitchEnv('target')
  152. self.TgtMap = Map(gisrc=self.target_gisrc)
  153. self.Map = self.SrcMap
  154. #
  155. # add layer to source map
  156. #
  157. if maptype == 'raster':
  158. rendertype = 'raster'
  159. cmdlist = ['d.rast', 'map=%s' % src_map]
  160. else: # -> vector layer
  161. rendertype = 'vector'
  162. cmdlist = ['d.vect', 'map=%s' % src_map]
  163. self.SwitchEnv('source')
  164. name, found = utils.GetLayerNameFromCmd(cmdlist)
  165. self.SrcMap.AddLayer(type=rendertype, command=cmdlist, l_active=True,
  166. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  167. self.SwitchEnv('target')
  168. if tgt_map['raster']:
  169. #
  170. # add raster layer to target map
  171. #
  172. rendertype = 'raster'
  173. cmdlist = ['d.rast', 'map=%s' % tgt_map['raster']]
  174. name, found = utils.GetLayerNameFromCmd(cmdlist)
  175. self.TgtMap.AddLayer(type=rendertype, command=cmdlist, l_active=True,
  176. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  177. if tgt_map['vector']:
  178. #
  179. # add raster layer to target map
  180. #
  181. rendertype = 'vector'
  182. cmdlist = ['d.vect', 'map=%s' % tgt_map['vector']]
  183. name, found = utils.GetLayerNameFromCmd(cmdlist)
  184. self.TgtMap.AddLayer(type=rendertype, command=cmdlist, l_active=True,
  185. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  186. #
  187. # start GCP Manager
  188. #
  189. self.gcpmgr = GCP(self.parent, grwiz=self, size=globalvar.MAP_WINDOW_SIZE,
  190. toolbars=["gcpdisp"],
  191. Map=self.SrcMap, lmgr=self.parent)
  192. # load GCPs
  193. self.gcpmgr.InitMapDisplay()
  194. self.gcpmgr.CenterOnScreen()
  195. self.gcpmgr.Show()
  196. # need to update AUI here for wingrass
  197. self.gcpmgr._mgr.Update()
  198. else:
  199. self.Cleanup()
  200. def SetSrcEnv(self, location, mapset):
  201. """!Create environment to use for location and mapset
  202. that are the source of the file(s) to georectify
  203. @param location source location
  204. @param mapset source mapset
  205. @return False on error
  206. @return True on success
  207. """
  208. self.newlocation = location
  209. self.newmapset = mapset
  210. # check to see if we are georectifying map in current working location/mapset
  211. if self.newlocation == self.currentlocation and self.newmapset == self.currentmapset:
  212. return False
  213. self.gisrc_dict['LOCATION_NAME'] = location
  214. self.gisrc_dict['MAPSET'] = mapset
  215. self.source_gisrc = utils.GetTempfile()
  216. try:
  217. f = open(self.source_gisrc, mode='w')
  218. for line in self.gisrc_dict.items():
  219. f.write(line[0] + ": " + line[1] + "\n")
  220. finally:
  221. f.close()
  222. return True
  223. def SwitchEnv(self, grc):
  224. """
  225. Switches between original working location/mapset and
  226. location/mapset that is source of file(s) to georectify
  227. """
  228. # check to see if we are georectifying map in current working location/mapset
  229. if self.newlocation == self.currentlocation and self.newmapset == self.currentmapset:
  230. return False
  231. if grc == 'target':
  232. os.environ['GISRC'] = str(self.target_gisrc)
  233. elif grc == 'source':
  234. os.environ['GISRC'] = str(self.source_gisrc)
  235. return True
  236. def OnWizFinished(self):
  237. # self.Cleanup()
  238. return True
  239. def OnGLMFocus(self, event):
  240. """!Layer Manager focus"""
  241. # self.SwitchEnv('target')
  242. event.Skip()
  243. def Cleanup(self):
  244. """!Return to current location and mapset"""
  245. self.SwitchEnv('target')
  246. self.parent.gcpmanagement = None
  247. self.wizard.Destroy()
  248. class LocationPage(TitledPage):
  249. """
  250. Set map type (raster or vector) to georectify and
  251. select location/mapset of map(s) to georectify.
  252. """
  253. def __init__(self, wizard, parent):
  254. TitledPage.__init__(self, wizard, _("Select map type and location/mapset"))
  255. self.parent = parent
  256. self.grassdatabase = self.parent.grassdatabase
  257. self.xylocation = ''
  258. self.xymapset = ''
  259. #
  260. # layout
  261. #
  262. self.sizer.AddGrowableCol(2)
  263. # map type
  264. self.rb_maptype = wx.RadioBox(parent=self, id=wx.ID_ANY,
  265. label=' %s ' % _("Map type to georectify"),
  266. choices=[_('raster'), _('vector')],
  267. majorDimension=wx.RA_SPECIFY_COLS)
  268. self.sizer.Add(item=self.rb_maptype,
  269. flag=wx.ALIGN_CENTER | wx.ALL | wx.EXPAND, border=5,
  270. pos=(1, 1), span=(1, 2))
  271. # location
  272. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select source location:')),
  273. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  274. pos=(2, 1))
  275. self.cb_location = LocationSelect(parent = self, gisdbase = self.grassdatabase)
  276. self.sizer.Add(item=self.cb_location,
  277. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  278. pos=(2, 2))
  279. # mapset
  280. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select source mapset:')),
  281. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  282. pos=(3, 1))
  283. self.cb_mapset = MapsetSelect(parent = self, gisdbase = self.grassdatabase,
  284. setItems = False)
  285. self.sizer.Add(item=self.cb_mapset,
  286. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  287. pos=(3,2))
  288. #
  289. # bindings
  290. #
  291. self.Bind(wx.EVT_RADIOBOX, self.OnMaptype, self.rb_maptype)
  292. self.Bind(wx.EVT_COMBOBOX, self.OnLocation, self.cb_location)
  293. self.Bind(wx.EVT_COMBOBOX, self.OnMapset, self.cb_mapset)
  294. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  295. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  296. # self.Bind(wx.EVT_CLOSE, self.parent.Cleanup)
  297. def OnMaptype(self,event):
  298. """!Change map type"""
  299. global maptype
  300. if event.GetInt() == 0:
  301. maptype = 'raster'
  302. else:
  303. maptype = 'vector'
  304. def OnLocation(self, event):
  305. """!Sets source location for map(s) to georectify"""
  306. self.xylocation = event.GetString()
  307. #create a list of valid mapsets
  308. tmplist = os.listdir(os.path.join(self.grassdatabase, self.xylocation))
  309. self.mapsetList = []
  310. for item in tmplist:
  311. if os.path.isdir(os.path.join(self.grassdatabase, self.xylocation, item)) and \
  312. os.path.exists(os.path.join(self.grassdatabase, self.xylocation, item, 'WIND')):
  313. if item != 'PERMANENT':
  314. self.mapsetList.append(item)
  315. self.xymapset = 'PERMANENT'
  316. utils.ListSortLower(self.mapsetList)
  317. self.mapsetList.insert(0, 'PERMANENT')
  318. self.cb_mapset.SetItems(self.mapsetList)
  319. self.cb_mapset.SetStringSelection(self.xymapset)
  320. if not wx.FindWindowById(wx.ID_FORWARD).IsEnabled():
  321. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  322. def OnMapset(self, event):
  323. """!Sets source mapset for map(s) to georectify"""
  324. if self.xylocation == '':
  325. GMessage(_('You must select a valid location '
  326. 'before selecting a mapset'),
  327. parent = self)
  328. return
  329. self.xymapset = event.GetString()
  330. if not wx.FindWindowById(wx.ID_FORWARD).IsEnabled():
  331. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  332. def OnPageChanging(self, event=None):
  333. if event.GetDirection() and \
  334. (self.xylocation == '' or self.xymapset == ''):
  335. GMessage(_('You must select a valid location '
  336. 'and mapset in order to continue'),
  337. parent = self)
  338. event.Veto()
  339. return
  340. self.parent.SetSrcEnv(self.xylocation, self.xymapset)
  341. def OnEnterPage(self, event=None):
  342. if self.xylocation == '' or self.xymapset == '':
  343. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  344. else:
  345. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  346. class GroupPage(TitledPage):
  347. """
  348. Set group to georectify. Create group if desired.
  349. """
  350. def __init__(self, wizard, parent):
  351. TitledPage.__init__(self, wizard, _("Select image/map group to georectify"))
  352. self.parent = parent
  353. self.grassdatabase = self.parent.grassdatabase
  354. self.groupList = []
  355. self.xylocation = ''
  356. self.xymapset = ''
  357. self.xygroup = ''
  358. # default extension
  359. self.extension = '_georect' + str(os.getpid())
  360. #
  361. # layout
  362. #
  363. self.sizer.AddGrowableCol(2)
  364. # group
  365. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select group:')),
  366. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  367. pos=(1, 1))
  368. self.cb_group = wx.ComboBox(parent=self, id=wx.ID_ANY,
  369. choices=self.groupList, size=(350, -1),
  370. style=wx.CB_DROPDOWN | wx.CB_READONLY)
  371. self.sizer.Add(item=self.cb_group,
  372. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  373. pos=(1, 2))
  374. # create group
  375. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Create group if none exists')),
  376. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  377. pos=(2, 1))
  378. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  379. self.btn_mkgroup = wx.Button(parent=self, id=wx.ID_ANY, label=_("Create/edit group..."))
  380. self.btn_vgroup = wx.Button(parent=self, id=wx.ID_ANY, label=_("Add vector map to group..."))
  381. btnSizer.Add(item=self.btn_mkgroup,
  382. flag=wx.RIGHT, border=5)
  383. btnSizer.Add(item=self.btn_vgroup,
  384. flag=wx.LEFT, border=5)
  385. self.sizer.Add(item=btnSizer,
  386. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  387. pos=(2, 2))
  388. # extension
  389. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Extension for output maps:')),
  390. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  391. pos=(3, 1))
  392. self.ext_txt = wx.TextCtrl(parent=self, id=wx.ID_ANY, value="", size=(350,-1))
  393. self.ext_txt.SetValue(self.extension)
  394. self.sizer.Add(item=self.ext_txt,
  395. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  396. pos=(3, 2))
  397. #
  398. # bindings
  399. #
  400. self.Bind(wx.EVT_COMBOBOX, self.OnGroup, self.cb_group)
  401. self.Bind(wx.EVT_TEXT, self.OnExtension, self.ext_txt)
  402. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  403. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  404. self.Bind(wx.EVT_CLOSE, self.parent.Cleanup)
  405. # hide vector group button by default
  406. self.btn_vgroup.Hide()
  407. def OnGroup(self, event):
  408. self.xygroup = event.GetString()
  409. def OnMkGroup(self, event):
  410. """!Create new group in source location/mapset"""
  411. dlg = GroupDialog(parent = self, defaultGroup = self.xygroup)
  412. dlg.ShowModal()
  413. gr = dlg.GetSelectedGroup()
  414. if gr in dlg.GetExistGroups():
  415. self.xygroup = gr
  416. else:
  417. gr = ''
  418. dlg.Destroy()
  419. self.OnEnterPage()
  420. self.Update()
  421. def OnVGroup(self, event):
  422. """!Add vector maps to group"""
  423. dlg = VectGroup(parent = self,
  424. id = wx.ID_ANY,
  425. grassdb = self.grassdatabase,
  426. location = self.xylocation,
  427. mapset = self.xymapset,
  428. group = self.xygroup)
  429. if dlg.ShowModal() != wx.ID_OK:
  430. return
  431. dlg.MakeVGroup()
  432. self.OnEnterPage()
  433. def OnExtension(self, event):
  434. self.extension = event.GetString()
  435. def OnPageChanging(self, event=None):
  436. if event.GetDirection() and self.xygroup == '':
  437. GMessage(_('You must select a valid image/map '
  438. 'group in order to continue'),
  439. parent = self)
  440. event.Veto()
  441. return
  442. if event.GetDirection() and self.extension == '':
  443. GMessage(_('You must enter an map name '
  444. 'extension in order to continue'),
  445. parent = self)
  446. event.Veto()
  447. return
  448. def OnEnterPage(self, event=None):
  449. global maptype
  450. self.groupList = []
  451. self.xylocation = self.parent.gisrc_dict['LOCATION_NAME']
  452. self.xymapset = self.parent.gisrc_dict['MAPSET']
  453. # create a list of groups in selected mapset
  454. if os.path.isdir(os.path.join(self.grassdatabase,
  455. self.xylocation,
  456. self.xymapset,
  457. 'group')):
  458. tmplist = os.listdir(os.path.join(self.grassdatabase,
  459. self.xylocation,
  460. self.xymapset,
  461. 'group'))
  462. for item in tmplist:
  463. if os.path.isdir(os.path.join(self.grassdatabase,
  464. self.xylocation,
  465. self.xymapset,
  466. 'group',
  467. item)):
  468. self.groupList.append(item)
  469. if maptype == 'raster':
  470. self.btn_vgroup.Hide()
  471. self.Bind(wx.EVT_BUTTON, self.OnMkGroup, self.btn_mkgroup)
  472. elif maptype == 'vector':
  473. self.btn_vgroup.Show()
  474. self.Bind(wx.EVT_BUTTON, self.OnMkGroup, self.btn_mkgroup)
  475. self.Bind(wx.EVT_BUTTON, self.OnVGroup, self.btn_vgroup)
  476. utils.ListSortLower(self.groupList)
  477. self.cb_group.SetItems(self.groupList)
  478. if len(self.groupList) > 0:
  479. if self.xygroup and self.xygroup in self.groupList:
  480. self.cb_group.SetStringSelection(self.xygroup)
  481. else:
  482. self.cb_group.SetSelection(0)
  483. self.xygroup = self.groupList[0]
  484. if self.xygroup == '' or \
  485. self.extension == '':
  486. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  487. else:
  488. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  489. # switch to source
  490. self.parent.SwitchEnv('source')
  491. class DispMapPage(TitledPage):
  492. """
  493. Select ungeoreferenced map to display for interactively
  494. setting ground control points (GCPs).
  495. """
  496. def __init__(self, wizard, parent):
  497. TitledPage.__init__(self, wizard,
  498. _("Select maps to display for ground control point (GCP) creation"))
  499. self.parent = parent
  500. global maptype
  501. #
  502. # layout
  503. #
  504. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select source map to display:')),
  505. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  506. pos=(1, 1))
  507. self.srcselection = Select(self, id=wx.ID_ANY,
  508. size=globalvar.DIALOG_GSELECT_SIZE, type=maptype, updateOnPopup = False)
  509. self.sizer.Add(item=self.srcselection,
  510. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  511. pos=(1, 2))
  512. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select target raster map to display:')),
  513. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  514. pos=(2, 1))
  515. self.tgtrastselection = Select(self, id=wx.ID_ANY,
  516. size=globalvar.DIALOG_GSELECT_SIZE, type='raster', updateOnPopup = False)
  517. self.sizer.Add(item=self.tgtrastselection,
  518. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  519. pos=(2, 2))
  520. self.sizer.Add(item=wx.StaticText(parent=self, id=wx.ID_ANY, label=_('Select target vector map to display:')),
  521. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  522. pos=(3, 1))
  523. self.tgtvectselection = Select(self, id=wx.ID_ANY,
  524. size=globalvar.DIALOG_GSELECT_SIZE, type='vector', updateOnPopup = False)
  525. self.sizer.Add(item=self.tgtvectselection,
  526. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5,
  527. pos=(3, 2))
  528. #
  529. # bindings
  530. #
  531. self.srcselection.Bind(wx.EVT_TEXT, self.OnSrcSelection)
  532. self.tgtrastselection.Bind(wx.EVT_TEXT, self.OnTgtRastSelection)
  533. self.tgtvectselection.Bind(wx.EVT_TEXT, self.OnTgtVectSelection)
  534. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  535. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  536. self.Bind(wx.EVT_CLOSE, self.parent.Cleanup)
  537. def OnSrcSelection(self,event):
  538. """!Source map to display selected"""
  539. global src_map
  540. global maptype
  541. src_map = event.GetString()
  542. if src_map == '':
  543. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  544. else:
  545. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  546. try:
  547. # set computational region to match selected map and zoom display to region
  548. if maptype == 'raster':
  549. p = RunCommand('g.region', 'rast=src_map')
  550. elif maptype == 'vector':
  551. p = RunCommand('g.region', 'vect=src_map')
  552. if p.returncode == 0:
  553. print 'returncode = ', str(p.returncode)
  554. self.parent.Map.region = self.parent.Map.GetRegion()
  555. except:
  556. pass
  557. def OnTgtRastSelection(self,event):
  558. """!Source map to display selected"""
  559. global tgt_map
  560. tgt_map['raster'] = event.GetString()
  561. def OnTgtVectSelection(self,event):
  562. """!Source map to display selected"""
  563. global tgt_map
  564. tgt_map['vector'] = event.GetString()
  565. def OnPageChanging(self, event=None):
  566. global src_map
  567. global tgt_map
  568. if event.GetDirection() and (src_map == ''):
  569. GMessage(_('You must select a source map '
  570. 'in order to continue'),
  571. parent = self)
  572. event.Veto()
  573. return
  574. self.parent.SwitchEnv('target')
  575. def OnEnterPage(self, event=None):
  576. global maptype
  577. global src_map
  578. global tgt_map
  579. self.srcselection.SetElementList(maptype)
  580. if maptype == 'raster':
  581. ret = RunCommand('i.group',
  582. parent = self,
  583. read = True,
  584. group = self.parent.grouppage.xygroup,
  585. flags = 'g')
  586. if ret:
  587. self.parent.src_maps = ret.splitlines()
  588. else:
  589. GError(parent = self,
  590. message = _('No maps in selected group <%s>.\n'
  591. 'Please edit group or select another group.') %
  592. self.parent.grouppage.xygroup)
  593. return
  594. elif maptype == 'vector':
  595. grassdatabase = self.parent.grassdatabase
  596. xylocation = self.parent.gisrc_dict['LOCATION_NAME']
  597. xymapset = self.parent.gisrc_dict['MAPSET']
  598. # make list of vectors to georectify from VREF
  599. vgrpfile = os.path.join(grassdatabase,
  600. xylocation,
  601. xymapset,
  602. 'group',
  603. self.parent.grouppage.xygroup,
  604. 'VREF')
  605. f = open(vgrpfile)
  606. try:
  607. for vect in f.readlines():
  608. vect = vect.strip('\n')
  609. if len(vect) < 1:
  610. continue
  611. self.parent.src_maps.append(vect)
  612. finally:
  613. f.close()
  614. if len(self.parent.src_maps) < 1:
  615. GError(parent = self,
  616. message = _('No maps in selected group <%s>.\n'
  617. 'Please edit group or select another group.') %
  618. self.parent.grouppage.xygroup)
  619. return
  620. # filter out all maps not in group
  621. self.srcselection.tcp.GetElementList(elements = self.parent.src_maps)
  622. src_map = self.parent.src_maps[0]
  623. self.srcselection.SetValue(src_map)
  624. self.parent.SwitchEnv('target')
  625. self.tgtrastselection.SetElementList('raster')
  626. self.tgtrastselection.GetElementList()
  627. self.tgtvectselection.SetElementList('vector')
  628. self.tgtvectselection.GetElementList()
  629. self.parent.SwitchEnv('source')
  630. if src_map == '':
  631. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  632. else:
  633. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  634. class GCP(MapFrame, ColumnSorterMixin):
  635. """!
  636. Manages ground control points for georectifying. Calculates RMS statistics.
  637. Calls i.rectify or v.rectify to georectify map.
  638. """
  639. def __init__(self, parent, grwiz = None, id = wx.ID_ANY,
  640. title = _("Manage Ground Control Points"),
  641. size = (700, 300), toolbars = ["gcpdisp"], Map = None, lmgr = None):
  642. self.grwiz = grwiz # GR Wizard
  643. if tgt_map['raster'] == '' and tgt_map['vector'] == '':
  644. self.show_target = False
  645. else:
  646. self.show_target = True
  647. #wx.Frame.__init__(self, parent, id, title, size = size, name = "GCPFrame")
  648. MapFrame.__init__(self, parent = parent, title = title, size = size,
  649. Map = Map, toolbars = toolbars, lmgr = lmgr, name = 'GCPMapWindow')
  650. #
  651. # init variables
  652. #
  653. self.parent = parent # GMFrame
  654. self.parent.gcpmanagement = self
  655. #
  656. # register data structures for drawing GCP's
  657. #
  658. self.pointsToDrawTgt = self.TgtMapWindow.RegisterGraphicsToDraw(graphicsType = "point", setStatusFunc = self.SetGCPSatus)
  659. self.pointsToDrawSrc = self.SrcMapWindow.RegisterGraphicsToDraw(graphicsType = "point", setStatusFunc = self.SetGCPSatus)
  660. # window resized
  661. self.resize = False
  662. self.grassdatabase = self.grwiz.grassdatabase
  663. self.currentlocation = self.grwiz.currentlocation
  664. self.currentmapset = self.grwiz.currentmapset
  665. self.newlocation = self.grwiz.newlocation
  666. self.newmapset = self.grwiz.newmapset
  667. self.xylocation = self.grwiz.gisrc_dict['LOCATION_NAME']
  668. self.xymapset = self.grwiz.gisrc_dict['MAPSET']
  669. self.xygroup = self.grwiz.grouppage.xygroup
  670. self.src_maps = self.grwiz.src_maps
  671. self.extension = self.grwiz.grouppage.extension
  672. self.outname = ''
  673. self.VectGRList = []
  674. self.file = {
  675. 'points' : os.path.join(self.grassdatabase,
  676. self.xylocation,
  677. self.xymapset,
  678. 'group',
  679. self.xygroup,
  680. 'POINTS'),
  681. 'points_bak' : os.path.join(self.grassdatabase,
  682. self.xylocation,
  683. self.xymapset,
  684. 'group',
  685. self.xygroup,
  686. 'POINTS_BAK'),
  687. 'rgrp' : os.path.join(self.grassdatabase,
  688. self.xylocation,
  689. self.xymapset,
  690. 'group',
  691. self.xygroup,
  692. 'REF'),
  693. 'vgrp' : os.path.join(self.grassdatabase,
  694. self.xylocation,
  695. self.xymapset,
  696. 'group',
  697. self.xygroup,
  698. 'VREF'),
  699. 'target' : os.path.join(self.grassdatabase,
  700. self.xylocation,
  701. self.xymapset,
  702. 'group',
  703. self.xygroup,
  704. 'TARGET'),
  705. }
  706. # make a backup of the current points file
  707. if os.path.exists(self.file['points']):
  708. shutil.copy(self.file['points'], self.file['points_bak'])
  709. # polynomial order transformation for georectification
  710. self.gr_order = 1
  711. # interpolation method for georectification
  712. self.gr_method = 'nearest'
  713. # region clipping for georectified map
  714. self.clip_to_region = False
  715. # number of GCPs selected to be used for georectification (checked)
  716. self.GCPcount = 0
  717. # forward RMS error
  718. self.fwd_rmserror = 0.0
  719. # backward RMS error
  720. self.bkw_rmserror = 0.0
  721. # list map coords and ID of map display they came from
  722. self.mapcoordlist = []
  723. self.mapcoordlist.append([ 0, # GCP number
  724. 0.0, # source east
  725. 0.0, # source north
  726. 0.0, # target east
  727. 0.0, # target north
  728. 0.0, # forward error
  729. 0.0 ] ) # backward error
  730. # init vars to highlight high RMS errors
  731. self.highest_only = True
  732. self.show_unused = True
  733. self.highest_key = -1
  734. self.rmsthresh = 0
  735. self.rmsmean = 0
  736. self.rmssd = 0
  737. self.SetTarget(self.xygroup, self.currentlocation, self.currentmapset)
  738. self.itemDataMap = None
  739. # images for column sorting
  740. # CheckListCtrlMixin must set an ImageList first
  741. self.il = self.list.GetImageList(wx.IMAGE_LIST_SMALL)
  742. SmallUpArrow = wx.BitmapFromImage(getSmallUpArrowImage())
  743. SmallDnArrow = wx.BitmapFromImage(getSmallDnArrowImage())
  744. self.sm_dn = self.il.Add(SmallDnArrow)
  745. self.sm_up = self.il.Add(SmallUpArrow)
  746. # set mouse characteristics
  747. self.mapwin = self.SrcMapWindow
  748. self.mapwin.mouse['box'] = 'point'
  749. self.mapwin.mouse["use"] == "pointer"
  750. self.mapwin.zoomtype = 0
  751. self.mapwin.pen = wx.Pen(colour='black', width=2, style=wx.SOLID)
  752. self.mapwin.SetCursor(self.cursors["cross"])
  753. self.mapwin = self.TgtMapWindow
  754. # set mouse characteristics
  755. self.mapwin.mouse['box'] = 'point'
  756. self.mapwin.mouse["use"] == "pointer"
  757. self.mapwin.zoomtype = 0
  758. self.mapwin.pen = wx.Pen(colour='black', width=2, style=wx.SOLID)
  759. self.mapwin.SetCursor(self.cursors["cross"])
  760. #
  761. # show new display & draw map
  762. #
  763. if self.show_target:
  764. self.MapWindow = self.TgtMapWindow
  765. self.Map = self.TgtMap
  766. self.OnZoomToMap(None)
  767. self.MapWindow = self.SrcMapWindow
  768. self.Map = self.SrcMap
  769. self.OnZoomToMap(None)
  770. #
  771. # bindings
  772. #
  773. self.Bind(wx.EVT_ACTIVATE, self.OnFocus)
  774. self.Bind(wx.EVT_SIZE, self.OnSize)
  775. self.Bind(wx.EVT_IDLE, self.OnIdle)
  776. self.Bind(wx.EVT_CLOSE, self.OnQuit)
  777. self.SetSettings()
  778. def __del__(self):
  779. """!Disable GCP manager mode"""
  780. self.parent.gcpmanagement = None
  781. def CreateGCPList(self):
  782. """!Create GCP List Control"""
  783. return GCPList(parent=self, gcp=self)
  784. # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
  785. def GetListCtrl(self):
  786. return self.list
  787. def GetMapCoordList(self):
  788. return self.mapcoordlist
  789. # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
  790. def GetSortImages(self):
  791. return (self.sm_dn, self.sm_up)
  792. def GetFwdError(self):
  793. return self.fwd_rmserror
  794. def GetBkwError(self):
  795. return self.bkw_rmserror
  796. def InitMapDisplay(self):
  797. self.list.LoadData()
  798. # initialize column sorter
  799. self.itemDataMap = self.mapcoordlist
  800. ncols = self.list.GetColumnCount()
  801. ColumnSorterMixin.__init__(self, ncols)
  802. # init to ascending sort on first click
  803. self._colSortFlag = [1] * ncols
  804. def SetTarget(self, tgroup, tlocation, tmapset):
  805. """
  806. Sets rectification target to current location and mapset
  807. """
  808. # check to see if we are georectifying map in current working location/mapset
  809. if self.newlocation == self.currentlocation and self.newmapset == self.currentmapset:
  810. RunCommand('i.target',
  811. parent = self,
  812. flags = 'c',
  813. group = tgroup)
  814. else:
  815. self.grwiz.SwitchEnv('source')
  816. RunCommand('i.target',
  817. parent = self,
  818. group = tgroup,
  819. location = tlocation,
  820. mapset = tmapset)
  821. self.grwiz.SwitchEnv('target')
  822. def AddGCP(self, event):
  823. """
  824. Appends an item to GCP list
  825. """
  826. keyval = self.list.AddGCPItem() + 1
  827. # source east, source north, target east, target north, forward error, backward error
  828. self.mapcoordlist.append([ keyval, # GCP number
  829. 0.0, # source east
  830. 0.0, # source north
  831. 0.0, # target east
  832. 0.0, # target north
  833. 0.0, # forward error
  834. 0.0 ] ) # backward error
  835. if self.statusbarManager.GetMode() == 8: # go to
  836. self.StatusbarUpdate()
  837. def DeleteGCP(self, event):
  838. """
  839. Deletes selected item in GCP list
  840. """
  841. minNumOfItems = self.OnGROrder(None)
  842. if self.list.GetItemCount() <= minNumOfItems:
  843. GMessage(parent = self,
  844. message=_("At least %d GCPs required. Operation canceled.") % minNumOfItems)
  845. return
  846. key = self.list.DeleteGCPItem()
  847. del self.mapcoordlist[key]
  848. # update key and GCP number
  849. for newkey in range(key, len(self.mapcoordlist)):
  850. index = self.list.FindItemData(-1, newkey + 1)
  851. self.mapcoordlist[newkey][0] = newkey
  852. self.list.SetStringItem(index, 0, str(newkey))
  853. self.list.SetItemData(index, newkey)
  854. # update selected
  855. if self.list.GetItemCount() > 0:
  856. if self.list.selected < self.list.GetItemCount():
  857. self.list.selectedkey = self.list.GetItemData(self.list.selected)
  858. else:
  859. self.list.selected = self.list.GetItemCount() - 1
  860. self.list.selectedkey = self.list.GetItemData(self.list.selected)
  861. self.list.SetItemState(self.list.selected,
  862. wx.LIST_STATE_SELECTED,
  863. wx.LIST_STATE_SELECTED)
  864. else:
  865. self.list.selected = wx.NOT_FOUND
  866. self.list.selectedkey = -1
  867. self.UpdateColours()
  868. if self.statusbarManager.GetMode() == 8: # go to
  869. self.StatusbarUpdate()
  870. if self.list.selectedkey > 0:
  871. self.statusbarManager.SetProperty('gotoGCP', self.list.selectedkey)
  872. def ClearGCP(self, event):
  873. """
  874. Clears all values in selected item of GCP list and unchecks it
  875. """
  876. index = self.list.GetSelected()
  877. key = self.list.GetItemData(index)
  878. for i in range(1, 5):
  879. self.list.SetStringItem(index, i, '0.0')
  880. self.list.SetStringItem(index, 5, '')
  881. self.list.SetStringItem(index, 6, '')
  882. self.list.CheckItem(index, False)
  883. # GCP number, source E, source N, target E, target N, fwd error, bkwd error
  884. self.mapcoordlist[key] = [key, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]
  885. def SetSettings(self):
  886. """!Sets settings for drawing of GCP's.
  887. """
  888. self.highest_only = UserSettings.Get(group='gcpman', key='rms', subkey='highestonly')
  889. self.show_unused = UserSettings.Get(group='gcpman', key='symbol', subkey='unused')
  890. colours = { "color" : "default",
  891. "hcolor" : "highest",
  892. "scolor" : "selected",
  893. "ucolor" : "unused" }
  894. wpx = UserSettings.Get(group = 'gcpman', key = 'symbol', subkey = 'width')
  895. for k, v in colours.iteritems():
  896. col = UserSettings.Get(group='gcpman', key='symbol', subkey= k)
  897. self.pointsToDrawSrc.GetPen(v).SetColour(wx.Colour(col[0], col[1], col[2], 255)) # TODO GetPen neni to spatne?
  898. self.pointsToDrawTgt.GetPen(v).SetColour(wx.Colour(col[0], col[1], col[2], 255))
  899. self.pointsToDrawSrc.GetPen(v).SetWidth(wpx)
  900. self.pointsToDrawTgt.GetPen(v).SetWidth(wpx)
  901. spx = UserSettings.Get(group = 'gcpman', key = 'symbol', subkey = 'size')
  902. self.pointsToDrawSrc.SetPropertyVal("size", int(spx))
  903. self.pointsToDrawTgt.SetPropertyVal("size", int(spx))
  904. font = self.GetFont()
  905. font.SetPointSize(int(spx) + 2)
  906. textProp = {}
  907. textProp['active'] = True
  908. textProp['font'] = font
  909. self.pointsToDrawSrc.SetPropertyVal("text", textProp)
  910. self.pointsToDrawTgt.SetPropertyVal("text", copy(textProp))
  911. def SetGCPSatus(self, item, itemIndex):
  912. """!Before GCP is drawn, decides it's colour and whether it
  913. will be drawed.
  914. """
  915. key = self.list.GetItemData(itemIndex)
  916. itemIndex += 1 # incremented because of itemDataMap (has one more item) - will be changed
  917. if not self.list.IsChecked(key - 1):
  918. wxPen = "unused"
  919. if not self.show_unused:
  920. item.SetPropertyVal('hide', True)
  921. else:
  922. item.SetPropertyVal('hide', False)
  923. else:
  924. item.SetPropertyVal('hide', False)
  925. if self.highest_only == True:
  926. if itemIndex == self.highest_key:
  927. wxPen = "highest"
  928. else:
  929. wxPen = "default"
  930. else:
  931. if (gcp[5] > self.rmsthresh):
  932. wxPen = "highest"
  933. else:
  934. wxPen = "default"
  935. if itemIndex == self.list.selectedkey:
  936. wxPen = "selected"
  937. item.SetPropertyVal('label', str(itemIndex))
  938. item.SetPropertyVal('penName', wxPen)
  939. def SetGCPData(self, coordtype, coord, mapdisp=None, confirm=False):
  940. """!Inserts coordinates from file, mouse click on map, or
  941. after editing into selected item of GCP list and checks it for
  942. use.
  943. """
  944. index = self.list.GetSelected()
  945. if index == wx.NOT_FOUND:
  946. return
  947. coord0 = coord[0]
  948. coord1 = coord[1]
  949. key = self.list.GetItemData(index)
  950. if confirm:
  951. if self.MapWindow == self.SrcMapWindow:
  952. currloc = _("source")
  953. else:
  954. currloc = _("target")
  955. ret = wx.MessageBox(parent = self,
  956. caption = _("Set GCP coordinates"),
  957. message = _('Set %(coor)s coordinates for GCP No. %(key)s? \n\n'
  958. 'East: %(coor0)s \n'
  959. 'North: %(coor1)s') % \
  960. { 'coor' : currloc,
  961. 'key' : str(key),
  962. 'coor0' : str(coord0),
  963. 'coor1' : str(coord1) },
  964. style = wx.ICON_QUESTION | wx.YES_NO | wx.CENTRE)
  965. # for wingrass
  966. if os.name == 'nt':
  967. self.MapWindow.SetFocus()
  968. if ret == wx.NO:
  969. return
  970. if coordtype == 'source':
  971. self.list.SetStringItem(index, 1, str(coord0))
  972. self.list.SetStringItem(index, 2, str(coord1))
  973. self.mapcoordlist[key][1] = coord[0]
  974. self.mapcoordlist[key][2] = coord[1]
  975. self.pointsToDrawSrc.GetItem(key - 1).SetCoords([coord0, coord1])
  976. elif coordtype == 'target':
  977. self.list.SetStringItem(index, 3, str(coord0))
  978. self.list.SetStringItem(index, 4, str(coord1))
  979. self.mapcoordlist[key][3] = coord[0]
  980. self.mapcoordlist[key][4] = coord[1]
  981. self.pointsToDrawTgt.GetItem(key - 1).SetCoords([coord0, coord1])
  982. self.list.SetStringItem(index, 5, '0')
  983. self.list.SetStringItem(index, 6, '0')
  984. self.mapcoordlist[key][5] = 0.0
  985. self.mapcoordlist[key][6] = 0.0
  986. # self.list.ResizeColumns()
  987. def SaveGCPs(self, event):
  988. """!Make a POINTS file or save GCP coordinates to existing
  989. POINTS file
  990. """
  991. self.GCPcount = 0
  992. try:
  993. f = open(self.file['points'], mode='w')
  994. # use os.linesep or '\n' here ???
  995. f.write('# Ground Control Points File\n')
  996. f.write("# \n")
  997. f.write("# target location: " + self.currentlocation + '\n')
  998. f.write("# target mapset: " + self.currentmapset + '\n')
  999. f.write("#\tsource\t\ttarget\t\tstatus\n")
  1000. f.write("#\teast\tnorth\teast\tnorth\t(1=ok, 0=ignore)\n")
  1001. f.write("#----------------------- ----------------------- ---------------\n")
  1002. for index in range(self.list.GetItemCount()):
  1003. if self.list.IsChecked(index) == True:
  1004. check = "1"
  1005. self.GCPcount += 1
  1006. else:
  1007. check = "0"
  1008. coord0 = self.list.GetItem(index, 1).GetText()
  1009. coord1 = self.list.GetItem(index, 2).GetText()
  1010. coord2 = self.list.GetItem(index, 3).GetText()
  1011. coord3 = self.list.GetItem(index, 4).GetText()
  1012. f.write(coord0 + ' ' + coord1 + ' ' + coord2 + ' ' + coord3 + ' ' + check + '\n')
  1013. except IOError, err:
  1014. GError(parent = self,
  1015. message="%s <%s>. %s%s" % (_("Writing POINTS file failed"),
  1016. self.file['points'], os.linesep, err))
  1017. return
  1018. f.close()
  1019. # if event != None save also to backup file
  1020. if event:
  1021. shutil.copy(self.file['points'], self.file['points_bak'])
  1022. self.parent.goutput.WriteLog(_('POINTS file saved for group <%s>') % self.xygroup)
  1023. #self.SetStatusText(_('POINTS file saved'))
  1024. def ReadGCPs(self):
  1025. """
  1026. Reads GCPs and georectified coordinates from POINTS file
  1027. """
  1028. self.GCPcount = 0
  1029. sourceMapWin = self.SrcMapWindow
  1030. targetMapWin = self.TgtMapWindow
  1031. if not sourceMapWin:
  1032. GError(parent = self,
  1033. message = "%s. %s%s" % (_("source mapwin not defined"),
  1034. os.linesep, err))
  1035. if not targetMapWin:
  1036. GError(parent = self,
  1037. message="%s. %s%s" % (_("target mapwin not defined"),
  1038. os.linesep, err))
  1039. try:
  1040. f = open(self.file['points'], 'r')
  1041. GCPcnt = 0
  1042. for line in f.readlines():
  1043. if line[0] == '#' or line =='':
  1044. continue
  1045. line = line.replace('\n', '').strip()
  1046. coords = map(float, line.split())
  1047. if coords[4] == 1:
  1048. check = True
  1049. self.GCPcount +=1
  1050. else:
  1051. check = False
  1052. self.AddGCP(event=None)
  1053. self.SetGCPData('source', (coords[0], coords[1]), sourceMapWin)
  1054. self.SetGCPData('target', (coords[2], coords[3]), targetMapWin)
  1055. index = self.list.GetSelected()
  1056. if index != wx.NOT_FOUND:
  1057. self.list.CheckItem(index, check)
  1058. GCPcnt += 1
  1059. except IOError, err:
  1060. GError(parent = self,
  1061. message = "%s <%s>. %s%s" % (_("Reading POINTS file failed"),
  1062. self.file['points'], os.linesep, err))
  1063. return
  1064. f.close()
  1065. if GCPcnt == 0:
  1066. # 3 gcp is minimum
  1067. for i in range(3):
  1068. self.AddGCP(None)
  1069. if self.CheckGCPcount():
  1070. # calculate RMS
  1071. self.RMSError(self.xygroup, self.gr_order)
  1072. def ReloadGCPs(self, event):
  1073. """!Reload data from file"""
  1074. # use backup
  1075. shutil.copy(self.file['points_bak'], self.file['points'])
  1076. # delete all items in mapcoordlist
  1077. self.mapcoordlist = []
  1078. self.mapcoordlist.append([ 0, # GCP number
  1079. 0.0, # source east
  1080. 0.0, # source north
  1081. 0.0, # target east
  1082. 0.0, # target north
  1083. 0.0, # forward error
  1084. 0.0 ] ) # backward error
  1085. self.list.LoadData()
  1086. self.itemDataMap = self.mapcoordlist
  1087. if self._col != -1:
  1088. self.list.ClearColumnImage(self._col)
  1089. self._colSortFlag = [1] * self.list.GetColumnCount()
  1090. # draw GCPs (source and target)
  1091. sourceMapWin = self.SrcMapWindow
  1092. sourceMapWin.UpdateMap(render=False, renderVector=False)
  1093. if self.show_target:
  1094. targetMapWin = self.TgtMapWindow
  1095. targetMapWin.UpdateMap(render=False, renderVector=False)
  1096. def OnFocus(self, event):
  1097. # self.grwiz.SwitchEnv('source')
  1098. pass
  1099. def OnRMS(self, event):
  1100. """
  1101. RMS button handler
  1102. """
  1103. self.RMSError(self.xygroup,self.gr_order)
  1104. sourceMapWin = self.SrcMapWindow
  1105. sourceMapWin.UpdateMap(render=False, renderVector=False)
  1106. if self.show_target:
  1107. targetMapWin = self.TgtMapWindow
  1108. targetMapWin.UpdateMap(render=False, renderVector=False)
  1109. def CheckGCPcount(self, msg=False):
  1110. """
  1111. Checks to make sure that the minimum number of GCPs have been defined and
  1112. are active for the selected transformation order
  1113. """
  1114. if (self.GCPcount < 3 and self.gr_order == 1) or \
  1115. (self.GCPcount < 6 and self.gr_order == 2) or \
  1116. (self.GCPcount < 10 and self.gr_order == 3):
  1117. if msg:
  1118. GWarning(parent = self,
  1119. message=_('Insufficient points defined and active (checked) '
  1120. 'for selected rectification method.\n'
  1121. '3+ points needed for 1st order,\n'
  1122. '6+ points for 2nd order, and\n'
  1123. '10+ points for 3rd order.'))
  1124. return False
  1125. else:
  1126. return True
  1127. def OnGeorect(self, event):
  1128. """
  1129. Georectifies map(s) in group using i.rectify or v.transform
  1130. """
  1131. global maptype
  1132. self.SaveGCPs(None)
  1133. if self.CheckGCPcount(msg=True) == False:
  1134. return
  1135. if maptype == 'raster':
  1136. self.grwiz.SwitchEnv('source')
  1137. if self.clip_to_region:
  1138. flags = "ac"
  1139. else:
  1140. flags = "a"
  1141. busy = wx.BusyInfo(message=_("Rectifying images, please wait..."),
  1142. parent=self)
  1143. wx.Yield()
  1144. ret, msg = RunCommand('i.rectify',
  1145. parent = self,
  1146. getErrorMsg = True,
  1147. quiet = True,
  1148. group = self.xygroup,
  1149. extension = self.extension,
  1150. order = self.gr_order,
  1151. method=self.gr_method,
  1152. flags = flags)
  1153. busy.Destroy()
  1154. # provide feedback on failure
  1155. if ret != 0:
  1156. print >> sys.stderr, msg
  1157. elif maptype == 'vector':
  1158. # loop through all vectors in VREF
  1159. self.grwiz.SwitchEnv('source')
  1160. # make list of vectors to georectify from VREF
  1161. f = open(self.file['vgrp'])
  1162. vectlist = []
  1163. try:
  1164. for vect in f.readlines():
  1165. vect = vect.strip('\n')
  1166. if len(vect) < 1:
  1167. continue
  1168. vectlist.append(vect)
  1169. finally:
  1170. f.close()
  1171. # georectify each vector in VREF using v.rectify
  1172. for vect in vectlist:
  1173. self.outname = str(vect.split('@')[0]) + self.extension
  1174. print self.outname
  1175. self.parent.goutput.WriteLog(text = _('Transforming <%s>...') % vect,
  1176. switchPage = True)
  1177. ret = msg = ''
  1178. busy = wx.BusyInfo(message=_("Rectifying vector map <%s>, please wait...") % vect,
  1179. parent=self)
  1180. wx.Yield()
  1181. ret, msg = RunCommand('v.rectify',
  1182. parent = self,
  1183. getErrorMsg = True,
  1184. quiet = True,
  1185. input = vect,
  1186. output = self.outname,
  1187. group = self.xygroup,
  1188. order = self.gr_order)
  1189. busy.Destroy()
  1190. # provide feedback on failure
  1191. if ret != 0:
  1192. print >> sys.stderr, msg
  1193. self.grwiz.SwitchEnv('target')
  1194. def OnGeorectDone(self, **kargs):
  1195. """!Print final message"""
  1196. global maptype
  1197. if maptype == 'raster':
  1198. return
  1199. returncode = kargs['returncode']
  1200. if returncode == 0:
  1201. self.VectGRList.append(self.outname)
  1202. print '*****vector list = ' + str(self.VectGRList)
  1203. else:
  1204. self.parent.goutput.WriteError(_('Georectification of vector map <%s> failed') %
  1205. self.outname)
  1206. def OnSettings(self, event):
  1207. """!GCP Manager settings"""
  1208. dlg = GrSettingsDialog(parent=self, id=wx.ID_ANY, title=_('GCP Manager settings'))
  1209. if dlg.ShowModal() == wx.ID_OK:
  1210. pass
  1211. dlg.Destroy()
  1212. def UpdateColours(self, srcrender=False, srcrenderVector=False,
  1213. tgtrender=False, tgtrenderVector=False):
  1214. """!update colours"""
  1215. highest_fwd_err = 0.0
  1216. self.highest_key = 0
  1217. highest_idx = 0
  1218. for index in range(self.list.GetItemCount()):
  1219. if self.list.IsChecked(index):
  1220. key = self.list.GetItemData(index)
  1221. fwd_err = self.mapcoordlist[key][5]
  1222. if self.highest_only == True:
  1223. self.list.SetItemTextColour(index, wx.BLACK)
  1224. if highest_fwd_err < fwd_err:
  1225. highest_fwd_err = fwd_err
  1226. self.highest_key = key
  1227. highest_idx = index
  1228. elif self.rmsthresh > 0:
  1229. if (fwd_err > self.rmsthresh):
  1230. self.list.SetItemTextColour(index, wx.RED)
  1231. else:
  1232. self.list.SetItemTextColour(index, wx.BLACK)
  1233. else:
  1234. self.list.SetItemTextColour(index, wx.BLACK)
  1235. if self.highest_only and highest_fwd_err > 0.0:
  1236. self.list.SetItemTextColour(highest_idx, wx.RED)
  1237. sourceMapWin = self.SrcMapWindow
  1238. sourceMapWin.UpdateMap(render=srcrender, renderVector=srcrenderVector)
  1239. if self.show_target:
  1240. targetMapWin = self.TgtMapWindow
  1241. targetMapWin.UpdateMap(render=tgtrender, renderVector=tgtrenderVector)
  1242. def OnQuit(self, event):
  1243. """!Quit georectifier"""
  1244. ret = wx.MessageBox(parent=self,
  1245. caption=_("Quit GCP Manager"),
  1246. message=_('Save ground control points?'),
  1247. style=wx.ICON_QUESTION | wx.YES_NO | wx.CANCEL | wx.CENTRE)
  1248. if ret != wx.CANCEL:
  1249. if ret == wx.YES:
  1250. self.SaveGCPs(None)
  1251. elif ret == wx.NO:
  1252. # restore POINTS file from backup
  1253. if os.path.exists(self.file['points_bak']):
  1254. shutil.copy(self.file['points_bak'], self.file['points'])
  1255. if os.path.exists(self.file['points_bak']):
  1256. os.unlink(self.file['points_bak'])
  1257. self.SrcMap.Clean()
  1258. self.TgtMap.Clean()
  1259. self.grwiz.Cleanup()
  1260. self.Destroy()
  1261. #event.Skip()
  1262. def OnGROrder(self, event):
  1263. """
  1264. sets transformation order for georectifying
  1265. """
  1266. if event:
  1267. self.gr_order = event.GetInt() + 1
  1268. numOfItems = self.list.GetItemCount()
  1269. minNumOfItems = numOfItems
  1270. if self.gr_order == 1:
  1271. minNumOfItems = 3
  1272. # self.SetStatusText(_('Insufficient points, 3+ points needed for 1st order'))
  1273. elif self.gr_order == 2:
  1274. minNumOfItems = 6
  1275. diff = 6 - numOfItems
  1276. # self.SetStatusText(_('Insufficient points, 6+ points needed for 2nd order'))
  1277. elif self.gr_order == 3:
  1278. minNumOfItems = 10
  1279. # self.SetStatusText(_('Insufficient points, 10+ points needed for 3rd order'))
  1280. for i in range(minNumOfItems - numOfItems):
  1281. self.AddGCP(None)
  1282. return minNumOfItems
  1283. def RMSError(self, xygroup, order):
  1284. """
  1285. Uses m.transform to calculate forward and backward error for each used GCP
  1286. in POINTS file and insert error values into GCP list.
  1287. Calculates total forward and backward RMS error for all used points
  1288. """
  1289. # save GCPs to points file to make sure that all checked GCPs are used
  1290. self.SaveGCPs(None)
  1291. #self.SetStatusText('')
  1292. if self.CheckGCPcount(msg=True) == False:
  1293. return
  1294. # get list of forward and reverse rms error values for each point
  1295. self.grwiz.SwitchEnv('source')
  1296. ret = RunCommand('m.transform',
  1297. parent = self,
  1298. read = True,
  1299. group = xygroup,
  1300. order = order)
  1301. self.grwiz.SwitchEnv('target')
  1302. if ret:
  1303. errlist = ret.splitlines()
  1304. else:
  1305. GError(parent = self,
  1306. message=_('Could not calculate RMS Error.\n'
  1307. 'Possible error with m.transform.'))
  1308. return
  1309. # insert error values into GCP list for checked items
  1310. sdfactor = float(UserSettings.Get(group='gcpman', key='rms', subkey='sdfactor'))
  1311. GCPcount = 0
  1312. sumsq_fwd_err = 0.0
  1313. sumsq_bkw_err = 0.0
  1314. sum_fwd_err = 0.0
  1315. highest_fwd_err = 0.0
  1316. self.highest_key = 0
  1317. highest_idx = 0
  1318. for index in range(self.list.GetItemCount()):
  1319. key = self.list.GetItemData(index)
  1320. if self.list.IsChecked(index):
  1321. fwd_err, bkw_err = errlist[GCPcount].split()
  1322. self.list.SetStringItem(index, 5, fwd_err)
  1323. self.list.SetStringItem(index, 6, bkw_err)
  1324. self.mapcoordlist[key][5] = float(fwd_err)
  1325. self.mapcoordlist[key][6] = float(bkw_err)
  1326. self.list.SetItemTextColour(index, wx.BLACK)
  1327. if self.highest_only:
  1328. if highest_fwd_err < float(fwd_err):
  1329. highest_fwd_err = float(fwd_err)
  1330. self.highest_key = key
  1331. highest_idx = index
  1332. sumsq_fwd_err += float(fwd_err)**2
  1333. sumsq_bkw_err += float(bkw_err)**2
  1334. sum_fwd_err += float(fwd_err)
  1335. GCPcount += 1
  1336. else:
  1337. self.list.SetStringItem(index, 5, '')
  1338. self.list.SetStringItem(index, 6, '')
  1339. self.mapcoordlist[key][5] = 0.0
  1340. self.mapcoordlist[key][6] = 0.0
  1341. self.list.SetItemTextColour(index, wx.BLACK)
  1342. # SD
  1343. if GCPcount > 0:
  1344. sum_fwd_err /= GCPcount
  1345. self.rmsmean = sum_fwd_err /GCPcount
  1346. self.rmssd = (((sumsq_fwd_err/GCPcount) - self.rmsmean**2)**0.5)
  1347. self.rmsthresh = self.rmsmean + sdfactor * self.rmssd
  1348. else:
  1349. self.rmsthresh = 0
  1350. self.rmsmean = 0
  1351. self.rmssd = 0
  1352. if self.highest_only and highest_fwd_err > 0.0:
  1353. self.list.SetItemTextColour(highest_idx, wx.RED)
  1354. elif GCPcount > 0 and self.rmsthresh > 0 and not self.highest_only:
  1355. for index in range(self.list.GetItemCount()):
  1356. if self.list.IsChecked(index):
  1357. key = self.list.GetItemData(index)
  1358. if (self.mapcoordlist[key][5] > self.rmsthresh):
  1359. self.list.SetItemTextColour(index, wx.RED)
  1360. # calculate global RMS error (geometric mean)
  1361. self.fwd_rmserror = round((sumsq_fwd_err/GCPcount)**0.5,4)
  1362. self.bkw_rmserror = round((sumsq_bkw_err/GCPcount)**0.5,4)
  1363. self.list.ResizeColumns()
  1364. def GetNewExtent(self, region, map = None):
  1365. coord_file = utils.GetTempfile()
  1366. newreg = { 'n' : 0.0, 's' : 0.0, 'e' : 0.0, 'w' : 0.0,}
  1367. try:
  1368. f = open(coord_file, mode='w')
  1369. # NW corner
  1370. f.write(str(region['e']) + " " + str(region['n']) + "\n")
  1371. # NE corner
  1372. f.write(str(region['e']) + " " + str(region['s']) + "\n")
  1373. # SW corner
  1374. f.write(str(region['w']) + " " + str(region['n']) + "\n")
  1375. # SE corner
  1376. f.write(str(region['w']) + " " + str(region['s']) + "\n")
  1377. finally:
  1378. f.close()
  1379. # save GCPs to points file to make sure that all checked GCPs are used
  1380. self.SaveGCPs(None)
  1381. order = self.gr_order
  1382. self.gr_order = 1
  1383. if self.CheckGCPcount(msg=True) == False:
  1384. self.gr_order = order
  1385. return
  1386. self.gr_order = order
  1387. # get list of forward and reverse rms error values for each point
  1388. self.grwiz.SwitchEnv('source')
  1389. if map == 'source':
  1390. ret = RunCommand('m.transform',
  1391. parent = self,
  1392. read = True,
  1393. group = self.xygroup,
  1394. order = 1,
  1395. format = 'dst',
  1396. coords = coord_file)
  1397. elif map == 'target':
  1398. ret = RunCommand('m.transform',
  1399. parent = self,
  1400. read = True,
  1401. group = self.xygroup,
  1402. order = 1,
  1403. flags = 'r',
  1404. format = 'src',
  1405. coords = coord_file)
  1406. os.unlink(coord_file)
  1407. self.grwiz.SwitchEnv('target')
  1408. if ret:
  1409. errlist = ret.splitlines()
  1410. else:
  1411. GError(parent = self,
  1412. message=_('Could not calculate new extends.\n'
  1413. 'Possible error with m.transform.'))
  1414. return
  1415. # fist corner
  1416. e, n = errlist[0].split()
  1417. fe = float(e)
  1418. fn = float(n)
  1419. newreg['n'] = fn
  1420. newreg['s'] = fn
  1421. newreg['e'] = fe
  1422. newreg['w'] = fe
  1423. # other three corners
  1424. for i in range(1, 4):
  1425. e, n = errlist[i].split()
  1426. fe = float(e)
  1427. fn = float(n)
  1428. if fe < newreg['w']:
  1429. newreg['w'] = fe
  1430. if fe > newreg['e']:
  1431. newreg['e'] = fe
  1432. if fn < newreg['s']:
  1433. newreg['s'] = fn
  1434. if fn > newreg['n']:
  1435. newreg['n'] = fn
  1436. return newreg
  1437. def OnHelp(self, event):
  1438. """!Show GCP Manager manual page"""
  1439. cmdlist = ['g.manual', 'entry=wxGUI.GCP_Manager']
  1440. self.parent.goutput.RunCmd(cmdlist, compReg=False,
  1441. switchPage=False)
  1442. def OnUpdateActive(self, event):
  1443. if self.activemap.GetSelection() == 0:
  1444. self.MapWindow = self.SrcMapWindow
  1445. self.Map = self.SrcMap
  1446. else:
  1447. self.MapWindow = self.TgtMapWindow
  1448. self.Map = self.TgtMap
  1449. self.UpdateActive(self.MapWindow)
  1450. # for wingrass
  1451. if os.name == 'nt':
  1452. self.MapWindow.SetFocus()
  1453. def UpdateActive(self, win):
  1454. # optionally disable tool zoomback tool
  1455. self.GetMapToolbar().Enable('zoomback', enable = (len(self.MapWindow.zoomhistory) > 1))
  1456. if self.activemap.GetSelection() != (win == self.TgtMapWindow):
  1457. self.activemap.SetSelection(win == self.TgtMapWindow)
  1458. self.StatusbarUpdate()
  1459. def AdjustMap(self, newreg):
  1460. """!Adjust map window to new extents
  1461. """
  1462. # adjust map window
  1463. self.Map.region['n'] = newreg['n']
  1464. self.Map.region['s'] = newreg['s']
  1465. self.Map.region['e'] = newreg['e']
  1466. self.Map.region['w'] = newreg['w']
  1467. self.MapWindow.ZoomHistory(self.Map.region['n'], self.Map.region['s'],
  1468. self.Map.region['e'], self.Map.region['w'])
  1469. # LL locations
  1470. if self.Map.projinfo['proj'] == 'll':
  1471. if newreg['n'] > 90.0:
  1472. newreg['n'] = 90.0
  1473. if newreg['s'] < -90.0:
  1474. newreg['s'] = -90.0
  1475. ce = newreg['w'] + (newreg['e'] - newreg['w']) / 2
  1476. cn = newreg['s'] + (newreg['n'] - newreg['s']) / 2
  1477. # calculate new center point and display resolution
  1478. self.Map.region['center_easting'] = ce
  1479. self.Map.region['center_northing'] = cn
  1480. self.Map.region["ewres"] = (newreg['e'] - newreg['w']) / self.Map.width
  1481. self.Map.region["nsres"] = (newreg['n'] - newreg['s']) / self.Map.height
  1482. self.Map.AlignExtentFromDisplay()
  1483. self.MapWindow.ZoomHistory(self.Map.region['n'], self.Map.region['s'],
  1484. self.Map.region['e'], self.Map.region['w'])
  1485. if self.MapWindow.redrawAll is False:
  1486. self.MapWindow.redrawAll = True
  1487. self.MapWindow.UpdateMap()
  1488. self.StatusbarUpdate()
  1489. def OnZoomToSource(self, event):
  1490. """!Set target map window to match extents of source map window
  1491. """
  1492. if not self.MapWindow == self.TgtMapWindow:
  1493. self.MapWindow = self.TgtMapWindow
  1494. self.Map = self.TgtMap
  1495. self.UpdateActive(self.TgtMapWindow)
  1496. # get new N, S, E, W for target
  1497. newreg = self.GetNewExtent(self.SrcMap.region, 'source')
  1498. if newreg:
  1499. self.AdjustMap(newreg)
  1500. def OnZoomToTarget(self, event):
  1501. """!Set source map window to match extents of target map window
  1502. """
  1503. if not self.MapWindow == self.SrcMapWindow:
  1504. self.MapWindow = self.SrcMapWindow
  1505. self.Map = self.SrcMap
  1506. self.UpdateActive(self.SrcMapWindow)
  1507. # get new N, S, E, W for target
  1508. newreg = self.GetNewExtent(self.TgtMap.region, 'target')
  1509. if newreg:
  1510. self.AdjustMap(newreg)
  1511. def OnZoomMenuGCP(self, event):
  1512. """!Popup Zoom menu
  1513. """
  1514. point = wx.GetMousePosition()
  1515. zoommenu = wx.Menu()
  1516. # Add items to the menu
  1517. zoomsource = wx.MenuItem(zoommenu, wx.ID_ANY, _('Adjust source display to target display'))
  1518. zoommenu.AppendItem(zoomsource)
  1519. self.Bind(wx.EVT_MENU, self.OnZoomToTarget, zoomsource)
  1520. zoomtarget = wx.MenuItem(zoommenu, wx.ID_ANY, _('Adjust target display to source display'))
  1521. zoommenu.AppendItem(zoomtarget)
  1522. self.Bind(wx.EVT_MENU, self.OnZoomToSource, zoomtarget)
  1523. # Popup the menu. If an item is selected then its handler
  1524. # will be called before PopupMenu returns.
  1525. self.PopupMenu(zoommenu)
  1526. zoommenu.Destroy()
  1527. def OnSize(self, event):
  1528. """!Adjust Map Windows after GCP Map Display has been resized
  1529. """
  1530. # re-render image on idle
  1531. self.resize = time.clock()
  1532. def OnIdle(self, event):
  1533. """!GCP Map Display resized, adjust Map Windows
  1534. """
  1535. if self.GetMapToolbar():
  1536. if self.resize and self.resize + 0.2 < time.clock():
  1537. srcwidth, srcheight = self.SrcMapWindow.GetSize()
  1538. tgtwidth, tgtheight = self.TgtMapWindow.GetSize()
  1539. srcwidth = (srcwidth + tgtwidth) / 2
  1540. if self.show_target:
  1541. self._mgr.GetPane("target").Hide()
  1542. self._mgr.Update()
  1543. self._mgr.GetPane("source").BestSize((srcwidth, srcheight))
  1544. self._mgr.GetPane("target").BestSize((srcwidth, tgtheight))
  1545. if self.show_target:
  1546. self._mgr.GetPane("target").Show()
  1547. self._mgr.Update()
  1548. self.resize = False
  1549. elif self.resize:
  1550. event.RequestMore()
  1551. pass
  1552. class GCPList(wx.ListCtrl,
  1553. CheckListCtrlMixin,
  1554. ListCtrlAutoWidthMixin):
  1555. def __init__(self, parent, gcp, id=wx.ID_ANY,
  1556. pos=wx.DefaultPosition, size=wx.DefaultSize,
  1557. style=wx.LC_REPORT | wx.SUNKEN_BORDER | wx.LC_HRULES |
  1558. wx.LC_SINGLE_SEL):
  1559. wx.ListCtrl.__init__(self, parent, id, pos, size, style)
  1560. self.gcp = gcp # GCP class
  1561. self.render = True
  1562. # Mixin settings
  1563. CheckListCtrlMixin.__init__(self)
  1564. ListCtrlAutoWidthMixin.__init__(self)
  1565. # TextEditMixin.__init__(self)
  1566. # tracks whether list items are checked or not
  1567. self.CheckList = []
  1568. self._Create()
  1569. self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected)
  1570. self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.OnItemActivated)
  1571. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick)
  1572. self.selected = wx.NOT_FOUND
  1573. self.selectedkey = -1
  1574. def _Create(self):
  1575. if 0:
  1576. # normal, simple columns
  1577. idx_col = 0
  1578. for col in (_('use'),
  1579. _('source E'),
  1580. _('source N'),
  1581. _('target E'),
  1582. _('target N'),
  1583. _('Forward error'),
  1584. _('Backward error')):
  1585. self.InsertColumn(idx_col, col)
  1586. idx_col += 1
  1587. else:
  1588. # the hard way: we want images on the column header
  1589. info = wx.ListItem()
  1590. info.SetMask(wx.LIST_MASK_TEXT | wx.LIST_MASK_IMAGE | wx.LIST_MASK_FORMAT)
  1591. info.SetImage(-1)
  1592. info.m_format = wx.LIST_FORMAT_LEFT
  1593. idx_col = 0
  1594. for lbl in (_('use'),
  1595. _('source E'),
  1596. _('source N'),
  1597. _('target E'),
  1598. _('target N'),
  1599. _('Forward error'),
  1600. _('Backward error')):
  1601. info.SetText(lbl)
  1602. self.InsertColumnInfo(idx_col, info)
  1603. idx_col += 1
  1604. def LoadData(self):
  1605. """!Load data into list"""
  1606. self.DeleteAllItems()
  1607. self.render = False
  1608. if os.path.isfile(self.gcp.file['points']):
  1609. self.gcp.ReadGCPs()
  1610. else:
  1611. # 3 gcp is minimum
  1612. for i in range(3):
  1613. self.gcp.AddGCP(None)
  1614. # select first point by default
  1615. self.selected = 0
  1616. self.selectedkey = self.GetItemData(self.selected)
  1617. self.SetItemState(self.selected,
  1618. wx.LIST_STATE_SELECTED,
  1619. wx.LIST_STATE_SELECTED)
  1620. self.ResizeColumns()
  1621. self.render = True
  1622. self.EnsureVisible(self.selected)
  1623. def OnCheckItem(self, index, flag):
  1624. """!Item is checked/unchecked"""
  1625. if self.render:
  1626. # redraw points
  1627. sourceMapWin = self.gcp.SrcMapWindow
  1628. sourceMapWin.UpdateMap(render=False, renderVector=False)
  1629. if self.gcp.show_target:
  1630. targetMapWin = self.gcp.TgtMapWindow
  1631. targetMapWin.UpdateMap(render=False, renderVector=False)
  1632. def AddGCPItem(self):
  1633. """
  1634. Appends an item to GCP list
  1635. """
  1636. self.selectedkey = self.GetItemCount() + 1
  1637. self.Append([str(self.selectedkey), # GCP number
  1638. '0.0', # source E
  1639. '0.0', # source N
  1640. '0.0', # target E
  1641. '0.0', # target N
  1642. '', # forward error
  1643. '']) # backward error
  1644. self.selected = self.GetItemCount() - 1
  1645. self.SetItemData(self.selected, self.selectedkey)
  1646. self.SetItemState(self.selected,
  1647. wx.LIST_STATE_SELECTED,
  1648. wx.LIST_STATE_SELECTED)
  1649. self.ResizeColumns()
  1650. self.gcp.pointsToDrawSrc.AddItem(coords = [0,0], label = str(self.selectedkey))
  1651. self.gcp.pointsToDrawTgt.AddItem(coords = [0,0], label = str(self.selectedkey))
  1652. self.EnsureVisible(self.selected)
  1653. return self.selected
  1654. def DeleteGCPItem(self):
  1655. """!Deletes selected item in GCP list.
  1656. """
  1657. if self.selected == wx.NOT_FOUND:
  1658. return
  1659. key = self.GetItemData(self.selected)
  1660. self.DeleteItem(self.selected)
  1661. if self.selected != wx.NOT_FOUND:
  1662. item = self.gcp.pointsToDrawSrc.GetItem(key - 1)
  1663. self.gcp.pointsToDrawSrc.DeleteItem(item)
  1664. item = self.gcp.pointsToDrawTgt.GetItem(key - 1)
  1665. self.gcp.pointsToDrawTgt.DeleteItem(item)
  1666. return key
  1667. def ResizeColumns(self):
  1668. """!Resize columns"""
  1669. minWidth = [90, 120]
  1670. for i in range(self.GetColumnCount()):
  1671. self.SetColumnWidth(i, wx.LIST_AUTOSIZE)
  1672. # first column is checkbox, don't set to minWidth
  1673. if i > 0 and self.GetColumnWidth(i) < minWidth[i > 4]:
  1674. self.SetColumnWidth(i, minWidth[i > 4])
  1675. self.SendSizeEvent()
  1676. def GetSelected(self):
  1677. """!Get index of selected item"""
  1678. return self.selected
  1679. def OnItemSelected(self, event):
  1680. """!Item selected
  1681. """
  1682. if self.render and self.selected != event.GetIndex():
  1683. self.selected = event.GetIndex()
  1684. self.selectedkey = self.GetItemData(self.selected)
  1685. sourceMapWin = self.gcp.SrcMapWindow
  1686. sourceMapWin.UpdateMap(render=False, renderVector=False)
  1687. if self.gcp.show_target:
  1688. targetMapWin = self.gcp.TgtMapWindow
  1689. targetMapWin.UpdateMap(render=False, renderVector=False)
  1690. event.Skip()
  1691. def OnItemActivated(self, event):
  1692. """
  1693. When item double clicked, open editor to update coordinate values
  1694. """
  1695. coords = []
  1696. index = event.GetIndex()
  1697. key = self.GetItemData(index)
  1698. changed = False
  1699. for i in range(1, 5):
  1700. coords.append(self.GetItem(index, i).GetText())
  1701. dlg = EditGCP(parent=self, id=wx.ID_ANY, data=coords, gcpno=key)
  1702. if dlg.ShowModal() == wx.ID_OK:
  1703. values = dlg.GetValues() # string
  1704. if len(values) == 0:
  1705. GError(parent = self,
  1706. message=_("Invalid coordinate value. Operation canceled."))
  1707. else:
  1708. for i in range(len(values)):
  1709. if values[i] != coords[i]:
  1710. self.SetStringItem(index, i + 1, values[i])
  1711. changed = True
  1712. if changed:
  1713. # reset RMS and update mapcoordlist
  1714. self.SetStringItem(index, 5, '')
  1715. self.SetStringItem(index, 6, '')
  1716. key = self.GetItemData(index)
  1717. self.gcp.mapcoordlist[key] = [key,
  1718. float(values[0]),
  1719. float(values[1]),
  1720. float(values[2]),
  1721. float(values[3]),
  1722. 0.0,
  1723. 0.0]
  1724. self.gcp.pointsToDrawSrc.GetItem(key - 1).SetCoords([float(values[0]),
  1725. float(values[1])])
  1726. self.gcp.pointsToDrawTgt.GetItem(key - 1).SetCoords([float(values[2]),
  1727. float(values[3])])
  1728. self.gcp.UpdateColours()
  1729. def OnColClick(self, event):
  1730. """!ListCtrl forgets selected item..."""
  1731. self.selected = self.FindItemData(-1, self.selectedkey)
  1732. self.SetItemState(self.selected,
  1733. wx.LIST_STATE_SELECTED,
  1734. wx.LIST_STATE_SELECTED)
  1735. event.Skip()
  1736. class VectGroup(wx.Dialog):
  1737. """!Dialog to create a vector group (VREF file) for georectifying
  1738. @todo Replace by g.group
  1739. """
  1740. def __init__(self, parent, id, grassdb, location, mapset, group,
  1741. style=wx.DEFAULT_DIALOG_STYLE):
  1742. wx.Dialog.__init__(self, parent, id, style=style,
  1743. title = _("Create vector map group"))
  1744. self.grassdatabase = grassdb
  1745. self.xylocation = location
  1746. self.xymapset = mapset
  1747. self.xygroup = group
  1748. #
  1749. # get list of valid vector directories
  1750. #
  1751. vectlist = os.listdir(os.path.join(self.grassdatabase,
  1752. self.xylocation,
  1753. self.xymapset,
  1754. 'vector'))
  1755. for dir in vectlist:
  1756. if not os.path.isfile(os.path.join(self.grassdatabase,
  1757. self.xylocation,
  1758. self.xymapset,
  1759. 'vector',
  1760. dir,
  1761. 'coor')):
  1762. vectlist.remove(dir)
  1763. utils.ListSortLower(vectlist)
  1764. # path to vref file
  1765. self.vgrpfile = os.path.join(self.grassdatabase,
  1766. self.xylocation,
  1767. self.xymapset,
  1768. 'group',
  1769. self.xygroup,
  1770. 'VREF')
  1771. #
  1772. # buttons
  1773. #
  1774. self.btnCancel = wx.Button(parent = self,
  1775. id = wx.ID_CANCEL)
  1776. self.btnOK = wx.Button(parent = self,
  1777. id = wx.ID_OK)
  1778. self.btnOK.SetDefault()
  1779. #
  1780. # list of vector maps
  1781. #
  1782. self.listMap = wx.CheckListBox(parent = self, id = wx.ID_ANY,
  1783. choices = vectlist)
  1784. if os.path.isfile(self.vgrpfile):
  1785. f = open(self.vgrpfile)
  1786. try:
  1787. checked = []
  1788. for line in f.readlines():
  1789. line = line.replace('\n', '')
  1790. if len(line) < 1:
  1791. continue
  1792. checked.append(line)
  1793. self.listMap.SetCheckedStrings(checked)
  1794. finally:
  1795. f.close()
  1796. line = wx.StaticLine(parent = self,
  1797. id = wx.ID_ANY, size = (20, -1),
  1798. style = wx.LI_HORIZONTAL)
  1799. #
  1800. # layout
  1801. #
  1802. sizer = wx.BoxSizer(wx.VERTICAL)
  1803. box = wx.BoxSizer(wx.HORIZONTAL)
  1804. box.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  1805. label = _('Select vector map(s) to add to group:')),
  1806. flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT | wx.LEFT,
  1807. border = 5)
  1808. box.Add(item = self.listMap,
  1809. flag = wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_RIGHT | wx.LEFT,
  1810. border = 5)
  1811. sizer.Add(box, flag = wx.ALIGN_RIGHT | wx.ALL,
  1812. border = 3)
  1813. sizer.Add(item = line, proportion = 0,
  1814. flag = wx.GROW | wx.ALIGN_CENTER_VERTICAL | wx.LEFT | wx.RIGHT,
  1815. border = 5)
  1816. # buttons
  1817. btnSizer = wx.StdDialogButtonSizer()
  1818. btnSizer.AddButton(self.btnCancel)
  1819. btnSizer.AddButton(self.btnOK)
  1820. btnSizer.Realize()
  1821. sizer.Add(item = btnSizer, proportion = 0,
  1822. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER,
  1823. border = 5)
  1824. self.SetSizer(sizer)
  1825. sizer.Fit(self)
  1826. self.Layout()
  1827. def MakeVGroup(self):
  1828. """!Create VREF file"""
  1829. vgrouplist = []
  1830. for item in range(self.listMap.GetCount()):
  1831. if not self.listMap.IsChecked(item):
  1832. continue
  1833. vgrouplist.append(self.listMap.GetString(item) + '@' + self.xymapset)
  1834. f = open(self.vgrpfile, mode='w')
  1835. try:
  1836. for vect in vgrouplist:
  1837. f.write(vect + '\n')
  1838. finally:
  1839. f.close()
  1840. class EditGCP(wx.Dialog):
  1841. def __init__(self, parent, data, gcpno, id=wx.ID_ANY,
  1842. title=_("Edit GCP"),
  1843. style=wx.DEFAULT_DIALOG_STYLE):
  1844. """!Dialog for editing GPC and map coordinates in list control"""
  1845. wx.Dialog.__init__(self, parent, id, title=title, style=style)
  1846. panel = wx.Panel(parent=self)
  1847. sizer = wx.BoxSizer(wx.VERTICAL)
  1848. box = wx.StaticBox (parent=panel, id=wx.ID_ANY,
  1849. label=" %s %s " % (_("Ground Control Point No."), str(gcpno)))
  1850. boxSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1851. # source coordinates
  1852. gridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  1853. self.xcoord = wx.TextCtrl(parent=panel, id=wx.ID_ANY, size=(150, -1))
  1854. self.ycoord = wx.TextCtrl(parent=panel, id=wx.ID_ANY, size=(150, -1))
  1855. self.ecoord = wx.TextCtrl(parent=panel, id=wx.ID_ANY, size=(150, -1))
  1856. self.ncoord = wx.TextCtrl(parent=panel, id=wx.ID_ANY, size=(150, -1))
  1857. # swap source N, target E
  1858. tmp_coord = data[1]
  1859. data[1] = data[2]
  1860. data[2] = tmp_coord
  1861. row = 0
  1862. col = 0
  1863. idx = 0
  1864. for label, win in ((_("source E:"), self.xcoord),
  1865. (_("target E:"), self.ecoord),
  1866. (_("source N:"), self.ycoord),
  1867. (_("target N:"), self.ncoord)):
  1868. label = wx.StaticText(parent=panel, id=wx.ID_ANY,
  1869. label=label)
  1870. gridSizer.Add(item=label,
  1871. flag=wx.ALIGN_CENTER_VERTICAL,
  1872. pos=(row, col))
  1873. col += 1
  1874. win.SetValue(str(data[idx]))
  1875. gridSizer.Add(item=win,
  1876. pos=(row, col))
  1877. col += 1
  1878. idx += 1
  1879. if col > 3:
  1880. row += 1
  1881. col = 0
  1882. boxSizer.Add(item=gridSizer, proportion=1,
  1883. flag=wx.EXPAND | wx.ALL, border=5)
  1884. sizer.Add(item=boxSizer, proportion=1,
  1885. flag=wx.EXPAND | wx.ALL, border=5)
  1886. #
  1887. # buttons
  1888. #
  1889. self.btnCancel = wx.Button(panel, wx.ID_CANCEL)
  1890. self.btnOk = wx.Button(panel, wx.ID_OK)
  1891. self.btnOk.SetDefault()
  1892. btnSizer = wx.StdDialogButtonSizer()
  1893. btnSizer.AddButton(self.btnCancel)
  1894. btnSizer.AddButton(self.btnOk)
  1895. btnSizer.Realize()
  1896. sizer.Add(item=btnSizer, proportion=0,
  1897. flag=wx.ALIGN_RIGHT | wx.ALL, border=5)
  1898. panel.SetSizer(sizer)
  1899. sizer.Fit(self)
  1900. def GetValues(self, columns=None):
  1901. """!Return list of values (as strings).
  1902. """
  1903. valuelist = []
  1904. try:
  1905. float(self.xcoord.GetValue())
  1906. float(self.ycoord.GetValue())
  1907. float(self.ecoord.GetValue())
  1908. float(self.ncoord.GetValue())
  1909. except ValueError:
  1910. return valuelist
  1911. valuelist.append(self.xcoord.GetValue())
  1912. valuelist.append(self.ycoord.GetValue())
  1913. valuelist.append(self.ecoord.GetValue())
  1914. valuelist.append(self.ncoord.GetValue())
  1915. return valuelist
  1916. class GrSettingsDialog(wx.Dialog):
  1917. def __init__(self, parent, id, title, pos=wx.DefaultPosition, size=wx.DefaultSize,
  1918. style=wx.DEFAULT_DIALOG_STYLE):
  1919. wx.Dialog.__init__(self, parent, id, title, pos, size, style)
  1920. """
  1921. Dialog to set profile text options: font, title
  1922. and font size, axis labels and font size
  1923. """
  1924. #
  1925. # initialize variables
  1926. #
  1927. self.parent = parent
  1928. self.new_src_map = src_map
  1929. self.new_tgt_map = { 'raster' : tgt_map['raster'],
  1930. 'vector' : tgt_map['vector'] }
  1931. self.sdfactor = 0
  1932. self.symbol = {}
  1933. self.methods = ["nearest",
  1934. "bilinear",
  1935. "bilinear_f",
  1936. "cubic",
  1937. "cubic_f",
  1938. "lanczos",
  1939. "lanczos_f"]
  1940. # notebook
  1941. notebook = wx.Notebook(parent=self, id=wx.ID_ANY, style=wx.BK_DEFAULT)
  1942. self.__CreateSymbologyPage(notebook)
  1943. self.__CreateRectificationPage(notebook)
  1944. # buttons
  1945. btnSave = wx.Button(self, wx.ID_SAVE)
  1946. btnApply = wx.Button(self, wx.ID_APPLY)
  1947. btnClose = wx.Button(self, wx.ID_CLOSE)
  1948. btnApply.SetDefault()
  1949. # bindings
  1950. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  1951. btnApply.SetToolTipString(_("Apply changes for the current session"))
  1952. btnSave.Bind(wx.EVT_BUTTON, self.OnSave)
  1953. btnSave.SetToolTipString(_("Apply and save changes to user settings file (default for next sessions)"))
  1954. btnClose.Bind(wx.EVT_BUTTON, self.OnClose)
  1955. btnClose.SetToolTipString(_("Close dialog"))
  1956. # sizers
  1957. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  1958. btnSizer.Add(btnApply, flag=wx.LEFT | wx.RIGHT, border=5)
  1959. btnSizer.Add(btnSave, flag=wx.LEFT | wx.RIGHT, border=5)
  1960. btnSizer.Add(btnClose, flag=wx.LEFT | wx.RIGHT, border=5)
  1961. # sizers
  1962. mainSizer = wx.BoxSizer(wx.VERTICAL)
  1963. mainSizer.Add(item=notebook, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  1964. mainSizer.Add(item=btnSizer, proportion=0,
  1965. flag=wx.ALIGN_RIGHT | wx.ALL, border=5)
  1966. # flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  1967. self.SetSizer(mainSizer)
  1968. mainSizer.Fit(self)
  1969. def __CreateSymbologyPage(self, notebook):
  1970. """!Create notebook page with symbology settings"""
  1971. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1972. notebook.AddPage(page=panel, text=_("Symbology"))
  1973. sizer = wx.BoxSizer(wx.VERTICAL)
  1974. rmsgridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  1975. rmsgridSizer.AddGrowableCol(1)
  1976. # highlight only highest forward RMS error
  1977. self.highlighthighest = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1978. label=_("Highlight highest RMS error only"))
  1979. hh = UserSettings.Get(group='gcpman', key='rms', subkey='highestonly')
  1980. self.highlighthighest.SetValue(hh)
  1981. rmsgridSizer.Add(item=self.highlighthighest, flag=wx.ALIGN_CENTER_VERTICAL, pos=(0, 0))
  1982. # RMS forward error threshold
  1983. rmslabel = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Highlight RMS error > M + SD * factor:"))
  1984. rmslabel.SetToolTip(wx.ToolTip(_("Highlight GCPs with an RMS error larger than \n"
  1985. "mean + standard deviation * given factor. \n"
  1986. "Recommended values for this factor are between 1 and 2.")))
  1987. rmsgridSizer.Add(item=rmslabel, flag=wx.ALIGN_CENTER_VERTICAL, pos=(1, 0))
  1988. sdfactor = UserSettings.Get(group='gcpman', key='rms', subkey='sdfactor')
  1989. self.rmsWin = wx.TextCtrl(parent=panel, id=wx.ID_ANY,
  1990. size=(70,-1), style=wx.TE_NOHIDESEL)
  1991. self.rmsWin.SetValue("%s" % str(sdfactor))
  1992. if (self.parent.highest_only == True):
  1993. self.rmsWin.Disable()
  1994. self.symbol['sdfactor'] = self.rmsWin.GetId()
  1995. rmsgridSizer.Add(item=self.rmsWin, flag=wx.ALIGN_RIGHT, pos=(1, 1))
  1996. sizer.Add(item=rmsgridSizer, flag=wx.EXPAND | wx.ALL, border=5)
  1997. box = wx.StaticBox(parent=panel, id=wx.ID_ANY,
  1998. label=" %s " % _("Symbol settings"))
  1999. boxSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  2000. gridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  2001. gridSizer.AddGrowableCol(1)
  2002. #
  2003. # general symbol color
  2004. #
  2005. row = 0
  2006. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Color:"))
  2007. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2008. col = UserSettings.Get(group='gcpman', key='symbol', subkey='color')
  2009. colWin = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  2010. colour=wx.Colour(col[0],
  2011. col[1],
  2012. col[2],
  2013. 255))
  2014. self.symbol['color'] = colWin.GetId()
  2015. gridSizer.Add(item=colWin,
  2016. flag=wx.ALIGN_RIGHT,
  2017. pos=(row, 1))
  2018. #
  2019. # symbol color for high forward RMS error
  2020. #
  2021. row += 1
  2022. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Color for high RMS error:"))
  2023. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2024. hcol = UserSettings.Get(group='gcpman', key='symbol', subkey='hcolor')
  2025. hcolWin = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  2026. colour=wx.Colour(hcol[0],
  2027. hcol[1],
  2028. hcol[2],
  2029. 255))
  2030. self.symbol['hcolor'] = hcolWin.GetId()
  2031. gridSizer.Add(item=hcolWin,
  2032. flag=wx.ALIGN_RIGHT,
  2033. pos=(row, 1))
  2034. #
  2035. # symbol color for selected GCP
  2036. #
  2037. row += 1
  2038. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Color for selected GCP:"))
  2039. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2040. scol = UserSettings.Get(group='gcpman', key='symbol', subkey='scolor')
  2041. scolWin = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  2042. colour=wx.Colour(scol[0],
  2043. scol[1],
  2044. scol[2],
  2045. 255))
  2046. self.symbol['scolor'] = scolWin.GetId()
  2047. gridSizer.Add(item=scolWin,
  2048. flag=wx.ALIGN_RIGHT,
  2049. pos=(row, 1))
  2050. #
  2051. # symbol color for unused GCP
  2052. #
  2053. row += 1
  2054. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Color for unused GCPs:"))
  2055. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2056. ucol = UserSettings.Get(group='gcpman', key='symbol', subkey='ucolor')
  2057. ucolWin = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  2058. colour=wx.Colour(ucol[0],
  2059. ucol[1],
  2060. ucol[2],
  2061. 255))
  2062. self.symbol['ucolor'] = ucolWin.GetId()
  2063. gridSizer.Add(item=ucolWin,
  2064. flag=wx.ALIGN_RIGHT,
  2065. pos=(row, 1))
  2066. # show unused GCPs
  2067. row += 1
  2068. self.showunused = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  2069. label=_("Show unused GCPs"))
  2070. shuu = UserSettings.Get(group='gcpman', key='symbol', subkey='unused')
  2071. self.showunused.SetValue(shuu)
  2072. gridSizer.Add(item=self.showunused, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2073. #
  2074. # symbol size
  2075. #
  2076. row += 1
  2077. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Symbol size:"))
  2078. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2079. symsize = int(UserSettings.Get(group='gcpman', key='symbol', subkey='size'))
  2080. sizeWin = wx.SpinCtrl(parent=panel, id=wx.ID_ANY,
  2081. min=1, max=20)
  2082. sizeWin.SetValue(symsize)
  2083. self.symbol['size'] = sizeWin.GetId()
  2084. gridSizer.Add(item=sizeWin,
  2085. flag=wx.ALIGN_RIGHT,
  2086. pos=(row, 1))
  2087. #
  2088. # symbol width
  2089. #
  2090. row += 1
  2091. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Line width:"))
  2092. gridSizer.Add(item=label, flag=wx.ALIGN_CENTER_VERTICAL, pos=(row, 0))
  2093. width = int(UserSettings.Get(group='gcpman', key='symbol', subkey='width'))
  2094. widWin = wx.SpinCtrl(parent=panel, id=wx.ID_ANY,
  2095. min=1, max=10)
  2096. widWin.SetValue(width)
  2097. self.symbol['width'] = widWin.GetId()
  2098. gridSizer.Add(item=widWin,
  2099. flag=wx.ALIGN_RIGHT,
  2100. pos=(row, 1))
  2101. boxSizer.Add(item=gridSizer, flag=wx.EXPAND)
  2102. sizer.Add(item=boxSizer, flag=wx.EXPAND | wx.ALL, border=5)
  2103. #
  2104. # maps to display
  2105. #
  2106. # source map to display
  2107. self.srcselection = Select(panel, id=wx.ID_ANY,
  2108. size=globalvar.DIALOG_GSELECT_SIZE, type='maptype', updateOnPopup = False)
  2109. self.parent.grwiz.SwitchEnv('source')
  2110. self.srcselection.SetElementList(maptype)
  2111. # filter out all maps not in group
  2112. self.srcselection.tcp.GetElementList(elements = self.parent.src_maps)
  2113. # target map(s) to display
  2114. self.parent.grwiz.SwitchEnv('target')
  2115. self.tgtrastselection = Select(panel, id=wx.ID_ANY,
  2116. size=globalvar.DIALOG_GSELECT_SIZE, type='raster', updateOnPopup = False)
  2117. self.tgtrastselection.SetElementList('cell')
  2118. self.tgtrastselection.GetElementList()
  2119. self.tgtvectselection = Select(panel, id=wx.ID_ANY,
  2120. size=globalvar.DIALOG_GSELECT_SIZE, type='vector', updateOnPopup = False)
  2121. self.tgtvectselection.SetElementList('vector')
  2122. self.tgtvectselection.GetElementList()
  2123. sizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY, label=_('Select source map to display:')),
  2124. proportion=0, flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2125. sizer.Add(item=self.srcselection, proportion=0,
  2126. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2127. self.srcselection.SetValue(src_map)
  2128. sizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY, label=_('Select target raster map to display:')),
  2129. proportion=0, flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2130. sizer.Add(item=self.tgtrastselection, proportion=0,
  2131. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2132. self.tgtrastselection.SetValue(tgt_map['raster'])
  2133. sizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY, label=_('Select target vector map to display:')),
  2134. proportion=0, flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2135. sizer.Add(item=self.tgtvectselection, proportion=0,
  2136. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2137. self.tgtvectselection.SetValue(tgt_map['vector'])
  2138. # bindings
  2139. self.highlighthighest.Bind(wx.EVT_CHECKBOX, self.OnHighlight)
  2140. self.rmsWin.Bind(wx.EVT_TEXT, self.OnSDFactor)
  2141. self.srcselection.Bind(wx.EVT_TEXT, self.OnSrcSelection)
  2142. self.tgtrastselection.Bind(wx.EVT_TEXT, self.OnTgtRastSelection)
  2143. self.tgtvectselection.Bind(wx.EVT_TEXT, self.OnTgtVectSelection)
  2144. panel.SetSizer(sizer)
  2145. return panel
  2146. def __CreateRectificationPage(self, notebook):
  2147. """!Create notebook page with symbology settings"""
  2148. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  2149. notebook.AddPage(page=panel, text=_("Rectification"))
  2150. sizer = wx.BoxSizer(wx.VERTICAL)
  2151. # transformation order
  2152. self.rb_grorder = wx.RadioBox(parent=panel, id=wx.ID_ANY,
  2153. label=" %s " % _("Select rectification order"),
  2154. choices=[_('1st order'), _('2nd order'), _('3rd order')],
  2155. majorDimension=wx.RA_SPECIFY_COLS)
  2156. sizer.Add(item=self.rb_grorder, proportion=0,
  2157. flag=wx.EXPAND | wx.ALL, border=5)
  2158. self.rb_grorder.SetSelection(self.parent.gr_order - 1)
  2159. # interpolation method
  2160. gridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  2161. gridSizer.AddGrowableCol(1)
  2162. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY, label=_('Select interpolation method:')),
  2163. pos=(0,0), flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2164. self.grmethod = wx.Choice(parent=panel, id=wx.ID_ANY,
  2165. choices = self.methods)
  2166. gridSizer.Add(item=self.grmethod, pos=(0,1),
  2167. flag=wx.ALIGN_RIGHT, border=5)
  2168. self.grmethod.SetStringSelection(self.parent.gr_method)
  2169. sizer.Add(item=gridSizer, flag=wx.EXPAND | wx.ALL, border=5)
  2170. # clip to region
  2171. self.check = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  2172. label=_("clip to computational region in target location"))
  2173. sizer.Add(item=self.check, proportion=0,
  2174. flag=wx.EXPAND | wx.ALL, border=5)
  2175. self.check.SetValue(self.parent.clip_to_region)
  2176. # extension
  2177. sizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY, label=_('Extension for output maps:')),
  2178. proportion=0, flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2179. self.ext_txt = wx.TextCtrl(parent=panel, id=wx.ID_ANY, value="", size=(350,-1))
  2180. self.ext_txt.SetValue(self.parent.extension)
  2181. sizer.Add(item=self.ext_txt,
  2182. proportion=0, flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=5)
  2183. # bindings
  2184. self.ext_txt.Bind(wx.EVT_TEXT, self.OnExtension)
  2185. self.Bind(wx.EVT_RADIOBOX, self.parent.OnGROrder, self.rb_grorder)
  2186. self.Bind(wx.EVT_CHOICE, self.OnMethod, self.grmethod)
  2187. self.Bind(wx.EVT_CHECKBOX, self.OnClipRegion, self.check)
  2188. panel.SetSizer(sizer)
  2189. return panel
  2190. def OnHighlight(self, event):
  2191. """!Checkbox 'highlighthighest' checked/unchecked"""
  2192. if self.highlighthighest.IsChecked():
  2193. self.parent.highest_only = True
  2194. self.rmsWin.Disable()
  2195. else:
  2196. self.parent.highest_only = False
  2197. self.rmsWin.Enable()
  2198. def OnSDFactor(self,event):
  2199. """!New factor for RMS threshold = M + SD * factor"""
  2200. self.sdfactor = float(event.GetString())
  2201. if self.sdfactor <= 0:
  2202. GError(parent = self,
  2203. message=_('RMS threshold factor must be > 0'))
  2204. elif self.sdfactor < 1:
  2205. GError(parent = self,
  2206. message=_('RMS threshold factor is < 1\n'
  2207. 'Too many points might be highlighted'))
  2208. def OnSrcSelection(self,event):
  2209. """!Source map to display selected"""
  2210. global src_map
  2211. tmp_map = event.GetString()
  2212. if not tmp_map == '' and not tmp_map == src_map:
  2213. self.new_src_map = tmp_map
  2214. def OnTgtRastSelection(self,event):
  2215. """!Target map to display selected"""
  2216. global tgt_map
  2217. self.new_tgt_map['raster'] = event.GetString()
  2218. if self.new_tgt_map['raster'] is None:
  2219. self.new_tgt_map['raster'] = ''
  2220. def OnTgtVectSelection(self,event):
  2221. """!Target map to display selected"""
  2222. global tgt_map
  2223. self.new_tgt_map['vector'] = event.GetString()
  2224. if self.new_tgt_map['vector'] is None:
  2225. self.new_tgt_map['vector'] = ''
  2226. def OnMethod(self, event):
  2227. self.parent.gr_method = self.methods[event.GetSelection()]
  2228. def OnClipRegion(self, event):
  2229. self.parent.clip_to_region = event.IsChecked()
  2230. def OnExtension(self, event):
  2231. self.parent.extension = event.GetString()
  2232. def UpdateSettings(self):
  2233. global src_map
  2234. global tgt_map
  2235. global maptype
  2236. layers = None
  2237. UserSettings.Set(group = 'gcpman', key = 'rms', subkey = 'highestonly',
  2238. value = self.highlighthighest.GetValue())
  2239. if self.sdfactor > 0:
  2240. UserSettings.Set(group='gcpman', key='rms', subkey='sdfactor',
  2241. value=self.sdfactor)
  2242. self.parent.sdfactor = self.sdfactor
  2243. if self.parent.rmsthresh > 0:
  2244. self.parent.rmsthresh = self.parent.rmsmean + self.parent.sdfactor * self.parent.rmssd
  2245. UserSettings.Set(group='gcpman', key='symbol', subkey='color',
  2246. value=tuple(wx.FindWindowById(self.symbol['color']).GetColour()))
  2247. UserSettings.Set(group='gcpman', key='symbol', subkey='hcolor',
  2248. value=tuple(wx.FindWindowById(self.symbol['hcolor']).GetColour()))
  2249. UserSettings.Set(group='gcpman', key='symbol', subkey='scolor',
  2250. value=tuple(wx.FindWindowById(self.symbol['scolor']).GetColour()))
  2251. UserSettings.Set(group='gcpman', key='symbol', subkey='ucolor',
  2252. value=tuple(wx.FindWindowById(self.symbol['ucolor']).GetColour()))
  2253. UserSettings.Set(group='gcpman', key='symbol', subkey='unused',
  2254. value=self.showunused.GetValue())
  2255. UserSettings.Set(group='gcpman', key='symbol', subkey='size',
  2256. value=wx.FindWindowById(self.symbol['size']).GetValue())
  2257. UserSettings.Set(group='gcpman', key='symbol', subkey='width',
  2258. value=wx.FindWindowById(self.symbol['width']).GetValue())
  2259. srcrender = False
  2260. srcrenderVector = False
  2261. tgtrender = False
  2262. tgtrenderVector = False
  2263. reload_target = False
  2264. if self.new_src_map != src_map:
  2265. # remove old layer
  2266. layers = self.parent.grwiz.SrcMap.GetListOfLayers()
  2267. self.parent.grwiz.SrcMap.DeleteLayer(layers[0])
  2268. src_map = self.new_src_map
  2269. if maptype == 'raster':
  2270. cmdlist = ['d.rast', 'map=%s' % src_map]
  2271. srcrender = True
  2272. else:
  2273. cmdlist = ['d.vect', 'map=%s' % src_map]
  2274. srcrenderVector = True
  2275. self.parent.grwiz.SwitchEnv('source')
  2276. name, found = utils.GetLayerNameFromCmd(cmdlist)
  2277. self.parent.grwiz.SrcMap.AddLayer(type=maptype, command=cmdlist, l_active=True,
  2278. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  2279. self.parent.grwiz.SwitchEnv('target')
  2280. if self.new_tgt_map['raster'] != tgt_map['raster'] or \
  2281. self.new_tgt_map['vector'] != tgt_map['vector']:
  2282. # remove all layers
  2283. layers = self.parent.grwiz.TgtMap.GetListOfLayers()
  2284. while layers:
  2285. self.parent.grwiz.TgtMap.DeleteLayer(layers[0])
  2286. del layers[0]
  2287. layers = self.parent.grwiz.TgtMap.GetListOfLayers()
  2288. #self.parent.grwiz.TgtMap.DeleteAllLayers()
  2289. reload_target = True
  2290. tgt_map['raster'] = self.new_tgt_map['raster']
  2291. tgt_map['vector'] = self.new_tgt_map['vector']
  2292. if tgt_map['raster'] != '':
  2293. cmdlist = ['d.rast', 'map=%s' % tgt_map['raster']]
  2294. name, found = utils.GetLayerNameFromCmd(cmdlist)
  2295. self.parent.grwiz.TgtMap.AddLayer(type='raster', command=cmdlist, l_active=True,
  2296. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  2297. tgtrender = True
  2298. if tgt_map['vector'] != '':
  2299. cmdlist = ['d.vect', 'map=%s' % tgt_map['vector']]
  2300. name, found = utils.GetLayerNameFromCmd(cmdlist)
  2301. self.parent.grwiz.TgtMap.AddLayer(type='vector', command=cmdlist, l_active=True,
  2302. name=name, l_hidden=False, l_opacity=1.0, l_render=False)
  2303. tgtrenderVector = True
  2304. if tgt_map['raster'] == '' and tgt_map['vector'] == '':
  2305. if self.parent.show_target == True:
  2306. self.parent.show_target = False
  2307. self.parent._mgr.GetPane("target").Hide()
  2308. self.parent._mgr.Update()
  2309. self.parent.activemap.SetSelection(0)
  2310. self.parent.activemap.Enable(False)
  2311. self.parent.GetMapToolbar().Enable('zoommenu', enable = False)
  2312. else:
  2313. if self.parent.show_target == False:
  2314. self.parent.show_target = True
  2315. self.parent._mgr.GetPane("target").Show()
  2316. self.parent._mgr.Update()
  2317. self.parent.activemap.SetSelection(0)
  2318. self.parent.activemap.Enable(True)
  2319. self.parent.GetMapToolbar().Enable('zoommenu', enable = True)
  2320. self.parent.TgtMapWindow.ZoomToMap(layers = self.parent.TgtMap.GetListOfLayers())
  2321. self.parent.UpdateColours(srcrender, srcrenderVector, tgtrender, tgtrenderVector)
  2322. self.parent.SetSettings()
  2323. def OnSave(self, event):
  2324. """!Button 'Save' pressed"""
  2325. self.UpdateSettings()
  2326. fileSettings = {}
  2327. UserSettings.ReadSettingsFile(settings=fileSettings)
  2328. fileSettings['gcpman'] = UserSettings.Get(group='gcpman')
  2329. file = UserSettings.SaveToFile(fileSettings)
  2330. self.parent.parent.goutput.WriteLog(_('GCP Manager settings saved to file \'%s\'.') % file)
  2331. #self.Close()
  2332. def OnApply(self, event):
  2333. """!Button 'Apply' pressed"""
  2334. self.UpdateSettings()
  2335. #self.Close()
  2336. def OnClose(self, event):
  2337. """!Button 'Cancel' pressed"""
  2338. self.Close()