Coverage for gws-app/gws/plugin/ows_client/wmts/layer.py: 0%
107 statements
« prev ^ index » next coverage.py v7.11.0, created at 2025-10-16 23:09 +0200
« prev ^ index » next coverage.py v7.11.0, created at 2025-10-16 23:09 +0200
1from typing import Optional
3import gws
4import gws.base.layer
5import gws.config.util
6import gws.lib.bounds
7import gws.lib.crs
8import gws.gis.source
9import gws.gis.zoom
11from . import provider
13gws.ext.new.layer('wmts')
16class Config(gws.base.layer.Config):
17 """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 if not self.sourceLayers:
49 raise gws.Error('no source layers found')
50 self.activeLayer = self.sourceLayers[0]
51 self.configure_tms()
52 self.configure_style()
54 def configure_source_layers(self):
55 return gws.config.util.configure_source_layers_for(self, self.serviceProvider.sourceLayers, is_image=True)
57 def configure_tms(self):
58 crs = self.serviceProvider.forceCrs
59 if not crs:
60 crs = gws.lib.crs.best_match(self.mapCrs, [tms.crs for tms in self.activeLayer.tileMatrixSets])
61 tms_list = [tms for tms in self.activeLayer.tileMatrixSets if tms.crs == crs]
62 if not tms_list:
63 raise gws.Error(f'no TMS for {crs} in {self.serviceProvider.url}')
64 self.activeTms = tms_list[0]
66 def configure_style(self):
67 p = self.cfg('styleName')
68 if p:
69 for style in self.activeLayer.styles:
70 if style.name == p:
71 self.activeStyle = style
72 return True
73 raise gws.Error(f'style {p!r} not found')
75 for style in self.activeLayer.styles:
76 if style.isDefault:
77 self.activeStyle = style
78 return True
80 self.activeStyle = gws.SourceStyle(name='default')
81 return True
83 #
84 # reprojecting the world doesn't make sense, just use the map extent here
85 # @TODO maybe look for more sensible grid alignment
86 #
87 # def configure_bounds(self):
88 # if super().configure_bounds():
89 # return True
90 # src_bounds = gws.Bounds(crs=self.activeTms.crs, extent=self.activeTms.matrices[0].extent)
91 # self.bounds = gws.lib.bounds.transform(src_bounds, self.mapCrs)
92 # return True
94 def configure_resolutions(self):
95 if super().configure_resolutions():
96 return True
97 res = [gws.lib.uom.scale_to_res(m.scale) for m in self.activeTms.matrices]
98 self.resolutions = sorted(res, reverse=True)
99 return True
101 def configure_grid(self):
102 p = self.cfg('grid', default=gws.Config())
103 self.grid = gws.TileGrid(
104 origin=p.origin or gws.Origin.nw,
105 tileSize=p.tileSize or self.activeTms.matrices[0].tileWidth,
106 )
107 if p.extent:
108 self.grid.bounds = gws.Bounds(crs=self.mapCrs, extent=p.extent)
109 elif self.activeTms.crs == self.mapCrs:
110 self.grid.bounds = gws.Bounds(crs=self.mapCrs, extent=self.activeTms.matrices[0].extent)
111 else:
112 self.grid.bounds = self.bounds
114 if p.resolutions:
115 self.grid.resolutions = p.resolutions
116 else:
117 self.grid.resolutions = gws.gis.zoom.resolutions_from_bounds(self.grid.bounds, self.grid.tileSize)
119 def configure_legend(self):
120 if super().configure_legend():
121 return True
122 url = self.activeStyle.legendUrl
123 if url:
124 self.legend = self.create_child(gws.ext.object.legend, type='remote', urls=[url])
125 return True
127 def configure_metadata(self):
128 if super().configure_metadata():
129 return True
130 self.metadata = self.serviceProvider.metadata
131 return True
133 def mapproxy_config(self, mc):
134 url = self.serviceProvider.tile_url_template(self.activeLayer, self.activeTms, self.activeStyle)
136 # mapproxy encoding
138 url = url.replace('{TileMatrix}', '%(z)02d')
139 url = url.replace('{TileCol}', '%(x)d')
140 url = url.replace('{TileRow}', '%(y)d')
142 source_grid = self.serviceProvider.grid_for_tms(self.activeTms)
144 if source_grid.origin == gws.Origin.nw:
145 origin = 'nw'
146 elif source_grid.origin == gws.Origin.sw:
147 origin = 'sw'
148 else:
149 raise gws.Error(f'invalid grid origin {source_grid.origin!r}')
151 source_grid_uid = mc.grid(
152 gws.u.compact(
153 {
154 'origin': origin,
155 'srs': source_grid.bounds.crs.epsg,
156 'bbox': source_grid.bounds.extent,
157 'res': source_grid.resolutions,
158 'tile_size': [source_grid.tileSize, source_grid.tileSize],
159 }
160 )
161 )
163 src_uid = gws.base.layer.util.mapproxy_back_cache_config(self, mc, url, source_grid_uid)
164 gws.base.layer.util.mapproxy_layer_config(self, mc, src_uid)
166 ##
168 def render(self, lri):
169 return gws.base.layer.util.mpx_raster_render(self, lri)