main.py 16 KB

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