simplelmgr.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. # -*- coding: utf-8 -*-
  2. """
  3. @package gui_core.simplelmgr
  4. @brief GUI class for simple layer management.
  5. Classes:
  6. - simplelmgr::SimpleLayerManager
  7. - simplelmgr::SimpleLmgrToolbar
  8. (C) 2013 by the GRASS Development Team
  9. This program is free software under the GNU General Public License
  10. (>=v2). Read the file COPYING that comes with GRASS for details.
  11. @author Anna Petrasova (kratochanna gmail.com)
  12. """
  13. import os
  14. import wx
  15. import wx.aui
  16. from grass.pydispatch.signal import Signal
  17. from gui_core.toolbars import BaseToolbar, BaseIcons
  18. from icons.icon import MetaIcon
  19. from gui_core.forms import GUI
  20. from gui_core.dialogs import SetOpacityDialog
  21. from core.utils import GetLayerNameFromCmd
  22. from core.gcmd import GError
  23. from core.layerlist import LayerList
  24. from core.utils import _
  25. SIMPLE_LMGR_RASTER = 1
  26. SIMPLE_LMGR_VECTOR = 2
  27. SIMPLE_LMGR_RASTER3D = 4
  28. SIMPLE_LMGR_RGB = 8
  29. SIMPLE_LMGR_TB_TOP = 16
  30. SIMPLE_LMGR_TB_BOTTOM = 32
  31. SIMPLE_LMGR_TB_LEFT = 64
  32. SIMPLE_LMGR_TB_RIGHT = 128
  33. class SimpleLayerManager(wx.Panel):
  34. """Simple layer manager class provides similar functionality to
  35. Layertree, but it's just list, not tree."""
  36. def __init__(
  37. self, parent, layerList, lmgrStyle=SIMPLE_LMGR_RASTER |
  38. SIMPLE_LMGR_VECTOR | SIMPLE_LMGR_TB_LEFT, toolbarCls=None,
  39. modal=False):
  40. wx.Panel.__init__(self, parent=parent, name='SimpleLayerManager')
  41. self._style = lmgrStyle
  42. self._layerList = layerList
  43. self._checkList = wx.CheckListBox(self, style=wx.LB_EXTENDED)
  44. if not toolbarCls:
  45. toolbarCls = SimpleLmgrToolbar
  46. self._toolbar = toolbarCls(self, lmgrStyle=self._style)
  47. self._auimgr = wx.aui.AuiManager(self)
  48. self._modal = modal
  49. # d.* dialogs are recreated each time, attempt to hide it resulted
  50. # in completely mysterious memory corruption and crash when opening
  51. # any dialog with stock labels (wx.ID_OK and so on)
  52. # needed in order not to change selection when moving layers
  53. self._blockSelectionChanged = False
  54. self._checkList.Bind(
  55. wx.EVT_LISTBOX,
  56. lambda evt: self._selectionChanged())
  57. self._checkList.Bind(
  58. wx.EVT_LISTBOX_DCLICK,
  59. self.OnLayerChangeProperties)
  60. self._checkList.Bind(wx.EVT_CHECKLISTBOX, self.OnLayerChecked)
  61. self._checkList.Bind(wx.EVT_CONTEXT_MENU, self.OnContextMenu)
  62. # signal emitted when somethin in layer list changes
  63. self.opacityChanged = Signal('SimpleLayerManager.opacityChanged')
  64. self.cmdChanged = Signal('SimpleLayerManager.cmdChanged')
  65. self.layerAdded = Signal('SimpleLayerManager.layerAdded')
  66. self.layerRemoved = Signal('SimpleLayerManager.layerRemoved')
  67. self.layerActivated = Signal('SimpleLayerManager.layerActivated')
  68. self.layerMovedUp = Signal('SimpleLayerManager.layerMovedUp')
  69. self.layerMovedDown = Signal('SimpleLayerManager.layerMovedDown')
  70. # emitted by any change (e.g. for rerendering)
  71. self.anyChange = Signal('SimpleLayerManager.layerChange')
  72. self._layout()
  73. self.SetMinSize((200, -1))
  74. self._update()
  75. def _layout(self):
  76. self._auimgr.AddPane(self._checkList,
  77. wx.aui.AuiPaneInfo().
  78. Name("checklist").
  79. CenterPane().
  80. CloseButton(False).
  81. BestSize((self._checkList.GetBestSize())))
  82. paneInfo = wx.aui.AuiPaneInfo(). \
  83. Name("toolbar").Caption(_("Toolbar")).ToolbarPane(). \
  84. CloseButton(False).Layer(1).Gripper(False). \
  85. BestSize((self._toolbar.GetBestSize()))
  86. if self._style & SIMPLE_LMGR_TB_LEFT:
  87. paneInfo.Left()
  88. elif self._style & SIMPLE_LMGR_TB_RIGHT:
  89. paneInfo.Right()
  90. elif self._style & SIMPLE_LMGR_TB_TOP:
  91. paneInfo.Top()
  92. else:
  93. paneInfo.Bottom()
  94. self._auimgr.AddPane(self._toolbar, paneInfo)
  95. self._auimgr.Update()
  96. def _selectionChanged(self):
  97. """Selection was changed externally,
  98. updates selection info in layers."""
  99. if self._blockSelectionChanged:
  100. return
  101. selected = self._checkList.GetSelections()
  102. for i, layer in enumerate(self._layerList):
  103. layer.Select(i in selected)
  104. def OnContextMenu(self, event):
  105. """Show context menu.
  106. So far offers only copying layer list to clipboard
  107. """
  108. if len(self._layerList) < 1:
  109. event.Skip()
  110. return
  111. menu = wx.Menu()
  112. llist = [layer.name for layer in self._layerList]
  113. texts = [','.join(llist), ','.join(reversed(llist))]
  114. labels = [_("Copy map names to clipboard (top to bottom)"),
  115. _("Copy map names to clipboard (bottom to top)")]
  116. for label, text in zip(labels, texts):
  117. id = wx.NewId()
  118. self.Bind(
  119. wx.EVT_MENU,
  120. lambda evt,
  121. t=text,
  122. id=id: self._copyText(t),
  123. id=id)
  124. menu.Append(id, label)
  125. # show the popup menu
  126. self.PopupMenu(menu)
  127. menu.Destroy()
  128. event.Skip()
  129. def _copyText(self, text):
  130. """Helper function for copying
  131. TODO: move to utils?
  132. """
  133. if wx.TheClipboard.Open():
  134. do = wx.TextDataObject()
  135. do.SetText(text)
  136. wx.TheClipboard.SetData(do)
  137. wx.TheClipboard.Close()
  138. def OnLayerChecked(self, event):
  139. """Layer was (un)checked, update layer's info."""
  140. checkedIdxs = self._checkList.GetChecked()
  141. for i, layer in enumerate(self._layerList):
  142. if i in checkedIdxs and not layer.IsActive():
  143. layer.Activate()
  144. self.layerActivated.emit(index=i, layer=layer)
  145. elif i not in checkedIdxs and layer.IsActive():
  146. layer.Activate(False)
  147. self.layerActivated.emit(index=i, layer=layer)
  148. self.anyChange.emit()
  149. event.Skip()
  150. def OnAddRaster(self, event):
  151. """Opens d.rast dialog and adds layer.
  152. Dummy layer is added first."""
  153. cmd = ['d.rast']
  154. layer = self.AddRaster(name='', cmd=cmd, hidden=True, dialog=None)
  155. GUI(parent=self, giface=None, modal=self._modal).ParseCommand(
  156. cmd=cmd, completed=(self.GetOptData, layer, ''))
  157. event.Skip()
  158. def OnAddVector(self, event):
  159. """Opens d.vect dialog and adds layer.
  160. Dummy layer is added first."""
  161. cmd = ['d.vect']
  162. layer = self.AddVector(name='', cmd=cmd, hidden=True, dialog=None)
  163. GUI(parent=self, giface=None, modal=self._modal).ParseCommand(
  164. cmd=cmd, completed=(self.GetOptData, layer, ''))
  165. event.Skip()
  166. def OnAddRast3d(self, event):
  167. """Opens d.rast3d dialog and adds layer.
  168. Dummy layer is added first."""
  169. cmd = ['d.rast3d']
  170. layer = self.AddRast3d(name='', cmd=cmd, hidden=True, dialog=None)
  171. GUI(parent=self, giface=None, modal=self._modal).ParseCommand(
  172. cmd=cmd, completed=(self.GetOptData, layer, ''))
  173. event.Skip()
  174. def OnAddRGB(self, event):
  175. """Opens d.rgb dialog and adds layer.
  176. Dummy layer is added first."""
  177. cmd = ['d.rgb']
  178. layer = self.AddRGB(name='', cmd=cmd, hidden=True, dialog=None)
  179. GUI(parent=self, giface=None, modal=self._modal).ParseCommand(
  180. cmd=cmd, completed=(self.GetOptData, layer, ''))
  181. event.Skip()
  182. def OnRemove(self, event):
  183. """Removes selected layers from list."""
  184. layers = self._layerList.GetSelectedLayers(activeOnly=False)
  185. for layer in layers:
  186. self.layerRemoved.emit(
  187. index=self._layerList.GetLayerIndex(layer), layer=layer)
  188. self._layerList.RemoveLayer(layer)
  189. self._update()
  190. self.anyChange.emit()
  191. event.Skip()
  192. def OnLayerUp(self, event):
  193. """Moves selected layers one step up.
  194. Note: not completely correct for multiple layers."""
  195. layers = self._layerList.GetSelectedLayers()
  196. self._blockSelectionChanged = True
  197. for layer in layers:
  198. idx = self._layerList.GetLayerIndex(layer)
  199. if idx > 0:
  200. self.layerMovedUp.emit(index=idx, layer=layer)
  201. self._layerList.MoveLayerUp(layer)
  202. self._update()
  203. self._blockSelectionChanged = False
  204. self.anyChange.emit()
  205. event.Skip()
  206. def OnLayerDown(self, event):
  207. """Moves selected layers one step down.
  208. Note: not completely correct for multiple layers."""
  209. layers = self._layerList.GetSelectedLayers()
  210. self._blockSelectionChanged = True
  211. for layer in layers:
  212. idx = self._layerList.GetLayerIndex(layer)
  213. if idx < len(self._layerList) - 1:
  214. self.layerMovedDown.emit(
  215. index=self._layerList.GetLayerIndex(layer), layer=layer)
  216. self._layerList.MoveLayerDown(layer)
  217. self._update()
  218. self._blockSelectionChanged = False
  219. self.anyChange.emit()
  220. event.Skip()
  221. def OnLayerChangeProperties(self, event):
  222. """Opens module dialog to edit layer command."""
  223. layers = self._layerList.GetSelectedLayers()
  224. if not layers or len(layers) > 1:
  225. return
  226. self._layerChangeProperties(layers[0])
  227. event.Skip()
  228. def _layerChangeProperties(self, layer):
  229. """Opens new module dialog or recycles it."""
  230. GUI(parent=self, giface=None, modal=self._modal).ParseCommand(
  231. cmd=layer.cmd, completed=(self.GetOptData, layer, ''))
  232. def OnLayerChangeOpacity(self, event):
  233. """Opacity of a layer is changing."""
  234. layers = self._layerList.GetSelectedLayers()
  235. if not layers or len(layers) > 1:
  236. return
  237. layer = layers[0]
  238. dlg = SetOpacityDialog(self, opacity=layer.opacity,
  239. title=_("Set opacity of <%s>") % layer.name)
  240. dlg.applyOpacity.connect(lambda value:
  241. self._setLayerOpacity(layer, value))
  242. dlg.CentreOnParent()
  243. if dlg.ShowModal() == wx.ID_OK:
  244. self._setLayerOpacity(layer, dlg.GetOpacity())
  245. dlg.Destroy()
  246. event.Skip()
  247. def _setLayerOpacity(self, layer, value):
  248. """Sets layer's opacity.'"""
  249. layer.opacity = value
  250. self._update()
  251. self.opacityChanged.emit(
  252. index=self._layerList.GetLayerIndex(layer),
  253. layer=layer)
  254. self.anyChange.emit()
  255. def _update(self):
  256. """Updates checklistbox according to layerList structure."""
  257. items = []
  258. active = []
  259. selected = []
  260. # remove hidden (temporary) layers first
  261. for layer in reversed(self._layerList):
  262. if layer.hidden:
  263. self._layerList.RemoveLayer(layer)
  264. for layer in self._layerList:
  265. if layer.opacity < 1:
  266. items.append(
  267. "{name} (opacity {opacity}%)".format(
  268. name=layer.name, opacity=int(
  269. layer.opacity * 100)))
  270. else:
  271. items.append(layer.name)
  272. active.append(layer.IsActive())
  273. selected.append(layer.IsSelected())
  274. self._checkList.SetItems(items)
  275. for i, check in enumerate(active):
  276. self._checkList.Check(i, check)
  277. for i, layer in enumerate(self._layerList):
  278. if selected[i]:
  279. self._checkList.Select(i)
  280. else:
  281. self._checkList.Deselect(i)
  282. def GetOptData(self, dcmd, layer, params, propwin):
  283. """Handler for module dialogs."""
  284. if dcmd:
  285. layer.cmd = dcmd
  286. layer.selected = True
  287. mapName, found = GetLayerNameFromCmd(dcmd)
  288. if found:
  289. try:
  290. if layer.hidden:
  291. layer.hidden = False
  292. signal = self.layerAdded
  293. else:
  294. signal = self.cmdChanged
  295. layer.name = mapName
  296. signal.emit(
  297. index=self._layerList.GetLayerIndex(layer),
  298. layer=layer)
  299. except ValueError as e:
  300. self._layerList.RemoveLayer(layer)
  301. GError(parent=self,
  302. message=str(e),
  303. showTraceback=False)
  304. self._update()
  305. self.anyChange.emit()
  306. def AddRaster(self, name, cmd, hidden, dialog):
  307. """Ads new raster layer."""
  308. layer = self._layerList.AddNewLayer(name=name, mapType='raster',
  309. active=True,
  310. cmd=cmd, hidden=hidden)
  311. return layer
  312. def AddRast3d(self, name, cmd, hidden, dialog):
  313. """Ads new raster3d layer."""
  314. layer = self._layerList.AddNewLayer(name=name, mapType='raster_3d',
  315. active=True,
  316. cmd=cmd, hidden=hidden)
  317. return layer
  318. def AddVector(self, name, cmd, hidden, dialog):
  319. """Ads new vector layer."""
  320. layer = self._layerList.AddNewLayer(name=name, mapType='vector',
  321. active=True,
  322. cmd=cmd, hidden=hidden)
  323. return layer
  324. def AddRGB(self, name, cmd, hidden, dialog):
  325. """Ads new vector layer."""
  326. layer = self._layerList.AddNewLayer(name=name, mapType='rgb',
  327. active=True,
  328. cmd=cmd, hidden=hidden)
  329. return layer
  330. def GetLayerInfo(self, layer, key):
  331. """Just for compatibility, should be removed in the future"""
  332. value = getattr(layer, key)
  333. # hack to return empty list, required in OnCancel in forms
  334. # not sure why it should be empty
  335. if key == 'cmd' and len(value) == 1:
  336. return []
  337. return value
  338. def Delete(self, layer):
  339. """Just for compatibility, should be removed in the future"""
  340. self._layerList.RemoveLayer(layer)
  341. class SimpleLmgrToolbar(BaseToolbar):
  342. """Toolbar of simple layer manager.
  343. Style of the toolbar can be changed (horizontal,
  344. vertical, which map types to include).
  345. """
  346. def __init__(self, parent, lmgrStyle):
  347. """Toolbar constructor
  348. """
  349. self._style = lmgrStyle
  350. if lmgrStyle & (SIMPLE_LMGR_TB_LEFT | SIMPLE_LMGR_TB_RIGHT):
  351. direction = wx.TB_VERTICAL
  352. else:
  353. direction = wx.TB_HORIZONTAL
  354. BaseToolbar.__init__(self, parent, style=wx.NO_BORDER | direction)
  355. self.InitToolbar(self._getToolbarData(self._toolbarData()))
  356. # realize the toolbar
  357. self.Realize()
  358. def _toolbarData(self):
  359. """Toolbar data"""
  360. data = [('edit', icons['edit'],
  361. self.parent.OnLayerChangeProperties),
  362. ('remove', icons['remove'],
  363. self.parent.OnRemove),
  364. (None, ),
  365. ('up', icons['up'],
  366. self.parent.OnLayerUp),
  367. ('down', icons['down'],
  368. self.parent.OnLayerDown),
  369. (None, ),
  370. ('opacity', icons['opacity'],
  371. self.parent.OnLayerChangeOpacity),
  372. ]
  373. if self._style & SIMPLE_LMGR_RASTER3D:
  374. data.insert(0, ('addRaster3d', icons['addRast3d'],
  375. self.parent.OnAddRast3d))
  376. if self._style & SIMPLE_LMGR_RGB:
  377. data.insert(0, ('addRGB', icons['addRGB'],
  378. self.parent.OnAddRGB))
  379. if self._style & SIMPLE_LMGR_VECTOR:
  380. data.insert(0, ('addVector', BaseIcons['addVect'],
  381. self.parent.OnAddVector))
  382. if self._style & SIMPLE_LMGR_RASTER:
  383. data.insert(0, ('addRaster', BaseIcons['addRast'],
  384. self.parent.OnAddRaster))
  385. return data
  386. icons = {
  387. 'remove': MetaIcon(img='layer-remove',
  388. label=_("Remove"),
  389. desc=_("Remove selected map(s) from list")),
  390. 'up': MetaIcon(img='layer-up',
  391. label=_("Layer up"),
  392. desc=_("Move selected layer(s) up")),
  393. 'down': MetaIcon(img='layer-down',
  394. label=_("Layer down"),
  395. desc=_("Move selected layer(s) down")),
  396. 'edit': MetaIcon(img='layer-edit',
  397. label=_("Edit layer properties"),
  398. desc=_("Edit layer properties")),
  399. 'opacity': MetaIcon(img='layer-opacity',
  400. label=_("Change opacity"),
  401. desc=_("Change layer opacity")),
  402. 'addRast3d': MetaIcon(img='layer-raster3d-add',
  403. label=_("Add 3D raster map layer"),
  404. desc=_("Add 3D raster map layer")),
  405. 'addRGB': MetaIcon(img='layer-rgb-add', label=_('Add RGB map layer'))
  406. }
  407. class TestFrame(wx.Frame):
  408. def __init__(self, parent):
  409. wx.Frame.__init__(self, parent=parent)
  410. SimpleLayerManager(parent=self, layerList=LayerList())
  411. def test():
  412. app = wx.App()
  413. frame = TestFrame(None)
  414. frame.Show()
  415. app.MainLoop()
  416. if __name__ == '__main__':
  417. test()