2025-02-27 18:46:32 +01:00
|
|
|
|
using System;
|
2025-11-21 16:37:06 +01:00
|
|
|
|
using System.Collections.Generic;
|
2024-05-22 11:25:32 +02:00
|
|
|
|
#if WPF
|
|
|
|
|
|
using System.Windows;
|
|
|
|
|
|
using System.Windows.Controls;
|
|
|
|
|
|
using System.Windows.Media;
|
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.Controls;
|
|
|
|
|
|
using Windows.UI.Xaml.Media;
|
2024-05-22 11:25:32 +02:00
|
|
|
|
#elif WINUI
|
|
|
|
|
|
using Windows.Foundation;
|
|
|
|
|
|
using Microsoft.UI.Xaml.Controls;
|
|
|
|
|
|
using Microsoft.UI.Xaml.Media;
|
2025-08-19 19:43:02 +02:00
|
|
|
|
#elif AVALONIA
|
|
|
|
|
|
using Avalonia;
|
|
|
|
|
|
using Avalonia.Controls;
|
|
|
|
|
|
using Avalonia.Media;
|
2020-03-20 18:12:56 +01:00
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
namespace MapControl
|
|
|
|
|
|
{
|
2025-11-14 23:59:24 +01:00
|
|
|
|
public class WmtsTileMatrixLayer : Panel
|
2020-03-20 18:12:56 +01:00
|
|
|
|
{
|
2025-11-19 17:49:49 +01:00
|
|
|
|
public WmtsTileMatrixLayer(WmtsTileMatrix wmtsTileMatrix, int zoomLevel)
|
2020-03-20 18:12:56 +01:00
|
|
|
|
{
|
2025-11-19 23:34:39 +01:00
|
|
|
|
this.SetRenderTransform(new MatrixTransform());
|
2025-11-19 17:49:49 +01:00
|
|
|
|
WmtsTileMatrix = wmtsTileMatrix;
|
2022-11-25 19:05:48 +01:00
|
|
|
|
TileMatrix = new TileMatrix(zoomLevel, 1, 1, 0, 0);
|
2020-03-20 18:12:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-11-25 19:05:48 +01:00
|
|
|
|
public WmtsTileMatrix WmtsTileMatrix { get; }
|
2021-11-13 00:28:27 +01:00
|
|
|
|
|
2022-11-25 19:05:48 +01:00
|
|
|
|
public TileMatrix TileMatrix { get; private set; }
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2025-11-21 16:37:06 +01:00
|
|
|
|
public IEnumerable<ImageTile> Tiles { get; private set; } = [];
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2025-10-30 19:49:06 +01:00
|
|
|
|
public void UpdateRenderTransform(ViewTransform viewTransform)
|
2020-03-20 18:12:56 +01:00
|
|
|
|
{
|
2022-11-30 22:18:45 +01:00
|
|
|
|
// Tile matrix origin in pixels.
|
2020-03-20 18:12:56 +01:00
|
|
|
|
//
|
2022-11-25 19:05:48 +01:00
|
|
|
|
var tileMatrixOrigin = new Point(WmtsTileMatrix.TileWidth * TileMatrix.XMin, WmtsTileMatrix.TileHeight * TileMatrix.YMin);
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2020-03-22 18:33:34 +01:00
|
|
|
|
((MatrixTransform)RenderTransform).Matrix =
|
2022-11-25 19:05:48 +01:00
|
|
|
|
viewTransform.GetTileLayerTransform(WmtsTileMatrix.Scale, WmtsTileMatrix.TopLeft, tileMatrixOrigin);
|
2020-03-22 18:33:34 +01:00
|
|
|
|
}
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2024-07-15 20:10:28 +02:00
|
|
|
|
public bool UpdateTiles(ViewTransform viewTransform, double viewWidth, double viewHeight)
|
2020-03-22 18:33:34 +01:00
|
|
|
|
{
|
2025-11-21 15:52:38 +01:00
|
|
|
|
// Tile matrix bounds in pixels.
|
2020-03-21 08:17:15 +01:00
|
|
|
|
//
|
2024-07-15 20:10:28 +02:00
|
|
|
|
var bounds = viewTransform.GetTileMatrixBounds(WmtsTileMatrix.Scale, WmtsTileMatrix.TopLeft, viewWidth, viewHeight);
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2022-11-30 22:18:45 +01:00
|
|
|
|
// Tile X and Y bounds.
|
2020-03-22 18:33:34 +01:00
|
|
|
|
//
|
2022-11-25 19:05:48 +01:00
|
|
|
|
var xMin = (int)Math.Floor(bounds.X / WmtsTileMatrix.TileWidth);
|
|
|
|
|
|
var yMin = (int)Math.Floor(bounds.Y / WmtsTileMatrix.TileHeight);
|
|
|
|
|
|
var xMax = (int)Math.Floor((bounds.X + bounds.Width) / WmtsTileMatrix.TileWidth);
|
|
|
|
|
|
var yMax = (int)Math.Floor((bounds.Y + bounds.Height) / WmtsTileMatrix.TileHeight);
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2025-11-26 23:34:22 +01:00
|
|
|
|
if (!WmtsTileMatrix.HasFullHorizontalCoverage)
|
2022-11-22 19:15:34 +01:00
|
|
|
|
{
|
2025-11-26 23:34:22 +01:00
|
|
|
|
// Set X range limits.
|
2022-11-22 19:15:34 +01:00
|
|
|
|
//
|
|
|
|
|
|
xMin = Math.Max(xMin, 0);
|
2022-11-25 19:05:48 +01:00
|
|
|
|
xMax = Math.Min(Math.Max(xMax, 0), WmtsTileMatrix.MatrixWidth - 1);
|
2022-11-22 19:15:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-26 23:34:22 +01:00
|
|
|
|
// Set Y range limits.
|
|
|
|
|
|
//
|
2020-03-20 18:12:56 +01:00
|
|
|
|
yMin = Math.Max(yMin, 0);
|
2022-11-25 19:05:48 +01:00
|
|
|
|
yMax = Math.Min(Math.Max(yMax, 0), WmtsTileMatrix.MatrixHeight - 1);
|
2020-03-20 18:12:56 +01:00
|
|
|
|
|
2022-11-25 19:05:48 +01:00
|
|
|
|
if (TileMatrix.XMin == xMin && TileMatrix.YMin == yMin &&
|
|
|
|
|
|
TileMatrix.XMax == xMax && TileMatrix.YMax == yMax)
|
2020-03-20 18:12:56 +01:00
|
|
|
|
{
|
2022-11-30 22:18:45 +01:00
|
|
|
|
// No change of the TileMatrix and the Tiles collection.
|
2022-11-25 19:05:48 +01:00
|
|
|
|
//
|
2020-03-20 18:12:56 +01:00
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-11-25 19:05:48 +01:00
|
|
|
|
TileMatrix = new TileMatrix(TileMatrix.ZoomLevel, xMin, yMin, xMax, yMax);
|
2025-11-20 00:14:52 +01:00
|
|
|
|
Tiles = new ImageTileList(Tiles, TileMatrix, WmtsTileMatrix.MatrixWidth);
|
2025-11-19 17:49:49 +01:00
|
|
|
|
|
2020-03-20 18:12:56 +01:00
|
|
|
|
Children.Clear();
|
|
|
|
|
|
|
2025-11-20 00:14:52 +01:00
|
|
|
|
foreach (var tile in Tiles)
|
2020-03-20 18:12:56 +01:00
|
|
|
|
{
|
|
|
|
|
|
Children.Add(tile.Image);
|
|
|
|
|
|
}
|
2022-11-25 19:05:48 +01:00
|
|
|
|
|
|
|
|
|
|
return true;
|
2020-03-20 18:12:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected override Size MeasureOverride(Size availableSize)
|
|
|
|
|
|
{
|
|
|
|
|
|
foreach (var tile in Tiles)
|
|
|
|
|
|
{
|
|
|
|
|
|
tile.Image.Measure(availableSize);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return new Size();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected override Size ArrangeOverride(Size finalSize)
|
|
|
|
|
|
{
|
|
|
|
|
|
foreach (var tile in Tiles)
|
|
|
|
|
|
{
|
2025-11-18 22:23:38 +01:00
|
|
|
|
// Arrange tiles relative to TileMatrix.XMin/YMin.
|
2020-03-20 18:12:56 +01:00
|
|
|
|
//
|
2025-11-14 15:02:48 +01:00
|
|
|
|
var tileWidth = WmtsTileMatrix.TileWidth;
|
|
|
|
|
|
var tileHeight = WmtsTileMatrix.TileHeight;
|
|
|
|
|
|
var x = tileWidth * (tile.X - TileMatrix.XMin);
|
|
|
|
|
|
var y = tileHeight * (tile.Y - TileMatrix.YMin);
|
|
|
|
|
|
|
|
|
|
|
|
tile.Image.Width = tileWidth;
|
|
|
|
|
|
tile.Image.Height = tileHeight;
|
|
|
|
|
|
tile.Image.Arrange(new Rect(x, y, tileWidth, tileHeight));
|
2020-03-20 18:12:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return finalSize;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|