Coverage for gws-app/gws/plugin/ows_client/wmts/layer.py: 0%
106 statements
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-17 01:37 +0200
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-17 01:37 +0200
1from typing import Optional
3import gws
4import gws.base.layer
5import gws.config.util
6import gws.gis.bounds
7import gws.gis.crs
8import gws.gis.source
9import gws.gis.zoom
10import gws.lib.uom as units
12from . import provider
14gws.ext.new.layer('wmts')
17class Config(gws.base.layer.Config):
18 """WMTS layer"""
19 provider: provider.Config
20 """WMTS provider"""
21 display: gws.LayerDisplayMode = gws.LayerDisplayMode.tile
22 """layer display mode"""
23 sourceLayers: Optional[gws.gis.source.LayerFilter]
24 """source layer filter"""
25 style: Optional[str]
26 """WMTS style name"""
29class Object(gws.base.layer.image.Object):
30 serviceProvider: provider.Object
31 sourceLayers: list[gws.SourceLayer]
33 activeLayer: gws.SourceLayer
34 activeStyle: gws.SourceStyle
35 activeTms: gws.TileMatrixSet
37 def configure(self):
38 self.configure_layer()
40 def configure_provider(self):
41 return gws.config.util.configure_service_provider_for(self, provider.Object)
43 def configure_sources(self):
44 if super().configure_sources():
45 return True
47 self.configure_source_layers()
48 self.activeLayer = self.sourceLayers[0]
49 self.configure_tms()
50 self.configure_style()
52 def configure_source_layers(self):
53 return gws.config.util.configure_source_layers_for(self, self.serviceProvider.sourceLayers, is_image=True)
55 def configure_tms(self):
56 crs = self.serviceProvider.forceCrs
57 if not crs:
58 crs = gws.gis.crs.best_match(self.mapCrs, [tms.crs for tms in self.activeLayer.tileMatrixSets])
59 tms_list = [tms for tms in self.activeLayer.tileMatrixSets if tms.crs == crs]
60 if not tms_list:
61 raise gws.Error(f'no TMS for {crs} in {self.serviceProvider.url}')
62 self.activeTms = tms_list[0]
64 def configure_style(self):
65 p = self.cfg('styleName')
66 if p:
67 for style in self.activeLayer.styles:
68 if style.name == p:
69 self.activeStyle = style
70 return True
71 raise gws.Error(f'style {p!r} not found')
73 for style in self.activeLayer.styles:
74 if style.isDefault:
75 self.activeStyle = style
76 return True
78 self.activeStyle = gws.SourceStyle(name='default')
79 return True
81 #
82 # reprojecting the world doesn't make sense, just use the map extent here
83 # @TODO maybe look for more sensible grid alignment
84 #
85 # def configure_bounds(self):
86 # if super().configure_bounds():
87 # return True
88 # src_bounds = gws.Bounds(crs=self.activeTms.crs, extent=self.activeTms.matrices[0].extent)
89 # self.bounds = gws.gis.bounds.transform(src_bounds, self.mapCrs)
90 # return True
92 def configure_resolutions(self):
93 if super().configure_resolutions():
94 return True
95 res = [gws.lib.uom.scale_to_res(m.scale) for m in self.activeTms.matrices]
96 self.resolutions = sorted(res, reverse=True)
97 return True
99 def configure_grid(self):
100 p = self.cfg('grid', default=gws.Config())
101 self.grid = gws.TileGrid(
102 origin=p.origin or gws.Origin.nw,
103 tileSize=p.tileSize or self.activeTms.matrices[0].tileWidth,
104 )
105 if p.extent:
106 self.grid.bounds = gws.Bounds(crs=self.mapCrs, extent=p.extent)
107 elif self.activeTms.crs == self.mapCrs:
108 self.grid.bounds = gws.Bounds(crs=self.mapCrs, extent=self.activeTms.matrices[0].extent)
109 else:
110 self.grid.bounds = self.bounds
112 if p.resolutions:
113 self.grid.resolutions = p.resolutions
114 else:
115 self.grid.resolutions = gws.gis.zoom.resolutions_from_bounds(self.grid.bounds, self.grid.tileSize)
117 def configure_legend(self):
118 if super().configure_legend():
119 return True
120 url = self.activeStyle.legendUrl
121 if url:
122 self.legend = self.create_child(gws.ext.object.legend, type='remote', urls=[url])
123 return True
125 def configure_metadata(self):
126 if super().configure_metadata():
127 return True
128 self.metadata = self.serviceProvider.metadata
129 return True
131 def mapproxy_config(self, mc):
132 url = self.serviceProvider.tile_url_template(self.activeLayer, self.activeTms, self.activeStyle)
134 # mapproxy encoding
136 url = url.replace('{TileMatrix}', '%(z)02d')
137 url = url.replace('{TileCol}', '%(x)d')
138 url = url.replace('{TileRow}', '%(y)d')
140 source_grid = self.serviceProvider.grid_for_tms(self.activeTms)
142 if source_grid.origin == gws.Origin.nw:
143 origin = 'nw'
144 elif source_grid.origin == gws.Origin.sw:
145 origin = 'sw'
146 else:
147 raise gws.Error(f'invalid grid origin {source_grid.origin!r}')
149 source_grid_uid = mc.grid(gws.u.compact({
150 'origin': origin,
151 'srs': source_grid.bounds.crs.epsg,
152 'bbox': source_grid.bounds.extent,
153 'res': source_grid.resolutions,
154 'tile_size': [source_grid.tileSize, source_grid.tileSize],
155 }))
157 src_uid = gws.base.layer.util.mapproxy_back_cache_config(self, mc, url, source_grid_uid)
158 gws.base.layer.util.mapproxy_layer_config(self, mc, src_uid)
160 ##
162 def render(self, lri):
163 return gws.base.layer.util.mpx_raster_render(self, lri)