mirror of
https://github.com/ClemensFischer/XAML-Map-Control.git
synced 2025-12-06 07:12:04 +01:00
416 lines
13 KiB
C#
416 lines
13 KiB
C#
// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
|
|
// Copyright © 2024 Clemens Fischer
|
|
// Licensed under the Microsoft Public License (Ms-PL)
|
|
|
|
using System;
|
|
using System.Diagnostics;
|
|
#if WPF
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Media;
|
|
#elif UWP
|
|
using Windows.Foundation;
|
|
using Windows.UI.Xaml;
|
|
using Windows.UI.Xaml.Controls;
|
|
using Windows.UI.Xaml.Media;
|
|
#elif WINUI
|
|
using Windows.Foundation;
|
|
using Microsoft.UI.Xaml;
|
|
using Microsoft.UI.Xaml.Controls;
|
|
using Microsoft.UI.Xaml.Media;
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// Arranges child elements on a Map at positions specified by the attached property Location,
|
|
/// or in rectangles specified by the attached property BoundingBox.
|
|
/// </summary>
|
|
namespace MapControl
|
|
{
|
|
/// <summary>
|
|
/// Optional interface to hold the value of the attached property MapPanel.ParentMap.
|
|
/// </summary>
|
|
public interface IMapElement
|
|
{
|
|
MapBase ParentMap { get; set; }
|
|
}
|
|
|
|
public partial class MapPanel : Panel, IMapElement
|
|
{
|
|
private static readonly DependencyProperty ViewPositionProperty =
|
|
DependencyPropertyHelper.RegisterAttached<MapPanel, Point?>("ViewPosition");
|
|
|
|
private static readonly DependencyProperty ParentMapProperty =
|
|
DependencyPropertyHelper.RegisterAttached<MapPanel, MapBase>("ParentMap", null,
|
|
(element, oldValue, newValue) =>
|
|
{
|
|
if (element is IMapElement mapElement)
|
|
{
|
|
mapElement.ParentMap = newValue;
|
|
}
|
|
},
|
|
true); // inherits
|
|
|
|
private MapBase parentMap;
|
|
|
|
/// <summary>
|
|
/// Implements IMapElement.ParentMap.
|
|
/// </summary>
|
|
public MapBase ParentMap
|
|
{
|
|
get => parentMap;
|
|
set => SetParentMap(value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a value that controls whether an element's Visibility is automatically
|
|
/// set to Collapsed when it is located outside the visible viewport area.
|
|
/// </summary>
|
|
public static bool GetAutoCollapse(FrameworkElement element)
|
|
{
|
|
return (bool)element.GetValue(AutoCollapseProperty);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the AutoCollapse property.
|
|
/// </summary>
|
|
public static void SetAutoCollapse(FrameworkElement element, bool value)
|
|
{
|
|
element.SetValue(AutoCollapseProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the Location of an element.
|
|
/// </summary>
|
|
public static Location GetLocation(FrameworkElement element)
|
|
{
|
|
return (Location)element.GetValue(LocationProperty);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the Location of an element.
|
|
/// </summary>
|
|
public static void SetLocation(FrameworkElement element, Location value)
|
|
{
|
|
element.SetValue(LocationProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the BoundingBox of an element.
|
|
/// </summary>
|
|
public static BoundingBox GetBoundingBox(FrameworkElement element)
|
|
{
|
|
return (BoundingBox)element.GetValue(BoundingBoxProperty);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the BoundingBox of an element.
|
|
/// </summary>
|
|
public static void SetBoundingBox(FrameworkElement element, BoundingBox value)
|
|
{
|
|
element.SetValue(BoundingBoxProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the view position of an element with Location.
|
|
/// </summary>
|
|
public static Point? GetViewPosition(FrameworkElement element)
|
|
{
|
|
return (Point?)element.GetValue(ViewPositionProperty);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the attached ViewPosition property of an element. The method is called during
|
|
/// ArrangeOverride and may be overridden to modify the actual view position value.
|
|
/// An overridden method should call this method to set the attached property.
|
|
/// </summary>
|
|
protected virtual void SetViewPosition(FrameworkElement element, ref Point? position)
|
|
{
|
|
element.SetValue(ViewPositionProperty, position);
|
|
}
|
|
|
|
protected virtual void SetParentMap(MapBase map)
|
|
{
|
|
if (parentMap != null && parentMap != this)
|
|
{
|
|
parentMap.ViewportChanged -= OnViewportChanged;
|
|
}
|
|
|
|
parentMap = map;
|
|
|
|
if (parentMap != null && parentMap != this)
|
|
{
|
|
parentMap.ViewportChanged += OnViewportChanged;
|
|
|
|
OnViewportChanged(new ViewportChangedEventArgs());
|
|
}
|
|
}
|
|
|
|
private void OnViewportChanged(object sender, ViewportChangedEventArgs e)
|
|
{
|
|
OnViewportChanged(e);
|
|
}
|
|
|
|
protected virtual void OnViewportChanged(ViewportChangedEventArgs e)
|
|
{
|
|
InvalidateArrange();
|
|
}
|
|
|
|
protected override Size MeasureOverride(Size availableSize)
|
|
{
|
|
availableSize = new Size(double.PositiveInfinity, double.PositiveInfinity);
|
|
|
|
foreach (var element in ChildElements)
|
|
{
|
|
element.Measure(availableSize);
|
|
}
|
|
|
|
return new Size();
|
|
}
|
|
|
|
protected override Size ArrangeOverride(Size finalSize)
|
|
{
|
|
if (parentMap != null)
|
|
{
|
|
foreach (var element in ChildElements)
|
|
{
|
|
try
|
|
{
|
|
ArrangeChildElement(element, finalSize);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Debug.WriteLine($"MapPanel.ArrangeOverride: {element}: {ex.Message}");
|
|
}
|
|
}
|
|
}
|
|
|
|
return finalSize;
|
|
}
|
|
|
|
protected Point? GetViewPosition(Location location)
|
|
{
|
|
var position = parentMap.LocationToView(location);
|
|
|
|
if (parentMap.MapProjection.Type <= MapProjectionType.NormalCylindrical &&
|
|
position.HasValue &&
|
|
!InsideViewport(position.Value))
|
|
{
|
|
position = parentMap.LocationToView(
|
|
new Location(location.Latitude, parentMap.CoerceLongitude(location.Longitude)));
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
protected ViewRect? GetViewRect(BoundingBox boundingBox)
|
|
{
|
|
var mapRect = parentMap.MapProjection.BoundingBoxToMap(boundingBox);
|
|
|
|
if (!mapRect.HasValue)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
return GetViewRect(mapRect.Value);
|
|
}
|
|
|
|
protected ViewRect GetViewRect(Rect mapRect)
|
|
{
|
|
var center = new Point(mapRect.X + mapRect.Width / 2d, mapRect.Y + mapRect.Height / 2d);
|
|
var position = parentMap.ViewTransform.MapToView(center);
|
|
var projection = parentMap.MapProjection;
|
|
|
|
if (projection.Type <= MapProjectionType.NormalCylindrical &&
|
|
!InsideViewport(position))
|
|
{
|
|
var location = projection.MapToLocation(center);
|
|
|
|
if (location != null)
|
|
{
|
|
var pos = parentMap.LocationToView(
|
|
new Location(location.Latitude, parentMap.CoerceLongitude(location.Longitude)));
|
|
|
|
if (pos.HasValue)
|
|
{
|
|
position = pos.Value;
|
|
}
|
|
}
|
|
}
|
|
|
|
var width = mapRect.Width * parentMap.ViewTransform.Scale;
|
|
var height = mapRect.Height * parentMap.ViewTransform.Scale;
|
|
var x = position.X - width / 2d;
|
|
var y = position.Y - height / 2d;
|
|
|
|
return new ViewRect(x, y, width, height, parentMap.ViewTransform.Rotation);
|
|
}
|
|
|
|
private bool InsideViewport(Point point)
|
|
{
|
|
return point.X >= 0d && point.X <= parentMap.ActualWidth
|
|
&& point.Y >= 0d && point.Y <= parentMap.ActualHeight;
|
|
}
|
|
|
|
private void ArrangeChildElement(FrameworkElement element, Size panelSize)
|
|
{
|
|
var location = GetLocation(element);
|
|
|
|
if (location != null)
|
|
{
|
|
var position = GetViewPosition(location);
|
|
|
|
SetViewPosition(element, ref position);
|
|
|
|
if (GetAutoCollapse(element))
|
|
{
|
|
SetVisible(element, position.HasValue && InsideViewport(position.Value));
|
|
}
|
|
|
|
if (position.HasValue)
|
|
{
|
|
ArrangeElement(element, position.Value);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
element.ClearValue(ViewPositionProperty);
|
|
|
|
var boundingBox = GetBoundingBox(element);
|
|
|
|
if (boundingBox != null)
|
|
{
|
|
var viewRect = GetViewRect(boundingBox);
|
|
|
|
if (viewRect.HasValue)
|
|
{
|
|
ArrangeElement(element, viewRect.Value);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ArrangeElement(element, panelSize);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void ArrangeElement(FrameworkElement element, ViewRect rect)
|
|
{
|
|
element.Width = rect.Rect.Width;
|
|
element.Height = rect.Rect.Height;
|
|
|
|
element.Arrange(rect.Rect);
|
|
|
|
if (element.RenderTransform is RotateTransform rotateTransform)
|
|
{
|
|
rotateTransform.Angle = rect.Rotation;
|
|
}
|
|
else if (rect.Rotation != 0d)
|
|
{
|
|
SetRenderTransform(element, new RotateTransform { Angle = rect.Rotation }, 0.5, 0.5);
|
|
}
|
|
}
|
|
|
|
private static void ArrangeElement(FrameworkElement element, Point position)
|
|
{
|
|
var size = GetDesiredSize(element);
|
|
var x = position.X;
|
|
var y = position.Y;
|
|
|
|
switch (element.HorizontalAlignment)
|
|
{
|
|
case HorizontalAlignment.Center:
|
|
x -= size.Width / 2d;
|
|
break;
|
|
|
|
case HorizontalAlignment.Right:
|
|
x -= size.Width;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (element.VerticalAlignment)
|
|
{
|
|
case VerticalAlignment.Center:
|
|
y -= size.Height / 2d;
|
|
break;
|
|
|
|
case VerticalAlignment.Bottom:
|
|
y -= size.Height;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
element.Arrange(new Rect(x, y, size.Width, size.Height));
|
|
}
|
|
|
|
private static void ArrangeElement(FrameworkElement element, Size panelSize)
|
|
{
|
|
var size = GetDesiredSize(element);
|
|
var x = 0d;
|
|
var y = 0d;
|
|
var width = size.Width;
|
|
var height = size.Height;
|
|
|
|
switch (element.HorizontalAlignment)
|
|
{
|
|
case HorizontalAlignment.Center:
|
|
x = (panelSize.Width - size.Width) / 2d;
|
|
break;
|
|
|
|
case HorizontalAlignment.Right:
|
|
x = panelSize.Width - size.Width;
|
|
break;
|
|
|
|
case HorizontalAlignment.Stretch:
|
|
width = panelSize.Width;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
switch (element.VerticalAlignment)
|
|
{
|
|
case VerticalAlignment.Center:
|
|
y = (panelSize.Height - size.Height) / 2d;
|
|
break;
|
|
|
|
case VerticalAlignment.Bottom:
|
|
y = panelSize.Height - size.Height;
|
|
break;
|
|
|
|
case VerticalAlignment.Stretch:
|
|
height = panelSize.Height;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
element.Arrange(new Rect(x, y, width, height));
|
|
}
|
|
|
|
internal static Size GetDesiredSize(FrameworkElement element)
|
|
{
|
|
var width = element.DesiredSize.Width;
|
|
var height = element.DesiredSize.Height;
|
|
|
|
if (width < 0d || width == double.PositiveInfinity)
|
|
{
|
|
width = 0d;
|
|
}
|
|
|
|
if (height < 0d || height == double.PositiveInfinity)
|
|
{
|
|
height = 0d;
|
|
}
|
|
|
|
return new Size(width, height);
|
|
}
|
|
}
|
|
}
|