main.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640
  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-2015 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. from __future__ import print_function
  23. import os
  24. import sys
  25. import time
  26. import shutil
  27. import fileinput
  28. from grass.script.setup import set_gui_path
  29. set_gui_path()
  30. from core import globalvar
  31. import wx
  32. from core import utils
  33. from core.giface import StandaloneGrassInterface
  34. from core.gcmd import RunCommand
  35. from core.render import Map, MapLayer, Overlay, RenderMapMgr
  36. from core.utils import _
  37. from mapdisp.frame import MapFrame
  38. from core.debug import Debug
  39. from core.settings import UserSettings
  40. from grass.script.utils import try_remove
  41. from grass.script import core as grass
  42. from grass.script.task import cmdtuple_to_list
  43. from grass.pydispatch.signal import Signal
  44. # for standalone app
  45. monFile = {'cmd': None,
  46. 'map': None,
  47. 'env': None,
  48. }
  49. monName = None
  50. monSize = list(globalvar.MAP_WINDOW_SIZE)
  51. monDecor = False
  52. class DMonMap(Map):
  53. def __init__(self, giface, cmdfile=None, mapfile=None):
  54. """Map composition (stack of map layers and overlays)
  55. :param cmdline: full path to the cmd file (defined by d.mon)
  56. :param mapfile: full path to the map file (defined by d.mon)
  57. """
  58. Map.__init__(self)
  59. self._giface = giface
  60. # environment settings
  61. self.env = dict()
  62. self.cmdfile = cmdfile
  63. # list of layers for rendering added from cmd file
  64. # TODO temporary solution, layer managment by different tools in GRASS
  65. # should be resovled
  66. self.ownedLayers = []
  67. self.oldOverlays = []
  68. if mapfile:
  69. self.mapfileCmd = mapfile
  70. self.maskfileCmd = os.path.splitext(mapfile)[0] + '.pgm'
  71. # generated file for g.pnmcomp output for rendering the map
  72. self.mapfile = monFile['map']
  73. if os.path.splitext(self.mapfile)[1] != '.ppm':
  74. self.mapfile += '.ppm'
  75. # signal sent when d.out.file/d.to.rast appears in cmd file, attribute
  76. # is cmd
  77. self.saveToFile = Signal('DMonMap.saveToFile')
  78. self.dToRast = Signal('DMonMap.dToRast')
  79. # signal sent when d.what.rast/vect appears in cmd file, attribute is
  80. # cmd
  81. self.query = Signal('DMonMap.query')
  82. self.renderMgr = RenderMapMgr(self)
  83. # update legend file variable with the one d.mon uses
  84. with open(monFile['env'], 'r') as f:
  85. lines = f.readlines()
  86. for line in lines:
  87. if 'GRASS_LEGEND_FILE' in line:
  88. legfile = line.split('=', 1)[1].strip()
  89. self.renderMgr.UpdateRenderEnv({'GRASS_LEGEND_FILE': legfile})
  90. break
  91. def GetLayersFromCmdFile(self):
  92. """Get list of map layers from cmdfile
  93. """
  94. if not self.cmdfile:
  95. return
  96. nlayers = 0
  97. try:
  98. fd = open(self.cmdfile, 'r')
  99. lines = fd.readlines()
  100. fd.close()
  101. # detect d.out.file, delete the line from the cmd file and export
  102. # graphics
  103. if len(lines) > 0:
  104. if lines[-1].startswith('d.out.file') or \
  105. lines[-1].startswith('d.to.rast'):
  106. dCmd = lines[-1].strip()
  107. fd = open(self.cmdfile, 'w')
  108. fd.writelines(lines[:-1])
  109. fd.close()
  110. if lines[-1].startswith('d.out.file'):
  111. self.saveToFile.emit(cmd=utils.split(dCmd))
  112. else:
  113. self.dToRast.emit(cmd=utils.split(dCmd))
  114. return
  115. if lines[-1].startswith('d.what'):
  116. dWhatCmd = lines[-1].strip()
  117. fd = open(self.cmdfile, 'w')
  118. fd.writelines(lines[:-1])
  119. fd.close()
  120. if '=' in utils.split(dWhatCmd)[1]:
  121. maps = utils.split(dWhatCmd)[1].split('=')[
  122. 1].split(',')
  123. else:
  124. maps = utils.split(dWhatCmd)[1].split(',')
  125. self.query.emit(ltype=utils.split(dWhatCmd)[
  126. 0].split('.')[-1], maps=maps)
  127. return
  128. else:
  129. # clean overlays after erase
  130. self.oldOverlays = []
  131. overlays = self._giface.GetMapDisplay().decorations.keys()
  132. for each in overlays:
  133. self._giface.GetMapDisplay().RemoveOverlay(each)
  134. existingLayers = self.GetListOfLayers()
  135. # holds new rendreing order for every layer in existingLayers
  136. layersOrder = [-1] * len(existingLayers)
  137. # next number in rendering order
  138. next_layer = 0
  139. mapFile = None
  140. render_env = dict()
  141. for line in lines:
  142. if line.startswith('#'):
  143. if 'GRASS_RENDER_FILE' in line:
  144. mapFile = line.split('=', 1)[1].strip()
  145. try:
  146. k, v = line[2:].strip().split('=', 1)
  147. except:
  148. pass
  149. render_env[k] = v
  150. continue
  151. cmd = utils.split(line.strip())
  152. ltype = None
  153. try:
  154. ltype = utils.command2ltype[cmd[0]]
  155. except KeyError:
  156. grass.warning(_("Unsupported command %s.") % cmd[0])
  157. continue
  158. name = utils.GetLayerNameFromCmd(cmd, fullyQualified=True,
  159. layerType=ltype)[0]
  160. args = {}
  161. if ltype in ('barscale', 'rastleg', 'northarrow', 'text', 'vectleg'):
  162. # TODO: this is still not optimal
  163. # it is there to prevent adding the same overlay multiple times
  164. if cmd in self.oldOverlays:
  165. continue
  166. if ltype == 'rastleg':
  167. self._giface.GetMapDisplay().AddLegendRast(cmd=cmd)
  168. elif ltype == 'barscale':
  169. self._giface.GetMapDisplay().AddBarscale(cmd=cmd)
  170. elif ltype == 'northarrow':
  171. self._giface.GetMapDisplay().AddArrow(cmd=cmd)
  172. elif ltype == 'text':
  173. self._giface.GetMapDisplay().AddDtext(cmd=cmd)
  174. elif ltype == 'vectleg':
  175. self._giface.GetMapDisplay().AddLegendVect(cmd=cmd)
  176. self.oldOverlays.append(cmd)
  177. continue
  178. classLayer = MapLayer
  179. args['ltype'] = ltype
  180. mapLayer = classLayer(name=name,
  181. cmd=cmd,
  182. Map=None,
  183. hidden=True,
  184. render=False,
  185. mapfile=mapFile,
  186. **args)
  187. mapLayer.GetRenderMgr().updateProgress.connect(self.GetRenderMgr().ReportProgress)
  188. if render_env:
  189. mapLayer.GetRenderMgr().UpdateRenderEnv(render_env)
  190. render_env = dict()
  191. exists = False
  192. for i, layer in enumerate(existingLayers):
  193. if layer.GetCmd(
  194. string=True) == mapLayer.GetCmd(
  195. string=True):
  196. exists = True
  197. if layersOrder[i] == -1:
  198. layersOrder[i] = next_layer
  199. next_layer += 1
  200. # layer must be put higher in render order (same cmd was insered more times)
  201. # TODO delete rendurant cmds from cmd file?
  202. else:
  203. for j, l_order in enumerate(layersOrder):
  204. if l_order > layersOrder[i]:
  205. layersOrder[j] -= 1
  206. layersOrder[i] = next_layer - 1
  207. break
  208. if exists:
  209. continue
  210. newLayer = self._addLayer(mapLayer)
  211. existingLayers.append(newLayer)
  212. self.ownedLayers.append(newLayer)
  213. layersOrder.append(next_layer)
  214. next_layer += 1
  215. nlayers += 1
  216. reorderedLayers = [-1] * next_layer
  217. for i, layer in enumerate(existingLayers):
  218. # owned layer was not found in cmd file -> is deleted
  219. if layersOrder[i] == -1 and layer in self.ownedLayers:
  220. self.ownedLayers.remove(layer)
  221. self.DeleteLayer(layer)
  222. # other layer e. g. added by wx.vnet are added to the top
  223. elif layersOrder[i] == -1 and layer not in self.ownedLayers:
  224. reorderedLayers.append(layer)
  225. # owned layer found in cmd file is added into proper rendering
  226. # position
  227. else:
  228. reorderedLayers[layersOrder[i]] = layer
  229. self.SetLayers(reorderedLayers)
  230. except IOError as e:
  231. grass.warning(
  232. _("Unable to read cmdfile '%(cmd)s'. Details: %(det)s") %
  233. {'cmd': self.cmdfile, 'det': e})
  234. return
  235. Debug.msg(1, "Map.GetLayersFromCmdFile(): cmdfile=%s, nlayers=%d" %
  236. (self.cmdfile, nlayers))
  237. self._giface.updateMap.emit(render=False)
  238. def Render(self, *args, **kwargs):
  239. """Render layer to image.
  240. For input params and returned data see overridden method in Map class.
  241. """
  242. return Map.Render(self, *args, **kwargs)
  243. def AddLayer(self, *args, **kwargs):
  244. """Adds generic map layer to list of layers.
  245. For input params and returned data see overridden method in Map class.
  246. """
  247. driver = UserSettings.Get(group='display', key='driver', subkey='type')
  248. if driver == 'png':
  249. os.environ["GRASS_RENDER_IMMEDIATE"] = "png"
  250. else:
  251. os.environ["GRASS_RENDER_IMMEDIATE"] = "cairo"
  252. layer = Map.AddLayer(self, *args, **kwargs)
  253. del os.environ["GRASS_RENDER_IMMEDIATE"]
  254. return layer
  255. class Layer(object):
  256. """@implements core::giface::Layer"""
  257. def __init__(self, maplayer):
  258. self._maplayer = maplayer
  259. def __getattr__(self, name):
  260. if name == 'cmd':
  261. return cmdtuple_to_list(self._maplayer.GetCmd())
  262. elif hasattr(self._maplayer, name):
  263. return getattr(self._maplayer, name)
  264. elif name == 'maplayer':
  265. return self._maplayer
  266. elif name == 'type':
  267. return self._maplayer.GetType()
  268. # elif name == 'ctrl':
  269. elif name == 'label':
  270. return self._maplayer.GetName()
  271. # elif name == 'propwin':
  272. class LayerList(object):
  273. """@implements core::giface::LayerList"""
  274. def __init__(self, map, giface):
  275. self._map = map
  276. self._giface = giface
  277. self._index = 0
  278. def __len__(self):
  279. return len(self._map.GetListOfLayers())
  280. def __iter__(self):
  281. return self
  282. def next(self):
  283. items = self._map.GetListOfLayers()
  284. try:
  285. result = items[self._index]
  286. except IndexError:
  287. raise StopIteration
  288. self._index += 1
  289. return result
  290. def GetSelectedLayers(self, checkedOnly=True):
  291. # hidden and selected vs checked and selected
  292. items = self._map.GetListOfLayers()
  293. layers = []
  294. for item in items:
  295. layer = Layer(item)
  296. layers.append(layer)
  297. return layers
  298. def GetSelectedLayer(self, checkedOnly=False):
  299. """Returns selected layer or None when there is no selected layer."""
  300. layers = self.GetSelectedLayers()
  301. if len(layers) > 0:
  302. return layers[0]
  303. else:
  304. return None
  305. def AddLayer(self, ltype, name=None, checked=None,
  306. opacity=1.0, cmd=None):
  307. """Adds a new layer to the layer list.
  308. Launches property dialog if needed (raster, vector, etc.)
  309. :param ltype: layer type (raster, vector, raster_3d, ...)
  310. :param name: layer name
  311. :param checked: if True layer is checked
  312. :param opacity: layer opacity level
  313. :param cmd: command (given as a list)
  314. """
  315. self._map.AddLayer(ltype=ltype, command=cmd,
  316. name=name, active=True,
  317. opacity=opacity, render=True,
  318. pos=-1)
  319. # TODO: this should be solved by signal
  320. # (which should be introduced everywhere,
  321. # alternative is some observer list)
  322. self._giface.updateMap.emit(render=True, renderVector=True)
  323. def GetLayersByName(self, name):
  324. items = self._map.GetListOfLayers()
  325. layers = []
  326. for item in items:
  327. if item.GetName() == name:
  328. layer = Layer(item)
  329. layers.append(layer)
  330. return layers
  331. def GetLayerByData(self, key, value):
  332. # TODO: implementation was not tested
  333. items = self._map.GetListOfLayers()
  334. for item in items:
  335. layer = Layer(item)
  336. try:
  337. if getattr(layer, key) == value:
  338. return layer
  339. except AttributeError:
  340. pass
  341. return None
  342. class DMonGrassInterface(StandaloneGrassInterface):
  343. """@implements GrassInterface"""
  344. def __init__(self, mapframe):
  345. StandaloneGrassInterface.__init__(self)
  346. self._mapframe = mapframe
  347. def GetLayerList(self):
  348. return LayerList(self._mapframe.GetMap(), giface=self)
  349. def GetMapWindow(self):
  350. return self._mapframe.GetMapWindow()
  351. def GetMapDisplay(self):
  352. return self._mapframe
  353. def GetProgress(self):
  354. return self._mapframe.GetProgressBar()
  355. def ShowStatusbar(self, show=True):
  356. if not self._mapframe.statusbarManager:
  357. self._mapframe.CreateStatusbar()
  358. self._mapframe.statusbarManager.Show(show)
  359. def IsStatusbarShown(self):
  360. if not self._mapframe.statusbarManager:
  361. return False
  362. return self._mapframe.statusbarManager.IsShown()
  363. def ShowAllToolbars(self, show=True):
  364. if not show: # hide
  365. action = self._mapframe.RemoveToolbar
  366. else:
  367. action = self._mapframe.AddToolbar
  368. toolbars = self._mapframe.GetToolbarNames()
  369. if not toolbars:
  370. toolbars.append('map')
  371. for toolbar in toolbars:
  372. action(toolbar)
  373. def AreAllToolbarsShown(self):
  374. toolbar = self._mapframe.GetMapToolbar()
  375. if toolbar is None:
  376. return False
  377. return toolbar.IsShown()
  378. class DMonFrame(MapFrame):
  379. def OnZoomToMap(self, event):
  380. layers = self.MapWindow.GetMap().GetListOfLayers()
  381. self.MapWindow.ZoomToMap(layers=layers)
  382. def OnSize(self, event):
  383. super(DMonFrame, self).OnSize(event)
  384. # update env file
  385. width, height = self.MapWindow.GetClientSize()
  386. for line in fileinput.input(monFile['env'], inplace=True):
  387. if 'GRASS_RENDER_WIDTH' in line:
  388. print('GRASS_RENDER_WIDTH={0}'.format(width))
  389. elif 'GRASS_RENDER_HEIGHT' in line:
  390. print('GRASS_RENDER_HEIGHT={0}'.format(height))
  391. else:
  392. print(line.rstrip('\n'))
  393. class MapApp(wx.App):
  394. def OnInit(self):
  395. if not globalvar.CheckWxVersion([2, 9]):
  396. wx.InitAllImageHandlers()
  397. grass.set_raise_on_error(True)
  398. # actual use of StandaloneGrassInterface not yet tested
  399. # needed for adding functionality in future
  400. self._giface = DMonGrassInterface(None)
  401. return True
  402. def CreateMapFrame(self, name, decorations=True):
  403. toolbars = []
  404. if decorations:
  405. toolbars.append('map')
  406. if __name__ == "__main__":
  407. self.cmdTimeStamp = os.path.getmtime(monFile['cmd'])
  408. self.Map = DMonMap(giface=self._giface, cmdfile=monFile['cmd'],
  409. mapfile=monFile['map'])
  410. self.timer = wx.PyTimer(self.watcher)
  411. # check each 0.5s
  412. global mtime
  413. mtime = 500
  414. self.timer.Start(mtime)
  415. else:
  416. self.Map = None
  417. self.mapFrm = DMonFrame(
  418. parent=None,
  419. id=wx.ID_ANY,
  420. title=name,
  421. Map=self.Map,
  422. giface=self._giface,
  423. size=monSize,
  424. toolbars=toolbars,
  425. statusbar=decorations)
  426. # FIXME: hack to solve dependency
  427. self._giface._mapframe = self.mapFrm
  428. self.mapFrm.GetMapWindow().SetAlwaysRenderEnabled(False)
  429. self.Map.saveToFile.connect(lambda cmd: self.mapFrm.DOutFile(cmd))
  430. self.Map.dToRast.connect(lambda cmd: self.mapFrm.DToRast(cmd))
  431. self.Map.query.connect(
  432. lambda ltype,
  433. maps: self.mapFrm.SetQueryLayersAndActivate(
  434. ltype=ltype,
  435. maps=maps))
  436. return self.mapFrm
  437. def OnExit(self):
  438. if __name__ == "__main__":
  439. # stop the timer
  440. # self.timer.Stop()
  441. # terminate thread
  442. for f in monFile.itervalues():
  443. try_remove(f)
  444. def watcher(self):
  445. """Redraw, if new layer appears (check's timestamp of
  446. cmdfile)
  447. """
  448. ###
  449. # TODO: find a better solution
  450. ###
  451. # the check below disabled, it's too much invasive to call
  452. # g.gisenv in the watcher...
  453. # try:
  454. # GISBASE and other system enviromental variables can not be used
  455. # since the process inherited them from GRASS
  456. # raises exception when vaiable does not exists
  457. # grass.gisenv()['GISDBASE']
  458. # except KeyError:
  459. # self.timer.Stop()
  460. # return
  461. # todo: events
  462. try:
  463. currentCmdFileTime = os.path.getmtime(monFile['cmd'])
  464. if currentCmdFileTime > self.cmdTimeStamp:
  465. self.timer.Stop()
  466. self.cmdTimeStamp = currentCmdFileTime
  467. self.mapFrm.GetMap().GetLayersFromCmdFile()
  468. self.timer.Start(mtime)
  469. except OSError as e:
  470. grass.warning("%s" % e)
  471. self.timer.Stop()
  472. def GetMapFrame(self):
  473. """Get Map Frame instance"""
  474. return self.mapFrm
  475. if __name__ == "__main__":
  476. if len(sys.argv) != 6:
  477. print(__doc__)
  478. sys.exit(0)
  479. # set command variable
  480. monName = sys.argv[1]
  481. monPath = sys.argv[2]
  482. monFile = {'map': os.path.join(monPath, 'map.ppm'),
  483. 'cmd': os.path.join(monPath, 'cmd'),
  484. 'env': os.path.join(monPath, 'env')}
  485. # monitor size
  486. monSize = (int(sys.argv[3]), int(sys.argv[4]))
  487. monDecor = not bool(int(sys.argv[5]))
  488. grass.verbose(_("Starting map display <%s>...") % (monName))
  489. # create pid file
  490. pidFile = os.path.join(monPath, "pid")
  491. fd = open(pidFile, 'w')
  492. if not fd:
  493. grass.fatal(_("Unable to create file <%s>") % pidFile)
  494. fd.write("%s\n" % os.getpid())
  495. fd.close()
  496. RunCommand('g.gisenv',
  497. set='MONITOR_%s_PID=%d' % (monName.upper(), os.getpid()))
  498. start = time.time()
  499. gmMap = MapApp(0)
  500. mapFrame = gmMap.CreateMapFrame(monName, monDecor)
  501. mapFrame.Show()
  502. Debug.msg(1, "WxMonitor started in %.6f sec" %
  503. (time.time() - start))
  504. gmMap.MainLoop()
  505. grass.verbose(_("Stopping map display <%s>...") % (monName))
  506. # clean up GRASS env variables
  507. try:
  508. shutil.rmtree(monPath)
  509. except OSError:
  510. pass
  511. RunCommand('g.gisenv',
  512. unset='MONITOR')
  513. sys.exit(0)