simplelmgr.py 17 KB

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