main.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531
  1. """
  2. @package mapdisp.main
  3. @brief Start Map Display as standalone application
  4. Classes:
  5. - mapdisp::DMonMap
  6. - mapdisp::Layer
  7. - mapdisp::LayerList
  8. - mapdisp::DMonGrassInterface
  9. - mapdisp::DMonFrame
  10. - mapdisp::MapApp
  11. Usage:
  12. python mapdisp/main.py monitor-identifier /path/to/map/file /path/to/command/file /path/to/env/file
  13. (C) 2006-2014 by the GRASS Development Team
  14. This program is free software under the GNU General Public License
  15. (>=v2). Read the file COPYING that comes with GRASS for details.
  16. @author Michael Barton
  17. @author Jachym Cepicky
  18. @author Martin Landa <landa.martin gmail.com>
  19. @author Vaclav Petras <wenzeslaus gmail.com> (MapFrameBase)
  20. @author Anna Kratochvilova <kratochanna gmail.com> (MapFrameBase)
  21. """
  22. import os
  23. import sys
  24. import time
  25. from core import globalvar
  26. import wx
  27. from core import utils
  28. from core.giface import StandaloneGrassInterface
  29. from core.gcmd import RunCommand
  30. from core.render import Map, MapLayer
  31. from core.utils import _
  32. from mapdisp.frame import MapFrame
  33. from core.debug import Debug
  34. from core.settings import UserSettings
  35. from grass.script.utils import try_remove
  36. from grass.script import core as grass
  37. from grass.script.task import cmdtuple_to_list
  38. from grass.pydispatch.signal import Signal
  39. # for standalone app
  40. monFile = { 'cmd' : None,
  41. 'map' : None,
  42. 'env' : None,
  43. }
  44. monName = None
  45. monSize = list(globalvar.MAP_WINDOW_SIZE)
  46. monDecor = False
  47. class DMonMap(Map):
  48. def __init__(self, giface, cmdfile=None, mapfile=None):
  49. """Map composition (stack of map layers and overlays)
  50. :param cmdline: full path to the cmd file (defined by d.mon)
  51. :param mapfile: full path to the map file (defined by d.mon)
  52. """
  53. Map.__init__(self)
  54. self._giface = giface
  55. # environment settings
  56. self.env = dict()
  57. self.cmdfile = cmdfile
  58. # list of layers for rendering added from cmd file
  59. # TODO temporary solution, layer managment by different tools in GRASS should be resovled
  60. self.ownedLayers = []
  61. if mapfile:
  62. self.mapfileCmd = mapfile
  63. self.maskfileCmd = os.path.splitext(mapfile)[0] + '.pgm'
  64. # generated file for g.pnmcomp output for rendering the map
  65. self.mapfile = monFile['map']
  66. if os.path.splitext(self.mapfile)[1] != '.ppm':
  67. self.mapfile += '.ppm'
  68. # signal sent when d.out.file/d.to.rast appears in cmd file, attribute is cmd
  69. self.saveToFile = Signal('DMonMap.saveToFile')
  70. self.dToRast = Signal('DMonMap.dToRast')
  71. # signal sent when d.what.rast/vect appears in cmd file, attribute is cmd
  72. self.query = Signal('DMonMap.query')
  73. def GetLayersFromCmdFile(self):
  74. """Get list of map layers from cmdfile
  75. """
  76. if not self.cmdfile:
  77. return
  78. nlayers = 0
  79. try:
  80. fd = open(self.cmdfile, 'r')
  81. lines = fd.readlines()
  82. fd.close()
  83. # detect d.out.file, delete the line from the cmd file and export graphics
  84. if len(lines) > 0:
  85. if lines[-1].startswith('d.out.file') or lines[-1].startswith('d.to.rast'):
  86. dCmd = lines[-1].strip()
  87. fd = open(self.cmdfile, 'w')
  88. fd.writelines(lines[:-1])
  89. fd.close()
  90. if lines[-1].startswith('d.out.file'):
  91. self.saveToFile.emit(cmd=utils.split(dCmd))
  92. else:
  93. self.dToRast.emit(cmd=utils.split(dCmd))
  94. return
  95. if lines[-1].startswith('d.what'):
  96. dWhatCmd = lines[-1].strip()
  97. fd = open(self.cmdfile, 'w')
  98. fd.writelines(lines[:-1])
  99. fd.close()
  100. if '=' in utils.split(dWhatCmd)[1]:
  101. maps = utils.split(dWhatCmd)[1].split('=')[1].split(',')
  102. else:
  103. maps = utils.split(dWhatCmd)[1].split(',')
  104. self.query.emit(ltype=utils.split(dWhatCmd)[0].split('.')[-1], maps=maps)
  105. return
  106. existingLayers = self.GetListOfLayers()
  107. # holds new rendreing order for every layer in existingLayers
  108. layersOrder = [-1] * len(self.GetListOfLayers())
  109. # next number in rendering order
  110. next_layer = 0
  111. for line in lines:
  112. cmd = utils.split(line.strip())
  113. ltype = None
  114. try:
  115. ltype = utils.command2ltype[cmd[0]]
  116. except KeyError:
  117. grass.warning(_("Unsupported command %s.") % cmd[0])
  118. continue
  119. name = utils.GetLayerNameFromCmd(cmd, fullyQualified = True,
  120. layerType = ltype)[0]
  121. # creating temporary layer object to compare commands
  122. # neccessary to get the same format
  123. # supposing that there are no side effects
  124. tmpMapLayer = MapLayer(ltype = ltype, name = name,
  125. cmd = cmd, Map = None,
  126. active = False, hidden = True,
  127. opacity = 0)
  128. exists = False
  129. for i, layer in enumerate(existingLayers):
  130. if layer.GetCmd(string=True) == tmpMapLayer.GetCmd(string=True):
  131. exists = True
  132. if layersOrder[i] == -1:
  133. layersOrder[i] = next_layer;
  134. next_layer += 1
  135. # layer must be put higher in render order (same cmd was insered more times)
  136. # TODO delete rendurant cmds from cmd file?
  137. else:
  138. for j, l_order in enumerate(layersOrder):
  139. if l_order > layersOrder[i]:
  140. layersOrder[j] -= 1;
  141. layersOrder[i] = next_layer - 1;
  142. break
  143. if exists:
  144. continue
  145. newLayer = Map.AddLayer(self, ltype = ltype, command = cmd, active = True, name = name)
  146. existingLayers.append(newLayer)
  147. self.ownedLayers.append(newLayer)
  148. layersOrder.append(next_layer)
  149. next_layer += 1
  150. nlayers += 1
  151. reorderedLayers = [-1] * next_layer
  152. for i, layer in enumerate(existingLayers):
  153. # owned layer was not found in cmd file -> is deleted
  154. if layersOrder[i] == -1 and layer in self.ownedLayers:
  155. self.ownedLayers.remove(layer)
  156. self.DeleteLayer(layer)
  157. # other layer e. g. added by wx.vnet are added to the top
  158. elif layersOrder[i] == -1 and layer not in self.ownedLayers:
  159. reorderedLayers.append(layer)
  160. # owned layer found in cmd file is added into proper rendering position
  161. else:
  162. reorderedLayers[layersOrder[i]] = layer
  163. self.SetLayers(reorderedLayers)
  164. except IOError as e:
  165. grass.warning(_("Unable to read cmdfile '%(cmd)s'. Details: %(det)s") % \
  166. { 'cmd' : self.cmdfile, 'det' : e })
  167. return
  168. self._giface.updateMap.emit()
  169. Debug.msg(1, "Map.GetLayersFromCmdFile(): cmdfile=%s" % self.cmdfile)
  170. Debug.msg(1, " nlayers=%d" % nlayers)
  171. def Render(self, *args, **kwargs):
  172. """Render layer to image.
  173. For input params and returned data see overridden method in Map class.
  174. """
  175. ret = Map.Render(self, *args, **kwargs)
  176. return ret
  177. def AddLayer(self, *args, **kwargs):
  178. """Adds generic map layer to list of layers.
  179. For input params and returned data see overridden method in Map class.
  180. """
  181. driver = UserSettings.Get(group = 'display', key = 'driver', subkey = 'type')
  182. if driver == 'png':
  183. os.environ["GRASS_RENDER_IMMEDIATE"] = "png"
  184. else:
  185. os.environ["GRASS_RENDER_IMMEDIATE"] = "cairo"
  186. layer = Map.AddLayer(self, *args, **kwargs)
  187. del os.environ["GRASS_RENDER_IMMEDIATE"]
  188. return layer
  189. class Layer(object):
  190. """@implements core::giface::Layer"""
  191. def __init__(self, maplayer):
  192. self._maplayer = maplayer
  193. def __getattr__(self, name):
  194. if name == 'cmd':
  195. return cmdtuple_to_list(self._maplayer.GetCmd())
  196. elif hasattr(self._maplayer, name):
  197. return getattr(self._maplayer, name)
  198. elif name == 'maplayer':
  199. return self._maplayer
  200. elif name == 'type':
  201. return self._maplayer.GetType()
  202. #elif name == 'ctrl':
  203. elif name == 'label':
  204. return self._maplayer.GetName()
  205. #elif name == 'propwin':
  206. class LayerList(object):
  207. def __init__(self, map, giface):
  208. """@implements core::giface::LayerList"""
  209. self._map = map
  210. self._giface = giface
  211. def GetSelectedLayers(self, checkedOnly=True):
  212. # hidden and selected vs checked and selected
  213. items = self._map.GetListOfLayers()
  214. layers = []
  215. for item in items:
  216. layer = Layer(item)
  217. layers.append(layer)
  218. return layers
  219. def GetSelectedLayer(self, checkedOnly=False):
  220. """Returns selected layer or None when there is no selected layer."""
  221. layers = self.GetSelectedLayers()
  222. if len(layers) > 0:
  223. return layers[0]
  224. else:
  225. return None
  226. def AddLayer(self, ltype, name=None, checked=None,
  227. opacity=1.0, cmd=None):
  228. """Adds a new layer to the layer list.
  229. Launches property dialog if needed (raster, vector, etc.)
  230. :param ltype: layer type (raster, vector, 3d-raster, ...)
  231. :param name: layer name
  232. :param checked: if True layer is checked
  233. :param opacity: layer opacity level
  234. :param cmd: command (given as a list)
  235. """
  236. self._map.AddLayer(ltype=ltype, command=cmd,
  237. name=name, active=True,
  238. opacity=opacity, render=True,
  239. pos=-1)
  240. # TODO: this should be solved by signal
  241. # (which should be introduced everywhere,
  242. # alternative is some observer list)
  243. self._giface.updateMap.emit(render=True, renderVector=True)
  244. def GetLayersByName(self, name):
  245. items = self._map.GetListOfLayers()
  246. layers = []
  247. for item in items:
  248. if item.GetName() == name:
  249. layer = Layer(item)
  250. layers.append(layer)
  251. return layers
  252. def GetLayerByData(self, key, value):
  253. # TODO: implementation was not tested
  254. items = self._map.GetListOfLayers()
  255. for item in items:
  256. layer = Layer(item)
  257. try:
  258. if getattr(layer, key) == value:
  259. return layer
  260. except AttributeError:
  261. pass
  262. return None
  263. class DMonGrassInterface(StandaloneGrassInterface):
  264. """@implements GrassInterface"""
  265. def __init__(self, mapframe):
  266. StandaloneGrassInterface.__init__(self)
  267. self._mapframe = mapframe
  268. def GetLayerList(self):
  269. return LayerList(self._mapframe.GetMap(), giface=self)
  270. def GetMapWindow(self):
  271. return self._mapframe.GetMapWindow()
  272. def GetProgress(self):
  273. return self._mapframe.GetProgressBar()
  274. def ShowStatusbar(self, show=True):
  275. if not self._mapframe.statusbarManager:
  276. self._mapframe.CreateStatusbar()
  277. self._mapframe.statusbarManager.Show(show)
  278. def IsStatusbarShown(self):
  279. if not self._mapframe.statusbarManager:
  280. return False
  281. return self._mapframe.statusbarManager.IsShown()
  282. def ShowAllToolbars(self, show=True):
  283. if not show: # hide
  284. action = self._mapframe.RemoveToolbar
  285. else:
  286. action = self._mapframe.AddToolbar
  287. toolbars = self._mapframe.GetToolbarNames()
  288. if not toolbars:
  289. toolbars.append('map')
  290. for toolbar in toolbars:
  291. action(toolbar)
  292. def AreAllToolbarsShown(self):
  293. toolbar = self._mapframe.GetMapToolbar()
  294. if toolbar is None:
  295. return False
  296. return toolbar.IsShown()
  297. class DMonFrame(MapFrame):
  298. def OnZoomToMap(self, event):
  299. layers = self.MapWindow.GetMap().GetListOfLayers()
  300. self.MapWindow.ZoomToMap(layers = layers)
  301. class MapApp(wx.App):
  302. def OnInit(self):
  303. if not globalvar.CheckWxVersion([2, 9]):
  304. wx.InitAllImageHandlers()
  305. grass.set_raise_on_error(True)
  306. # actual use of StandaloneGrassInterface not yet tested
  307. # needed for adding functionality in future
  308. self._giface = DMonGrassInterface(None)
  309. if __name__ == "__main__":
  310. self.cmdTimeStamp = os.path.getmtime(monFile['cmd'])
  311. self.Map = DMonMap(giface=self._giface, cmdfile=monFile['cmd'],
  312. mapfile = monFile['map'])
  313. self.timer = wx.PyTimer(self.watcher)
  314. #check each 0.5s
  315. global mtime
  316. mtime = 500
  317. self.timer.Start(mtime)
  318. else:
  319. self.Map = None
  320. return True
  321. def CreateMapFrame(self, name, decorations=True):
  322. if decorations:
  323. toolbars = ['map']
  324. statusbar = True
  325. else:
  326. toolbars = []
  327. statusbar = False
  328. self.mapFrm = DMonFrame(parent=None, id=wx.ID_ANY, title=name, Map=self.Map,
  329. giface=self._giface, size=monSize,
  330. toolbars=toolbars, statusbar=statusbar)
  331. # FIXME: hack to solve dependency
  332. self._giface._mapframe = self.mapFrm
  333. self.mapFrm.GetMapWindow().SetAlwaysRenderEnabled(True)
  334. self.Map.saveToFile.connect(lambda cmd: self.mapFrm.DOutFile(cmd))
  335. self.Map.dToRast.connect(lambda cmd: self.mapFrm.DToRast(cmd))
  336. self.Map.query.connect(lambda ltype, maps: \
  337. self.mapFrm.SetQueryLayersAndActivate(ltype=ltype, maps=maps))
  338. return self.mapFrm
  339. def OnExit(self):
  340. if __name__ == "__main__":
  341. # stop the timer
  342. # self.timer.Stop()
  343. # terminate thread
  344. for f in monFile.itervalues():
  345. try_remove(f)
  346. def watcher(self):
  347. """Redraw, if new layer appears (check's timestamp of
  348. cmdfile)
  349. """
  350. ###
  351. ### TODO: find a better solution
  352. ###
  353. ### the check bellow disabled, it's too much invasive to call
  354. ### g.gisenv in the watcher...
  355. # try:
  356. # GISBASE and other system enviromental variables can not be used
  357. # since the process inherited them from GRASS
  358. # raises exception when vaiable does not exists
  359. ### grass.gisenv()['GISDBASE']
  360. # except KeyError:
  361. # self.timer.Stop()
  362. # return
  363. # todo: events
  364. try:
  365. currentCmdFileTime = os.path.getmtime(monFile['cmd'])
  366. if currentCmdFileTime > self.cmdTimeStamp:
  367. self.timer.Stop()
  368. self.cmdTimeStamp = currentCmdFileTime
  369. self.mapFrm.GetMap().GetLayersFromCmdFile()
  370. self.timer.Start(mtime)
  371. except OSError as e:
  372. grass.warning("%s" % e)
  373. self.timer.Stop()
  374. def GetMapFrame(self):
  375. """Get Map Frame instance"""
  376. return self.mapFrm
  377. if __name__ == "__main__":
  378. # set command variable
  379. if len(sys.argv) < 5:
  380. print __doc__
  381. sys.exit(1)
  382. monName = sys.argv[1]
  383. monFile = { 'map' : sys.argv[2],
  384. 'cmd' : sys.argv[3],
  385. 'env' : sys.argv[4],
  386. }
  387. if len(sys.argv) >= 6:
  388. try:
  389. monSize[0] = int(sys.argv[5])
  390. except ValueError:
  391. pass
  392. if len(sys.argv) >= 7:
  393. try:
  394. monSize[1] = int(sys.argv[6])
  395. except ValueError:
  396. pass
  397. if len(sys.argv) == 8:
  398. try:
  399. monDecor = True if sys.argv[7] == "0" else False
  400. except ValueError:
  401. monDecor = True
  402. grass.verbose(_("Starting map display <%s>...") % (monName))
  403. RunCommand('g.gisenv',
  404. set = 'MONITOR_%s_PID=%d' % (monName.upper(), os.getpid()))
  405. start = time.time()
  406. gmMap = MapApp(0)
  407. mapFrame = gmMap.CreateMapFrame(monName, monDecor)
  408. mapFrame.Show()
  409. Debug.msg(1, "WxMonitor started in %.6f sec" % \
  410. (time.time() - start))
  411. gmMap.MainLoop()
  412. grass.verbose(_("Stopping map display <%s>...") % (monName))
  413. # clean up GRASS env variables
  414. env = grass.gisenv()
  415. env_name = 'MONITOR_%s' % monName.upper()
  416. unset = list()
  417. for key in env.keys():
  418. if key.find(env_name) == 0 or \
  419. (key == 'MONITOR' and env[key] == monName):
  420. unset.append(key)
  421. if unset:
  422. RunCommand('g.gisenv',
  423. unset = '%s' % ','.join(unset))
  424. sys.exit(0)