XAML-Map-Control/MapControl/Shared/WmtsTileLayer.cs

221 lines
7.6 KiB
C#
Raw Normal View History

2025-03-31 21:33:52 +02:00
using Microsoft.Extensions.Logging;
using System;
2020-03-20 18:12:56 +01:00
using System.Collections.Generic;
using System.Linq;
2024-05-22 11:25:32 +02:00
#if WPF
using System.Windows;
2021-11-17 23:17:11 +01:00
#elif UWP
2020-03-20 18:12:56 +01:00
using Windows.Foundation;
using Windows.UI.Xaml;
2024-05-22 11:25:32 +02:00
#elif WINUI
using Windows.Foundation;
using Microsoft.UI.Xaml;
2025-08-19 19:43:02 +02:00
#elif AVALONIA
using Avalonia;
using Avalonia.Interactivity;
2020-03-20 18:12:56 +01:00
#endif
namespace MapControl
{
/// <summary>
/// Displays map tiles from a Web Map Tile Service (WMTS).
/// </summary>
2025-09-21 10:57:11 +02:00
public partial class WmtsTileLayer : MapTilePyramidLayer
2020-03-20 18:12:56 +01:00
{
2025-08-22 11:06:37 +02:00
private static ILogger logger;
2025-09-19 18:49:12 +02:00
private static ILogger Logger => logger ??= ImageLoader.LoggerFactory?.CreateLogger(typeof(WmtsTileLayer));
2025-08-22 11:06:37 +02:00
2024-05-22 11:47:57 +02:00
public static readonly DependencyProperty CapabilitiesUriProperty =
2024-05-23 18:22:52 +02:00
DependencyPropertyHelper.Register<WmtsTileLayer, Uri>(nameof(CapabilitiesUri), null,
2024-05-22 11:47:57 +02:00
(layer, oldValue, newValue) => layer.TileMatrixSets.Clear());
2020-03-20 18:12:56 +01:00
2024-05-22 11:47:57 +02:00
public static readonly DependencyProperty LayerProperty =
DependencyPropertyHelper.Register<WmtsTileLayer, string>(nameof(Layer));
2024-05-22 11:47:57 +02:00
public static readonly DependencyProperty PreferredTileMatrixSetsProperty =
DependencyPropertyHelper.Register<WmtsTileLayer, string[]>(nameof(PreferredTileMatrixSets));
2020-03-20 18:12:56 +01:00
public WmtsTileLayer()
{
Loaded += OnLoaded;
}
/// <summary>
/// The Uri of a XML file or web response that contains the service capabilities.
/// </summary>
2020-03-20 20:40:13 +01:00
public Uri CapabilitiesUri
2020-03-20 18:12:56 +01:00
{
2022-08-06 10:40:59 +02:00
get => (Uri)GetValue(CapabilitiesUriProperty);
set => SetValue(CapabilitiesUriProperty, value);
2020-03-20 18:12:56 +01:00
}
/// <summary>
/// The Identifier of the Layer that should be displayed. If not set, the first Layer is displayed.
/// </summary>
public string Layer
2020-03-20 18:12:56 +01:00
{
get => (string)GetValue(LayerProperty);
set => SetValue(LayerProperty, value);
}
/// <summary>
/// In case there are TileMatrixSets with identical SupportedCRS values,
/// the ones with Identifiers contained in this collection take precedence.
/// </summary>
public string[] PreferredTileMatrixSets
{
get => (string[])GetValue(PreferredTileMatrixSetsProperty);
set => SetValue(PreferredTileMatrixSetsProperty, value);
2020-03-20 18:12:56 +01:00
}
2025-09-19 18:49:12 +02:00
/// <summary>
/// Gets a dictionary of all tile matrix sets supported by a WMTS, with their CRS as dictionary key.
/// </summary>
2025-09-15 17:46:31 +02:00
public Dictionary<string, WmtsTileMatrixSet> TileMatrixSets { get; } = [];
2020-03-20 18:12:56 +01:00
2025-09-19 18:49:12 +02:00
/// <summary>
/// Gets a collection of all CRSs supported by a WMTS.
/// </summary>
2025-09-19 20:22:10 +02:00
public override IReadOnlyCollection<string> SupportedCrsIds => TileMatrixSets.Keys;
2025-09-19 18:49:12 +02:00
protected IEnumerable<WmtsTileMatrixLayer> ChildLayers => Children.Cast<WmtsTileMatrixLayer>();
2020-03-20 18:12:56 +01:00
protected override Size MeasureOverride(Size availableSize)
{
2020-03-23 17:13:50 +01:00
foreach (var layer in ChildLayers)
2020-03-20 18:12:56 +01:00
{
layer.Measure(availableSize);
}
return new Size();
}
protected override Size ArrangeOverride(Size finalSize)
{
2020-03-23 17:13:50 +01:00
foreach (var layer in ChildLayers)
2020-03-20 18:12:56 +01:00
{
layer.Arrange(new Rect(0, 0, finalSize.Width, finalSize.Height));
}
return finalSize;
}
2025-09-21 12:58:16 +02:00
protected override void UpdateTiles(bool resetTiles)
2020-03-20 18:12:56 +01:00
{
2025-08-18 22:23:19 +02:00
// resetTiles is ignored here because it is always false.
2023-08-12 17:36:37 +02:00
2020-03-20 18:12:56 +01:00
if (ParentMap == null ||
2020-04-16 19:24:38 +02:00
!TileMatrixSets.TryGetValue(ParentMap.MapProjection.CrsId, out WmtsTileMatrixSet tileMatrixSet))
2020-03-20 18:12:56 +01:00
{
Children.Clear();
2021-07-05 00:03:44 +02:00
2025-08-19 23:21:51 +02:00
CancelLoadTiles();
2020-03-20 18:12:56 +01:00
}
2024-08-31 11:55:59 +02:00
else if (UpdateChildLayers(tileMatrixSet))
2022-11-25 19:05:48 +01:00
{
2025-01-17 08:25:26 +01:00
((WmtsTileSource)TileSource).TileMatrixSet = tileMatrixSet;
2025-01-17 08:25:10 +01:00
var cacheName = SourceName;
2025-01-17 08:25:26 +01:00
if (!string.IsNullOrEmpty(cacheName))
2025-01-17 08:25:10 +01:00
{
2025-01-17 08:25:26 +01:00
if (!string.IsNullOrEmpty(Layer))
2025-01-17 08:25:10 +01:00
{
2025-01-17 08:25:26 +01:00
cacheName += "/" + Layer.Replace(':', '_');
2025-01-17 08:25:10 +01:00
}
2025-01-17 08:25:26 +01:00
2025-10-29 18:59:19 +01:00
if (!string.IsNullOrEmpty(tileMatrixSet.Identifier))
{
cacheName += "/" + tileMatrixSet.Identifier.Replace(':', '_');
}
2025-01-17 08:25:10 +01:00
}
2025-08-22 22:41:57 +02:00
BeginLoadTiles(ChildLayers.SelectMany(layer => layer.Tiles), cacheName);
2022-11-25 19:05:48 +01:00
}
2020-03-20 18:12:56 +01:00
}
protected override void SetRenderTransform()
{
2020-03-23 17:13:50 +01:00
foreach (var layer in ChildLayers)
2020-03-20 18:12:56 +01:00
{
layer.SetRenderTransform(ParentMap.ViewTransform);
2020-03-20 18:12:56 +01:00
}
}
2020-03-21 01:52:17 +01:00
private bool UpdateChildLayers(WmtsTileMatrixSet tileMatrixSet)
2020-03-20 18:12:56 +01:00
{
2022-11-30 22:18:45 +01:00
// Multiply scale by 1.001 to avoid rounding issues.
//
var maxScale = 1.001 * ParentMap.ViewTransform.Scale;
2020-03-20 18:12:56 +01:00
2022-11-30 22:18:45 +01:00
// Show all WmtsTileMatrix layers with Scale <= maxScale, at least the first layer.
2020-03-21 01:52:17 +01:00
//
var currentMatrixes = tileMatrixSet.TileMatrixes
2020-03-22 18:33:34 +01:00
.Where((matrix, i) => i == 0 || matrix.Scale <= maxScale)
2020-03-21 01:52:17 +01:00
.ToList();
2022-11-30 22:18:45 +01:00
if (!IsBaseMapLayer)
2020-03-20 18:12:56 +01:00
{
2022-11-30 22:18:45 +01:00
// Show only the last layer.
//
2021-11-11 22:27:57 +01:00
currentMatrixes = currentMatrixes.Skip(currentMatrixes.Count - 1).ToList();
2020-03-21 01:52:17 +01:00
}
else if (currentMatrixes.Count > MaxBackgroundLevels + 1)
{
2022-11-30 22:18:45 +01:00
// Show not more than MaxBackgroundLevels + 1 layers.
//
2020-03-21 01:52:17 +01:00
currentMatrixes = currentMatrixes.Skip(currentMatrixes.Count - MaxBackgroundLevels - 1).ToList();
}
2020-03-20 18:12:56 +01:00
2022-11-25 19:05:48 +01:00
var currentLayers = ChildLayers.Where(layer => currentMatrixes.Contains(layer.WmtsTileMatrix)).ToList();
var tilesChanged = false;
2020-03-20 18:12:56 +01:00
2020-03-21 01:52:17 +01:00
Children.Clear();
2020-03-20 18:12:56 +01:00
2020-03-21 01:52:17 +01:00
foreach (var tileMatrix in currentMatrixes)
{
2022-12-13 22:12:49 +01:00
var layer = currentLayers.FirstOrDefault(l => l.WmtsTileMatrix == tileMatrix) ??
2025-09-21 10:57:11 +02:00
new WmtsTileMatrixLayer(tileMatrix, tileMatrixSet.TileMatrixes.IndexOf(tileMatrix));
2020-03-20 18:12:56 +01:00
if (layer.UpdateTiles(ParentMap.ViewTransform, ParentMap.ActualWidth, ParentMap.ActualHeight))
2020-03-20 18:12:56 +01:00
{
2022-11-25 19:05:48 +01:00
tilesChanged = true;
2020-03-21 01:52:17 +01:00
}
2020-03-20 18:12:56 +01:00
2022-11-25 19:05:48 +01:00
layer.SetRenderTransform(ParentMap.ViewTransform);
2020-03-21 01:52:17 +01:00
Children.Add(layer);
2020-03-20 18:12:56 +01:00
}
2022-11-25 19:05:48 +01:00
return tilesChanged;
2020-03-20 18:12:56 +01:00
}
private async void OnLoaded(object sender, RoutedEventArgs e)
{
Loaded -= OnLoaded;
2020-03-21 01:52:17 +01:00
if (TileMatrixSets.Count == 0 && CapabilitiesUri != null)
2020-03-20 18:12:56 +01:00
{
try
{
var capabilities = await WmtsCapabilities.ReadCapabilitiesAsync(CapabilitiesUri, Layer);
2020-03-21 08:17:15 +01:00
2020-06-18 18:30:57 +02:00
foreach (var tileMatrixSet in capabilities.TileMatrixSets
2025-09-19 18:57:54 +02:00
.Where(s => !TileMatrixSets.ContainsKey(s.SupportedCrs) ||
PreferredTileMatrixSets != null && PreferredTileMatrixSets.Contains(s.Identifier)))
2020-06-18 18:30:57 +02:00
{
2025-09-19 18:57:54 +02:00
TileMatrixSets[tileMatrixSet.SupportedCrs] = tileMatrixSet;
2020-06-18 18:30:57 +02:00
}
2020-03-20 18:12:56 +01:00
Layer = capabilities.Layer;
2025-10-29 18:59:19 +01:00
TileSource = new WmtsTileSource { UriTemplate = capabilities.UriTemplate };
2020-03-20 18:12:56 +01:00
}
2020-04-15 20:46:16 +02:00
catch (Exception ex)
2020-03-20 18:12:56 +01:00
{
2025-08-22 11:06:37 +02:00
Logger?.LogError(ex, "Failed reading capabilities from {uri}", CapabilitiesUri);
2020-03-20 18:12:56 +01:00
}
}
}
}
}