XAML-Map-Control/MapControl/WPF/DrawingTileMatrixLayer.cs

162 lines
5.7 KiB
C#
Raw Normal View History

2025-12-05 00:17:50 +01:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Media;
2025-12-05 14:44:33 +01:00
using System.Windows.Media.Animation;
2025-12-05 00:17:50 +01:00
namespace MapControl
{
2025-12-05 14:44:33 +01:00
public class ImageDrawingTile : Tile
2025-12-05 00:17:50 +01:00
{
2025-12-05 14:44:33 +01:00
private readonly ImageDrawing imageDrawing = new ImageDrawing();
public ImageDrawingTile(int zoomLevel, int x, int y, int columnCount)
: base(zoomLevel, x, y, columnCount)
{
Drawing.Children.Add(imageDrawing);
}
public DrawingGroup Drawing { get; } = new DrawingGroup();
public ImageSource ImageSource
{
get => imageDrawing.ImageSource;
set => imageDrawing.ImageSource = value;
}
public void SetRect(int xMin, int yMin, int tileWidth, int tileHeight)
{
imageDrawing.Rect = new Rect(tileWidth * (X - xMin), tileHeight * (Y - yMin), tileWidth, tileHeight);
}
2025-12-05 00:17:50 +01:00
public override async Task LoadImageAsync(Func<Task<ImageSource>> loadImageFunc)
{
2025-12-05 00:50:28 +01:00
var image = await loadImageFunc().ConfigureAwait(false);
2025-12-05 00:17:50 +01:00
2025-12-05 14:44:33 +01:00
void SetImageSource()
2025-12-05 00:50:28 +01:00
{
2025-12-05 14:44:33 +01:00
imageDrawing.ImageSource = image;
if (image != null && MapBase.ImageFadeDuration > TimeSpan.Zero)
{
var fadeInAnimation = new DoubleAnimation
{
From = 0d,
Duration = MapBase.ImageFadeDuration,
FillBehavior = FillBehavior.Stop
};
Drawing.BeginAnimation(DrawingGroup.OpacityProperty, fadeInAnimation);
}
2025-12-05 00:50:28 +01:00
}
2025-12-05 14:44:33 +01:00
await Drawing.Dispatcher.InvokeAsync(SetImageSource);
2025-12-05 00:17:50 +01:00
}
}
public class DrawingTileMatrixLayer(WmtsTileMatrix wmtsTileMatrix, int zoomLevel) : UIElement
{
public WmtsTileMatrix WmtsTileMatrix => wmtsTileMatrix;
public TileMatrix TileMatrix { get; private set; } = new TileMatrix(zoomLevel, 1, 1, 0, 0);
2025-12-05 00:50:28 +01:00
public IEnumerable<ImageDrawingTile> Tiles { get; private set; } = [];
2025-12-05 00:17:50 +01:00
public DrawingGroup Drawing { get; } = new DrawingGroup { Transform = new MatrixTransform() };
protected override void OnRender(DrawingContext drawingContext)
{
drawingContext.DrawDrawing(Drawing);
}
public void UpdateRenderTransform(ViewTransform viewTransform)
{
// Tile matrix origin in pixels.
//
var tileMatrixOrigin = new Point(WmtsTileMatrix.TileWidth * TileMatrix.XMin, WmtsTileMatrix.TileHeight * TileMatrix.YMin);
((MatrixTransform)Drawing.Transform).Matrix =
viewTransform.GetTileLayerTransform(WmtsTileMatrix.Scale, WmtsTileMatrix.TopLeft, tileMatrixOrigin);
}
public bool UpdateTiles(ViewTransform viewTransform, double viewWidth, double viewHeight)
{
// Tile matrix bounds in pixels.
//
var bounds = viewTransform.GetTileMatrixBounds(WmtsTileMatrix.Scale, WmtsTileMatrix.TopLeft, viewWidth, viewHeight);
// Tile X and Y bounds.
//
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);
if (!WmtsTileMatrix.HasFullHorizontalCoverage)
{
// Set X range limits.
//
xMin = Math.Max(xMin, 0);
xMax = Math.Min(Math.Max(xMax, 0), WmtsTileMatrix.MatrixWidth - 1);
}
// Set Y range limits.
//
yMin = Math.Max(yMin, 0);
yMax = Math.Min(Math.Max(yMax, 0), WmtsTileMatrix.MatrixHeight - 1);
if (TileMatrix.XMin == xMin && TileMatrix.YMin == yMin &&
TileMatrix.XMax == xMax && TileMatrix.YMax == yMax)
{
// No change of the TileMatrix and the Tiles collection.
//
return false;
}
TileMatrix = new TileMatrix(TileMatrix.ZoomLevel, xMin, yMin, xMax, yMax);
CreateTiles();
return true;
}
private void CreateTiles()
{
2025-12-05 08:10:14 +01:00
var tileCount = TileMatrix.Width * TileMatrix.Height;
var tiles = new List<ImageDrawingTile>(tileCount);
var drawings = new DrawingCollection(tileCount);
2025-12-05 00:17:50 +01:00
for (var y = TileMatrix.YMin; y <= TileMatrix.YMax; y++)
{
for (var x = TileMatrix.XMin; x <= TileMatrix.XMax; x++)
{
var tile = Tiles.FirstOrDefault(t => t.X == x && t.Y == y);
if (tile == null)
{
2025-12-05 00:50:28 +01:00
tile = new ImageDrawingTile(TileMatrix.ZoomLevel, x, y, WmtsTileMatrix.MatrixWidth);
2025-12-05 00:17:50 +01:00
2025-12-05 14:44:33 +01:00
var equivalentTile = Tiles.FirstOrDefault(t => t.ImageSource != null && t.Column == tile.Column && t.Row == tile.Row);
2025-12-05 00:17:50 +01:00
if (equivalentTile != null)
{
tile.IsPending = false;
2025-12-05 14:44:33 +01:00
tile.ImageSource = equivalentTile.ImageSource;
2025-12-05 00:17:50 +01:00
}
}
2025-12-05 14:44:33 +01:00
tile.SetRect(TileMatrix.XMin, TileMatrix.YMin, WmtsTileMatrix.TileWidth, WmtsTileMatrix.TileHeight);
2025-12-05 00:17:50 +01:00
tiles.Add(tile);
2025-12-05 00:58:30 +01:00
drawings.Add(tile.Drawing);
2025-12-05 00:17:50 +01:00
}
}
Tiles = tiles;
2025-12-05 00:58:30 +01:00
Drawing.Children = drawings;
2025-12-05 00:17:50 +01:00
}
}
}