using System; using System.Collections; using System.Collections.Generic; using System.Collections.Specialized; using System.Linq; using System.Windows; using System.Windows.Media; namespace MapControl { /// /// Base class of MapPolyline, MapPolygon and MapMultiPolygon. /// public class MapPolypoint : MapPath, IWeakEventListener { public static readonly DependencyProperty FillRuleProperty = DependencyPropertyHelper.Register(nameof(FillRule), FillRule.EvenOdd, (polypoint, oldValue, newValue) => ((StreamGeometry)polypoint.Data).FillRule = newValue); public FillRule FillRule { get => (FillRule)GetValue(FillRuleProperty); set => SetValue(FillRuleProperty, value); } protected MapPolypoint() { Data = new StreamGeometry(); } protected void DataCollectionPropertyChanged(IEnumerable oldValue, IEnumerable newValue) { if (oldValue is INotifyCollectionChanged oldCollection) { CollectionChangedEventManager.RemoveListener(oldCollection, this); } if (newValue is INotifyCollectionChanged newCollection) { CollectionChangedEventManager.AddListener(newCollection, this); } UpdateData(); InvalidateVisual(); // necessary for StreamGeometry } bool IWeakEventListener.ReceiveWeakEvent(Type managerType, object sender, EventArgs e) { UpdateData(); InvalidateVisual(); // necessary for StreamGeometry return true; } protected void UpdateData(IEnumerable locations, bool closed) { using var context = ((StreamGeometry)Data).Open(); if (ParentMap != null && locations != null) { var longitudeOffset = GetLongitudeOffset(Location ?? locations.FirstOrDefault()); AddPolylinePoints(context, locations, longitudeOffset, closed); } } protected void UpdateData(IEnumerable> polygons) { using var context = ((StreamGeometry)Data).Open(); if (ParentMap != null && polygons != null) { var longitudeOffset = GetLongitudeOffset(Location); foreach (var locations in polygons) { AddPolylinePoints(context, locations, longitudeOffset, true); } } } private void AddPolylinePoints(StreamGeometryContext context, IEnumerable locations, double longitudeOffset, bool closed) { var points = locations .Select(location => LocationToView(location, longitudeOffset)) .Where(point => point.HasValue) .Select(point => point.Value); if (points.Any()) { var start = points.First(); var polyline = points.Skip(1).ToList(); var minX = start.X; var maxX = start.X; var minY = start.Y; var maxY = start.Y; foreach (var point in polyline) { 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) { context.BeginFigure(start, true, closed); context.PolyLineTo(polyline, true, true); } } } } }