widgets.py 38 KB

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