widgets.py 38 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174
  1. """
  2. @package web_services.widgets
  3. @brief Widgets for web services (WMS, WMTS, NasaOnEarh)
  4. List of classes:
  5. - widgets::WSPanel
  6. - widgets::LayersList
  7. (C) 2012-2013 by the GRASS Development Team
  8. This program is free software under the GNU General Public License
  9. (>=v2). Read the file COPYING that comes with GRASS for details.
  10. @author Martin Landa <landa.martin gmail.com>
  11. @author Stepan Turek <stepan.turek seznam.cz>
  12. """
  13. import re
  14. import os
  15. import sys
  16. import six
  17. import shutil
  18. from copy import deepcopy
  19. from core import globalvar
  20. try:
  21. from xml.etree.ElementTree import ParseError
  22. except ImportError: # < Python 2.7
  23. from xml.parsers.expat import ExpatError as ParseError
  24. import wx
  25. if globalvar.wxPythonPhoenix:
  26. try:
  27. import agw.flatnotebook as FN
  28. except ImportError: # if it's not there locally, try the wxPython lib.
  29. import wx.lib.agw.flatnotebook as FN
  30. else:
  31. import wx.lib.flatnotebook as FN
  32. import wx.lib.colourselect as csel
  33. from core.debug import Debug
  34. from core.gcmd import GWarning, GMessage
  35. from core.gconsole import CmdThread, GStderr, EVT_CMD_DONE, EVT_CMD_OUTPUT
  36. from web_services.cap_interface import WMSCapabilities, WMTSCapabilities, OnEarthCapabilities
  37. from gui_core.widgets import GNotebook
  38. from gui_core.widgets import ManageSettingsWidget
  39. from gui_core.wrap import Button, ScrolledPanel, SpinCtrl, StaticBox, \
  40. StaticText, TextCtrl, TreeCtrl
  41. import grass.script as grass
  42. rinwms_path = os.path.join(os.getenv("GISBASE"), "etc", "r.in.wms")
  43. if rinwms_path not in sys.path:
  44. sys.path.append(rinwms_path)
  45. from wms_base import WMSDriversInfo
  46. from srs import Srs
  47. from grass.pydispatch.signal import Signal
  48. class WSPanel(wx.Panel):
  49. def __init__(self, parent, web_service, **kwargs):
  50. """Show data from capabilities file.
  51. Signal: capParsed - this signal is emitted when capabilities file is downloaded
  52. (after ConnectToServer method was called)
  53. :param parent: parent widget
  54. :param web_service: web service to be panel generated for
  55. """
  56. wx.Panel.__init__(self, parent=parent, id=wx.ID_ANY)
  57. self.parent = parent
  58. self.ws = web_service
  59. self.capParsed = Signal('WSPanel.capParsed')
  60. # stores widgets, which represents parameters/flags of d.wms
  61. self.params = {}
  62. self.flags = {}
  63. self.o_layer_name = ''
  64. # stores err output from r.in.wms during getting capabilities
  65. self.cmd_err_str = ''
  66. # stores selected layer from layer list
  67. self.sel_layers = []
  68. # downloaded and parsed data from server successfully?
  69. self.is_connected = False
  70. # common part of command for r.in.wms -c and d.wms
  71. self.ws_cmdl = None
  72. # provides information about driver parameters
  73. self.drv_info = WMSDriversInfo()
  74. self.drv_props = self.drv_info.GetDrvProperties(self.ws)
  75. self.ws_drvs = {
  76. 'WMS_1.1.1': {
  77. 'cmd': [
  78. 'wms_version=1.1.1',
  79. 'driver=WMS_GRASS'],
  80. 'cap_parser': lambda temp_file: WMSCapabilities(
  81. temp_file,
  82. '1.1.1'),
  83. },
  84. 'WMS_1.3.0': {
  85. 'cmd': [
  86. 'wms_version=1.3.0',
  87. 'driver=WMS_GRASS'],
  88. 'cap_parser': lambda temp_file: WMSCapabilities(
  89. temp_file,
  90. '1.3.0'),
  91. },
  92. 'WMTS': {
  93. 'cmd': ['driver=WMTS_GRASS'],
  94. 'cap_parser': WMTSCapabilities,
  95. },
  96. 'OnEarth': {
  97. 'cmd': ['driver=OnEarth_GRASS'],
  98. 'cap_parser': OnEarthCapabilities,
  99. }}
  100. self.cmdStdErr = GStderr(self)
  101. self.cmd_thread = CmdThread(self)
  102. self.cap_file = grass.tempfile()
  103. reqDataBox = StaticBox(
  104. parent=self, label=_(" Requested data settings "))
  105. self._nb_sizer = wx.StaticBoxSizer(reqDataBox, wx.VERTICAL)
  106. self.notebook = GNotebook(parent=self,
  107. style=FN.FNB_FANCY_TABS | FN.FNB_NO_X_BUTTON)
  108. self._requestPage()
  109. self._advancedSettsPage()
  110. self._layout()
  111. self.layerSelected = self.list.layerSelected
  112. self.Bind(EVT_CMD_DONE, self.OnCapDownloadDone)
  113. self.Bind(EVT_CMD_OUTPUT, self.OnCmdOutput)
  114. self.SetMinSize((-1, 300))
  115. def __del__(self):
  116. self.cmd_thread.abort(abortall=True)
  117. grass.try_remove(self.cap_file)
  118. def _layout(self):
  119. self._nb_sizer.Add(self.notebook, proportion=1, flag=wx.EXPAND)
  120. self.SetSizer(self._nb_sizer)
  121. def _requestPage(self):
  122. """Create request page"""
  123. self.req_page_panel = wx.Panel(parent=self, id=wx.ID_ANY)
  124. self.notebook.AddPage(page=self.req_page_panel,
  125. text=_('Request'),
  126. name='request')
  127. # list of layers
  128. self.layersBox = StaticBox(parent=self.req_page_panel, id=wx.ID_ANY,
  129. label=_("List of layers "))
  130. style = wx.TR_DEFAULT_STYLE | wx.TR_HAS_BUTTONS | wx.TR_FULL_ROW_HIGHLIGHT
  131. if self.drv_props['req_multiple_layers']:
  132. style = style | wx.TR_MULTIPLE
  133. if 'WMS' not in self.ws:
  134. style = style | wx.TR_HIDE_ROOT
  135. self.list = LayersList(parent=self.req_page_panel,
  136. web_service=self.ws,
  137. style=style)
  138. self.params['format'] = None
  139. self.params['srs'] = None
  140. if 'srs' not in self.drv_props['ignored_params']:
  141. projText = StaticText(
  142. parent=self.req_page_panel, id=wx.ID_ANY,
  143. label=_("Source projection:"))
  144. self.params['srs'] = wx.Choice(
  145. parent=self.req_page_panel, id=wx.ID_ANY)
  146. self.list.Bind(wx.EVT_TREE_SEL_CHANGED, self.OnListSelChanged)
  147. # layout
  148. self.req_page_sizer = wx.BoxSizer(wx.VERTICAL)
  149. layersSizer = wx.StaticBoxSizer(self.layersBox, wx.HORIZONTAL)
  150. layersSizer.Add(
  151. self.list,
  152. proportion=1,
  153. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND,
  154. border=5)
  155. self.req_page_sizer.Add(
  156. layersSizer,
  157. proportion=1,
  158. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND,
  159. border=5)
  160. self.source_sizer = wx.BoxSizer(wx.HORIZONTAL)
  161. if self.params['format'] is not None:
  162. self.source_sizer.Add(
  163. self.params['format'],
  164. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM,
  165. border=5)
  166. if self.params['srs'] is not None:
  167. self.source_sizer.Add(
  168. projText,
  169. flag=wx.ALIGN_CENTER_VERTICAL | wx.ALL,
  170. border=5)
  171. self.source_sizer.Add(
  172. self.params['srs'],
  173. flag=wx.ALIGN_CENTER_VERTICAL | wx.RIGHT | wx.TOP | wx.BOTTOM,
  174. border=5)
  175. self.req_page_sizer.Add(
  176. self.source_sizer,
  177. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND,
  178. border=5)
  179. self.req_page_panel.SetSizer(self.req_page_sizer)
  180. def enableButtons(self, enable=True):
  181. """Enable/disable up, down, buttons
  182. """
  183. self.btnUp.Enable(enable)
  184. self.btnDown.Enable(enable)
  185. def _advancedSettsPage(self):
  186. """Create advanced settings page
  187. """
  188. # TODO parse maxcol, maxrow, settings from d.wms module?
  189. # TODO OnEarth driver - add selection of time
  190. adv_setts_panel = ScrolledPanel(parent=self, id=wx.ID_ANY,
  191. style=wx.TAB_TRAVERSAL | wx.SUNKEN_BORDER)
  192. self.notebook.AddPage(page=adv_setts_panel,
  193. text=_('Advanced request settings'),
  194. name='adv_req_setts')
  195. labels = {}
  196. self.l_odrder_list = None
  197. if 'WMS' in self.ws:
  198. labels['l_order'] = StaticBox(
  199. parent=adv_setts_panel, id=wx.ID_ANY,
  200. label=_("Order of layers in raster"))
  201. self.l_odrder_list = wx.ListBox(
  202. adv_setts_panel, id=wx.ID_ANY, choices=[],
  203. style=wx.LB_SINGLE | wx.LB_NEEDED_SB)
  204. self.btnUp = Button(
  205. adv_setts_panel, id=wx.ID_ANY, label=_("Up"))
  206. self.btnDown = Button(
  207. adv_setts_panel, id=wx.ID_ANY, label=_("Down"))
  208. self.btnUp.Bind(wx.EVT_BUTTON, self.OnUp)
  209. self.btnDown.Bind(wx.EVT_BUTTON, self.OnDown)
  210. labels['method'] = StaticText(parent=adv_setts_panel, id=wx.ID_ANY,
  211. label=_("Reprojection method:"))
  212. self.reproj_methods = ['nearest', 'linear', 'cubic', 'cubicspline']
  213. self.params['method'] = wx.Choice(
  214. parent=adv_setts_panel,
  215. id=wx.ID_ANY,
  216. choices=[
  217. _('Nearest neighbor'),
  218. _('Linear interpolation'),
  219. _('Cubic interpolation'),
  220. _('Cubic spline interpolation')])
  221. labels['maxcols'] = StaticText(
  222. parent=adv_setts_panel, id=wx.ID_ANY,
  223. label=_("Maximum columns to request from server at time:"))
  224. self.params['maxcols'] = SpinCtrl(
  225. parent=adv_setts_panel, id=wx.ID_ANY, size=(100, -1))
  226. labels['maxrows'] = StaticText(
  227. parent=adv_setts_panel, id=wx.ID_ANY,
  228. label=_("Maximum rows to request from server at time:"))
  229. self.params['maxrows'] = SpinCtrl(
  230. parent=adv_setts_panel, id=wx.ID_ANY, size=(100, -1))
  231. min = 100
  232. max = 10000
  233. self.params['maxcols'].SetRange(min, max)
  234. self.params['maxrows'].SetRange(min, max)
  235. val = 500
  236. self.params['maxcols'].SetValue(val)
  237. self.params['maxrows'].SetValue(val)
  238. self.flags['o'] = self.params['bgcolor'] = None
  239. if not 'o' in self.drv_props['ignored_flags']:
  240. self.flags['o'] = wx.CheckBox(
  241. parent=adv_setts_panel, id=wx.ID_ANY,
  242. label=_("Do not request transparent data"))
  243. self.flags['o'].Bind(wx.EVT_CHECKBOX, self.OnTransparent)
  244. labels['bgcolor'] = StaticText(
  245. parent=adv_setts_panel, id=wx.ID_ANY,
  246. label=_("Background color:"))
  247. self.params['bgcolor'] = csel.ColourSelect(
  248. parent=adv_setts_panel, id=wx.ID_ANY, colour=(
  249. 255, 255, 255), size=globalvar.DIALOG_COLOR_SIZE)
  250. self.params['bgcolor'].Enable(False)
  251. self.params['urlparams'] = None
  252. if self.params['urlparams'] not in self.drv_props['ignored_params']:
  253. labels['urlparams'] = StaticText(
  254. parent=adv_setts_panel, id=wx.ID_ANY,
  255. label=_("Additional query parameters for server:"))
  256. self.params['urlparams'] = TextCtrl(
  257. parent=adv_setts_panel, id=wx.ID_ANY)
  258. # layout
  259. border = wx.BoxSizer(wx.VERTICAL)
  260. if 'WMS' in self.ws:
  261. boxSizer = wx.StaticBoxSizer(labels['l_order'], wx.VERTICAL)
  262. gridSizer = wx.GridBagSizer(hgap=3, vgap=3)
  263. gridSizer.Add(self.l_odrder_list,
  264. pos=(0, 0),
  265. span=(4, 1),
  266. flag=wx.ALIGN_CENTER_VERTICAL | wx.EXPAND,
  267. border=0)
  268. gridSizer.Add(self.btnUp,
  269. pos=(0, 1),
  270. flag=wx.ALIGN_CENTER_VERTICAL,
  271. border=0)
  272. gridSizer.Add(self.btnDown,
  273. pos=(1, 1),
  274. flag=wx.ALIGN_CENTER_VERTICAL,
  275. border=0)
  276. gridSizer.AddGrowableCol(0)
  277. boxSizer.Add(gridSizer,
  278. flag=wx.EXPAND | wx.ALL,
  279. border=5)
  280. border.Add(boxSizer,
  281. flag=wx.LEFT | wx.RIGHT | wx.UP | wx.EXPAND,
  282. border=5)
  283. gridSizer = wx.GridBagSizer(hgap=3, vgap=3)
  284. row = 0
  285. for k in ['method', 'maxcols', 'maxrows', 'o', 'bgcolor']:
  286. if k in self.params:
  287. param = self.params[k]
  288. elif k in self.flags:
  289. param = self.flags[k]
  290. if param is None:
  291. continue
  292. if k in labels or k == 'o':
  293. if k != 'o':
  294. label = labels[k]
  295. else:
  296. label = param
  297. gridSizer.Add(label,
  298. flag=wx.ALIGN_LEFT |
  299. wx.ALIGN_CENTER_VERTICAL,
  300. pos=(row, 0))
  301. if k != 'o':
  302. gridSizer.Add(param,
  303. flag=wx.ALIGN_RIGHT |
  304. wx.ALIGN_CENTER_VERTICAL,
  305. pos=(row, 1))
  306. row += 1
  307. gridSizer.AddGrowableCol(0)
  308. border.Add(gridSizer,
  309. flag=wx.LEFT | wx.RIGHT | wx.TOP | wx.EXPAND,
  310. border=5)
  311. if self.params['urlparams']:
  312. gridSizer = wx.GridBagSizer(hgap=3, vgap=3)
  313. row = 0
  314. gridSizer.Add(labels['urlparams'],
  315. flag=wx.ALIGN_LEFT |
  316. wx.ALIGN_CENTER_VERTICAL,
  317. pos=(row, 0))
  318. gridSizer.Add(self.params['urlparams'],
  319. flag=wx.ALIGN_RIGHT |
  320. wx.ALIGN_CENTER_VERTICAL | wx.EXPAND,
  321. pos=(row, 1))
  322. gridSizer.AddGrowableCol(1)
  323. border.Add(gridSizer,
  324. flag=wx.LEFT | wx.RIGHT | wx.TOP | wx.EXPAND,
  325. border=5)
  326. adv_setts_panel.SetSizer(border)
  327. adv_setts_panel.SetAutoLayout(True)
  328. adv_setts_panel.SetupScrolling()
  329. def OnUp(self, event):
  330. """Move selected layer up
  331. """
  332. if self.l_odrder_list.GetSelections():
  333. pos = self.l_odrder_list.GetSelection()
  334. if pos:
  335. self.sel_layers.insert(pos - 1, self.sel_layers.pop(pos))
  336. if pos > 0:
  337. self._updateLayerOrderList(selected=(pos - 1))
  338. else:
  339. self._updateLayerOrderList(selected=0)
  340. def OnDown(self, event):
  341. """Move selected to down
  342. """
  343. if self.l_odrder_list.GetSelections():
  344. pos = self.l_odrder_list.GetSelection()
  345. if pos != len(self.sel_layers) - 1:
  346. self.sel_layers.insert(pos + 1, self.sel_layers.pop(pos))
  347. if pos < len(self.sel_layers) - 1:
  348. self._updateLayerOrderList(selected=(pos + 1))
  349. else:
  350. self._updateLayerOrderList(selected=len(self.sel_layers) - 1)
  351. def _updateLayerOrderList(self, selected=None):
  352. """Update order in list.
  353. """
  354. def getlayercaption(l):
  355. if l['title']:
  356. cap = (l['title'])
  357. else:
  358. cap = (l['name'])
  359. if l['style']:
  360. if l['style']['title']:
  361. cap += ' / ' + l['style']['title']
  362. else:
  363. cap += ' / ' + l['style']['name']
  364. return cap
  365. layer_capts = [getlayercaption(l) for l in self.sel_layers]
  366. self.l_odrder_list.Set(layer_capts)
  367. if self.l_odrder_list.IsEmpty():
  368. self.enableButtons(False)
  369. else:
  370. self.enableButtons(True)
  371. if selected is not None:
  372. self.l_odrder_list.SetSelection(selected)
  373. self.l_odrder_list.EnsureVisible(selected)
  374. def OnTransparent(self, event):
  375. checked = event.IsChecked()
  376. if checked:
  377. self.params['bgcolor'].Enable(True)
  378. else:
  379. self.params['bgcolor'].Enable(False)
  380. def ConnectToServer(self, url, username, password):
  381. """Download and parse data from capabilities file.
  382. :param url: server url
  383. :type url: str
  384. :param username: username for connection
  385. :type username: str
  386. :param password: password for connection
  387. :type password: str
  388. """
  389. self._prepareForNewConn(url, username, password)
  390. cap_cmd = [
  391. 'r.in.wms',
  392. '-c',
  393. ('capfile_output=%s' % self.cap_file),
  394. '--overwrite'] + self.ws_cmdl
  395. self.currentPid = self.cmd_thread.GetId()
  396. self.cmd_thread.RunCmd(cap_cmd, stderr=self.cmdStdErr)
  397. def OnCmdOutput(self, event):
  398. """Manage cmd output.
  399. """
  400. if Debug.GetLevel() != 0:
  401. Debug.msg(1, event.text)
  402. elif event.type != 'message' and event.type != 'warning':
  403. self.cmd_err_str += event.text + os.linesep
  404. def _prepareForNewConn(self, url, username, password):
  405. """Prepare panel for new connection
  406. """
  407. self.is_connected = False
  408. self.sel_layers = []
  409. self.formats_list = []
  410. self.projs_list = []
  411. self.conn = {
  412. 'url': url,
  413. 'password': password,
  414. 'username': username
  415. }
  416. conn_cmd = []
  417. for k, v in six.iteritems(self.conn):
  418. if v:
  419. conn_cmd.append("%s=%s" % (k, v))
  420. self.ws_cmdl = self.ws_drvs[self.ws]['cmd'] + conn_cmd
  421. def OnCapDownloadDone(self, event):
  422. """Process donwloaded capabilities file and emits capParsed
  423. signal (see class constructor).
  424. """
  425. if event.pid != self.currentPid:
  426. return
  427. if event.returncode != 0:
  428. if self.cmd_err_str:
  429. self.cmd_err_str = _(
  430. "Unable to download %s capabilities file\nfrom <%s>:\n" %
  431. (self.ws.replace('_', ' '),
  432. self.conn['url'])) + self.cmd_err_str
  433. self._postCapParsedEvt(error_msg=self.cmd_err_str)
  434. self.cmd_err_str = ''
  435. return
  436. self._parseCapFile(self.cap_file)
  437. def _parseCapFile(self, cap_file):
  438. """Parse capabilities data and emits capParsed signal
  439. (see class constructor).
  440. """
  441. try:
  442. self.cap = self.ws_drvs[self.ws]['cap_parser'](cap_file)
  443. except (IOError, ParseError) as error:
  444. error_msg = _(
  445. "%s web service was not found in fetched capabilities file from <%s>:\n%s\n" %
  446. (self.ws, self.conn['url'], str(error)))
  447. if Debug.GetLevel() != 0:
  448. Debug.msg(1, error_msg)
  449. self._postCapParsedEvt(None)
  450. else:
  451. self._postCapParsedEvt(error_msg=error_msg)
  452. return
  453. self.is_connected = True
  454. # WMS standard has formats defined for all layers
  455. if 'WMS' in self.ws:
  456. self.formats_list = sorted(self._getFormats())
  457. self._updateFormatRadioBox(self.formats_list)
  458. self._setDefaultFormatVal()
  459. self.list.LoadData(self.cap)
  460. self.OnListSelChanged(event=None)
  461. self._postCapParsedEvt(None)
  462. def ParseCapFile(self, url, username, password, cap_file=None,):
  463. """Parse capabilities data and emits capParsed signal
  464. (see class constructor).
  465. """
  466. self._prepareForNewConn(url, username, password)
  467. if cap_file is None or not url:
  468. self._postCapParsedEvt(None)
  469. return
  470. shutil.copyfile(cap_file, self.cap_file)
  471. self._parseCapFile(self.cap_file)
  472. def UpdateWidgetsByCmd(self, cmd):
  473. """Update panel widgets accordnig to passed cmd tuple
  474. :param cmd: cmd in tuple
  475. """
  476. dcmd = cmd[1]
  477. layers = []
  478. if 'layers' in dcmd:
  479. layers = dcmd['layers']
  480. styles = []
  481. if 'styles' in dcmd:
  482. styles = dcmd['styles']
  483. if 'WMS' in self.ws:
  484. layers = layers.split(',')
  485. styles = styles.split(',')
  486. else:
  487. layers = [layers]
  488. styles = [styles]
  489. if len(layers) != len(styles):
  490. styles = [''] * len(layers)
  491. l_st_list = []
  492. for i in range(len(layers)):
  493. l_st_list.append({'style': styles[i],
  494. 'layer': layers[i]})
  495. # WMS standard - first layer in params is most bottom...
  496. # therefore layers order need to be reversed
  497. l_st_list = [l for l in reversed(l_st_list)]
  498. self.list.SelectLayers(l_st_list)
  499. params = {}
  500. if 'format' in dcmd:
  501. params['format'] = dcmd['format']
  502. if 'srs' in dcmd:
  503. params['srs'] = 'EPSG:' + dcmd['srs']
  504. if 'method' in dcmd:
  505. params['method'] = dcmd['method']
  506. for p, v in six.iteritems(params):
  507. if self.params[p]:
  508. self.params[p].SetStringSelection(v)
  509. for p, conv_f in [
  510. ('urlparams', None),
  511. ('maxcols', int),
  512. ('maxrows', int)]:
  513. if p in dcmd:
  514. v = dcmd[p]
  515. if conv_f:
  516. v = conv_f(v)
  517. self.params[p].SetValue(v)
  518. if 'flags' in dcmd and \
  519. 'o' in dcmd['flags']:
  520. self.flags['o'].SetValue(1)
  521. self.params['bgcolor'].Enable(True)
  522. if 'bgcolor' in dcmd and \
  523. self.params['bgcolor']:
  524. bgcolor = dcmd['bgcolor'].strip().lower()
  525. if len(bgcolor) == 8 and \
  526. '0x' == bgcolor[:2]:
  527. colour = '#' + bgcolor[2:]
  528. self.params['bgcolor'].SetColour(colour)
  529. def IsConnected(self):
  530. """Was successful in downloading and parsing capabilities data?
  531. """
  532. return self.is_connected
  533. def _postCapParsedEvt(self, error_msg):
  534. """Helper function
  535. """
  536. self.capParsed.emit(error_msg=error_msg)
  537. def CreateCmd(self):
  538. """Create d.wms cmd from values of panels widgets
  539. :return: cmd list
  540. :return: None if required widgets do not have selected/filled values.
  541. """
  542. # check required widgets
  543. if not self._checkImportValues():
  544. return None
  545. # create d.wms command
  546. lcmd = self.ws_cmdl
  547. lcmd = ['d.wms'] + lcmd
  548. layers = "layers="
  549. styles = 'styles='
  550. first = True
  551. # WMS standard - first layer in params is most bottom...
  552. # therefore layers order need to be reversed
  553. for layer in reversed(self.sel_layers):
  554. if not first:
  555. layers += ','
  556. styles += ','
  557. first = False
  558. layers += layer['name']
  559. if layer['style'] is not None:
  560. styles += layer['style']['name']
  561. lcmd.append(layers)
  562. lcmd.append(styles)
  563. if 'format' not in self.drv_props['ignored_params']:
  564. i_format = self.params['format'].GetSelection()
  565. lcmd.append("format=%s" % self.formats_list[i_format])
  566. if 'srs' not in self.drv_props['ignored_params']:
  567. i_srs = self.params['srs'].GetSelection()
  568. srs = self.projs_list[i_srs].split(':')[-1]
  569. epsg_num = int(''.join(re.findall(r'\d+', srs)))
  570. lcmd.append("srs=%s" % epsg_num)
  571. for k in ['maxcols', 'maxrows', 'urlparams']:
  572. lcmd.append(k + '=' + str(self.params[k].GetValue()))
  573. i_method = self.params['method'].GetSelection()
  574. lcmd.append('method=' + self.reproj_methods[i_method])
  575. if not 'o' in self.drv_props['ignored_flags'] and \
  576. self.flags['o'].IsChecked():
  577. lcmd.append('-o')
  578. c = self.params['bgcolor'].GetColour()
  579. hex_color = wx.Colour(
  580. c[0],
  581. c[1],
  582. c[2]).GetAsString(
  583. wx.C2S_HTML_SYNTAX)
  584. lcmd.append("bgcolor=" + '0x' + hex_color[1:])
  585. lcmd.append("map=" + self.o_layer_name)
  586. return lcmd
  587. def OnListSelChanged(self, event):
  588. """Update widgets according to selected layer in list.
  589. """
  590. curr_sel_ls = self.list.GetSelectedLayers()
  591. # update self.sel_layers (selected layer list)
  592. if 'WMS' in self.ws:
  593. for sel_l in self.sel_layers[:]:
  594. if sel_l not in curr_sel_ls:
  595. self.sel_layers.remove(sel_l)
  596. for l in curr_sel_ls:
  597. if l not in self.sel_layers:
  598. self.sel_layers.append(l)
  599. self._updateLayerOrderList()
  600. else:
  601. self.sel_layers = curr_sel_ls
  602. # update projection
  603. self.projs_list = []
  604. projs_list = []
  605. intersect_proj = []
  606. first = True
  607. for l in curr_sel_ls:
  608. layer_projs = l['cap_intf_l'].GetLayerData('srs')
  609. if first:
  610. projs_list = layer_projs
  611. first = False
  612. continue
  613. projs_list = set(projs_list).intersection(layer_projs)
  614. if 'srs' not in self.drv_props['ignored_params']:
  615. for proj in projs_list:
  616. proj_code = Srs(proj.strip()).getcode()
  617. proj_spl = proj_code.split(':')
  618. if proj_spl[0].strip().lower() in self.drv_info.GetSrs():
  619. # accept ogc:crs code
  620. self.projs_list.append(proj_code)
  621. cur_sel = self.params['srs'].GetStringSelection()
  622. self.projs_list = sorted(self.projs_list)
  623. self.params['srs'].SetItems(self.projs_list)
  624. if cur_sel:
  625. self.params['srs'].SetStringSelection(cur_sel)
  626. else:
  627. try:
  628. i = self.projs_list.index('EPSG:4326')
  629. self.params['srs'].SetSelection(i)
  630. except ValueError:
  631. if len(self.projs_list) > 0:
  632. self.params['srs'].SetSelection(0)
  633. # update format
  634. if 'WMS' not in self.ws and \
  635. 'format' not in self.drv_props['ignored_params']:
  636. self.formats_list = []
  637. cur_sel = None
  638. if self.params['format']:
  639. cur_sel = self.params['format'].GetStringSelection()
  640. if len(curr_sel_ls) > 0:
  641. self.formats_list = sorted(
  642. self._getFormats(
  643. curr_sel_ls[0]['cap_intf_l']))
  644. self._updateFormatRadioBox(self.formats_list)
  645. if cur_sel:
  646. if self.params['format']:
  647. self.params['format'].SetStringSelection(cur_sel)
  648. else:
  649. self._setDefaultFormatVal()
  650. self.Layout()
  651. def _setDefaultFormatVal(self):
  652. """Set default format value.
  653. """
  654. try:
  655. i = self.formats_list.index('png')
  656. self.params['format'].SetSelection(i)
  657. except ValueError:
  658. pass
  659. def _updateFormatRadioBox(self, formats_list):
  660. """Helper function
  661. """
  662. if self.params['format']:
  663. self.req_page_sizer.Detach(self.params['format'])
  664. self.params['format'].Destroy()
  665. if len(self.formats_list) > 0:
  666. self.params['format'] = wx.RadioBox(
  667. parent=self.req_page_panel,
  668. id=wx.ID_ANY,
  669. label=_("Source image format"),
  670. pos=wx.DefaultPosition,
  671. choices=formats_list,
  672. majorDimension=4,
  673. style=wx.RA_SPECIFY_COLS)
  674. self.source_sizer.Insert(2, window=self.params['format'],
  675. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM,
  676. border=5)
  677. def _getFormats(self, layer=None):
  678. """Get formats
  679. WMS has formats defined generally for whole cap.
  680. In WMTS and NASA OnEarh formats are defined for layer.
  681. """
  682. formats_label = []
  683. if layer is None:
  684. formats_list = self.cap.GetFormats()
  685. else:
  686. formats_list = layer.GetLayerData('format')
  687. for frmt in formats_list:
  688. frmt = frmt.strip()
  689. label = self.drv_info.GetFormatLabel(frmt)
  690. if label:
  691. formats_label.append(label)
  692. return formats_label
  693. def _checkImportValues(self,):
  694. """Check if required widgets are selected/filled
  695. """
  696. warning_str = ""
  697. show_war = False
  698. if not self.list or not self.list.GetSelectedLayers():
  699. warning_str += _("Select layer in layer list.\n")
  700. show_war = True
  701. if self.params['format'] is not None and \
  702. self.params['format'].GetSelection() == -1:
  703. warning_str += _("Select source image format.\n")
  704. show_war = True
  705. if self.params['srs'] is not None and \
  706. self.params['srs'].GetSelection() == -1:
  707. warning_str += _("Select source projection.\n")
  708. show_war = True
  709. if not self.o_layer_name:
  710. warning_str += _("Choose output layer name.\n")
  711. show_war = True
  712. if show_war:
  713. GMessage(parent=self.parent, message=warning_str)
  714. return False
  715. return True
  716. def SetOutputLayerName(self, name):
  717. """Set name of layer to be added to layer tree
  718. """
  719. self.o_layer_name = name
  720. def GetOutputLayerName(self):
  721. return self.o_layer_name
  722. def GetCapFile(self):
  723. """Get path to file where capabilities are saved
  724. """
  725. return self.cap_file
  726. def GetWebService(self):
  727. """Get web service
  728. """
  729. return self.ws
  730. class LayersList(TreeCtrl):
  731. def __init__(self, parent, web_service, style, pos=wx.DefaultPosition):
  732. """List of layers and styles available in capabilities file
  733. """
  734. self.parent = parent
  735. self.ws = web_service
  736. TreeCtrl.__init__(self, parent=parent, id=wx.ID_ANY, style=style)
  737. self.root = None
  738. self.Bind(wx.EVT_TREE_SEL_CHANGING, self.OnListSelChanging)
  739. self.layerSelected = Signal('LayersList.layerSelected')
  740. def LoadData(self, cap=None):
  741. """Load data into list
  742. """
  743. # detete first all items
  744. self.DeleteAllItems()
  745. if not cap:
  746. return
  747. def AddLayerChildrenToTree(parent_layer, parent_item):
  748. """Recursive function which adds all capabilities
  749. layers/styles to the LayersList.
  750. """
  751. def gettitle(layer):
  752. """Helper function"""
  753. if layer.GetLayerData('title') is not None:
  754. layer_title = layer.GetLayerData('title')
  755. elif layer.GetLayerData('name') is not None:
  756. layer_title = layer.GetLayerData('name')
  757. else:
  758. layer_title = str(layer.GetId())
  759. return layer_title
  760. def addlayer(layer, item):
  761. styles = layer.GetLayerData('styles')
  762. def_st = None
  763. for st in styles:
  764. if st['name']:
  765. style_name = st['name']
  766. else:
  767. continue
  768. if st['title']:
  769. style_name = st['title']
  770. if st['isDefault']:
  771. def_st = st
  772. style_item = self.AppendItem(item, style_name)
  773. self.SetItemData(style_item, {'type': 'style',
  774. 'layer': layer, # it is parent layer of style
  775. 'style': st})
  776. self.SetItemData(item, {'type': 'layer', # is it layer or style?
  777. 'layer': layer, # Layer instance from web_services.cap_interface
  778. 'style': def_st}) # layer can have assigned default style
  779. if parent_layer is None:
  780. parent_layer = cap.GetRootLayer()
  781. layer_title = gettitle(parent_layer)
  782. parent_item = self.AddRoot(layer_title)
  783. addlayer(parent_layer, parent_item)
  784. for layer in parent_layer.GetChildren():
  785. item = self.AppendItem(parent_item, gettitle(layer))
  786. addlayer(layer, item)
  787. AddLayerChildrenToTree(layer, item)
  788. AddLayerChildrenToTree(None, None)
  789. # self.ExpandAll(self.GetRootItem())
  790. def GetSelectedLayers(self):
  791. """Get selected layers/styles in LayersList
  792. :return: dict with these items:
  793. * 'name' : layer name used for request
  794. if it is style, it is name of parent layer
  795. * 'title' : layer title
  796. * 'style' : {'name' : 'style name', title : 'style title'}
  797. * 'cap_intf_l' : \*Layer instance from web_services.cap_interface
  798. """
  799. sel_layers = self.GetSelections()
  800. sel_layers_dict = []
  801. for s in sel_layers:
  802. try:
  803. layer = self.GetItemData(s)['layer']
  804. except ValueError:
  805. continue
  806. sel_layers_dict.append({
  807. 'name': layer.GetLayerData('name'),
  808. 'title': layer.GetLayerData('title'),
  809. 'style': self.GetItemData(s)['style'],
  810. 'cap_intf_l': layer
  811. })
  812. return sel_layers_dict
  813. def OnListSelChanging(self, event):
  814. """Do not allow selecting items, which cannot be requested from server.
  815. """
  816. def _emitSelected(layer):
  817. title = layer.GetLayerData('title')
  818. self.layerSelected.emit(title=title)
  819. def _selectRequestableChildren(item, list_to_check, items_to_sel):
  820. self.Expand(item)
  821. child_item, cookie = self.GetFirstChild(item)
  822. while child_item and child_item.IsOk():
  823. if self.GetItemData(child_item)['layer'].IsRequestable() \
  824. and not self.IsSelected(child_item):
  825. items_to_sel.append(child_item)
  826. elif not self.GetItemData(child_item)['layer'].IsRequestable():
  827. list_to_check.append(child_item)
  828. child_item, cookie = self.GetNextChild(item, cookie)
  829. cur_item = event.GetItem()
  830. if not self.GetItemData(cur_item)['layer'].IsRequestable():
  831. event.Veto()
  832. if not self.HasFlag(wx.TR_MULTIPLE):
  833. return
  834. _emitSelected(self.GetItemData(cur_item)['layer'])
  835. items_to_chck = []
  836. items_to_sel = []
  837. chck_item = cur_item
  838. while True:
  839. _selectRequestableChildren(
  840. chck_item, items_to_chck, items_to_sel)
  841. if items_to_chck:
  842. chck_item = items_to_chck.pop()
  843. else:
  844. break
  845. while items_to_sel:
  846. self.SelectItem(items_to_sel.pop(), select=True)
  847. else:
  848. _emitSelected(self.GetItemData(cur_item)['layer'])
  849. def GetItemCount(self):
  850. """Required for listmix.ListCtrlAutoWidthMixin
  851. """
  852. return 0
  853. def GetCountPerPage(self):
  854. """Required for listmix.ListCtrlAutoWidthMixin
  855. """
  856. return 0
  857. def SelectLayers(self, l_st_list):
  858. """Select layers/styles in LayersList
  859. :param l_st_list: [{style : 'style_name', layer : 'layer_name'}, ...]
  860. :return: items from l_st_list which were not found
  861. """
  862. def checknext(root_item, l_st_list, items_to_sel):
  863. def compare(item, l_name, st_name):
  864. it_l_name = self.GetItemData(item)['layer'].GetLayerData('name')
  865. it_st = self.GetItemData(item)['style']
  866. it_type = self.GetItemData(item)['type']
  867. if it_l_name == l_name and ((not it_st and not st_name) or (
  868. it_st and it_st['name'] == st_name and it_type == 'style')):
  869. return True
  870. return False
  871. (child, cookie) = self.GetFirstChild(root_item)
  872. while child.IsOk():
  873. for i, l_st in enumerate(l_st_list):
  874. l_name = l_st['layer']
  875. st_name = l_st['style']
  876. if compare(child, l_name, st_name):
  877. items_to_sel[i] = [child, l_st]
  878. break
  879. if len(items_to_sel) == len(l_st_list):
  880. if self.ItemHasChildren(child):
  881. checknext(child, l_st_list, items_to_sel)
  882. child = self.GetNextSibling(child)
  883. self.UnselectAll()
  884. l_st_list = deepcopy(l_st_list)
  885. root_item = self.GetRootItem()
  886. items_to_sel = [None] * len(l_st_list)
  887. checknext(root_item, l_st_list, items_to_sel)
  888. self.CollapseAll()
  889. # items are selected according to position in l_st_list
  890. # to be added to Layers order list in right order
  891. for i in items_to_sel:
  892. if not i:
  893. continue
  894. item, l_st = i
  895. keep = False
  896. if self.HasFlag(wx.TR_MULTIPLE):
  897. keep = True
  898. self.SelectItem(item, select=keep)
  899. self.SetFocusedItem(item)
  900. self.Expand(item)
  901. l_st_list.remove(l_st)
  902. return l_st_list
  903. class WSManageSettingsWidget(ManageSettingsWidget):
  904. def __init__(self, parent, settingsFile, default_servers):
  905. ManageSettingsWidget.__init__(self, parent, settingsFile)
  906. self.default_servers = default_servers
  907. def _layout(self):
  908. self.btnAddDefaultServers = Button(parent=self, id=wx.ID_ANY,
  909. label=_("Add default"))
  910. self.btnAddDefaultServers.Bind(wx.EVT_BUTTON, self.OnAddDefaultServers)
  911. ManageSettingsWidget._layout(self)
  912. self.settingsSizer.Add(self.btnAddDefaultServers,
  913. flag=wx.RIGHT,
  914. border=5)
  915. def OnAddDefaultServers(self, event):
  916. setts = self.GetSettings()
  917. self.servers_to_add = {}
  918. for k, v in six.iteritems(self.default_servers):
  919. if k not in six.iterkeys(setts):
  920. self.servers_to_add[k] = v
  921. elif v != setts[k]:
  922. GMessage(parent=self,
  923. message=_("User defined server with same name "
  924. "as default server <%s> already exists.\n"
  925. "Keeping user defined server") % (k))
  926. if self.servers_to_add:
  927. self.AddSettings(self.servers_to_add)