frame.py 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341
  1. """!
  2. @package iclass.frame
  3. @brief wxIClass frame with toolbar for digitizing training areas and
  4. for spectral signature analysis.
  5. Classes:
  6. - frame::IClassMapFrame
  7. - frame::MapManager
  8. (C) 2006-2013 by the GRASS Development Team
  9. This program is free software under the GNU General Public
  10. License (>=v2). Read the file COPYING that comes with GRASS
  11. for details.
  12. @author Vaclav Petras <wenzeslaus gmail.com>
  13. @author Anna Kratochvilova <kratochanna gmail.com>
  14. """
  15. import os
  16. import sys
  17. import copy
  18. import tempfile
  19. from core.utils import _
  20. if __name__ == "__main__":
  21. gui_wx_path = os.path.join(os.getenv('GISBASE'), 'etc', 'gui', 'wxpython')
  22. if gui_wx_path not in sys.path:
  23. sys.path.append(gui_wx_path)
  24. from core import globalvar
  25. import wx
  26. from ctypes import *
  27. try:
  28. from grass.lib.imagery import *
  29. from grass.lib.vector import *
  30. haveIClass = True
  31. errMsg = ''
  32. except ImportError, e:
  33. haveIClass = False
  34. errMsg = _("Loading imagery lib failed.\n%s") % e
  35. import grass.script as grass
  36. from mapdisp import statusbar as sb
  37. from mapwin.buffered import BufferedMapWindow
  38. from vdigit.toolbars import VDigitToolbar
  39. from gui_core.mapdisp import DoubleMapFrame
  40. from core.render import Map, MapLayer
  41. from core.gcmd import RunCommand, GMessage, GError, GWarning
  42. from gui_core.dialogs import SetOpacityDialog
  43. from mapwin.base import MapWindowProperties
  44. from dbmgr.vinfo import VectorDBInfo
  45. import grass.script as grass
  46. from iclass.digit import IClassVDigitWindow, IClassVDigit
  47. from iclass.toolbars import IClassMapToolbar, IClassMiscToolbar,\
  48. IClassToolbar, IClassMapManagerToolbar
  49. from iclass.statistics import StatisticsData, Statistics, BandStatistics
  50. from iclass.dialogs import CategoryListCtrl, IClassCategoryManagerDialog,\
  51. IClassGroupDialog, IClassSignatureFileDialog,\
  52. IClassExportAreasDialog, IClassMapDialog
  53. from iclass.plots import PlotPanel
  54. from grass.pydispatch.signal import Signal
  55. class IClassMapFrame(DoubleMapFrame):
  56. """! wxIClass main frame
  57. It has two map windows one for digitizing training areas and one for
  58. result preview.
  59. It generates histograms, raster maps and signature files using
  60. @c I_iclass_* functions from C imagery library.
  61. It is wxGUI counterpart of old i.class module.
  62. """
  63. def __init__(self, parent = None, giface = None, title = _("GRASS GIS Supervised Classification Tool"),
  64. toolbars = ["iClassMisc", "iClassMap", "vdigit", "iClass"],
  65. size = (875, 600), name = 'IClassWindow', **kwargs):
  66. """!
  67. @param parent (no parent is expected)
  68. @param title window title
  69. @param toolbars dictionary of active toolbars (defalult value represents all toolbars)
  70. @param size default size
  71. """
  72. DoubleMapFrame.__init__(self, parent = parent, title = title,
  73. name = name,
  74. firstMap = Map(), secondMap = Map(),
  75. **kwargs)
  76. self._giface = giface
  77. self.mapWindowProperties = MapWindowProperties()
  78. self.mapWindowProperties.setValuesFromUserSettings()
  79. # show computation region by defaut
  80. self.mapWindowProperties.showRegion = True
  81. self.firstMapWindow = IClassVDigitWindow(parent = self, giface = self._giface,
  82. properties=self.mapWindowProperties,
  83. map = self.firstMap)
  84. self.secondMapWindow = BufferedMapWindow(parent=self, giface=self._giface,
  85. properties=self.mapWindowProperties,
  86. Map=self.secondMap)
  87. self.MapWindow = self.firstMapWindow # current by default
  88. self._bindWindowsActivation()
  89. self._setUpMapWindow(self.firstMapWindow)
  90. self._setUpMapWindow(self.secondMapWindow)
  91. self.firstMapWindow.InitZoomHistory()
  92. self.secondMapWindow.InitZoomHistory()
  93. # TODO: for vdigit: it does nothing here because areas do not produce this info
  94. self.firstMapWindow.digitizingInfo.connect(
  95. lambda text:
  96. self.statusbarManager.statusbarItems['coordinates'].SetAdditionalInfo(text))
  97. self.firstMapWindow.digitizingInfoUnavailable.connect(
  98. lambda:
  99. self.statusbarManager.statusbarItems['coordinates'].SetAdditionalInfo(None))
  100. self.SetSize(size)
  101. #
  102. #Signals
  103. #
  104. self.groupSet = Signal("IClassMapFrame.groupSet")
  105. self.categoryChanged = Signal('IClassMapFrame.categoryChanged')
  106. self.InitStatistics()
  107. #
  108. # Add toolbars
  109. #
  110. toolbarsCopy = toolbars[:]
  111. if sys.platform == 'win32':
  112. self.AddToolbar(toolbarsCopy.pop(1))
  113. toolbarsCopy.reverse()
  114. else:
  115. self.AddToolbar(toolbarsCopy.pop(0))
  116. for toolb in toolbarsCopy:
  117. self.AddToolbar(toolb)
  118. self.firstMapWindow.SetToolbar(self.toolbars['vdigit'])
  119. self.GetMapToolbar().GetActiveMapTool().Bind(wx.EVT_CHOICE, self.OnUpdateActive)
  120. #
  121. # Add statusbar
  122. #
  123. # items for choice
  124. self.statusbarItems = [sb.SbCoordinates,
  125. sb.SbRegionExtent,
  126. sb.SbCompRegionExtent,
  127. sb.SbShowRegion,
  128. sb.SbAlignExtent,
  129. sb.SbResolution,
  130. sb.SbDisplayGeometry,
  131. sb.SbMapScale,
  132. sb.SbGoTo,
  133. sb.SbProjection]
  134. # create statusbar and its manager
  135. statusbar = self.CreateStatusBar(number = 4, style = 0)
  136. statusbar.SetStatusWidths([-5, -2, -1, -1])
  137. self.statusbarManager = sb.SbManager(mapframe = self, statusbar = statusbar)
  138. # fill statusbar manager
  139. self.statusbarManager.AddStatusbarItemsByClass(self.statusbarItems, mapframe = self, statusbar = statusbar)
  140. self.statusbarManager.AddStatusbarItem(sb.SbMask(self, statusbar = statusbar, position = 2))
  141. self.statusbarManager.AddStatusbarItem(sb.SbRender(self, statusbar = statusbar, position = 3))
  142. self.statusbarManager.Update()
  143. self.trainingMapManager = MapManager(self, mapWindow = self.GetFirstWindow(),
  144. Map = self.GetFirstMap())
  145. self.previewMapManager = MapManager(self, mapWindow = self.GetSecondWindow(),
  146. Map = self.GetSecondMap())
  147. self.changes = False
  148. self.exportVector = None
  149. # dialogs
  150. self.dialogs = dict()
  151. self.dialogs['classManager'] = None
  152. self.dialogs['scatt_plot'] = None
  153. # just to make digitizer happy
  154. self.dialogs['attributes'] = None
  155. self.dialogs['category'] = None
  156. # PyPlot init
  157. self.plotPanel = PlotPanel(self, giface = self._giface, stats_data = self.stats_data)
  158. self._addPanes()
  159. self._mgr.Update()
  160. self.trainingMapManager.SetToolbar(self.toolbars['iClassTrainingMapManager'])
  161. self.previewMapManager.SetToolbar(self.toolbars['iClassPreviewMapManager'])
  162. # default action
  163. self.GetMapToolbar().SelectDefault()
  164. wx.CallAfter(self.AddTrainingAreaMap)
  165. self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
  166. self.Bind(wx.EVT_SIZE, self.OnSize)
  167. self.SendSizeEvent()
  168. def OnCloseWindow(self, event):
  169. self.GetFirstWindow().GetDigit().CloseMap()
  170. self.plotPanel.CloseWindow()
  171. self.Destroy()
  172. def __del__(self):
  173. """! Frees C structs and removes vector map and all raster maps."""
  174. I_free_signatures(self.signatures)
  175. I_free_group_ref(self.refer)
  176. for st in self.cStatisticsDict.values():
  177. I_iclass_free_statistics(st)
  178. self.RemoveTempVector()
  179. for i in self.stats_data.GetCategories():
  180. self.RemoveTempRaster(self.stats_data.GetStatistics(i).rasterName)
  181. def OnHelp(self, event):
  182. """!Show help page"""
  183. self._giface.Help(entry = 'wxGUI.iclass')
  184. def _getTempVectorName(self):
  185. """!Return new name for temporary vector map (training areas)"""
  186. vectorPath = grass.tempfile(create = False)
  187. return 'trAreas' + os.path.basename(vectorPath).replace('.','')
  188. def SetGroup(self, group, subgroup):
  189. """!Set group and subgroup manually"""
  190. self.g = {'group' : group, 'subgroup' : subgroup}
  191. def CreateTempVector(self):
  192. """!Create temporary vector map for training areas"""
  193. vectorName = self._getTempVectorName()
  194. env = os.environ.copy()
  195. env['GRASS_VECTOR_TEMPORARY'] = '1' # create temporary map
  196. cmd = ('v.edit', {'tool': 'create',
  197. 'map': vectorName})
  198. ret = RunCommand(prog = cmd[0],
  199. parent = self, env = env,
  200. **cmd[1])
  201. if ret != 0:
  202. return False
  203. return vectorName
  204. def RemoveTempVector(self):
  205. """!Removes temporary vector map with training areas"""
  206. ret = RunCommand(prog = 'g.remove',
  207. parent = self,
  208. vect = self.trainingAreaVector)
  209. if ret != 0:
  210. return False
  211. return True
  212. def RemoveTempRaster(self, raster):
  213. """!Removes temporary raster maps"""
  214. self.GetFirstMap().Clean()
  215. self.GetSecondMap().Clean()
  216. ret = RunCommand(prog = 'g.remove',
  217. parent = self,
  218. rast = raster)
  219. if ret != 0:
  220. return False
  221. return True
  222. def AddToolbar(self, name):
  223. """!Add defined toolbar to the window
  224. Currently known toolbars are:
  225. - 'iClassMap' - basic map toolbar
  226. - 'iClass' - iclass tools
  227. - 'iClassMisc' - miscellaneous (help)
  228. - 'vdigit' - digitizer toolbar (areas)
  229. Toolbars 'iClassPreviewMapManager' are added in _addPanes().
  230. """
  231. if name == "iClassMap":
  232. self.toolbars[name] = IClassMapToolbar(self, self._toolSwitcher)
  233. self._mgr.AddPane(self.toolbars[name],
  234. wx.aui.AuiPaneInfo().
  235. Name(name).Caption(_("Map Toolbar")).
  236. ToolbarPane().Top().
  237. LeftDockable(False).RightDockable(False).
  238. BottomDockable(False).TopDockable(True).
  239. CloseButton(False).Layer(2).Row(1).
  240. BestSize((self.toolbars[name].GetBestSize())))
  241. if name == "iClass":
  242. self.toolbars[name] = IClassToolbar(self, stats_data=self.stats_data)
  243. self._mgr.AddPane(self.toolbars[name],
  244. wx.aui.AuiPaneInfo().
  245. Name(name).Caption(_("IClass Toolbar")).
  246. ToolbarPane().Top().
  247. LeftDockable(False).RightDockable(False).
  248. BottomDockable(False).TopDockable(True).
  249. CloseButton(False).Layer(2).Row(2).
  250. BestSize((self.toolbars[name].GetBestSize())))
  251. if name == "iClassMisc":
  252. self.toolbars[name] = IClassMiscToolbar(self)
  253. self._mgr.AddPane(self.toolbars[name],
  254. wx.aui.AuiPaneInfo().
  255. Name(name).Caption(_("IClass Misc Toolbar")).
  256. ToolbarPane().Top().
  257. LeftDockable(False).RightDockable(False).
  258. BottomDockable(False).TopDockable(True).
  259. CloseButton(False).Layer(2).Row(1).
  260. BestSize((self.toolbars[name].GetBestSize())))
  261. if name == "vdigit":
  262. self.toolbars[name] = VDigitToolbar(parent=self, toolSwitcher=self._toolSwitcher,
  263. MapWindow=self.GetFirstWindow(),
  264. digitClass=IClassVDigit, giface=self._giface,
  265. tools = ['addArea', 'moveVertex', 'addVertex',
  266. 'removeVertex', 'editLine', 'moveLine',
  267. 'deleteArea',
  268. 'undo', 'redo'])
  269. self._mgr.AddPane(self.toolbars[name],
  270. wx.aui.AuiPaneInfo().
  271. Name(name).Caption(_("Digitization Toolbar")).
  272. ToolbarPane().Top().
  273. LeftDockable(False).RightDockable(False).
  274. BottomDockable(False).TopDockable(True).
  275. CloseButton(False).Layer(2).Row(2).
  276. BestSize((self.toolbars[name].GetBestSize())))
  277. def _addPanes(self):
  278. """!Add mapwindows and toolbars to aui manager"""
  279. if sys.platform == 'win32':
  280. self._addPaneMapWindow(name = 'training')
  281. self._addPaneToolbar(name = 'iClassTrainingMapManager')
  282. self._addPaneMapWindow(name = 'preview')
  283. self._addPaneToolbar(name = 'iClassPreviewMapManager')
  284. else:
  285. self._addPaneToolbar(name = 'iClassPreviewMapManager')
  286. self._addPaneMapWindow(name = 'preview')
  287. self._addPaneToolbar(name = 'iClassTrainingMapManager')
  288. self._addPaneMapWindow(name = 'training')
  289. # otherwise best size was ignored
  290. self._mgr.SetDockSizeConstraint(0.5, 0.5)
  291. self._mgr.AddPane(self.plotPanel, wx.aui.AuiPaneInfo().
  292. Name("plots").Caption(_("Plots")).
  293. Dockable(False).Floatable(False).CloseButton(False).
  294. Left().Layer(1).BestSize((335, -1)))
  295. def _addPaneToolbar(self, name):
  296. if name == 'iClassPreviewMapManager':
  297. parent = self.previewMapManager
  298. else:
  299. parent = self.trainingMapManager
  300. self.toolbars[name] = IClassMapManagerToolbar(self, parent)
  301. self._mgr.AddPane(self.toolbars[name],
  302. wx.aui.AuiPaneInfo().ToolbarPane().Movable().
  303. Name(name).
  304. CloseButton(False).Center().Layer(0).
  305. BestSize((self.toolbars[name].GetBestSize())))
  306. def _addPaneMapWindow(self, name):
  307. if name == 'preview':
  308. window = self.GetSecondWindow()
  309. caption = _("Preview Display")
  310. else:
  311. window = self.GetFirstWindow()
  312. caption = _("Training Areas Display")
  313. self._mgr.AddPane(window, wx.aui.AuiPaneInfo().
  314. Name(name).Caption(caption).
  315. Dockable(False).Floatable(False).CloseButton(False).
  316. Center().Layer(0))
  317. def IsStandalone(self):
  318. """!Check if Map display is standalone"""
  319. return True
  320. def OnUpdateActive(self, event):
  321. """!
  322. @todo move to DoubleMapFrame?
  323. """
  324. if self.GetMapToolbar().GetActiveMap() == 0:
  325. self.MapWindow = self.firstMapWindow
  326. self.Map = self.firstMap
  327. else:
  328. self.MapWindow = self.secondMapWindow
  329. self.Map = self.secondMap
  330. self.UpdateActive(self.MapWindow)
  331. # for wingrass
  332. if os.name == 'nt':
  333. self.MapWindow.SetFocus()
  334. def UpdateActive(self, win):
  335. """!
  336. @todo move to DoubleMapFrame?
  337. """
  338. mapTb = self.GetMapToolbar()
  339. # optionally disable tool zoomback tool
  340. mapTb.Enable('zoomBack', enable = (len(self.MapWindow.zoomhistory) > 1))
  341. if mapTb.GetActiveMap() != (win == self.secondMapWindow):
  342. mapTb.SetActiveMap((win == self.secondMapWindow))
  343. self.StatusbarUpdate()
  344. def ActivateFirstMap(self, event=None):
  345. DoubleMapFrame.ActivateFirstMap(self, event)
  346. self.GetMapToolbar().Enable('zoomBack', enable=(len(self.MapWindow.zoomhistory) > 1))
  347. def ActivateSecondMap(self, event=None):
  348. DoubleMapFrame.ActivateSecondMap(self, event)
  349. self.GetMapToolbar().Enable('zoomBack', enable=(len(self.MapWindow.zoomhistory) > 1))
  350. def GetMapToolbar(self):
  351. """!Returns toolbar with zooming tools"""
  352. return self.toolbars['iClassMap']
  353. def GetClassColor(self, cat):
  354. """!Get class color as string
  355. @param cat class category
  356. @return 'R:G:B'
  357. """
  358. if cat in self.stats_data.GetCategories():
  359. return self.stats_data.GetStatistics(cat).color
  360. return '0:0:0'
  361. def OnZoomMenu(self, event):
  362. """!Popup Zoom menu """
  363. zoommenu = wx.Menu()
  364. # Add items to the menu
  365. i = 0
  366. for label, handler in ((_('Zoom to computational region'), self.OnZoomToWind),
  367. (None, None),
  368. (_('Adjust Training Area Display to Preview Display'), self.OnZoomToPreview),
  369. (_('Adjust Preview display to Training Area Display'), self.OnZoomToTraining),
  370. (_("Display synchronization ON"), lambda event: self.SetBindRegions(True)),
  371. (_("Display synchronization OFF"), lambda event: self.SetBindRegions(False))):
  372. if label is None:
  373. zoommenu.AppendSeparator()
  374. continue
  375. item = wx.MenuItem(zoommenu, wx.ID_ANY, label)
  376. zoommenu.AppendItem(item)
  377. self.Bind(wx.EVT_MENU, handler, item)
  378. if i == 3:
  379. item.Enable(not self._bindRegions)
  380. elif i == 4:
  381. item.Enable(self._bindRegions)
  382. i += 1
  383. # Popup the menu. If an item is selected then its handler
  384. # will be called before PopupMenu returns.
  385. self.PopupMenu(zoommenu)
  386. zoommenu.Destroy()
  387. def OnZoomToTraining(self, event):
  388. """!Set preview display to match extents of training display """
  389. if not self.MapWindow == self.GetSecondWindow():
  390. self.MapWindow = self.GetSecondWindow()
  391. self.Map = self.GetSecondMap()
  392. self.UpdateActive(self.GetSecondWindow())
  393. newreg = self.firstMap.GetCurrentRegion()
  394. self.GetSecondMap().region = copy.copy(newreg)
  395. self.Render(self.GetSecondWindow())
  396. def OnZoomToPreview(self, event):
  397. """!Set preview display to match extents of training display """
  398. if not self.MapWindow == self.GetFirstWindow():
  399. self.MapWindow = self.GetFirstWindow()
  400. self.Map = self.GetFirstMap()
  401. self.UpdateActive(self.GetFirstWindow())
  402. newreg = self.GetSecondMap().GetCurrentRegion()
  403. self.GetFirstMap().region = copy.copy(newreg)
  404. self.Render(self.GetFirstWindow())
  405. def AddBands(self):
  406. """!Add imagery group"""
  407. dlg = IClassGroupDialog(self, group=self.g['group'])
  408. while True:
  409. if dlg.ShowModal() == wx.ID_OK:
  410. if dlg.GetGroupBandsErr(parent=self):
  411. g, s = dlg.GetData()
  412. group = grass.find_file(name=g, element='group')
  413. self.g['group'] = group['name']
  414. self.g['subgroup'] = s
  415. self.groupSet.emit(group=self.g['group'],
  416. subgroup=self.g['subgroup'])
  417. break
  418. else:
  419. break
  420. dlg.Destroy()
  421. def OnImportAreas(self, event):
  422. """!Import training areas"""
  423. # check if we have any changes
  424. if self.GetAreasCount() or self.stats_data.GetCategories():
  425. qdlg = wx.MessageDialog(parent = self,
  426. message = _("All changes will be lost. "
  427. "Do you want to continue?") ,
  428. style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION | wx.CENTRE)
  429. if qdlg.ShowModal() == wx.ID_NO:
  430. qdlg.Destroy()
  431. return
  432. qdlg.Destroy()
  433. dlg = IClassMapDialog(self, title = _("Import vector map"), element = 'vector')
  434. if dlg.ShowModal() == wx.ID_OK:
  435. vName = dlg.GetMap()
  436. self.ImportAreas(vName)
  437. dlg.Destroy()
  438. def _checkImportedTopo(self, vector):
  439. """!Check if imported vector map has areas
  440. @param vector vector map name
  441. @return warning message (empty if topology is ok)
  442. """
  443. topo = grass.vector_info_topo(map = vector)
  444. warning = ''
  445. if topo['areas'] == 0:
  446. warning = _("No areas in vector map <%s>.\n" % vector)
  447. if topo['points'] or topo['lines']:
  448. warning +=_("Vector map <%s> contains points or lines, "
  449. "these features are ignored." % vector)
  450. return warning
  451. def ImportAreas(self, vector):
  452. """!Import training areas.
  453. If table connected, try load certain columns to class manager
  454. @param vector vector map name
  455. """
  456. warning = self._checkImportedTopo(vector)
  457. if warning:
  458. GMessage(parent = self, message = warning)
  459. return
  460. wx.BeginBusyCursor()
  461. wx.Yield()
  462. # close, build, copy and open again the temporary vector
  463. digitClass = self.GetFirstWindow().GetDigit()
  464. # open vector map to be imported
  465. if digitClass.OpenMap(vector) is None:
  466. GError(parent = self, message = _("Unable to open vector map <%s>") % vector)
  467. return
  468. # copy features to the temporary map
  469. vname = self._getTempVectorName()
  470. os.environ['GRASS_VECTOR_TEMPORARY'] = '1' # avoid deleting temporary map
  471. if digitClass.CopyMap(vname, tmp = True) == -1:
  472. GError(parent = self, message = _("Unable to copy vector features from <%s>") % vector)
  473. return
  474. del os.environ['GRASS_VECTOR_TEMPORARY']
  475. # close map
  476. digitClass.CloseMap()
  477. # open temporary map (copy of imported map)
  478. self.poMapInfo = digitClass.OpenMap(vname, tmp = True)
  479. if self.poMapInfo is None:
  480. GError(parent = self, message = _("Unable to open temporary vector map"))
  481. return
  482. # remove temporary rasters
  483. for cat in self.stats_data.GetCategories():
  484. self.RemoveTempRaster(self.stats_data.GetStatistics(cat).rasterName)
  485. # clear current statistics
  486. self.stats_data.DeleteAllStatistics()
  487. # reset plots
  488. self.plotPanel.Reset()
  489. self.GetFirstWindow().UpdateMap(render = False, renderVector = True)
  490. self.ImportClasses(vector)
  491. # should be saved in attribute table?
  492. self.toolbars['iClass'].UpdateStddev(1.5)
  493. wx.EndBusyCursor()
  494. return True
  495. def ImportClasses(self, vector):
  496. """!If imported map has table, try to import certain columns to class manager"""
  497. # check connection
  498. dbInfo = VectorDBInfo(vector)
  499. connected = (len(dbInfo.layers.keys()) > 0)
  500. # remove attribute table of temporary vector, we don't need it
  501. if connected:
  502. RunCommand('v.db.droptable',
  503. flags = 'f',
  504. map = self.trainingAreaVector)
  505. # we use first layer with table, TODO: user should choose
  506. layer = None
  507. for key in dbInfo.layers.keys():
  508. if dbInfo.GetTable(key):
  509. layer = key
  510. # get columns to check if we can use them
  511. # TODO: let user choose which columns mean what
  512. if layer is not None:
  513. columns = dbInfo.GetColumns(table = dbInfo.GetTable(layer))
  514. else:
  515. columns = []
  516. # get class manager
  517. if self.dialogs['classManager'] is None:
  518. self.dialogs['classManager'] = IClassCategoryManagerDialog(self)
  519. listCtrl = self.dialogs['classManager'].GetListCtrl()
  520. # unable to load data (no connection, table, right columns)
  521. if not connected or layer is None or \
  522. 'class' not in columns or \
  523. 'color' not in columns:
  524. # no table connected
  525. cats = RunCommand('v.category',
  526. input = vector,
  527. layer = 1, # set layer?
  528. # type = ['centroid', 'area'] ?
  529. option = "print",
  530. read = True)
  531. cats = map(int, cats.strip().split())
  532. cats = sorted(list(set(cats)))
  533. for cat in cats:
  534. listCtrl.AddCategory(cat = cat, name = 'class_%d' % cat, color = "0:0:0")
  535. # connection, table and columns exists
  536. else:
  537. columns = ['cat', 'class', 'color']
  538. ret = RunCommand('v.db.select',
  539. quiet = True,
  540. parent = self,
  541. flags = 'c',
  542. map = vector,
  543. layer = 1,
  544. columns = ','.join(columns),
  545. read = True)
  546. records = ret.strip().split('\n')
  547. for record in records:
  548. record = record.split('|')
  549. listCtrl.AddCategory(cat = int(record[0]), name = record[1], color = record[2])
  550. def OnExportAreas(self, event):
  551. """!Export training areas"""
  552. if self.GetAreasCount() == 0:
  553. GMessage(parent = self, message = _("No training areas to export."))
  554. return
  555. dlg = IClassExportAreasDialog(self, vectorName = self.exportVector)
  556. if dlg.ShowModal() == wx.ID_OK:
  557. vName = dlg.GetVectorName()
  558. self.exportVector = vName
  559. withTable = dlg.WithTable()
  560. dlg.Destroy()
  561. if self.ExportAreas(vectorName = vName, withTable = withTable):
  562. GMessage(_("%d training areas (%d classes) exported to vector map <%s>.") % \
  563. (self.GetAreasCount(), len(self.stats_data.GetCategories()),
  564. self.exportVector), parent = self)
  565. def ExportAreas(self, vectorName, withTable):
  566. """!Export training areas to new vector map (with attribute table).
  567. @param vectorName name of exported vector map
  568. @param withTable true if attribute table is required
  569. """
  570. wx.BeginBusyCursor()
  571. wx.Yield()
  572. # close, build, copy and open again the temporary vector
  573. digitClass = self.GetFirstWindow().GetDigit()
  574. if '@' in vectorName:
  575. vectorName = vectorName.split('@')[0]
  576. if digitClass.CopyMap(vectorName) < 0:
  577. return False
  578. if not withTable:
  579. wx.EndBusyCursor()
  580. return False
  581. # add new table
  582. columns = ["class varchar(30)",
  583. "color varchar(11)",
  584. "n_cells integer",]
  585. nbands = len(self.GetGroupLayers(self.g['group'], self.g['subgroup']))
  586. for statistic, format in (("min", "integer"), ("mean", "double precision"), ("max", "integer")):
  587. for i in range(nbands):
  588. # 10 characters limit?
  589. columns.append("band%(band)d_%(stat)s %(format)s" % {'band' : i + 1,
  590. 'stat' : statistic,
  591. 'format' : format})
  592. if 0 != RunCommand('v.db.addtable',
  593. map = vectorName,
  594. columns = columns,
  595. parent = self):
  596. wx.EndBusyCursor()
  597. return False
  598. # populate table
  599. for cat in self.stats_data.GetCategories():
  600. stat = self.stats_data.GetStatistics(cat)
  601. self._runDBUpdate(map = vectorName, column = "class", value = stat.name, cat = cat)
  602. self._runDBUpdate(map = vectorName, column = "color", value = stat.color, cat = cat)
  603. if not stat.IsReady():
  604. continue
  605. self._runDBUpdate(map = vectorName, column = "n_cells",value = stat.ncells, cat = cat)
  606. for i in range(nbands):
  607. self._runDBUpdate(map = vectorName, column = "band%d_min" % (i + 1), value = stat.bands[i].min, cat = cat)
  608. self._runDBUpdate(map = vectorName, column = "band%d_mean" % (i + 1), value = stat.bands[i].mean, cat = cat)
  609. self._runDBUpdate(map = vectorName, column = "band%d_max" % (i + 1), value = stat.bands[i].max, cat = cat)
  610. wx.EndBusyCursor()
  611. return True
  612. def _runDBUpdate(self, map, column, value, cat):
  613. """!Helper function for calling v.db.update.
  614. @param map vector map name
  615. @param column name of updated column
  616. @param value new value
  617. @param cat which category to update
  618. @return returncode (0 is OK)
  619. """
  620. ret = RunCommand('v.db.update',
  621. map = map,
  622. layer = 1,
  623. column = column,
  624. value = value,
  625. where = "cat = %d" % cat)
  626. return ret
  627. def OnCategoryManager(self, event):
  628. """!Show category management dialog"""
  629. if self.dialogs['classManager'] is None:
  630. dlg = IClassCategoryManagerDialog(self)
  631. dlg.Show()
  632. self.dialogs['classManager'] = dlg
  633. else:
  634. if not self.dialogs['classManager'].IsShown():
  635. self.dialogs['classManager'].Show()
  636. def CategoryChanged(self, currentCat):
  637. """!Updates everything which depends on current category.
  638. Updates number of stddev, histograms, layer in preview display.
  639. """
  640. if currentCat:
  641. stat = self.stats_data.GetStatistics(currentCat)
  642. nstd = stat.nstd
  643. self.toolbars['iClass'].UpdateStddev(nstd)
  644. self.plotPanel.UpdateCategory(currentCat)
  645. self.plotPanel.OnPlotTypeSelected(None)
  646. name = stat.rasterName
  647. name = self.previewMapManager.GetAlias(name)
  648. if name:
  649. self.previewMapManager.SelectLayer(name)
  650. self.categoryChanged.emit(cat = currentCat)
  651. def DeleteAreas(self, cats):
  652. """!Removes all training areas of given categories
  653. @param cats list of categories to be deleted
  654. """
  655. self.firstMapWindow.GetDigit().DeleteAreasByCat(cats)
  656. self.firstMapWindow.UpdateMap(render=False, renderVector=True)
  657. def HighlightCategory(self, cats):
  658. """!Highlight araes given by category"""
  659. self.firstMapWindow.GetDigit().GetDisplay().SetSelected(cats, layer = 1)
  660. self.firstMapWindow.UpdateMap(render=False, renderVector=True)
  661. def ZoomToAreasByCat(self, cat):
  662. """!Zoom to areas given by category"""
  663. n, s, w, e = self.GetFirstWindow().GetDigit().GetDisplay().GetRegionSelected()
  664. self.GetFirstMap().GetRegion(n = n, s = s, w = w, e = e, update = True)
  665. self.GetFirstMap().AdjustRegion()
  666. self.GetFirstMap().AlignExtentFromDisplay()
  667. self.GetFirstWindow().UpdateMap(render = True, renderVector = True)
  668. def UpdateRasterName(self, newName, cat):
  669. """!Update alias of raster map when category name is changed"""
  670. origName = self.stats_data.GetStatistics(cat).rasterName
  671. self.previewMapManager.SetAlias(origName, self._addSuffix(newName))
  672. def StddevChanged(self, cat, nstd):
  673. """!Standard deviation multiplier changed, rerender map, histograms"""
  674. stat = self.stats_data.GetStatistics(cat)
  675. stat.SetStatistics({"nstd" : nstd})
  676. if not stat.IsReady():
  677. return
  678. raster = stat.rasterName
  679. cstat = self.cStatisticsDict[cat]
  680. I_iclass_statistics_set_nstd(cstat, nstd)
  681. I_iclass_create_raster(cstat, self.refer, raster)
  682. self.Render(self.GetSecondWindow())
  683. stat.SetBandStatistics(cstat)
  684. self.plotPanel.StddevChanged()
  685. def UpdateChangeState(self, changes):
  686. """!Informs if any important changes happened
  687. since last analysis computation.
  688. """
  689. self.changes = changes
  690. def AddRasterMap(self, name, firstMap = True, secondMap = True):
  691. """!Add raster map to Map"""
  692. cmdlist = ['d.rast', 'map=%s' % name]
  693. if firstMap:
  694. self.GetFirstMap().AddLayer(ltype='raster', command=cmdlist, active=True,
  695. name=name, hidden=False, opacity=1.0, render=False)
  696. self.Render(self.GetFirstWindow())
  697. if secondMap:
  698. self.GetSecondMap().AddLayer(ltype='raster', command=cmdlist, active=True,
  699. name=name, hidden=False, opacity=1.0, render=False)
  700. self.Render(self.GetSecondWindow())
  701. def AddTrainingAreaMap(self):
  702. """!Add vector map with training areas to Map (training
  703. sub-display)"""
  704. vname = self.CreateTempVector()
  705. if vname:
  706. self.trainingAreaVector = vname
  707. else:
  708. GMessage(parent = self, message = _("Failed to create temporary vector map."))
  709. return
  710. # use 'hidden' for temporary maps (TODO: do it better)
  711. mapLayer = self.GetFirstMap().AddLayer(ltype = 'vector',
  712. command = ['d.vect', 'map=%s' % vname],
  713. name = vname, active = False, hidden = True)
  714. self.toolbars['vdigit'].StartEditing(mapLayer)
  715. self.poMapInfo = self.GetFirstWindow().GetDigit().GetMapInfo()
  716. self.Render(self.GetFirstWindow())
  717. def OnRunAnalysis(self, event):
  718. """!Run analysis and update plots"""
  719. if self.RunAnalysis():
  720. currentCat = self.GetCurrentCategoryIdx()
  721. self.plotPanel.UpdatePlots(group = self.g['group'], subgroup = self.g['subgroup'],
  722. currentCat = currentCat, stats_data = self.stats_data)
  723. def RunAnalysis(self):
  724. """!Run analysis
  725. Calls C functions to compute all statistics and creates raster maps.
  726. Signatures are created but signature file is not.
  727. """
  728. if not self.CheckInput(group = self.g['group'], vector = self.trainingAreaVector):
  729. return
  730. for statistic in self.cStatisticsDict.values():
  731. I_iclass_free_statistics(statistic)
  732. self.cStatisticsDict = {}
  733. # init Ref struct with the files in group */
  734. I_free_group_ref(self.refer)
  735. if (not I_iclass_init_group(self.g['group'], self.g["subgroup"], self.refer)):
  736. return False
  737. I_free_signatures(self.signatures)
  738. I_iclass_init_signatures(self.signatures, self.refer)
  739. # why create copy
  740. #cats = self.statisticsList[:]
  741. cats = self.stats_data.GetCategories()
  742. for i in cats:
  743. stats = self.stats_data.GetStatistics(i)
  744. statistics_obj = IClass_statistics()
  745. statistics = pointer(statistics_obj)
  746. I_iclass_init_statistics(statistics,
  747. stats.category,
  748. stats.name,
  749. stats.color,
  750. stats.nstd)
  751. ret = I_iclass_analysis(statistics, self.refer, self.poMapInfo, "1",
  752. self.g['group'], stats.rasterName)
  753. if ret > 0:
  754. # tests
  755. self.cStatisticsDict[i] = statistics
  756. stats.SetFromcStatistics(statistics)
  757. stats.SetReady()
  758. # stat is already part of stats_data?
  759. #self.statisticsDict[stats.category] = stats
  760. self.ConvertToNull(name = stats.rasterName)
  761. self.previewMapManager.AddLayer(name = stats.rasterName,
  762. alias = self._addSuffix(stats.name), resultsLayer = True)
  763. # write statistics
  764. I_iclass_add_signature(self.signatures, statistics)
  765. elif ret == 0:
  766. GMessage(parent = self, message = _("No area in category %s. Category skipped.") % stats.category)
  767. I_iclass_free_statistics(statistics)
  768. else:
  769. GMessage(parent = self, message = _("Analysis failed."))
  770. I_iclass_free_statistics(statistics)
  771. self.UpdateChangeState(changes = False)
  772. return True
  773. def _addSuffix(self, name):
  774. suffix = _('results')
  775. return '_'.join((name, suffix))
  776. def OnSaveSigFile(self, event):
  777. """!Asks for signature file name and saves it."""
  778. if not self.g['group']:
  779. GMessage(parent = self, message = _("No imagery group selected."))
  780. return
  781. if self.changes:
  782. qdlg = wx.MessageDialog(parent = self,
  783. message = _("Due to recent changes in classes, "
  784. "signatures can be outdated and should be recalculated. "
  785. "Do you still want to continue?") ,
  786. caption = _("Outdated signatures"),
  787. style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION | wx.CENTRE)
  788. if qdlg.ShowModal() == wx.ID_YES:
  789. qdlg.Destroy()
  790. else:
  791. qdlg.Destroy()
  792. return
  793. dlg = IClassSignatureFileDialog(self,
  794. group = self.g['group'],
  795. subgroup = self.g['subgroup'],
  796. file = self.sigFile)
  797. if dlg.ShowModal() == wx.ID_OK:
  798. if os.path.exists(dlg.GetFileName(fullPath = True)):
  799. qdlg = wx.MessageDialog(parent = self,
  800. message = _("A signature file named %s already exists.\n"
  801. "Do you want to replace it?") % dlg.GetFileName(),
  802. caption = _("File already exists"),
  803. style = wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION | wx.CENTRE)
  804. if qdlg.ShowModal() == wx.ID_YES:
  805. qdlg.Destroy()
  806. else:
  807. qdlg.Destroy()
  808. return
  809. self.sigFile = dlg.GetFileName()
  810. self.WriteSignatures(self.signatures, self.g['group'], self.g['subgroup'], self.sigFile)
  811. dlg.Destroy()
  812. def InitStatistics(self):
  813. """!Initialize variables and c structures neccessary for
  814. computing statistics.
  815. """
  816. self.g = {'group' : None, 'subgroup' : None}
  817. self.sigFile = None
  818. self.stats_data = StatisticsData()
  819. self.cStatisticsDict = {}
  820. self.signatures_obj = Signature()
  821. self.signatures = pointer(self.signatures_obj)
  822. I_init_signatures(self.signatures, 0) # must be freed on exit
  823. refer_obj = Ref()
  824. self.refer = pointer(refer_obj)
  825. I_init_group_ref(self.refer) # must be freed on exit
  826. def WriteSignatures(self, signatures, group, subgroup, filename):
  827. """!Writes current signatures to signature file
  828. @param signatures signature (c structure)
  829. @param group imagery group
  830. @param filename signature file name
  831. """
  832. I_iclass_write_signatures(signatures, group, subgroup, filename)
  833. def CheckInput(self, group, vector):
  834. """!Check if input is valid"""
  835. # check if group is ok
  836. #TODO check subgroup
  837. if not group:
  838. GMessage(parent = self,
  839. message = _("No imagery group selected. "
  840. "Operation canceled."))
  841. return False
  842. groupLayers = self.GetGroupLayers(self.g['group'], self.g['subgroup'])
  843. nLayers = len(groupLayers)
  844. if nLayers <= 1:
  845. GMessage(parent = self,
  846. message = _("Group <%(group)s> does not have enough files "
  847. "(it has %(files)d files). Operation canceled.") % \
  848. { 'group' : group,
  849. 'files' : nLayers })
  850. return False
  851. #check if vector has any areas
  852. if self.GetAreasCount() == 0:
  853. GMessage(parent = self,
  854. message = _("No areas given. "
  855. "Operation canceled."))
  856. return False
  857. # check if vector is inside raster
  858. regionBox = bound_box()
  859. Vect_get_map_box(self.poMapInfo, byref(regionBox))
  860. rasterInfo = grass.raster_info(groupLayers[0])
  861. if regionBox.N > rasterInfo['north'] or \
  862. regionBox.S < rasterInfo['south'] or \
  863. regionBox.E > rasterInfo['east'] or \
  864. regionBox.W < rasterInfo['west']:
  865. GMessage(parent = self,
  866. message = _("Vector features are outside raster layers. "
  867. "Operation canceled."))
  868. return False
  869. return True
  870. def GetAreasCount(self):
  871. """!Returns number of not dead areas"""
  872. count = 0
  873. numAreas = Vect_get_num_areas(self.poMapInfo)
  874. for i in range(numAreas):
  875. if Vect_area_alive(self.poMapInfo, i + 1):
  876. count += 1
  877. return count
  878. def GetGroupLayers(self, group, subgroup=None):
  879. """!Get layers in subgroup (expecting same name for group and subgroup)
  880. @todo consider moving this function to core module for convenient
  881. """
  882. kwargs = {}
  883. if subgroup:
  884. kwargs['subgroup'] = subgroup
  885. res = RunCommand('i.group',
  886. flags = 'g',
  887. group = group,
  888. read = True, **kwargs).strip()
  889. if res.split('\n')[0]:
  890. return res.split('\n')
  891. return []
  892. def ConvertToNull(self, name):
  893. """! Sets value which represents null values for given raster map.
  894. @param name raster map name
  895. """
  896. RunCommand('r.null',
  897. map = name,
  898. setnull = 0)
  899. def GetCurrentCategoryIdx(self):
  900. """!Returns current category number"""
  901. return self.toolbars['iClass'].GetSelectedCategoryIdx()
  902. def OnZoomIn(self, event):
  903. """!Enable zooming for plots"""
  904. super(IClassMapFrame, self).OnZoomIn(event)
  905. self.plotPanel.EnableZoom(type = 1)
  906. def OnZoomOut(self, event):
  907. """!Enable zooming for plots"""
  908. super(IClassMapFrame, self).OnZoomOut(event)
  909. self.plotPanel.EnableZoom(type = -1)
  910. def OnPan(self, event):
  911. """!Enable panning for plots"""
  912. super(IClassMapFrame, self).OnPan(event)
  913. self.plotPanel.EnablePan()
  914. def OnPointer(self, event):
  915. """!Set pointer mode.
  916. @todo: pointers need refactoring
  917. """
  918. self.GetFirstWindow().SetModePointer()
  919. self.GetSecondWindow().SetModePointer()
  920. def GetMapManagers(self):
  921. """!Get map managers of wxIClass
  922. @return trainingMapManager, previewMapManager
  923. """
  924. return self.trainingMapManager, self.previewMapManager
  925. class MapManager:
  926. """! Class for managing map renderer.
  927. It is connected with iClassMapManagerToolbar.
  928. """
  929. def __init__(self, frame, mapWindow, Map):
  930. """!
  931. It is expected that \a mapWindow is conected with \a Map.
  932. @param frame application main window
  933. @param mapWindow map window instance
  934. @param Map map renderer instance
  935. """
  936. self.map = Map
  937. self.frame = frame
  938. self.mapWindow = mapWindow
  939. self.toolbar = None
  940. self.layerName = {}
  941. def SetToolbar(self, toolbar):
  942. self.toolbar = toolbar
  943. def AddLayer(self, name, alias = None, resultsLayer = False):
  944. """!Adds layer to Map and update toolbar
  945. @param name layer (raster) name
  946. @param resultsLayer True if layer is temp. raster showing the results of computation
  947. """
  948. if (resultsLayer and
  949. name in [l.GetName() for l in self.map.GetListOfLayers(name = name)]):
  950. self.frame.Render(self.mapWindow)
  951. return
  952. cmdlist = ['d.rast', 'map=%s' % name]
  953. self.map.AddLayer(ltype = 'raster', command = cmdlist, active = True,
  954. name = name, hidden = False, opacity = 1.0, render = True)
  955. self.frame.Render(self.mapWindow)
  956. if alias is not None:
  957. self.layerName[alias] = name
  958. name = alias
  959. else:
  960. self.layerName[name] = name
  961. self.toolbar.choice.Insert(name, 0)
  962. self.toolbar.choice.SetSelection(0)
  963. def AddLayerRGB(self, cmd):
  964. """!Adds RGB layer and update toolbar.
  965. @param cmd d.rgb command as a list
  966. """
  967. name = []
  968. for param in cmd:
  969. if '=' in param:
  970. name.append(param.split('=')[1])
  971. name = ','.join(name)
  972. self.map.AddLayer(ltype = 'rgb', command = cmd, active = True,
  973. name = name, hidden = False, opacity = 1.0, render = True)
  974. self.frame.Render(self.mapWindow)
  975. self.layerName[name] = name
  976. self.toolbar.choice.Insert(name, 0)
  977. self.toolbar.choice.SetSelection(0)
  978. def RemoveTemporaryLayer(self, name):
  979. """!Removes temporary layer (if exists) from Map and and updates toolbar.
  980. @param name real name of layer
  981. """
  982. # check if layer is loaded
  983. layers = self.map.GetListOfLayers(ltype = 'raster')
  984. idx = None
  985. for i, layer in enumerate(layers):
  986. if name == layer.GetName():
  987. idx = i
  988. break
  989. if idx is None:
  990. return
  991. # remove it from Map
  992. self.map.RemoveLayer(name = name)
  993. # update inner list of layers
  994. alias = self.GetAlias(name)
  995. if alias not in self.layerName:
  996. return
  997. del self.layerName[alias]
  998. # update choice
  999. idx = self.toolbar.choice.FindString(alias)
  1000. if idx != wx.NOT_FOUND:
  1001. self.toolbar.choice.Delete(idx)
  1002. if not self.toolbar.choice.IsEmpty():
  1003. self.toolbar.choice.SetSelection(0)
  1004. self.frame.Render(self.mapWindow)
  1005. def Render(self):
  1006. """@todo giface shoud be used instead of this method"""
  1007. self.frame.Render(self.mapWindow)
  1008. def RemoveLayer(self, name, idx):
  1009. """!Removes layer from Map and update toolbar"""
  1010. name = self.layerName[name]
  1011. self.map.RemoveLayer(name = name)
  1012. del self.layerName[name]
  1013. self.toolbar.choice.Delete(idx)
  1014. if not self.toolbar.choice.IsEmpty():
  1015. self.toolbar.choice.SetSelection(0)
  1016. self.frame.Render(self.mapWindow)
  1017. def SelectLayer(self, name):
  1018. """!Moves selected layer to top"""
  1019. layers = self.map.GetListOfLayers(ltype = 'rgb') + \
  1020. self.map.GetListOfLayers(ltype = 'raster')
  1021. idx = None
  1022. for i, layer in enumerate(layers):
  1023. if self.layerName[name] == layer.GetName():
  1024. idx = i
  1025. break
  1026. if idx is not None: # should not happen
  1027. layers.append(layers.pop(idx))
  1028. choice = self.toolbar.choice
  1029. idx = choice.FindString(name)
  1030. choice.Delete(idx)
  1031. choice.Insert(name, 0)
  1032. choice.SetSelection(0)
  1033. #layers.reverse()
  1034. self.map.SetLayers(layers)
  1035. self.frame.Render(self.mapWindow)
  1036. def SetOpacity(self, name):
  1037. """!Sets opacity of layers."""
  1038. name = self.layerName[name]
  1039. layers = self.map.GetListOfLayers(name = name)
  1040. if not layers:
  1041. return
  1042. # works for first layer only
  1043. oldOpacity = layers[0].GetOpacity()
  1044. dlg = SetOpacityDialog(self.frame, opacity = oldOpacity)
  1045. dlg.applyOpacity.connect(lambda value:
  1046. self._changeOpacity(layer=layers[0], opacity=value))
  1047. if dlg.ShowModal() == wx.ID_OK:
  1048. self._changeOpacity(layer=layers[0], opacity=dlg.GetOpacity())
  1049. dlg.Destroy()
  1050. def _changeOpacity(self, layer, opacity):
  1051. self.map.ChangeOpacity(layer=layer, opacity=opacity)
  1052. self.frame.Render(self.mapWindow)
  1053. def GetAlias(self, name):
  1054. """!Returns alias for layer"""
  1055. name = [k for k, v in self.layerName.iteritems() if v == name]
  1056. if name:
  1057. return name[0]
  1058. return None
  1059. def SetAlias(self, original, alias):
  1060. name = self.GetAlias(original)
  1061. if name:
  1062. self.layerName[alias] = original
  1063. del self.layerName[name]
  1064. idx = self.toolbar.choice.FindString(name)
  1065. if idx != wx.NOT_FOUND:
  1066. self.toolbar.choice.SetString(idx, alias)
  1067. def test():
  1068. import core.render as render
  1069. app = wx.PySimpleApp()
  1070. wx.InitAllImageHandlers()
  1071. frame = IClassMapFrame()
  1072. frame.Show()
  1073. app.MainLoop()
  1074. if __name__ == "__main__":
  1075. test()