2024-05-26 09:49:00 +02:00
|
|
|
|
// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
|
|
|
|
|
|
// Copyright © 2024 Clemens Fischer
|
|
|
|
|
|
// Licensed under the Microsoft Public License (Ms-PL)
|
|
|
|
|
|
|
2024-07-16 10:41:44 +02:00
|
|
|
|
using System;
|
2024-05-26 09:49:00 +02:00
|
|
|
|
using System.Collections;
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.Collections.Specialized;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
|
|
namespace MapControl
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// Base class of MapPolyline and MapPolygon.
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public class MapPolypoint : MapPath
|
|
|
|
|
|
{
|
|
|
|
|
|
public static readonly StyledProperty<FillRule> FillRuleProperty =
|
|
|
|
|
|
DependencyPropertyHelper.Register<MapPolygon, FillRule>(nameof(FillRule), FillRule.EvenOdd,
|
|
|
|
|
|
(polypoint, oldValue, newValue) => ((PathGeometry)polypoint.Data).FillRule = newValue);
|
|
|
|
|
|
|
|
|
|
|
|
public FillRule FillRule
|
|
|
|
|
|
{
|
|
|
|
|
|
get => GetValue(FillRuleProperty);
|
|
|
|
|
|
set => SetValue(FillRuleProperty, value);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected MapPolypoint()
|
|
|
|
|
|
{
|
|
|
|
|
|
Data = new PathGeometry();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected void DataCollectionPropertyChanged(IEnumerable oldValue, IEnumerable newValue)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (oldValue is INotifyCollectionChanged oldCollection)
|
|
|
|
|
|
{
|
|
|
|
|
|
oldCollection.CollectionChanged -= DataCollectionChanged;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (newValue is INotifyCollectionChanged newCollection)
|
|
|
|
|
|
{
|
|
|
|
|
|
newCollection.CollectionChanged += DataCollectionChanged;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UpdateData();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected void DataCollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
|
|
|
|
|
|
{
|
|
|
|
|
|
UpdateData();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected void UpdateData(IEnumerable<Location> locations, bool closed)
|
|
|
|
|
|
{
|
2024-07-15 13:41:52 +02:00
|
|
|
|
var pathFigures = new PathFigures();
|
2024-05-26 09:49:00 +02:00
|
|
|
|
|
2024-07-15 13:41:52 +02:00
|
|
|
|
if (ParentMap != null && locations != null)
|
2024-05-26 09:49:00 +02:00
|
|
|
|
{
|
2024-07-15 13:41:52 +02:00
|
|
|
|
var longitudeOffset = GetLongitudeOffset(Location ?? locations.FirstOrDefault());
|
2024-05-26 09:49:00 +02:00
|
|
|
|
|
2024-07-15 13:41:52 +02:00
|
|
|
|
AddPolylinePoints(pathFigures, locations, longitudeOffset, closed);
|
2024-05-26 09:49:00 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-07-16 13:59:00 +02:00
|
|
|
|
SetPathFigures(pathFigures);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected void SetPathFigures(PathFigures pathFigures)
|
|
|
|
|
|
{
|
2024-07-16 10:41:44 +02:00
|
|
|
|
if (pathFigures.Count == 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
// Avalonia Shape seems to ignore PathGeometry with empty Figures collection
|
|
|
|
|
|
|
|
|
|
|
|
pathFigures.Add(new PathFigure { StartPoint = new Point(-1000, -1000) });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-07-15 13:41:52 +02:00
|
|
|
|
((PathGeometry)Data).Figures = pathFigures;
|
|
|
|
|
|
InvalidateGeometry();
|
2024-05-26 09:49:00 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-07-15 13:41:52 +02:00
|
|
|
|
private void AddPolylinePoints(PathFigures pathFigures, IEnumerable<Location> locations, double longitudeOffset, bool closed)
|
2024-05-26 09:49:00 +02:00
|
|
|
|
{
|
2024-07-15 09:09:15 +02:00
|
|
|
|
var points = locations
|
|
|
|
|
|
.Select(location => LocationToView(location, longitudeOffset))
|
|
|
|
|
|
.Where(point => point.HasValue)
|
|
|
|
|
|
.Select(point => point.Value);
|
|
|
|
|
|
|
2024-07-15 13:41:52 +02:00
|
|
|
|
if (points.Any())
|
|
|
|
|
|
{
|
2024-07-16 10:41:44 +02:00
|
|
|
|
var startPoint = points.First();
|
|
|
|
|
|
var polylineSegment = new PolyLineSegment(points.Skip(1));
|
|
|
|
|
|
var minX = startPoint.X;
|
|
|
|
|
|
var maxX = startPoint.X;
|
|
|
|
|
|
var minY = startPoint.Y;
|
|
|
|
|
|
var maxY = startPoint.Y;
|
|
|
|
|
|
|
|
|
|
|
|
foreach (var point in polylineSegment.Points)
|
2024-07-15 13:41:52 +02:00
|
|
|
|
{
|
2024-07-16 10:41:44 +02:00
|
|
|
|
minX = Math.Min(minX, point.X);
|
|
|
|
|
|
maxX = Math.Max(maxX, point.X);
|
|
|
|
|
|
minY = Math.Min(minY, point.Y);
|
|
|
|
|
|
maxY = Math.Max(maxY, point.Y);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (maxX >= 0 && minX <= ParentMap.ActualWidth &&
|
|
|
|
|
|
maxY >= 0 && minY <= ParentMap.ActualHeight)
|
|
|
|
|
|
{
|
|
|
|
|
|
var pathFigure = new PathFigure
|
|
|
|
|
|
{
|
|
|
|
|
|
StartPoint = startPoint,
|
|
|
|
|
|
IsClosed = closed,
|
|
|
|
|
|
IsFilled = true
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
pathFigure.Segments.Add(polylineSegment);
|
|
|
|
|
|
pathFigures.Add(pathFigure);
|
|
|
|
|
|
}
|
2024-07-15 13:41:52 +02:00
|
|
|
|
}
|
2024-05-26 09:49:00 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|