location_wizard.py 100 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718
  1. """
  2. @package location_wizard.py
  3. @brief Location wizard - creates a new GRASS Location. User can choose
  4. from multiple methods.
  5. Classes:
  6. - BaseClass
  7. - TitledPage
  8. - DatabasePage
  9. - CoordinateSystemPage
  10. - ProjectionsPage
  11. - ItemList
  12. - ProjTypePage
  13. - DatumPage
  14. - EllipsePage
  15. - GeoreferencedFilePage
  16. - EPSGPage
  17. - CustomPage
  18. - SummaryPage
  19. - RegionDef
  20. - LocationWizard
  21. - SelectDatumDialog
  22. COPYRIGHT: (C) 2007-2009 by the GRASS Development Team
  23. This program is free software under the GNU General Public
  24. License (>=v2). Read the file COPYING that comes with GRASS
  25. for details.
  26. @author Michael Barton
  27. @author Jachym Cepicky
  28. @author Martin Landa <landa.martin gmail.com>
  29. """
  30. import os
  31. import shutil
  32. import re
  33. import string
  34. import sys
  35. import locale
  36. import platform
  37. import wx
  38. import wx.lib.mixins.listctrl as listmix
  39. import wx.wizard as wiz
  40. import gcmd
  41. import globalvar
  42. import utils
  43. import grass
  44. try:
  45. import subprocess
  46. except:
  47. CompatPath = os.path.join(globalvar.ETCWXDIR)
  48. sys.path.append(CompatPath)
  49. from compat import subprocess
  50. global coordsys
  51. global north
  52. global south
  53. global east
  54. global west
  55. global resolution
  56. global wizerror
  57. coordsys = ''
  58. north = ''
  59. south = ''
  60. east = ''
  61. west = ''
  62. resolution = ''
  63. class BaseClass(wx.Object):
  64. """Base class providing basic methods"""
  65. def __init__(self):
  66. pass
  67. def MakeLabel(self, text="", style=wx.ALIGN_LEFT, parent=None):
  68. """Make aligned label"""
  69. if not parent:
  70. parent = self
  71. return wx.StaticText(parent=parent, id=wx.ID_ANY, label=text,
  72. style=style)
  73. def MakeTextCtrl(self, text='', size=(100,-1), style=0, parent=None):
  74. """Generic text control"""
  75. if not parent:
  76. parent = self
  77. return wx.TextCtrl(parent=parent, id=wx.ID_ANY, value=text,
  78. size=size, style=style)
  79. def MakeButton(self, text, id=wx.ID_ANY, size=(-1,-1), parent=None):
  80. """Generic button"""
  81. if not parent:
  82. parent = self
  83. return wx.Button(parent=parent, id=id, label=text,
  84. size=size)
  85. class TitledPage(BaseClass, wiz.WizardPageSimple):
  86. """
  87. Class to make wizard pages. Generic methods to make
  88. labels, text entries, and buttons.
  89. """
  90. def __init__(self, parent, title):
  91. self.page = wiz.WizardPageSimple.__init__(self, parent)
  92. # page title
  93. self.title = wx.StaticText(parent=self, id=wx.ID_ANY, label=title)
  94. self.title.SetFont(wx.Font(13, wx.SWISS, wx.NORMAL, wx.BOLD))
  95. # main sizer
  96. self.sizer = wx.GridBagSizer(vgap=0, hgap=0)
  97. def DoLayout(self):
  98. """Do page layout"""
  99. tmpsizer = wx.BoxSizer(wx.VERTICAL)
  100. tmpsizer.Add(item=self.title, proportion=0,
  101. flag=wx.ALIGN_CENTRE | wx.ALL,
  102. border=5)
  103. tmpsizer.Add(item=wx.StaticLine(self, -1), proportion=0,
  104. flag=wx.EXPAND | wx.ALL,
  105. border=0)
  106. tmpsizer.Add(item=self.sizer, proportion=1,
  107. flag=wx.EXPAND | wx.ALL,
  108. border=5)
  109. self.SetAutoLayout(True)
  110. self.SetSizer(tmpsizer)
  111. # tmpsizer.Fit(self)
  112. self.Layout()
  113. class DatabasePage(TitledPage):
  114. """
  115. Wizard page for setting GIS data directory and location name
  116. """
  117. def __init__(self, wizard, parent, grassdatabase):
  118. TitledPage.__init__(self, wizard, _("Define GRASS Database and Location Name"))
  119. self.grassdatabase = grassdatabase
  120. self.location = ''
  121. # buttons
  122. self.bbrowse = self.MakeButton(_("Browse"))
  123. # text controls
  124. self.tgisdbase = self.MakeTextCtrl(grassdatabase, size=(300, -1))
  125. self.tlocation = self.MakeTextCtrl("newLocation", size=(300, -1))
  126. # layout
  127. self.sizer.AddGrowableCol(3)
  128. self.sizer.Add(item=self.MakeLabel(_("GIS Data Directory:")),
  129. flag=wx.ALIGN_RIGHT |
  130. wx.ALIGN_CENTER_VERTICAL |
  131. wx.ALL, border=5,
  132. pos=(1, 1))
  133. self.sizer.Add(item=self.tgisdbase,
  134. flag=wx.ALIGN_LEFT |
  135. wx.ALIGN_CENTER_VERTICAL |
  136. wx.ALL, border=5,
  137. pos=(1, 2))
  138. self.sizer.Add(item=self.bbrowse,
  139. flag=wx.ALIGN_LEFT |
  140. wx.ALIGN_CENTER_VERTICAL |
  141. wx.ALL, border=5,
  142. pos=(1, 3))
  143. #
  144. self.sizer.Add(item=self.MakeLabel("%s:" % _("Project Location")),
  145. flag=wx.ALIGN_RIGHT |
  146. wx.ALIGN_CENTER_VERTICAL |
  147. wx.ALL, border=5,
  148. pos=(2, 1))
  149. self.sizer.Add(item=self.tlocation,
  150. flag=wx.ALIGN_LEFT |
  151. wx.ALIGN_CENTER_VERTICAL |
  152. wx.ALL, border=5,
  153. pos=(2, 2))
  154. # self.sizer.Add(item=self.MakeLabel(_("(projection/coordinate system)")),
  155. # flag=wx.ALIGN_LEFT |
  156. # wx.ALIGN_CENTER_VERTICAL |
  157. # wx.ALL, border=5,
  158. # pos=(2, 4))
  159. # bindings
  160. self.Bind(wx.EVT_BUTTON, self.OnBrowse, self.bbrowse)
  161. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  162. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  163. self.tgisdbase.Bind(wx.EVT_TEXT, self.OnChangeName)
  164. self.tlocation.Bind(wx.EVT_TEXT, self.OnChangeName)
  165. # do page layout
  166. # self.DoLayout()
  167. def OnChangeName(self, event):
  168. """Name for new location was changed"""
  169. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  170. if len(event.GetString()) > 0:
  171. if not nextButton.IsEnabled():
  172. nextButton.Enable()
  173. else:
  174. nextButton.Disable()
  175. event.Skip()
  176. def OnBrowse(self, event):
  177. dlg = wx.DirDialog(self, _("Choose GRASS data directory:"),
  178. os.getcwd(), wx.DD_DEFAULT_STYLE)
  179. if dlg.ShowModal() == wx.ID_OK:
  180. self.grassdatabase = dlg.GetPath()
  181. self.tgisdbase.SetValue(self.grassdatabase)
  182. dlg.Destroy()
  183. def OnPageChanging(self,event=None):
  184. error = ''
  185. if os.path.isdir(os.path.join(self.tgisdbase.GetValue(), self.tlocation.GetValue())):
  186. error = _("Location already exists in GRASS Database.")
  187. if error != '':
  188. dlg = wx.MessageDialog(parent=self, message="%s <%s>.%s%s" % (_("Unable to create location"),
  189. str(self.tlocation.GetValue()),
  190. os.linesep,
  191. error),
  192. caption=_("Error"), style=wx.OK | wx.ICON_ERROR)
  193. dlg.ShowModal()
  194. dlg.Destroy()
  195. event.Veto()
  196. return
  197. self.location = self.tlocation.GetValue()
  198. self.grassdatabase = self.tgisdbase.GetValue()
  199. def OnEnterPage(self, event):
  200. """Wizard page changed"""
  201. self.grassdatabase = self.tgisdbase.GetValue()
  202. self.location = self.tlocation.GetValue()
  203. event.Skip()
  204. class CoordinateSystemPage(TitledPage):
  205. """
  206. Wizard page for choosing method for location creation
  207. """
  208. def __init__(self, wizard, parent):
  209. TitledPage.__init__(self, wizard, _("Choose method for creating a new location"))
  210. self.parent = parent
  211. global coordsys
  212. # toggles
  213. self.radio1 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  214. label=_("Select coordinate system"),
  215. style = wx.RB_GROUP)
  216. self.radio2 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  217. label=_("Select EPSG code of coordinate system"))
  218. self.radio3 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  219. label=_("Use coordinate system of selected "
  220. "georeferenced file"))
  221. self.radio4 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  222. label=_("Use coordinate system of selected "
  223. "WKT or PRJ file"))
  224. self.radio5 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  225. label=_("Create custom PROJ.4 parameters "
  226. "string for coordinate system"))
  227. self.radio6 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  228. label=_("Use arbitrary non-earth "
  229. "coordinate system (XY)"))
  230. # layout
  231. self.sizer.AddGrowableCol(1)
  232. self.sizer.SetVGap(10)
  233. self.sizer.Add(item=self.radio1,
  234. flag=wx.ALIGN_LEFT, pos=(1, 1))
  235. self.sizer.Add(item=self.radio2,
  236. flag=wx.ALIGN_LEFT, pos=(2, 1))
  237. self.sizer.Add(item=self.radio3,
  238. flag=wx.ALIGN_LEFT, pos=(3, 1))
  239. self.sizer.Add(item=self.radio4,
  240. flag=wx.ALIGN_LEFT, pos=(4, 1))
  241. self.sizer.Add(item=self.radio5,
  242. flag=wx.ALIGN_LEFT, pos=(5, 1))
  243. self.sizer.Add(item=self.radio6,
  244. flag=wx.ALIGN_LEFT, pos=(6, 1))
  245. # bindings
  246. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio1.GetId())
  247. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio2.GetId())
  248. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio3.GetId())
  249. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio4.GetId())
  250. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio5.GetId())
  251. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio6.GetId())
  252. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  253. # do page layout
  254. # self.DoLayout()
  255. def OnEnterPage(self, event):
  256. global coordsys
  257. if event.GetDirection() and not coordsys:
  258. coordsys = "proj"
  259. self.SetNext(self.parent.projpage)
  260. self.parent.sumpage.SetPrev(self.parent.datumpage)
  261. if not wx.FindWindowById(wx.ID_FORWARD).IsEnabled():
  262. wx.FindWindowById(wx.ID_FORWARD).Enable()
  263. def SetVal(self, event):
  264. """Choose method"""
  265. global coordsys
  266. if event.GetId() == self.radio1.GetId():
  267. coordsys = "proj"
  268. self.SetNext(self.parent.projpage)
  269. self.parent.sumpage.SetPrev(self.parent.datumpage)
  270. elif event.GetId() == self.radio2.GetId():
  271. coordsys = "epsg"
  272. self.SetNext(self.parent.epsgpage)
  273. self.parent.sumpage.SetPrev(self.parent.epsgpage)
  274. elif event.GetId() == self.radio3.GetId():
  275. coordsys = "file"
  276. self.SetNext(self.parent.filepage)
  277. self.parent.sumpage.SetPrev(self.parent.filepage)
  278. elif event.GetId() == self.radio4.GetId():
  279. coordsys = "wkt"
  280. self.SetNext(self.parent.wktpage)
  281. self.parent.sumpage.SetPrev(self.parent.wktpage)
  282. elif event.GetId() == self.radio5.GetId():
  283. coordsys = "custom"
  284. self.SetNext(self.parent.custompage)
  285. self.parent.sumpage.SetPrev(self.parent.custompage)
  286. elif event.GetId() == self.radio6.GetId():
  287. coordsys = "xy"
  288. self.SetNext(self.parent.sumpage)
  289. self.parent.sumpage.SetPrev(self.parent.csystemspage)
  290. class ProjectionsPage(TitledPage):
  291. """
  292. Wizard page for selecting projection (select coordinate system option)
  293. """
  294. def __init__(self, wizard, parent):
  295. TitledPage.__init__(self, wizard, _("Choose projection"))
  296. self.parent = parent
  297. self.proj = ''
  298. self.projdesc = ''
  299. # text input
  300. self.tproj = self.MakeTextCtrl("", size=(200,-1))
  301. # search box
  302. self.searchb = wx.SearchCtrl(self, size=(200,-1),
  303. style=wx.TE_PROCESS_ENTER)
  304. # projection list
  305. self.projlist = ItemList(self, data=self.parent.projections.items(),
  306. columns=[_('Code'), _('Description')])
  307. # layout
  308. self.sizer.AddGrowableCol(3)
  309. self.sizer.Add(item=self.MakeLabel(_("Projection code:")),
  310. flag=wx.ALIGN_LEFT |
  311. wx.ALIGN_CENTER_VERTICAL |
  312. wx.ALL, border=5, pos=(1, 1))
  313. self.sizer.Add(item=self.tproj,
  314. flag=wx.ALIGN_RIGHT | wx.EXPAND | wx.ALL,
  315. border=5, pos=(1, 2))
  316. self.sizer.Add(item=self.MakeLabel(_("Search in description:")),
  317. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
  318. border=5, pos=(2, 1))
  319. self.sizer.Add(item=self.searchb,
  320. flag=wx.ALIGN_RIGHT | wx.EXPAND | wx.ALL,
  321. border=5, pos=(2, 2))
  322. self.sizer.AddGrowableRow(3)
  323. self.sizer.Add(item=self.projlist,
  324. flag=wx.EXPAND |
  325. wx.ALIGN_LEFT |
  326. wx.ALL, border=5, pos=(3, 1), span=(1, 3))
  327. # events
  328. self.tproj.Bind(wx.EVT_TEXT, self.OnText)
  329. self.tproj.Bind(wx.EVT_TEXT_ENTER, self.OnText)
  330. self.searchb.Bind(wx.EVT_TEXT_ENTER, self.OnSearch)
  331. self.projlist.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected)
  332. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  333. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  334. # do layout
  335. # self.Layout()
  336. def OnPageChanging(self,event):
  337. if event.GetDirection() and self.proj not in self.parent.projections:
  338. event.Veto()
  339. if self.proj == 'utm':
  340. self.parent.projtypepage.text_utm.SetEditable(True)
  341. self.parent.projtypepage.hemischoices = ['north','south']
  342. else:
  343. self.parent.projtypepage.text_utm.SetValue('')
  344. self.parent.projtypepage.text_utm.SetEditable(False)
  345. self.parent.projtypepage.hemischoices = []
  346. def OnText(self, event):
  347. """Projection name changed"""
  348. self.proj = event.GetString()
  349. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  350. if len(self.proj) == 0 and nextButton.IsEnabled():
  351. nextButton.Enable(False)
  352. if self.proj in self.parent.projections:
  353. self.projdesc = self.parent.projections[self.proj]
  354. if not nextButton.IsEnabled():
  355. nextButton.Enable()
  356. def OnEnterPage(self, event):
  357. if len(self.proj) == 0:
  358. # disable 'next' button by default
  359. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  360. else:
  361. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  362. event.Skip()
  363. def OnSearch(self, event):
  364. """Search projection by desc"""
  365. str = event.GetString()
  366. try:
  367. self.proj, self.projdesc = self.projlist.Search(index=1, str=event.GetString())
  368. except:
  369. self.proj = self.projdesc = ''
  370. event.Skip()
  371. def OnItemSelected(self, event):
  372. """Projection selected"""
  373. index = event.m_itemIndex
  374. # set values
  375. self.proj = self.projlist.GetItem(index, 0).GetText()
  376. self.projdesc = self.projlist.GetItem(index, 0).GetText()
  377. self.tproj.SetValue(self.proj)
  378. class ItemList(wx.ListCtrl,
  379. listmix.ListCtrlAutoWidthMixin,
  380. listmix.ColumnSorterMixin):
  381. """Generic list (for projections, ellipsoids, etc.)"""
  382. def __init__(self, parent, columns, data=None):
  383. wx.ListCtrl.__init__(self, parent=parent, id=wx.ID_ANY,
  384. style=wx.LC_REPORT |
  385. wx.LC_VIRTUAL |
  386. wx.LC_HRULES |
  387. wx.LC_VRULES |
  388. wx.LC_SINGLE_SEL |
  389. wx.LC_SORT_ASCENDING, size=(550, 125))
  390. # original data or None
  391. self.sourceData = data
  392. #
  393. # insert columns
  394. #
  395. i = 0
  396. for column in columns:
  397. self.InsertColumn(i, column)
  398. i += 1
  399. if self.sourceData:
  400. self.Populate()
  401. else:
  402. for i in range(self.GetColumnCount()):
  403. self.SetColumnWidth(i, wx.LIST_AUTOSIZE_USEHEADER)
  404. #
  405. # listmix
  406. #
  407. listmix.ListCtrlAutoWidthMixin.__init__(self)
  408. listmix.ColumnSorterMixin.__init__(self, self.GetColumnCount())
  409. #
  410. # add some attributes
  411. #
  412. self.attr1 = wx.ListItemAttr()
  413. self.attr1.SetBackgroundColour(wx.Colour(238,238,238))
  414. self.attr2 = wx.ListItemAttr()
  415. self.attr2.SetBackgroundColour("white")
  416. self.il = wx.ImageList(16, 16)
  417. self.sm_up = self.il.Add(wx.ArtProvider_GetBitmap(wx.ART_GO_UP, wx.ART_TOOLBAR,
  418. (16,16)))
  419. self.sm_dn = self.il.Add(wx.ArtProvider_GetBitmap(wx.ART_GO_DOWN, wx.ART_TOOLBAR,
  420. (16,16)))
  421. self.SetImageList(self.il, wx.IMAGE_LIST_SMALL)
  422. #
  423. # sort by first column
  424. #
  425. if self.sourceData:
  426. self.SortListItems(col=0, ascending=True)
  427. #
  428. # bindings
  429. #
  430. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColumnClick)
  431. def Populate(self, data=None, update=False):
  432. """Populate list"""
  433. self.itemDataMap = {}
  434. self.itemIndexMap = []
  435. if data is None:
  436. data = self.sourceData
  437. elif update:
  438. self.sourceData = data
  439. try:
  440. data.sort()
  441. self.DeleteAllItems()
  442. row = 0
  443. for value in data:
  444. # index = self.InsertStringItem(sys.maxint, str(value[0]))
  445. self.itemDataMap[row] = [value[0]]
  446. for i in range(1, len(value)):
  447. # try:
  448. # self.SetStringItem(index, i, str(value[i]))
  449. # except:
  450. # self.SetStringItem(index, i, unicode(str(value[i]), 'latin1'))
  451. self.itemDataMap[row].append(value[i])
  452. # self.SetItemData(index, row)
  453. self.itemIndexMap.append(row)
  454. row += 1
  455. self.SetItemCount(row)
  456. # set column width
  457. # for i in range(self.GetColumnCount()):
  458. # self.SetColumnWidth(i, wx.LIST_AUTOSIZE)
  459. # for i in range(self.GetColumnCount()):
  460. # if self.GetColumnWidth(i) < 80:
  461. # self.SetColumnWidth(i, 80)
  462. self.SetColumnWidth(0, 80)
  463. self.SetColumnWidth(1, 300)
  464. self.SendSizeEvent()
  465. except StandardError, e:
  466. wx.MessageBox(parent=self,
  467. message=_("Unable to read list: %s") % e,
  468. caption=_("Error"), style=wx.OK | wx.ICON_ERROR)
  469. def OnColumnClick(self, event):
  470. """Sort by column"""
  471. self._col = event.GetColumn()
  472. # remove duplicated arrow symbol from column header
  473. # FIXME: should be done automatically
  474. info = wx.ListItem()
  475. info.m_mask = wx.LIST_MASK_TEXT | wx.LIST_MASK_IMAGE
  476. info.m_image = -1
  477. for column in range(self.GetColumnCount()):
  478. info.m_text = self.GetColumn(column).GetText()
  479. self.SetColumn(column, info)
  480. event.Skip()
  481. def GetSortImages(self):
  482. """Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py"""
  483. return (self.sm_dn, self.sm_up)
  484. def OnGetItemText(self, item, col):
  485. """Get item text"""
  486. index = self.itemIndexMap[item]
  487. s = str(self.itemDataMap[index][col])
  488. return s
  489. def OnGetItemAttr(self, item):
  490. """Get item attributes"""
  491. index = self.itemIndexMap[item]
  492. if ( index % 2) == 0:
  493. return self.attr2
  494. else:
  495. return self.attr1
  496. def SortItems(self, sorter=cmp):
  497. """Sort items"""
  498. items = list(self.itemDataMap.keys())
  499. items.sort(self.Sorter)
  500. self.itemIndexMap = items
  501. # redraw the list
  502. self.Refresh()
  503. def Sorter(self, key1, key2):
  504. colName = self.GetColumn(self._col).GetText()
  505. ascending = self._colSortFlag[self._col]
  506. # convert always string
  507. item1 = self.itemDataMap[key1][self._col]
  508. item2 = self.itemDataMap[key2][self._col]
  509. if type(item1) == type('') or type(item2) == type(''):
  510. cmpVal = locale.strcoll(str(item1), str(item2))
  511. else:
  512. cmpVal = cmp(item1, item2)
  513. # If the items are equal then pick something else to make the sort value unique
  514. if cmpVal == 0:
  515. cmpVal = apply(cmp, self.GetSecondarySortValues(self._col, key1, key2))
  516. if ascending:
  517. return cmpVal
  518. else:
  519. return -cmpVal
  520. def GetListCtrl(self):
  521. """Used by listmix.ColumnSorterMixin"""
  522. return self
  523. def Search (self, index, str):
  524. """Search projection by description
  525. Return first found item or None
  526. """
  527. if str == '':
  528. self.Populate(self.sourceData)
  529. return []
  530. data = []
  531. str = str.lower()
  532. for i in range(len(self.sourceData)):
  533. try:
  534. value = self.sourceData[i][index].lower()
  535. if str in value:
  536. data.append(self.sourceData[i])
  537. except UnicodeDecodeError:
  538. # osgeo4w problem (should be fixed)
  539. pass
  540. if len(data) > 0:
  541. self.Populate(data)
  542. return data[0]
  543. else:
  544. self.Populate(self.sourceData)
  545. return []
  546. class ProjTypePage(TitledPage):
  547. """
  548. Wizard page for selecting method of setting coordinate system parameters
  549. (select coordinate system option)
  550. """
  551. def __init__(self, wizard, parent):
  552. TitledPage.__init__(self, wizard, _("Choose method of specifying georeferencing parameters"))
  553. global coordsys
  554. self.utmzone = ''
  555. self.utmhemisphere = ''
  556. self.hemischoices = ["north","south"]
  557. self.parent = parent
  558. self.radio1 = wx.RadioButton(parent=self, id=wx.ID_ANY, label=_("Select datum with associated ellipsoid"),
  559. style = wx.RB_GROUP)
  560. self.radio2 = wx.RadioButton(parent=self, id=wx.ID_ANY,
  561. label=_("Select ellipsoid"))
  562. self.title_utm = self.MakeLabel(_("Set zone for UTM projection:"))
  563. self.text_utm = self.MakeTextCtrl(size=(100,-1))
  564. self.label_utm = self.MakeLabel(_("Zone:"))
  565. self.hemisphere = wx.Choice(parent=self, id=wx.ID_ANY, size=(100, -1),
  566. choices=self.hemischoices)
  567. self.label_hemisphere = self.MakeLabel(_("Hemisphere for zone:"))
  568. # layout
  569. self.sizer.AddGrowableCol(2)
  570. self.sizer.Add(item=self.radio1,
  571. flag=wx.ALIGN_LEFT, pos=(1, 1), span=(1, 2))
  572. self.sizer.Add(item=self.radio2,
  573. flag=wx.ALIGN_LEFT, pos=(2, 1), span=(1, 2))
  574. self.sizer.Add(item=self.title_utm,
  575. flag=wx.ALIGN_LEFT | wx.ALL,
  576. border=5, pos=(4, 1), span=(1, 2))
  577. self.sizer.Add(item=self.label_utm,
  578. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
  579. border=5, pos=(5, 1), span=(1, 1))
  580. self.sizer.Add(item=self.text_utm,
  581. flag=wx.ALIGN_LEFT | wx.ALL, border=5,
  582. pos=(5, 2), span=(1, 1))
  583. self.sizer.Add(item=self.label_hemisphere,
  584. flag=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
  585. border=5, pos=(6, 1), span=(1, 1))
  586. self.sizer.Add(item=self.hemisphere,
  587. flag=wx.ALIGN_LEFT|wx.ALL,
  588. border=5, pos=(6, 2), span=(1, 1))
  589. self.title_utm.Hide()
  590. self.text_utm.Hide()
  591. self.label_utm.Hide()
  592. self.hemisphere.Hide()
  593. self.label_hemisphere.Hide()
  594. # bindings
  595. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio1.GetId())
  596. self.Bind(wx.EVT_RADIOBUTTON, self.SetVal, id=self.radio2.GetId())
  597. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChange)
  598. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  599. # do page layout
  600. # self.DoLayout()
  601. def OnPageChange(self,event=None):
  602. if event.GetDirection() and self.parent.projpage.proj == 'utm' and self.utmzone == '':
  603. wx.MessageBox('You must set a zone for a UTM projection')
  604. event.Veto()
  605. self.title_utm.Hide()
  606. self.text_utm.Hide()
  607. self.label_utm.Hide()
  608. self.hemisphere.Hide()
  609. self.label_hemisphere.Hide()
  610. def OnEnterPage(self,event):
  611. if self.parent.projpage.proj == 'utm':
  612. self.title_utm.Show()
  613. self.text_utm.Show()
  614. self.label_utm.Show()
  615. self.hemisphere.Show()
  616. self.label_hemisphere.Show()
  617. self.Bind(wx.EVT_CHOICE, self.OnHemisphere, self.hemisphere)
  618. self.Bind(wx.EVT_TEXT, self.GetUTM, self.text_utm)
  619. if not wx.FindWindowById(wx.ID_FORWARD).IsEnabled():
  620. wx.FindWindowById(wx.ID_FORWARD).Enable()
  621. event.Skip()
  622. def SetVal(self, event):
  623. global coordsys
  624. if event.GetId() == self.radio1.GetId():
  625. self.SetNext(self.parent.datumpage)
  626. self.parent.sumpage.SetPrev(self.parent.datumpage)
  627. elif event.GetId() == self.radio2.GetId():
  628. self.SetNext(self.parent.ellipsepage)
  629. self.parent.sumpage.SetPrev(self.parent.ellipsepage)
  630. def GetUTM(self, event):
  631. self.utmzone = event.GetString()
  632. def OnHemisphere(self, event):
  633. self.utmhemisphere = event.GetString()
  634. class DatumPage(TitledPage):
  635. """
  636. Wizard page for selecting datum (with associated ellipsoid)
  637. and datum transformation parameters (select coordinate system option)
  638. """
  639. def __init__(self, wizard, parent):
  640. TitledPage.__init__(self, wizard, _("Specify geodetic datum"))
  641. self.parent = parent
  642. self.datum = ''
  643. self.datumdesc = ''
  644. self.ellipsoid = ''
  645. self.datumparams = ''
  646. self.transform = ''
  647. self.transregion = ''
  648. self.transparams = ''
  649. self.hastransform = False
  650. self.proj4params = ''
  651. # text input
  652. self.tdatum = self.MakeTextCtrl("", size=(200,-1))
  653. self.ttrans = self.MakeTextCtrl("", size=(200,-1))
  654. # search box
  655. self.searchb = wx.SearchCtrl(self, size=(200,-1),
  656. style=wx.TE_PROCESS_ENTER)
  657. # create list control for datum/elipsoid list
  658. data = []
  659. for key in self.parent.datums.keys():
  660. data.append([key, self.parent.datums[key][0], self.parent.datums[key][1]])
  661. self.datumlist = ItemList(self,
  662. data=data,
  663. columns=[_('Code'), _('Description'), _('Ellipsoid')])
  664. # create list control for datum transformation parameters list
  665. data = []
  666. for key in self.parent.transforms.keys():
  667. data.append([key, self.parent.transforms[key][0], self.parent.transforms[key][1]])
  668. self.transformlist = ItemList(self,
  669. data=None,
  670. columns=[_('Code'), _('Datum'), _('Description')])
  671. self.transformlist.sourceData = data
  672. # layout
  673. self.sizer.AddGrowableCol(4)
  674. self.sizer.Add(item=self.MakeLabel(_("Datum code:")),
  675. flag=wx.ALIGN_LEFT |
  676. wx.ALIGN_CENTER_VERTICAL |
  677. wx.ALL, border=5, pos=(1, 1))
  678. self.sizer.Add(item=self.tdatum,
  679. flag=wx.ALIGN_LEFT |
  680. wx.ALIGN_CENTER_VERTICAL |
  681. wx.ALL, border=5, pos=(1, 2))
  682. self.sizer.Add(item=self.MakeLabel(_("Search in description:")),
  683. flag=wx.ALIGN_LEFT |
  684. wx.ALIGN_CENTER_VERTICAL |
  685. wx.ALL, border=5, pos=(2, 1))
  686. self.sizer.Add(item=self.searchb,
  687. flag=wx.ALIGN_LEFT |
  688. wx.ALIGN_CENTER_VERTICAL |
  689. wx.ALL, border=5, pos=(2, 2))
  690. self.sizer.AddGrowableRow(3)
  691. self.sizer.Add(item=self.datumlist,
  692. flag=wx.EXPAND |
  693. wx.ALIGN_LEFT |
  694. wx.ALL, border=5, pos=(3, 1), span=(1, 4))
  695. self.sizer.Add(item=self.MakeLabel(_("Transformation parameters:")),
  696. flag=wx.ALIGN_RIGHT |
  697. wx.ALIGN_CENTER_VERTICAL |
  698. wx.ALL, border=5, pos=(5, 1))
  699. self.sizer.Add(item=self.ttrans,
  700. flag=wx.ALIGN_LEFT |
  701. wx.ALIGN_CENTER_VERTICAL |
  702. wx.ALL, border=5, pos=(5, 2))
  703. self.sizer.Add(item=self.transformlist,
  704. flag=wx.EXPAND |
  705. wx.ALIGN_LEFT |
  706. wx.ALL, border=5, pos=(6, 1), span=(1, 4))
  707. # events
  708. self.datumlist.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnDatumSelected)
  709. self.transformlist.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnTransformSelected)
  710. self.searchb.Bind(wx.EVT_TEXT_ENTER, self.OnDSearch)
  711. self.tdatum.Bind(wx.EVT_TEXT, self.OnDText)
  712. self.tdatum.Bind(wx.EVT_TEXT_ENTER, self.OnDText)
  713. self.ttrans.Bind(wx.EVT_TEXT, self.OnTText)
  714. self.ttrans.Bind(wx.EVT_TEXT_ENTER, self.OnTText)
  715. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  716. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  717. # do page layout
  718. # self.DoLayout()
  719. def OnPageChanging(self, event):
  720. self.proj4params = ''
  721. if event.GetDirection() and self.datum not in self.parent.datums:
  722. event.Veto()
  723. if self.hastransform == True and self.transform == '':
  724. event.Veto()
  725. self.GetNext().SetPrev(self)
  726. self.parent.ellipsepage.ellipseparams = self.parent.ellipsoids[self.ellipsoid][1]
  727. def OnEnterPage(self,event):
  728. if len(self.datum) == 0 or \
  729. (self.hastransform == True and self.transform == ''):
  730. # disable 'next' button by default
  731. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  732. else:
  733. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  734. event.Skip()
  735. def OnDText(self, event):
  736. self.datum = event.GetString()
  737. self.transformlist.Search(index=1, str=self.datum)
  738. if self.transformlist.GetItemCount() > 0:
  739. self.hastransform = True
  740. else:
  741. self.hastransform = False
  742. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  743. if len(self.datum) == 0 and nextButton.IsEnabled():
  744. nextButton.Enable(False)
  745. elif self.datum in self.parent.datums:
  746. self.datumdesc = self.parent.datums[self.datum][0]
  747. self.ellipsoid = self.parent.datums[self.datum][1]
  748. self.datumparams = self.parent.datums[self.datum][2]
  749. if self.hastransform == False or \
  750. (self.hastransform == True and self.transform != ''):
  751. if not nextButton.IsEnabled():
  752. nextButton.Enable(True)
  753. else:
  754. if nextButton.IsEnabled():
  755. nextButton.Enable(False)
  756. event.Skip()
  757. def OnTText(self, event):
  758. if self.hastransform == False:
  759. event.Skip()
  760. return
  761. self.transform = event.GetString()
  762. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  763. if len(self.transform) == 0 and nextButton.IsEnabled():
  764. nextButton.Enable(False)
  765. elif self.transform in self.parent.transforms:
  766. self.transdatum = self.parent.transforms[self.transform][0]
  767. self.transregion = self.parent.transforms[self.transform][1]
  768. self.transparams = self.parent.transforms[self.transform][2]
  769. if not nextButton.IsEnabled():
  770. nextButton.Enable()
  771. def OnDSearch(self, event):
  772. str = self.searchb.GetValue()
  773. try:
  774. self.datum, self.datumdesc, self.ellipsoid = self.datumlist.Search(index=1, str=str)
  775. self.transformlist.Search(index=1, str=self.datum)
  776. except:
  777. self.datum = self.datumdesc = self.ellipsoid = ''
  778. if str == '' or self.datum == '':
  779. self.transformlist.DeleteAllItems()
  780. self.transformlist.Refresh()
  781. self.tdatum.SetValue(self.datum)
  782. event.Skip()
  783. def OnTransformSelected(self,event):
  784. index = event.m_itemIndex
  785. item = event.GetItem()
  786. self.transform = self.transformlist.GetItem(index, 0).GetText()
  787. self.transdatum = self.parent.transforms[self.transform][0]
  788. self.transregion = self.parent.transforms[self.transform][1]
  789. self.transparams = self.parent.transforms[self.transform][2]
  790. self.ttrans.SetValue(str(self.transform))
  791. def OnDatumSelected(self,event):
  792. index = event.m_itemIndex
  793. item = event.GetItem()
  794. self.datum = self.datumlist.GetItem(index, 0).GetText()
  795. self.datumdesc = self.parent.datums[self.datum][0]
  796. self.ellipsoid = self.parent.datums[self.datum][1]
  797. self.datumparams = self.parent.datums[self.datum][2]
  798. self.tdatum.SetValue(self.datum)
  799. event.Skip()
  800. class EllipsePage(TitledPage):
  801. """
  802. Wizard page for selecting ellipsoid (select coordinate system option)
  803. """
  804. def __init__(self, wizard, parent):
  805. TitledPage.__init__(self, wizard, _("Specify ellipsoid"))
  806. self.parent = parent
  807. self.ellipse = ''
  808. self.ellipsedesc = ''
  809. self.ellipseparams = ''
  810. self.proj4params = ''
  811. # text input
  812. self.tellipse = self.MakeTextCtrl("", size=(200,-1))
  813. # search box
  814. self.searchb = wx.SearchCtrl(self, size=(200,-1),
  815. style=wx.TE_PROCESS_ENTER)
  816. # create list control for ellipse list
  817. data = []
  818. # extract code, desc
  819. for key in self.parent.ellipsoids.keys():
  820. data.append([key, self.parent.ellipsoids[key][0]])
  821. self.ellipselist = ItemList(self, data=data,
  822. columns=[_('Code'), _('Description')])
  823. # layout
  824. self.sizer.AddGrowableCol(4)
  825. self.sizer.Add(item=self.MakeLabel(_("Ellipsoid code:")),
  826. flag=wx.ALIGN_RIGHT |
  827. wx.ALIGN_CENTER_VERTICAL |
  828. wx.ALL, border=5, pos=(1, 1))
  829. self.sizer.Add(item=self.tellipse,
  830. flag=wx.ALIGN_LEFT |
  831. wx.ALIGN_CENTER_VERTICAL |
  832. wx.ALL, border=5, pos=(1, 2))
  833. self.sizer.Add(item=self.MakeLabel(_("Search in description:")),
  834. flag=wx.ALIGN_RIGHT |
  835. wx.ALIGN_CENTER_VERTICAL |
  836. wx.ALL, border=5, pos=(2, 1))
  837. self.sizer.Add(item=self.searchb,
  838. flag=wx.ALIGN_LEFT |
  839. wx.ALIGN_CENTER_VERTICAL |
  840. wx.ALL, border=5, pos=(2, 2))
  841. self.sizer.AddGrowableRow(3)
  842. self.sizer.Add(item=self.ellipselist,
  843. flag=wx.EXPAND |
  844. wx.ALIGN_LEFT |
  845. wx.ALL, border=5, pos=(3, 1), span=(1, 4))
  846. # events
  847. self.ellipselist.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected)
  848. self.tellipse.Bind(wx.EVT_TEXT, self.OnText)
  849. self.tellipse.Bind(wx.EVT_TEXT_ENTER, self.OnText)
  850. self.searchb.Bind(wx.EVT_TEXT_ENTER, self.OnSearch)
  851. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  852. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  853. # do page layout
  854. # self.DoLayout()
  855. def OnEnterPage(self,event):
  856. if len(self.ellipse) == 0:
  857. # disable 'next' button by default
  858. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  859. else:
  860. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  861. event.Skip()
  862. def OnPageChanging(self, event):
  863. if event.GetDirection() and self.ellipse not in self.parent.ellipsoids:
  864. event.Veto()
  865. self.proj4params = ''
  866. self.GetNext().SetPrev(self)
  867. self.parent.datumpage.datumparams = ''
  868. self.parent.datumpage.transparams = ''
  869. # self.GetNext().SetPrev(self) (???)
  870. def OnText(self, event):
  871. """Ellipspoid code changed"""
  872. self.ellipse = event.GetString()
  873. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  874. if len(self.ellipse) == 0 and nextButton.IsEnabled():
  875. nextButton.Enable(False)
  876. self.ellipsedesc = ''
  877. self.ellipseparams = ''
  878. self.proj4params = ''
  879. elif self.ellipse in self.parent.ellipsoids:
  880. self.ellipsedesc = self.parent.ellipsoids[self.ellipse][0]
  881. self.ellipseparams = self.parent.ellipsoids[self.ellipse][1]
  882. if not nextButton.IsEnabled():
  883. nextButton.Enable()
  884. def OnSearch(self, event):
  885. """Search ellipsoid by desc"""
  886. str = event.GetString()
  887. try:
  888. self.ellipse, self.ellipsedesc = \
  889. self.ellipselist.Search(index=1, str=event.GetString())
  890. self.ellipseparams = self.parent.ellipsoids[self.ellipse][1]
  891. self.proj4params = self.parent.ellipsoids[self.ellipse][2]
  892. except:
  893. self.ellipse = self.ellipsedesc = ''
  894. self.ellipseparams = self.proj4params = ''
  895. self.tellipse.SetValue(self.ellipse)
  896. event.Skip()
  897. def OnItemSelected(self,event):
  898. index = event.m_itemIndex
  899. item = event.GetItem()
  900. self.ellipse = self.ellipselist.GetItem(index, 0).GetText()
  901. self.ellipsedesc = self.parent.ellipsoids[self.ellipse][0]
  902. self.ellipseparams = self.parent.ellipsoids[self.ellipse][1]
  903. self.tellipse.SetValue(self.ellipse)
  904. class GeoreferencedFilePage(TitledPage):
  905. """
  906. Wizard page for selecting georeferenced file to use
  907. for setting coordinate system parameters
  908. """
  909. def __init__(self, wizard, parent):
  910. TitledPage.__init__(self, wizard, _("Select georeferenced file"))
  911. self.georeffile = ''
  912. # create controls
  913. self.lfile= self.MakeLabel(_("Georeferenced file:"))
  914. self.tfile = self.MakeTextCtrl(size=(300,-1))
  915. self.bbrowse = self.MakeButton(_("Browse"))
  916. # do layout
  917. self.sizer.AddGrowableCol(3)
  918. self.sizer.Add(item=self.lfile, flag=wx.ALIGN_LEFT |
  919. wx.ALIGN_CENTRE_VERTICAL |
  920. wx.ALL, border=5, pos=(1, 1))
  921. self.sizer.Add(item=self.tfile, flag=wx.ALIGN_LEFT |
  922. wx.ALIGN_CENTRE_VERTICAL |
  923. wx.ALL, border=5, pos=(1, 2))
  924. self.sizer.Add(item=self.bbrowse, flag=wx.ALIGN_LEFT |
  925. wx.ALL, border=5, pos=(1, 3))
  926. self.bbrowse.Bind(wx.EVT_BUTTON, self.OnBrowse)
  927. self.tfile.Bind(wx.EVT_TEXT, self.OnText)
  928. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  929. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  930. # do page layout
  931. # self.DoLayout()
  932. def OnEnterPage(self, event):
  933. if len(self.georeffile) == 0:
  934. # disable 'next' button by default
  935. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  936. else:
  937. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  938. event.Skip()
  939. def OnPageChanging(self, event):
  940. if event.GetDirection() and self.georeffile == '':
  941. event.Veto()
  942. self.GetNext().SetPrev(self)
  943. event.Skip()
  944. def OnText(self, event):
  945. self.georeffile = event.GetString()
  946. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  947. if len(self.georeffile) > 0 and os.path.isfile(self.georeffile):
  948. if not nextButton.IsEnabled():
  949. nextButton.Enable(True)
  950. else:
  951. if nextButton.IsEnabled():
  952. nextButton.Enable(False)
  953. event.Skip()
  954. def OnBrowse(self, event):
  955. """Choose file"""
  956. dlg = wx.FileDialog(self,
  957. _("Select georeferenced file"),
  958. os.getcwd(), "", "*.*", wx.OPEN)
  959. if dlg.ShowModal() == wx.ID_OK:
  960. path = dlg.GetPath()
  961. self.tfile.SetValue(path)
  962. dlg.Destroy()
  963. event.Skip()
  964. def OnCreate(self, event):
  965. pass
  966. class WKTPage(TitledPage):
  967. """
  968. Wizard page for selecting WKT file to use
  969. for setting coordinate system parameters
  970. """
  971. def __init__(self, wizard, parent):
  972. TitledPage.__init__(self, wizard, _("Select WKT file"))
  973. self.wktfile = ''
  974. # create controls
  975. self.lfile= self.MakeLabel(_("WKT file:"))
  976. self.tfile = self.MakeTextCtrl(size=(300,-1))
  977. self.bbrowse = self.MakeButton(_("Browse"))
  978. # do layout
  979. self.sizer.AddGrowableCol(3)
  980. self.sizer.Add(item=self.lfile, flag=wx.ALIGN_LEFT |
  981. wx.ALIGN_CENTRE_VERTICAL |
  982. wx.ALL, border=5, pos=(1, 1))
  983. self.sizer.Add(item=self.tfile, flag=wx.ALIGN_LEFT |
  984. wx.ALIGN_CENTRE_VERTICAL |
  985. wx.ALL, border=5, pos=(1, 2))
  986. self.sizer.Add(item=self.bbrowse, flag=wx.ALIGN_LEFT |
  987. wx.ALL, border=5, pos=(1, 3))
  988. self.bbrowse.Bind(wx.EVT_BUTTON, self.OnBrowse)
  989. self.tfile.Bind(wx.EVT_TEXT, self.OnText)
  990. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  991. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  992. # do page layout
  993. # self.DoLayout()
  994. def OnEnterPage(self, event):
  995. if len(self.wktfile) == 0:
  996. # disable 'next' button by default
  997. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  998. else:
  999. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  1000. event.Skip()
  1001. def OnPageChanging(self, event):
  1002. if event.GetDirection() and self.wktfile == '':
  1003. event.Veto()
  1004. self.GetNext().SetPrev(self)
  1005. event.Skip()
  1006. def OnText(self, event):
  1007. self.wktfile = event.GetString()
  1008. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  1009. if len(self.wktfile) > 0 and os.path.isfile(self.wktfile):
  1010. if not nextButton.IsEnabled():
  1011. nextButton.Enable(True)
  1012. else:
  1013. if nextButton.IsEnabled():
  1014. nextButton.Enable(False)
  1015. event.Skip()
  1016. def OnBrowse(self, event):
  1017. """Choose file"""
  1018. dlg = wx.FileDialog(self,
  1019. _("Select WKT file"),
  1020. os.getcwd(), "", "*.*", wx.OPEN)
  1021. if dlg.ShowModal() == wx.ID_OK:
  1022. path = dlg.GetPath()
  1023. self.tfile.SetValue(path)
  1024. dlg.Destroy()
  1025. event.Skip()
  1026. def OnCreate(self, event):
  1027. pass
  1028. class EPSGPage(TitledPage):
  1029. """
  1030. Wizard page for selecting EPSG code for
  1031. setting coordinate system parameters
  1032. """
  1033. def __init__(self, wizard, parent):
  1034. TitledPage.__init__(self, wizard, _("Choose EPSG Code"))
  1035. self.parent = parent
  1036. self.epsgCodeDict = {}
  1037. self.epsgcode = None
  1038. self.epsgdesc = ''
  1039. self.epsgparams = ''
  1040. # labels
  1041. self.lfile= self.MakeLabel(_("Path to the EPSG-codes file:"),
  1042. style=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL)
  1043. self.lcode= self.MakeLabel(_("EPSG code:"),
  1044. style=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL)
  1045. self.lsearch = self.MakeLabel(_("Search in description:"),
  1046. style=wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL)
  1047. # text input
  1048. epsgdir = utils.PathJoin(os.environ["GRASS_PROJSHARE"], 'epsg')
  1049. self.tfile = self.MakeTextCtrl(text=epsgdir, size=(200,-1))
  1050. self.tcode = self.MakeTextCtrl(size=(200,-1))
  1051. # buttons
  1052. self.bbrowse = self.MakeButton(_("Browse"))
  1053. self.bbcodes = self.MakeButton(_("Reload EPSG Codes"))
  1054. # search box
  1055. self.searchb = wx.SearchCtrl(self, size=(200,-1),
  1056. style=wx.TE_PROCESS_ENTER)
  1057. self.epsglist = ItemList(self, data=None,
  1058. columns=[_('Code'), _('Description'), _('Parameters')])
  1059. # layout
  1060. self.sizer.AddGrowableCol(3)
  1061. self.sizer.Add(item=self.lfile,
  1062. flag=wx.ALIGN_LEFT |
  1063. wx.ALIGN_CENTER_VERTICAL |
  1064. wx.ALL, border=5, pos=(1, 1))
  1065. self.sizer.Add(item=self.tfile,
  1066. flag=wx.ALIGN_LEFT |
  1067. wx.ALIGN_CENTER_VERTICAL |
  1068. wx.ALL, border=5, pos=(1, 2))
  1069. self.sizer.Add(item=self.bbrowse,
  1070. flag=wx.ALIGN_LEFT |
  1071. wx.ALIGN_CENTER_VERTICAL |
  1072. wx.ALL, border=5, pos=(1, 3))
  1073. self.sizer.Add(item=self.lcode,
  1074. flag=wx.ALIGN_LEFT |
  1075. wx.ALIGN_CENTER_VERTICAL |
  1076. wx.ALL, border=5, pos=(2, 1))
  1077. self.sizer.Add(item=self.tcode,
  1078. flag=wx.ALIGN_LEFT |
  1079. wx.ALIGN_CENTER_VERTICAL |
  1080. wx.ALL, border=5, pos=(2, 2))
  1081. self.sizer.Add(item=self.lsearch,
  1082. flag=wx.ALIGN_LEFT |
  1083. wx.ALIGN_CENTER_VERTICAL |
  1084. wx.ALL, border=5, pos=(3, 1))
  1085. self.sizer.Add(item=self.searchb,
  1086. flag=wx.ALIGN_LEFT |
  1087. wx.ALIGN_CENTER_VERTICAL |
  1088. wx.ALL, border=5, pos=(3, 2))
  1089. self.sizer.Add(item=self.bbcodes,
  1090. flag=wx.ALIGN_LEFT |
  1091. wx.ALIGN_CENTER_VERTICAL |
  1092. wx.ALL, border=5, pos=(3, 3))
  1093. self.sizer.AddGrowableRow(4)
  1094. self.sizer.Add(item=self.epsglist,
  1095. flag=wx.ALIGN_LEFT | wx.EXPAND, pos=(4, 1),
  1096. span=(1, 3))
  1097. # events
  1098. self.bbrowse.Bind(wx.EVT_BUTTON, self.OnBrowse)
  1099. self.bbcodes.Bind(wx.EVT_BUTTON, self.OnBrowseCodes)
  1100. self.tcode.Bind(wx.EVT_TEXT, self.OnText)
  1101. self.tcode.Bind(wx.EVT_TEXT_ENTER, self.OnText)
  1102. self.epsglist.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected)
  1103. self.searchb.Bind(wx.EVT_TEXT_ENTER, self.OnSearch)
  1104. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  1105. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  1106. # do page layout
  1107. # self.DoLayout()
  1108. def OnEnterPage(self, event):
  1109. if not self.epsgcode:
  1110. # disable 'next' button by default
  1111. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  1112. else:
  1113. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  1114. # load default epsg database file
  1115. self.OnBrowseCodes(None)
  1116. event.Skip()
  1117. def OnPageChanging(self, event):
  1118. if event.GetDirection() and not self.epsgcode:
  1119. event.Veto()
  1120. self.GetNext().SetPrev(self)
  1121. def OnText(self, event):
  1122. self.epsgcode = event.GetString()
  1123. try:
  1124. self.epsgcode = int(self.epsgcode)
  1125. except:
  1126. self.epsgcode = None
  1127. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  1128. if self.epsgcode and self.epsgcode in self.epsgCodeDict.keys():
  1129. self.epsgdesc = self.epsgCodeDict[self.epsgcode][0]
  1130. self.epsgparams = self.epsgCodeDict[self.epsgcode][1]
  1131. if not nextButton.IsEnabled():
  1132. nextButton.Enable(True)
  1133. else:
  1134. self.epsgcode = None # not found
  1135. if nextButton.IsEnabled():
  1136. nextButton.Enable(False)
  1137. self.epsgdesc = self.epsgparams = ''
  1138. def OnSearch(self, event):
  1139. value = self.searchb.GetValue()
  1140. if self.epsglist.GetItemCount() == 0:
  1141. event.Skip()
  1142. return
  1143. try:
  1144. self.epsgcode = self.epsglist.Search(index=1, str=value)[0]
  1145. self.tcode.SetValue(str(self.epsgcode))
  1146. except IndexError: # -> no item found
  1147. self.epsgcode = None
  1148. self.tcode.SetValue('')
  1149. self.searchb.SetValue('')
  1150. event.Skip()
  1151. def OnBrowse(self, event):
  1152. """Define path for EPSG code file"""
  1153. path = os.path.dirname(self.tfile.GetValue())
  1154. if not path:
  1155. path = os.getcwd()
  1156. dlg = wx.FileDialog(parent=self, message=_("Choose EPSG codes file"),
  1157. defaultDir=path, defaultFile="", wildcard="*", style=wx.OPEN)
  1158. if dlg.ShowModal() == wx.ID_OK:
  1159. path = dlg.GetPath()
  1160. self.tfile.SetValue(path)
  1161. self.OnBrowseCodes(None)
  1162. dlg.Destroy()
  1163. event.Skip()
  1164. def OnItemSelected(self, event):
  1165. """EPSG code selected from the list"""
  1166. index = event.m_itemIndex
  1167. item = event.GetItem()
  1168. self.epsgcode = int(self.epsglist.GetItem(index, 0).GetText())
  1169. self.epsgdesc = self.epsglist.GetItem(index, 1).GetText()
  1170. self.tcode.SetValue(str(self.epsgcode))
  1171. event.Skip()
  1172. def OnBrowseCodes(self, event, search=None):
  1173. """Browse EPSG codes"""
  1174. # try:
  1175. if True:
  1176. data = []
  1177. self.epsgCodeDict = {}
  1178. f = open(self.tfile.GetValue(), "r")
  1179. i = 0
  1180. code = None
  1181. for line in f.readlines():
  1182. line = line.strip()
  1183. if len(line) < 1:
  1184. continue
  1185. if line[0] == '#':
  1186. descr = line[1:].strip()
  1187. elif line[0] == '<':
  1188. code, params = line.split(" ", 1)
  1189. code = int(code.replace('<', '').replace('>', ''))
  1190. if code is not None:
  1191. data.append((code, descr, params))
  1192. self.epsgCodeDict[code] = (descr, params)
  1193. code = None
  1194. i += 1
  1195. f.close()
  1196. self.epsglist.Populate(data, update=True)
  1197. # except StandardError, e:
  1198. # wx.MessageBox(parent=self,
  1199. # message=_("Unable to read EPGS codes: %s") % e,
  1200. # caption=_("Error"), style=wx.OK | wx.ICON_ERROR | wx.CENTRE)
  1201. # self.epsglist.Populate([], update=True)
  1202. class CustomPage(TitledPage):
  1203. """
  1204. Wizard page for entering custom PROJ.4 string
  1205. for setting coordinate system parameters
  1206. """
  1207. def __init__(self, wizard, parent):
  1208. TitledPage.__init__(self, wizard,
  1209. _("Choose method of specifying georeferencing parameters"))
  1210. global coordsys
  1211. self.customstring = ''
  1212. self.parent = parent
  1213. # widgets
  1214. self.text_proj4string = self.MakeTextCtrl(size=(400, 200),
  1215. style=wx.TE_MULTILINE)
  1216. self.label_proj4string = self.MakeLabel(_("Enter PROJ.4 parameters string:"))
  1217. # layout
  1218. self.sizer.AddGrowableCol(2)
  1219. self.sizer.Add(self.label_proj4string,
  1220. flag=wx.ALIGN_LEFT | wx.ALL,
  1221. border=5, pos=(1, 1))
  1222. self.sizer.AddGrowableRow(2)
  1223. self.sizer.Add(self.text_proj4string,
  1224. flag=wx.ALIGN_LEFT | wx.ALL | wx.EXPAND,
  1225. border=5, pos=(2, 1), span=(1, 2))
  1226. self.text_proj4string.Bind(wx.EVT_TEXT, self.GetProjstring)
  1227. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGING, self.OnPageChanging)
  1228. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  1229. # do page layout
  1230. # self.DoLayout()
  1231. def OnEnterPage(self, event):
  1232. if len(self.customstring) == 0:
  1233. # disable 'next' button by default
  1234. wx.FindWindowById(wx.ID_FORWARD).Enable(False)
  1235. else:
  1236. wx.FindWindowById(wx.ID_FORWARD).Enable(True)
  1237. event.Skip()
  1238. def OnPageChanging(self, event):
  1239. if event.GetDirection() and not self.customstring:
  1240. event.Veto()
  1241. self.GetNext().SetPrev(self)
  1242. def GetProjstring(self, event):
  1243. """Change proj string"""
  1244. # TODO: check PROJ.4 syntax
  1245. self.customstring = event.GetString()
  1246. nextButton = wx.FindWindowById(wx.ID_FORWARD)
  1247. if len(self.customstring) == 0:
  1248. if nextButton.IsEnabled():
  1249. nextButton.Enable(False)
  1250. else:
  1251. if not nextButton.IsEnabled():
  1252. nextButton.Enable()
  1253. class SummaryPage(TitledPage):
  1254. """
  1255. Shows summary result of choosing coordinate system parameters
  1256. prior to creating location
  1257. """
  1258. def __init__(self, wizard, parent):
  1259. TitledPage.__init__(self, wizard, _("Summary"))
  1260. self.parent = parent
  1261. # labels
  1262. self.ldatabase = self.MakeLabel("")
  1263. self.llocation = self.MakeLabel("")
  1264. self.lprojection = self.MakeLabel("")
  1265. self.lproj4string = self.MakeLabel("")
  1266. self.lproj4stringLabel = self.MakeLabel("")
  1267. self.lprojection.Wrap(400)
  1268. self.Bind(wiz.EVT_WIZARD_PAGE_CHANGED, self.OnEnterPage)
  1269. # self.Bind(wx.EVT_BUTTON, self.OnFinish, wx.ID_FINISH)
  1270. # do sub-page layout
  1271. self.__DoLayout()
  1272. def __DoLayout(self):
  1273. """Do page layout"""
  1274. self.sizer.AddGrowableCol(1)
  1275. self.sizer.Add(item=self.MakeLabel(_("GRASS Database:")),
  1276. flag=wx.ALIGN_LEFT | wx.ALL,
  1277. border=5, pos=(1, 0))
  1278. self.sizer.Add(item=self.ldatabase,
  1279. flag=wx.ALIGN_LEFT | wx.ALL,
  1280. border=5, pos=(1, 1))
  1281. self.sizer.Add(item=self.MakeLabel(_("Location Name:")),
  1282. flag=wx.ALIGN_LEFT | wx.ALL,
  1283. border=5, pos=(2, 0))
  1284. self.sizer.Add(item=self.llocation,
  1285. flag=wx.ALIGN_LEFT | wx.ALL,
  1286. border=5, pos=(2, 1))
  1287. self.sizer.Add(item=self.MakeLabel(_("Projection:")),
  1288. flag=wx.ALIGN_LEFT | wx.ALL,
  1289. border=5, pos=(3, 0))
  1290. self.sizer.Add(item=self.lprojection,
  1291. flag=wx.ALIGN_LEFT | wx.ALL,
  1292. border=5, pos=(3, 1))
  1293. self.sizer.Add(item=self.lproj4stringLabel,
  1294. flag=wx.ALIGN_LEFT | wx.ALL,
  1295. border=5, pos=(4, 0))
  1296. self.sizer.Add(item=self.lproj4string,
  1297. flag=wx.ALIGN_LEFT | wx.ALL,
  1298. border=5, pos=(4, 1))
  1299. self.sizer.Add(item=(10,20),
  1300. flag=wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
  1301. border=5, pos=(5, 0), span=(1, 2))
  1302. # self.sizer.AddGrowableRow(6)
  1303. #self.sizer.Add(item=self.MakeLabel(_("You can set the default extents "
  1304. # "and resolution after creating new location%s"
  1305. # "or you can set them during a working session.") % os.linesep,
  1306. # style=wx.ALIGN_CENTER),
  1307. # flag=wx.ALIGN_CENTRE | wx.ALL, border=5, pos=(6, 0),
  1308. # span=(1, 2))
  1309. def OnEnterPage(self,event):
  1310. """
  1311. Insert values into text controls for summary of location creation options
  1312. """
  1313. database = self.parent.startpage.grassdatabase
  1314. location = self.parent.startpage.location
  1315. global coordsys
  1316. if coordsys not in ['proj', 'epsg']:
  1317. self.lproj4stringLabel.Hide()
  1318. self.lproj4string.Hide()
  1319. self.lproj4stringLabel.SetLabel('')
  1320. self.lproj4string.SetLabel('')
  1321. else:
  1322. self.lproj4string.Show()
  1323. self.lproj4stringLabel.SetLabel(_("PROJ.4 definition:"))
  1324. if coordsys == 'proj':
  1325. self.lproj4string.SetLabel(self.parent.CreateProj4String())
  1326. else:
  1327. self.lproj4string.SetLabel(self.parent.epsgpage.epsgCodeDict[self.parent.epsgpage.epsgcode][1])
  1328. self.lproj4string.Wrap(400)
  1329. projection = self.parent.projpage.proj
  1330. projdesc = self.parent.projpage.projdesc
  1331. utmzone = self.parent.projtypepage.utmzone
  1332. utmhemisphere = self.parent.projtypepage.utmhemisphere
  1333. ellipse = self.parent.ellipsepage.ellipse
  1334. ellipsedesc = self.parent.ellipsepage.ellipsedesc
  1335. datum = self.parent.datumpage.datum
  1336. datumdesc = self.parent.datumpage.datumdesc
  1337. ellipsoid = self.parent.datumpage.ellipsoid
  1338. datumparams = self.parent.datumpage.datumparams
  1339. transform = self.parent.datumpage.transform
  1340. transregion = self.parent.datumpage.transregion
  1341. transparams = self.parent.datumpage.transparams
  1342. self.ldatabase.SetLabel(str(database))
  1343. self.llocation.SetLabel(str(location))
  1344. label = ''
  1345. if coordsys == 'epsg':
  1346. label = 'EPSG code %s (%s)' % (self.parent.epsgpage.epsgcode, self.parent.epsgpage.epsgdesc)
  1347. self.lprojection.SetLabel(label)
  1348. elif coordsys == 'file':
  1349. label = 'matches file %s' % self.parent.filepage.georeffile
  1350. self.lprojection.SetLabel(label)
  1351. elif coordsys == 'wkt':
  1352. label = 'matches file %s' % self.parent.wktpage.wktfile
  1353. self.lprojection.SetLabel(label)
  1354. elif coordsys == 'proj':
  1355. label = ('%s, %s%s' % (projdesc, datumdesc, ellipsedesc))
  1356. self.lprojection.SetLabel(label)
  1357. elif coordsys == 'xy':
  1358. label = ('XY coordinate system (not projected).')
  1359. self.lprojection.SetLabel(label)
  1360. elif coordsys == 'custom':
  1361. label = ('%s' % self.parent.custompage.customstring)
  1362. self.lprojection.SetLabel(label)
  1363. def OnFinish(self, event):
  1364. dlg = wx.MessageDialog(parent=self.wizard,
  1365. message=_("Do you want to create GRASS location <%s>?") % location,
  1366. caption=_("Create new location?"),
  1367. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  1368. if dlg.ShowModal() == wx.ID_NO:
  1369. dlg.Destroy()
  1370. event.Veto()
  1371. else:
  1372. dlg.Destroy()
  1373. event.Skip()
  1374. class LocationWizard(wx.Object):
  1375. """
  1376. Start wizard here and finish wizard here
  1377. """
  1378. def __init__(self, parent, grassdatabase):
  1379. global coordsys
  1380. self.parent = parent
  1381. #
  1382. # define wizard image
  1383. #
  1384. # file = "loc_wizard.png"
  1385. file = "loc_wizard_qgis.png"
  1386. imagePath = os.path.join(globalvar.ETCWXDIR, "images", file)
  1387. wizbmp = wx.Image(imagePath, wx.BITMAP_TYPE_PNG)
  1388. # wizbmp.Rescale(250,600)
  1389. wizbmp = wizbmp.ConvertToBitmap()
  1390. #
  1391. # get georeferencing information from tables in $GISBASE/etc
  1392. #
  1393. self.__readData()
  1394. #
  1395. # define wizard pages
  1396. #
  1397. self.wizard = wiz.Wizard(parent, id=wx.ID_ANY, title=_("Define new GRASS Location"),
  1398. bitmap=wizbmp)
  1399. self.startpage = DatabasePage(self.wizard, self, grassdatabase)
  1400. self.csystemspage = CoordinateSystemPage(self.wizard, self)
  1401. self.projpage = ProjectionsPage(self.wizard, self)
  1402. self.datumpage = DatumPage(self.wizard, self)
  1403. self.projtypepage = ProjTypePage(self.wizard,self)
  1404. self.epsgpage = EPSGPage(self.wizard, self)
  1405. self.filepage = GeoreferencedFilePage(self.wizard, self)
  1406. self.wktpage = WKTPage(self.wizard, self)
  1407. self.ellipsepage = EllipsePage(self.wizard, self)
  1408. self.custompage = CustomPage(self.wizard, self)
  1409. self.sumpage = SummaryPage(self.wizard, self)
  1410. #
  1411. # set the initial order of the pages
  1412. # (should follow the epsg line)
  1413. #
  1414. self.startpage.SetNext(self.csystemspage)
  1415. self.csystemspage.SetPrev(self.startpage)
  1416. self.csystemspage.SetNext(self.sumpage)
  1417. self.projpage.SetPrev(self.csystemspage)
  1418. self.projpage.SetNext(self.projtypepage)
  1419. self.projtypepage.SetPrev(self.projpage)
  1420. self.projtypepage.SetNext(self.datumpage)
  1421. self.datumpage.SetPrev(self.projtypepage)
  1422. self.datumpage.SetNext(self.sumpage)
  1423. self.ellipsepage.SetPrev(self.projtypepage)
  1424. self.ellipsepage.SetNext(self.sumpage)
  1425. self.epsgpage.SetPrev(self.csystemspage)
  1426. self.epsgpage.SetNext(self.sumpage)
  1427. self.filepage.SetPrev(self.csystemspage)
  1428. self.filepage.SetNext(self.sumpage)
  1429. self.wktpage.SetPrev(self.csystemspage)
  1430. self.wktpage.SetNext(self.sumpage)
  1431. self.custompage.SetPrev(self.csystemspage)
  1432. self.custompage.SetNext(self.sumpage)
  1433. self.sumpage.SetPrev(self.csystemspage)
  1434. #
  1435. # do pages layout
  1436. #
  1437. self.startpage.DoLayout()
  1438. self.csystemspage.DoLayout()
  1439. self.projpage.DoLayout()
  1440. self.datumpage.DoLayout()
  1441. self.projtypepage.DoLayout()
  1442. self.epsgpage.DoLayout()
  1443. self.filepage.DoLayout()
  1444. self.wktpage.DoLayout()
  1445. self.ellipsepage.DoLayout()
  1446. self.custompage.DoLayout()
  1447. self.sumpage.DoLayout()
  1448. self.wizard.FitToPage(self.datumpage)
  1449. # new location created?
  1450. self.location = None
  1451. success = False
  1452. #
  1453. # run wizard...
  1454. #
  1455. if self.wizard.RunWizard(self.startpage):
  1456. success = self.OnWizFinished()
  1457. if success == True:
  1458. self.wizard.Destroy()
  1459. self.location = self.startpage.location
  1460. dlg = wx.MessageDialog(parent=self.parent,
  1461. message=_("Do you want to set the default "
  1462. "region extents and resolution now?"),
  1463. caption=_("Location <%s> created") % self.location,
  1464. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  1465. dlg.CenterOnScreen()
  1466. if dlg.ShowModal() == wx.ID_YES:
  1467. dlg.Destroy()
  1468. defineRegion = RegionDef(self.parent, location=self.location)
  1469. defineRegion.CenterOnScreen()
  1470. defineRegion.Show()
  1471. else:
  1472. dlg.Destroy()
  1473. elif success == False:
  1474. self.wizard.Destroy()
  1475. wx.MessageBox(parent=self.parent,
  1476. message="%s" % _("Unable to create new location. "
  1477. "Location <%s> not created.") % self.startpage.location,
  1478. caption=_("Location wizard"),
  1479. style=wx.OK | wx.ICON_ERROR | wx.CENTRE)
  1480. else: # None
  1481. pass
  1482. else:
  1483. win = wx.MessageBox(parent=self.parent,
  1484. message=_("Location wizard canceled. "
  1485. "Location not created."),
  1486. caption=_("Location wizard"))
  1487. def __readData(self):
  1488. """Get georeferencing information from tables in $GISBASE/etc"""
  1489. # read projection definitions
  1490. f = open(os.path.join(globalvar.ETCDIR, "projections"), "r")
  1491. self.projections = {}
  1492. for line in f.readlines():
  1493. line = line.expandtabs(1)
  1494. line = line.strip()
  1495. if line == '' or line[0] == "#":
  1496. continue
  1497. proj, projdesc = line.split(":", 1)
  1498. self.projections[proj.strip()] = projdesc.strip()
  1499. f.close()
  1500. # read datum definitions
  1501. f = open(os.path.join(globalvar.ETCDIR, "datum.table"), "r")
  1502. self.datums = {}
  1503. paramslist = []
  1504. for line in f.readlines():
  1505. line = line.expandtabs(1)
  1506. line = line.strip()
  1507. if line == '' or line[0] == "#":
  1508. continue
  1509. datum, info = line.split(" ", 1)
  1510. info = info.strip()
  1511. datumdesc, params = info.split(" ", 1)
  1512. datumdesc = datumdesc.strip('"')
  1513. paramlist = params.split()
  1514. ellipsoid = paramlist.pop(0)
  1515. self.datums[datum] = (datumdesc.replace('_', ' '), ellipsoid, paramlist)
  1516. f.close()
  1517. # read datum transforms parameters
  1518. f = open(os.path.join(globalvar.ETCDIR, "datumtransform.table"), "r")
  1519. self.transforms = {}
  1520. j = 1
  1521. for line in f.readlines():
  1522. if j < 10:
  1523. transcode = 'T0' + str(j)
  1524. else:
  1525. transcode = 'T' + str(j)
  1526. line = line.expandtabs(1)
  1527. line = line.strip()
  1528. if line == '' or line[0] == "#":
  1529. continue
  1530. datum, rest = line.split(" ", 1)
  1531. rest = rest.strip('" ')
  1532. params, rest = rest.split('"', 1)
  1533. params = params.strip()
  1534. rest = rest.strip('" ')
  1535. try:
  1536. region, info = rest.split('"', 1)
  1537. info = info.strip('" ')
  1538. info = region + ': ' + info
  1539. except:
  1540. info = rest
  1541. self.transforms[transcode] = (datum, info, params)
  1542. j += 1
  1543. f.close()
  1544. # read ellipsiod definitions
  1545. f = open(os.path.join(globalvar.ETCDIR, "ellipse.table"), "r")
  1546. self.ellipsoids = {}
  1547. for line in f.readlines():
  1548. line = line.expandtabs(1)
  1549. line = line.strip()
  1550. if line == '' or line[0] == "#":
  1551. continue
  1552. ellipse, rest = line.split(" ", 1)
  1553. rest = rest.strip('" ')
  1554. desc, params = rest.split('"', 1)
  1555. desc = desc.strip('" ')
  1556. paramslist = params.split()
  1557. self.ellipsoids[ellipse] = (desc, paramslist)
  1558. f.close()
  1559. def OnWizFinished(self):
  1560. database = self.startpage.grassdatabase
  1561. location = self.startpage.location
  1562. global coordsys
  1563. success = False
  1564. # location already exists?
  1565. if os.path.isdir(os.path.join(database,location)):
  1566. dlg = wx.MessageDialog(parent=self.wizard,
  1567. message="%s <%s>: %s" % \
  1568. (_("Unable to create new location"),
  1569. os.path.join(database, location),
  1570. _("Location already exists in GRASS Database.")),
  1571. caption=_("Error"),
  1572. style=wx.OK | wx.ICON_ERROR)
  1573. dlg.ShowModal()
  1574. dlg.Destroy()
  1575. return False
  1576. # current GISDbase or a new one?
  1577. current_gdb = grass.gisenv()['GISDBASE']
  1578. if current_gdb != database:
  1579. # change to new GISDbase or create new one
  1580. if os.path.isdir(database) != True:
  1581. # create new directory
  1582. os.mkdir(database)
  1583. # change to new GISDbase directory
  1584. gcmd.RunCommand('g.gisenv',
  1585. parent = self.wizard,
  1586. set='GISDBASE=%s' % database)
  1587. wx.MessageBox(parent=self.wizard,
  1588. message=_("Location <%(loc)s> will be created "
  1589. "in GIS data directory <%(dir)s>."
  1590. "You will need to change the default GIS "
  1591. "data directory in the GRASS startup screen.") % \
  1592. { 'loc' : location, 'dir' : database},
  1593. caption=_("New GIS data directory"),
  1594. style=wx.OK | wx.ICON_INFORMATION | wx.CENTRE)
  1595. if coordsys == "xy":
  1596. success = self.XYCreate()
  1597. elif coordsys == "latlong":
  1598. rows = int(round((float(north) - float(south)) / float(resolution)))
  1599. cols = int(round((float(east) - float(west)) / float(resolution)))
  1600. cells = int(rows * cols)
  1601. success = self.LatlongCreate()
  1602. elif coordsys == "proj":
  1603. proj4string = self.CreateProj4String()
  1604. success = self.Proj4Create(proj4string)
  1605. elif coordsys == 'custom':
  1606. success = self.CustomCreate()
  1607. elif coordsys == "epsg":
  1608. success = self.EPSGCreate()
  1609. elif coordsys == "file":
  1610. success = self.FileCreate()
  1611. elif coordsys == "wkt":
  1612. success = self.WKTCreate()
  1613. return success
  1614. def XYCreate(self):
  1615. """Create an XY location"""
  1616. database = self.startpage.grassdatabase
  1617. location = self.startpage.location
  1618. # create location directory and PERMANENT mapset
  1619. try:
  1620. os.mkdir(os.path.join(database, location))
  1621. os.mkdir(os.path.join(database, location, 'PERMANENT'))
  1622. # create DEFAULT_WIND and WIND files
  1623. regioninfo = ['proj: 0',
  1624. 'zone: 0',
  1625. 'north: 1',
  1626. 'south: 0',
  1627. 'east: 1',
  1628. 'west: 0',
  1629. 'cols: 1',
  1630. 'rows: 1',
  1631. 'e-w resol: 1',
  1632. 'n-s resol: 1',
  1633. 'top: 1',
  1634. 'bottom: 0',
  1635. 'cols3: 1',
  1636. 'rows3: 1',
  1637. 'depths: 1',
  1638. 'e-w resol3: 1',
  1639. 'n-s resol3: 1',
  1640. 't-b resol: 1']
  1641. defwind = open(os.path.join(database, location,
  1642. "PERMANENT", "DEFAULT_WIND"), 'w')
  1643. for param in regioninfo:
  1644. defwind.write(param + '%s' % os.linesep)
  1645. defwind.close()
  1646. shutil.copy(os.path.join(database, location, "PERMANENT", "DEFAULT_WIND"),
  1647. os.path.join(database, location, "PERMANENT", "WIND"))
  1648. # create MYNAME file
  1649. myname = open(os.path.join(database, location, "PERMANENT",
  1650. "MYNAME"), 'w')
  1651. myname.write('%s' % os.linesep)
  1652. myname.close()
  1653. return True
  1654. except OSError, e:
  1655. dlg = wx.MessageDialog(parent=self.wizard,
  1656. message="%s: %s" % (_("Unable to create new location"), e),
  1657. caption=_("Error"),
  1658. style=wx.OK | wx.ICON_ERROR)
  1659. dlg.ShowModal()
  1660. dlg.Destroy()
  1661. return False
  1662. def CreateProj4String(self):
  1663. """Constract PROJ.4 string"""
  1664. location = self.startpage.location
  1665. proj = self.projpage.proj
  1666. projdesc = self.projpage.projdesc
  1667. utmzone = self.projtypepage.utmzone
  1668. utmhemisphere = self.projtypepage.utmhemisphere
  1669. datum = self.datumpage.datum
  1670. if self.datumpage.datumdesc:
  1671. datumdesc = self.datumpage.datumdesc +' - ' + self.datumpage.ellipsoid
  1672. else:
  1673. datumdesc = ''
  1674. datumparams = self.datumpage.datumparams
  1675. transparams = self.datumpage.transparams
  1676. ellipse = self.ellipsepage.ellipse
  1677. ellipsedesc = self.ellipsepage.ellipsedesc
  1678. ellipseparams = self.ellipsepage.ellipseparams
  1679. #
  1680. # creating PROJ.4 string
  1681. #
  1682. if proj == 'll':
  1683. proj4string = '+proj=longlat'
  1684. elif proj == 'utm':
  1685. proj4string = '+proj=%s +zone=%s' % (proj, utmzone)
  1686. if utmhemisphere == 'south':
  1687. proj4string += '+south'
  1688. else:
  1689. proj4string = '+proj=%s ' % (proj)
  1690. proj4params = ''
  1691. # set ellipsoid parameters
  1692. for item in ellipseparams:
  1693. if item[:4] == 'f=1/':
  1694. item = '+rf='+item[4:]
  1695. else:
  1696. item = '+'+item
  1697. proj4params = '%s %s' % (proj4params, item)
  1698. # set datum and transform parameters if relevant
  1699. if datumparams:
  1700. for item in datumparams:
  1701. proj4params = '%s +%s' % (proj4params,item)
  1702. if transparams:
  1703. proj4params = '%s +no_defs +%s' % (proj4params,transparams)
  1704. else:
  1705. proj4params = '%s +no_defs' % proj4params
  1706. else:
  1707. proj4params = '%s +no_defs' % proj4params
  1708. return '%s %s' % (proj4string, proj4params)
  1709. def Proj4Create(self, proj4string):
  1710. """
  1711. Create a new location for selected projection
  1712. """
  1713. # creating location from PROJ.4 string passed to g.proj
  1714. ret = gcmd.RunCommand('g.proj',
  1715. flags = 'c',
  1716. proj4 = proj4string,
  1717. location = self.startpage.location)
  1718. if ret == 0:
  1719. return True
  1720. return False
  1721. def CustomCreate(self):
  1722. """Create a new location based on given proj4 string"""
  1723. proj4string = self.custompage.customstring
  1724. location = self.startpage.location
  1725. ret = gcmd.RunCommand('g.proj',
  1726. flags = 'c',
  1727. proj4 = proj4string,
  1728. location = location)
  1729. if ret == 0:
  1730. return True
  1731. return False
  1732. def EPSGCreate(self):
  1733. """
  1734. Create a new location from an EPSG code.
  1735. """
  1736. epsgcode = self.epsgpage.epsgcode
  1737. epsgdesc = self.epsgpage.epsgdesc
  1738. location = self.startpage.location
  1739. # should not happend
  1740. if epsgcode == '':
  1741. wx.MessageBox(parent=self,
  1742. message="%s: %s" % (_("Unable to create new location"), _("EPSG code missing.")),
  1743. caption=_("Error"), style=wx.OK | wx.ICON_ERROR)
  1744. return False
  1745. # creating location
  1746. ret = gcmd.RunCommand('g.proj',
  1747. read = True,
  1748. epsg = epsgcode,
  1749. datumtrans = '-1')
  1750. dtoptions = {}
  1751. if ret:
  1752. line = ret.splitlines()
  1753. i = 0
  1754. while i < len(line):
  1755. if line[i] == '---':
  1756. for j in range(3):
  1757. dtoptions[line[i+1]] = (line[i+2],
  1758. line[i+3],
  1759. line[i+4])
  1760. i += 5
  1761. if dtoptions != {}:
  1762. dtrans = ''
  1763. # open a dialog to select datum transform number
  1764. dlg = SelectDatumDialog(self.parent, datums=dtoptions)
  1765. if dlg.ShowModal() == wx.ID_OK:
  1766. dtrans = dlg.GetDatum()
  1767. if dtrans == '':
  1768. wx.MessageBox(parent=self.parent,
  1769. message=_('Datum transform is required.'),
  1770. caption=_("Error"),
  1771. style=wx.OK | wx.ICON_ERROR | wx.CENTRE)
  1772. dlg.Destroy()
  1773. return False
  1774. dlg.Destroy()
  1775. else:
  1776. dlg.Destroy()
  1777. return False
  1778. datumtrans = dtrans
  1779. else:
  1780. datumtrans = '1'
  1781. ret = gcmd.RunCommand('g.proj',
  1782. flags = 'c',
  1783. epsg = epsgcode,
  1784. location = location,
  1785. datumtrans = datumtrans)
  1786. if ret == 0:
  1787. return True
  1788. return False
  1789. def FileCreate(self):
  1790. """
  1791. Create a new location from a georeferenced file
  1792. """
  1793. georeffile = self.filepage.georeffile
  1794. location = self.startpage.location
  1795. # this should not happen
  1796. if not georeffile or not os.path.isfile(georeffile):
  1797. dlg = wx.MessageBox(parent=self.wizard,
  1798. message="%s: %s ('%s')" % \
  1799. (_("Unable to create new location"),
  1800. _("file not found"),
  1801. georeffile),
  1802. caption=("Error"), style=wx.OK | wx.ICON_ERROR)
  1803. return False
  1804. # creating location
  1805. ret = gcmd.RunCommand('g.proj',
  1806. flags = 'c',
  1807. georef = georeffile,
  1808. location = location)
  1809. if ret == 0:
  1810. return True
  1811. return False
  1812. def WKTCreate(self):
  1813. """
  1814. Create a new location from a WKT file
  1815. """
  1816. wktfile = self.wktpage.wktfile
  1817. location = self.startpage.location
  1818. # this should not happen
  1819. if not wktfile or not os.path.isfile(wktfile):
  1820. dlg = wx.MessageBox(parent=self.wizard,
  1821. message="%s: %s ('%s')" % \
  1822. (_("Unable to create new location"),
  1823. _("file not found"),
  1824. wktfile),
  1825. caption=("Error"), style=wx.OK | wx.ICON_ERROR)
  1826. return False
  1827. # creating location
  1828. ret = gcmd.RunCommand('g.proj',
  1829. flags = 'c',
  1830. wkt = wktfile,
  1831. location = location)
  1832. if ret == 0:
  1833. return True
  1834. return False
  1835. class RegionDef(BaseClass, wx.Frame):
  1836. """
  1837. Page for setting default region extents and resolution
  1838. """
  1839. def __init__(self, parent, id=wx.ID_ANY,
  1840. title=_("Set default region extent and resolution"), location=None):
  1841. wx.Frame.__init__(self, parent, id, title, size=(650,300))
  1842. panel = wx.Panel(self, id=wx.ID_ANY)
  1843. self.SetIcon(wx.Icon(os.path.join(globalvar.ETCICONDIR, 'grass.ico'), wx.BITMAP_TYPE_ICO))
  1844. self.parent = parent
  1845. self.location = location
  1846. #
  1847. # default values
  1848. #
  1849. # 2D
  1850. self.north = 1.0
  1851. self.south = 0.0
  1852. self.east = 1.0
  1853. self.west = 0.0
  1854. self.nsres = 1.0
  1855. self.ewres = 1.0
  1856. # 3D
  1857. self.top = 1.0
  1858. self.bottom = 0.0
  1859. # self.nsres3 = 1.0
  1860. # self.ewres3 = 1.0
  1861. self.tbres = 1.0
  1862. #
  1863. # inputs
  1864. #
  1865. # 2D
  1866. self.tnorth = self.MakeTextCtrl(text=str(self.north), size=(150, -1), parent=panel)
  1867. self.tsouth = self.MakeTextCtrl(str(self.south), size=(150, -1), parent=panel)
  1868. self.twest = self.MakeTextCtrl(str(self.west), size=(150, -1), parent=panel)
  1869. self.teast = self.MakeTextCtrl(str(self.east), size=(150, -1), parent=panel)
  1870. self.tnsres = self.MakeTextCtrl(str(self.nsres), size=(150, -1), parent=panel)
  1871. self.tewres = self.MakeTextCtrl(str(self.ewres), size=(150, -1), parent=panel)
  1872. #
  1873. # labels
  1874. #
  1875. self.lrows = self.MakeLabel(parent=panel)
  1876. self.lcols = self.MakeLabel(parent=panel)
  1877. self.lcells = self.MakeLabel(parent=panel)
  1878. #
  1879. # buttons
  1880. #
  1881. self.bset = self.MakeButton(text=_("&Set region"), id=wx.ID_OK, parent=panel)
  1882. self.bcancel = wx.Button(panel, id=wx.ID_CANCEL)
  1883. self.bset.SetDefault()
  1884. #
  1885. # image
  1886. #
  1887. self.img = wx.Image(os.path.join(globalvar.ETCWXDIR, "images",
  1888. "qgis_world.png"), wx.BITMAP_TYPE_PNG).ConvertToBitmap()
  1889. #
  1890. # set current working environment to PERMANENT mapset
  1891. # in selected location in order to set default region (WIND)
  1892. #
  1893. envval = {}
  1894. ret = gcmd.RunCommand('g.gisenv',
  1895. read = True)
  1896. if ret:
  1897. for line in ret.splitlines():
  1898. key, val = line.split('=')
  1899. envval[key] = val
  1900. self.currlocation = envval['LOCATION_NAME'].strip("';")
  1901. self.currmapset = envval['MAPSET'].strip("';")
  1902. if self.currlocation != self.location or self.currmapset != 'PERMANENT':
  1903. gcmd.RunCommand('g.gisenv',
  1904. set = 'LOCATION_NAME=%s' % self.location)
  1905. gcmd.RunCommand('g.gisenv',
  1906. set = 'MAPSET=PERMANENT')
  1907. else:
  1908. dlg = wx.MessageBox(parent=self,
  1909. message=_('Invalid location selected.'),
  1910. caption=_("Error"), style=wx.ID_OK | wx.ICON_ERROR)
  1911. return
  1912. #
  1913. # get current region settings
  1914. #
  1915. region = {}
  1916. ret = gcmd.RunCommand('g.region',
  1917. read = True,
  1918. flags = 'gp3')
  1919. if ret:
  1920. for line in ret.splitlines():
  1921. key, val = line.split('=')
  1922. region[key] = float(val)
  1923. else:
  1924. dlg = wx.MessageBox(parent=self,
  1925. message=_("Invalid region"),
  1926. caption=_("Error"), style=wx.ID_OK | wx.ICON_ERROR)
  1927. dlg.ShowModal()
  1928. dlg.Destroy()
  1929. return
  1930. #
  1931. # update values
  1932. # 2D
  1933. self.north = float(region['n'])
  1934. self.south = float(region['s'])
  1935. self.east = float(region['e'])
  1936. self.west = float(region['w'])
  1937. self.nsres = float(region['nsres'])
  1938. self.ewres = float(region['ewres'])
  1939. self.rows = int(region['rows'])
  1940. self.cols = int(region['cols'])
  1941. self.cells = int(region['cells'])
  1942. # 3D
  1943. self.top = float(region['t'])
  1944. self.bottom = float(region['b'])
  1945. # self.nsres3 = float(region['nsres3'])
  1946. # self.ewres3 = float(region['ewres3'])
  1947. self.tbres = float(region['tbres'])
  1948. self.depth = int(region['depths'])
  1949. self.cells3 = int(region['3dcells'])
  1950. #
  1951. # 3D box collapsable
  1952. #
  1953. self.infoCollapseLabelExp = _("Click here to show 3D settings")
  1954. self.infoCollapseLabelCol = _("Click here to hide 3D settings")
  1955. self.settings3D = wx.CollapsiblePane(parent=panel,
  1956. label=self.infoCollapseLabelExp,
  1957. style=wx.CP_DEFAULT_STYLE |
  1958. wx.CP_NO_TLW_RESIZE | wx.EXPAND)
  1959. self.MakeSettings3DPaneContent(self.settings3D.GetPane())
  1960. self.settings3D.Collapse(False) # FIXME
  1961. self.Bind(wx.EVT_COLLAPSIBLEPANE_CHANGED, self.OnSettings3DPaneChanged, self.settings3D)
  1962. #
  1963. # set current region settings
  1964. #
  1965. self.tnorth.SetValue(str(self.north))
  1966. self.tsouth.SetValue(str(self.south))
  1967. self.twest.SetValue(str(self.west))
  1968. self.teast.SetValue(str(self.east))
  1969. self.tnsres.SetValue(str(self.nsres))
  1970. self.tewres.SetValue(str(self.ewres))
  1971. self.ttop.SetValue(str(self.top))
  1972. self.tbottom.SetValue(str(self.bottom))
  1973. # self.tnsres3.SetValue(str(self.nsres3))
  1974. # self.tewres3.SetValue(str(self.ewres3))
  1975. self.ttbres.SetValue(str(self.tbres))
  1976. self.lrows.SetLabel(_("Rows: %d" % self.rows))
  1977. self.lcols.SetLabel(_("Cols: %d" % self.cols))
  1978. self.lcells.SetLabel(_("Cells: %d" % self.cells))
  1979. #
  1980. # bindings
  1981. #
  1982. self.Bind(wx.EVT_BUTTON, self.OnSetButton, self.bset)
  1983. self.Bind(wx.EVT_BUTTON, self.OnCancel, self.bcancel)
  1984. self.tnorth.Bind(wx.EVT_TEXT, self.OnValue)
  1985. self.tsouth.Bind(wx.EVT_TEXT, self.OnValue)
  1986. self.teast.Bind(wx.EVT_TEXT, self.OnValue)
  1987. self.twest.Bind(wx.EVT_TEXT, self.OnValue)
  1988. self.tnsres.Bind(wx.EVT_TEXT, self.OnValue)
  1989. self.tewres.Bind(wx.EVT_TEXT, self.OnValue)
  1990. self.ttop.Bind(wx.EVT_TEXT, self.OnValue)
  1991. self.tbottom.Bind(wx.EVT_TEXT, self.OnValue)
  1992. # self.tnsres3.Bind(wx.EVT_TEXT, self.OnValue)
  1993. # self.tewres3.Bind(wx.EVT_TEXT, self.OnValue)
  1994. self.ttbres.Bind(wx.EVT_TEXT, self.OnValue)
  1995. self.__DoLayout(panel)
  1996. self.SetMinSize(self.GetBestSize())
  1997. self.minWindowSize = self.GetMinSize()
  1998. def MakeSettings3DPaneContent(self, pane):
  1999. """Create 3D region settings pane"""
  2000. border = wx.BoxSizer(wx.VERTICAL)
  2001. gridSizer = wx.GridBagSizer(vgap=0, hgap=0)
  2002. # inputs
  2003. self.ttop = wx.TextCtrl(parent=pane, id=wx.ID_ANY, value=str(self.top),
  2004. size=(150, -1))
  2005. self.tbottom = wx.TextCtrl(parent=pane, id=wx.ID_ANY, value=str(self.bottom),
  2006. size=(150, -1))
  2007. self.ttbres = wx.TextCtrl(parent=pane, id=wx.ID_ANY, value=str(self.tbres),
  2008. size=(150, -1))
  2009. # self.tnsres3 = wx.TextCtrl(parent=pane, id=wx.ID_ANY, value=str(self.nsres3),
  2010. # size=(150, -1))
  2011. # self.tewres3 = wx.TextCtrl(parent=pane, id=wx.ID_ANY, value=str(self.ewres3),
  2012. # size=(150, -1))
  2013. #labels
  2014. self.ldepth = wx.StaticText(parent=pane, label=_("Depth: %d") % self.depth)
  2015. self.lcells3 = wx.StaticText(parent=pane, label=_("3D Cells: %d") % self.cells3)
  2016. # top
  2017. gridSizer.Add(item=wx.StaticText(parent=pane, label=_("Top")),
  2018. flag=wx.ALIGN_CENTER |
  2019. wx.LEFT | wx.RIGHT | wx.TOP, border=5,
  2020. pos=(0, 1))
  2021. gridSizer.Add(item=self.ttop,
  2022. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2023. wx.ALL, border=5, pos=(1, 1))
  2024. # bottom
  2025. gridSizer.Add(item=wx.StaticText(parent=pane, label=_("Bottom")),
  2026. flag=wx.ALIGN_CENTER |
  2027. wx.LEFT | wx.RIGHT | wx.TOP, border=5,
  2028. pos=(0, 2))
  2029. gridSizer.Add(item=self.tbottom,
  2030. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2031. wx.ALL, border=5, pos=(1, 2))
  2032. # tbres
  2033. gridSizer.Add(item=wx.StaticText(parent=pane, label=_("T-B resolution")),
  2034. flag=wx.ALIGN_CENTER |
  2035. wx.LEFT | wx.RIGHT | wx.TOP, border=5,
  2036. pos=(0, 3))
  2037. gridSizer.Add(item=self.ttbres,
  2038. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2039. wx.ALL, border=5, pos=(1, 3))
  2040. # res
  2041. # gridSizer.Add(item=wx.StaticText(parent=pane, label=_("3D N-S resolution")),
  2042. # flag=wx.ALIGN_CENTER |
  2043. # wx.LEFT | wx.RIGHT | wx.TOP, border=5,
  2044. # pos=(2, 1))
  2045. # gridSizer.Add(item=self.tnsres3,
  2046. # flag=wx.ALIGN_CENTER_HORIZONTAL |
  2047. # wx.ALL, border=5, pos=(3, 1))
  2048. # gridSizer.Add(item=wx.StaticText(parent=pane, label=_("3D E-W resolution")),
  2049. # flag=wx.ALIGN_CENTER |
  2050. # wx.LEFT | wx.RIGHT | wx.TOP, border=5,
  2051. # pos=(2, 3))
  2052. # gridSizer.Add(item=self.tewres3,
  2053. # flag=wx.ALIGN_CENTER_HORIZONTAL |
  2054. # wx.ALL, border=5, pos=(3, 3))
  2055. # rows/cols/cells
  2056. gridSizer.Add(item=self.ldepth,
  2057. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2058. wx.ALL, border=5, pos=(2, 1))
  2059. gridSizer.Add(item=self.lcells3,
  2060. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2061. wx.ALL, border=5, pos=(2, 2))
  2062. border.Add(item=gridSizer, proportion=1,
  2063. flag=wx.ALL | wx.ALIGN_CENTER | wx.EXPAND, border=5)
  2064. pane.SetSizer(border)
  2065. border.Fit(pane)
  2066. def OnSettings3DPaneChanged(self, event):
  2067. """Collapse 3D settings box"""
  2068. if self.settings3D.IsExpanded():
  2069. self.settings3D.SetLabel(self.infoCollapseLabelCol)
  2070. self.Layout()
  2071. self.SetSize(self.GetBestSize())
  2072. self.SetMinSize(self.GetSize())
  2073. else:
  2074. self.settings3D.SetLabel(self.infoCollapseLabelExp)
  2075. self.Layout()
  2076. self.SetSize(self.minWindowSize)
  2077. self.SetMinSize(self.minWindowSize)
  2078. self.SendSizeEvent()
  2079. def __DoLayout(self, panel):
  2080. """Window layout"""
  2081. frameSizer = wx.BoxSizer(wx.VERTICAL)
  2082. gridSizer = wx.GridBagSizer(vgap=0, hgap=0)
  2083. settings3DSizer = wx.BoxSizer(wx.VERTICAL)
  2084. buttonSizer = wx.BoxSizer(wx.HORIZONTAL)
  2085. # north
  2086. gridSizer.Add(item=self.MakeLabel(text=_("North"), parent=panel),
  2087. flag=wx.ALIGN_BOTTOM | wx.ALIGN_CENTER_HORIZONTAL |
  2088. wx.TOP | wx.LEFT | wx.RIGHT, border=5, pos=(0, 2))
  2089. gridSizer.Add(item=self.tnorth,
  2090. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2091. wx.ALIGN_CENTER_VERTICAL |
  2092. wx.ALL, border=5, pos=(1, 2))
  2093. # west
  2094. gridSizer.Add(item=self.MakeLabel(text=_("West"), parent=panel),
  2095. flag=wx.ALIGN_RIGHT |
  2096. wx.ALIGN_CENTER_VERTICAL |
  2097. wx.LEFT | wx.TOP | wx.BOTTOM, border=5, pos=(2, 0))
  2098. gridSizer.Add(item=self.twest,
  2099. flag=wx.ALIGN_RIGHT |
  2100. wx.ALIGN_CENTER_VERTICAL |
  2101. wx.ALL, border=5, pos=(2, 1))
  2102. gridSizer.Add(item=wx.StaticBitmap(panel, wx.ID_ANY, self.img, (-1, -1),
  2103. (self.img.GetWidth(), self.img.GetHeight())),
  2104. flag=wx.ALIGN_CENTER |
  2105. wx.ALIGN_CENTER_VERTICAL |
  2106. wx.ALL, border=5, pos=(2, 2))
  2107. # east
  2108. gridSizer.Add(item=self.teast,
  2109. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2110. wx.ALIGN_CENTER_VERTICAL |
  2111. wx.ALL, border=5, pos=(2, 3))
  2112. gridSizer.Add(item=self.MakeLabel(text=_("East"), parent=panel),
  2113. flag=wx.ALIGN_LEFT |
  2114. wx.ALIGN_CENTER_VERTICAL |
  2115. wx.RIGHT | wx.TOP | wx.BOTTOM, border=5, pos=(2, 4))
  2116. # south
  2117. gridSizer.Add(item=self.tsouth,
  2118. flag=wx.ALIGN_CENTER_HORIZONTAL |
  2119. wx.ALIGN_CENTER_VERTICAL |
  2120. wx.ALL, border=5, pos=(3, 2))
  2121. gridSizer.Add(item=self.MakeLabel(text=_("South"), parent=panel),
  2122. flag=wx.ALIGN_TOP | wx.ALIGN_CENTER_HORIZONTAL |
  2123. wx.LEFT | wx.RIGHT | wx.BOTTOM, border=5, pos=(4, 2))
  2124. # ns-res
  2125. gridSizer.Add(item=self.MakeLabel(text=_("N-S resolution"), parent=panel),
  2126. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2127. wx.TOP | wx.LEFT | wx.RIGHT, border=5, pos=(5, 1))
  2128. gridSizer.Add(item=self.tnsres,
  2129. flag=wx.ALIGN_RIGHT |
  2130. wx.ALIGN_CENTER_VERTICAL |
  2131. wx.ALL, border=5, pos=(6, 1))
  2132. # ew-res
  2133. gridSizer.Add(item=self.MakeLabel(text=_("E-W resolution"), parent=panel),
  2134. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2135. wx.TOP | wx.LEFT | wx.RIGHT, border=5, pos=(5, 3))
  2136. gridSizer.Add(item=self.tewres,
  2137. flag=wx.ALIGN_RIGHT |
  2138. wx.ALIGN_CENTER_VERTICAL |
  2139. wx.ALL, border=5, pos=(6, 3))
  2140. # rows/cols/cells
  2141. gridSizer.Add(item=self.lrows,
  2142. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2143. wx.ALL, border=5, pos=(7, 1))
  2144. gridSizer.Add(item=self.lcells,
  2145. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2146. wx.ALL, border=5, pos=(7, 2))
  2147. gridSizer.Add(item=self.lcols,
  2148. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER |
  2149. wx.ALL, border=5, pos=(7, 3))
  2150. # 3D
  2151. settings3DSizer.Add(item=self.settings3D,
  2152. flag=wx.ALL,
  2153. border=5)
  2154. # buttons
  2155. buttonSizer.Add(item=self.bcancel, proportion=1,
  2156. flag=wx.ALIGN_RIGHT |
  2157. wx.ALIGN_CENTER_VERTICAL |
  2158. wx.ALL, border=10)
  2159. buttonSizer.Add(item=self.bset, proportion=1,
  2160. flag=wx.ALIGN_CENTER |
  2161. wx.ALIGN_CENTER_VERTICAL |
  2162. wx.ALL, border=10)
  2163. frameSizer.Add(item=gridSizer, proportion=1,
  2164. flag=wx.ALL | wx.ALIGN_CENTER, border=5)
  2165. frameSizer.Add(item=settings3DSizer, proportion=0,
  2166. flag=wx.ALL | wx.ALIGN_CENTER, border=5)
  2167. frameSizer.Add(item=buttonSizer, proportion=0,
  2168. flag=wx.ALL | wx.ALIGN_RIGHT, border=5)
  2169. self.SetAutoLayout(True)
  2170. panel.SetSizer(frameSizer)
  2171. frameSizer.Fit(panel)
  2172. self.Layout()
  2173. def OnValue(self, event):
  2174. """Set given value"""
  2175. try:
  2176. if event.GetId() == self.tnorth.GetId():
  2177. self.north = float(event.GetString())
  2178. elif event.GetId() == self.tsouth.GetId():
  2179. self.south = float(event.GetString())
  2180. elif event.GetId() == self.teast.GetId():
  2181. self.east = float(event.GetString())
  2182. elif event.GetId() == self.twest.GetId():
  2183. self.west = float(event.GetString())
  2184. elif event.GetId() == self.tnsres.GetId():
  2185. self.nsres = float(event.GetString())
  2186. elif event.GetId() == self.tewres.GetId():
  2187. self.ewres = float(event.GetString())
  2188. elif event.GetId() == self.ttop.GetId():
  2189. self.top = float(event.GetString())
  2190. elif event.GetId() == self.tbottom.GetId():
  2191. self.bottom = float(event.GetString())
  2192. # elif event.GetId() == self.tnsres3.GetId():
  2193. # self.nsres3 = float(event.GetString())
  2194. # elif event.GetId() == self.tewres3.GetId():
  2195. # self.ewres3 = float(event.GetString())
  2196. elif event.GetId() == self.ttbres.GetId():
  2197. self.tbres = float(event.GetString())
  2198. self.__UpdateInfo()
  2199. except ValueError, e:
  2200. if len(event.GetString()) > 0 and event.GetString() != '-':
  2201. dlg = wx.MessageBox(parent=self,
  2202. message=_("Invalid value: %s") % e,
  2203. caption=_("Error"),
  2204. style=wx.OK | wx.ICON_ERROR)
  2205. # reset values
  2206. self.tnorth.SetValue(str(self.north))
  2207. self.tsouth.SetValue(str(self.south))
  2208. self.teast.SetValue(str(self.east))
  2209. self.twest.SetValue(str(self.west))
  2210. self.tnsres.SetValue(str(self.nsres))
  2211. self.tewres.SetValue(str(self.ewres))
  2212. self.ttop.SetValue(str(self.top))
  2213. self.tbottom.SetValue(str(self.bottom))
  2214. self.ttbres.SetValue(str(self.tbres))
  2215. # self.tnsres3.SetValue(str(self.nsres3))
  2216. # self.tewres3.SetValue(str(self.ewres3))
  2217. event.Skip()
  2218. def __UpdateInfo(self):
  2219. """Update number of rows/cols/cells"""
  2220. self.rows = int((self.north - self.south) / self.nsres)
  2221. self.cols = int((self.east - self.west) / self.ewres)
  2222. self.cells = self.rows * self.cols
  2223. self.depth = int((self.top - self.bottom) / self.tbres)
  2224. self.cells3 = self.rows * self.cols * self.depth
  2225. # 2D
  2226. self.lrows.SetLabel(_("Rows: %d") % self.rows)
  2227. self.lcols.SetLabel(_("Cols: %d") % self.cols)
  2228. self.lcells.SetLabel(_("Cells: %d") % self.cells)
  2229. # 3D
  2230. self.ldepth.SetLabel(_("Depth: %d" % self.depth))
  2231. self.lcells3.SetLabel(_("3D Cells: %d" % self.cells3))
  2232. def OnSetButton(self, event=None):
  2233. """Set default region"""
  2234. ret = gcmd.RunCommand('g.region',
  2235. flags = 'sgpa',
  2236. n = self.north,
  2237. s = self.south,
  2238. e = self.east,
  2239. w = self.west,
  2240. nsres = self.nsres,
  2241. ewres = self.ewres,
  2242. t = self.top,
  2243. b = self.bottom,
  2244. tbres = self.tbres)
  2245. if ret == 0:
  2246. self.Destroy()
  2247. def OnCancel(self, event):
  2248. self.Destroy()
  2249. class SelectDatumDialog(wx.Dialog):
  2250. """Dialog for selecting datum transformations"""
  2251. def __init__(self, parent, datums, title=_("Select datum transformation"),
  2252. pos=wx.DefaultPosition, size=wx.DefaultSize, style=wx.DEFAULT_DIALOG_STYLE):
  2253. self.datums = datums
  2254. wx.Dialog.__init__(self, parent, wx.ID_ANY, title, pos, size, style)
  2255. panel = wx.Panel(self, wx.ID_ANY)
  2256. sizer = wx.BoxSizer(wx.VERTICAL)
  2257. #
  2258. # dialog body
  2259. #
  2260. bodyBox = wx.StaticBox(parent=panel, id=wx.ID_ANY,
  2261. label=" %s " % _("List of datum transformations"))
  2262. bodySizer = wx.StaticBoxSizer(bodyBox, wx.HORIZONTAL)
  2263. gridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  2264. gridSizer.Add(item=wx.StaticText(parent=panel, label=_("Datums (select to see description):")),
  2265. flag=wx.ALIGN_CENTER_VERTICAL, pos=(0, 0))
  2266. items = self.datums.keys()
  2267. utils.ListSortLower(items)
  2268. self.cdatums = wx.ComboBox(parent=panel, id=wx.ID_ANY,
  2269. style=wx.CB_SIMPLE | wx.CB_READONLY,
  2270. choices=items,
  2271. size=(60,-1))
  2272. self.cdatums.SetSelection(0)
  2273. self.cdatums.Bind(wx.EVT_COMBOBOX, self.OnChangeDatum)
  2274. gridSizer.Add(item=self.cdatums, pos=(0, 1))
  2275. self.textWidth = self.GetSize()[0]
  2276. self.datumDesc = wx.StaticText(parent=panel, size=(self.textWidth,-1),
  2277. label='\n'.join(self.datums[self.cdatums.GetStringSelection()]))
  2278. self.datumDesc.Wrap(self.textWidth)
  2279. gridSizer.Add(item=self.datumDesc, flag=wx.EXPAND,
  2280. pos=(1, 0), span=(1, 2))
  2281. bodySizer.Add(item=gridSizer, proportion=1,
  2282. flag=wx.ALL | wx.ALIGN_CENTER | wx.EXPAND, border=5)
  2283. #
  2284. # buttons
  2285. #
  2286. btnsizer = wx.StdDialogButtonSizer()
  2287. btn = wx.Button(parent=panel, id=wx.ID_OK)
  2288. btn.SetDefault()
  2289. btnsizer.AddButton(btn)
  2290. btn = wx.Button(parent=panel, id=wx.ID_CANCEL)
  2291. btnsizer.AddButton(btn)
  2292. btnsizer.Realize()
  2293. sizer.Add(item=bodySizer, proportion=1,
  2294. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  2295. sizer.Add(item=btnsizer, proportion=0,
  2296. flag=wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border=5)
  2297. #
  2298. # set panel sizer
  2299. #
  2300. panel.SetSizer(sizer)
  2301. sizer.Fit(panel)
  2302. self.SetSize(self.GetBestSize())
  2303. def OnChangeDatum(self, event):
  2304. """Datum changed, update description text"""
  2305. self.datumDesc.SetLabel('\n'.join(self.datums[event.GetString()]))
  2306. # self.textWidth = self.GetSize()[0]
  2307. # self.datumDesc.Wrap(self.textWidth)
  2308. event.Skip()
  2309. def GetDatum(self):
  2310. """Get selected datum"""
  2311. return self.cdatums.GetStringSelection()
  2312. if __name__ == "__main__":
  2313. app = wx.PySimpleApp()
  2314. # gWizard = LocationWizard(None, "")
  2315. gWizard = RegionDef(None)
  2316. gWizzard.Show()
  2317. app.MainLoop()