// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control // © 2018 Clemens Fischer // Licensed under the Microsoft Public License (Ms-PL) using System; using System.Collections.Generic; using System.Collections.Specialized; using System.Linq; using System.Windows; using System.Windows.Media; using System.Windows.Shapes; namespace MapControl { public abstract partial class MapShape : Shape, IWeakEventListener { protected Geometry Data { get; } protected override Geometry DefiningGeometry { get { return Data; } } partial void SetDataTransform() { if (parentMap != null) { var transform = new TransformGroup(); var offsetX = GetLongitudeOffset() * parentMap.MapProjection.TrueScale; transform.Children.Add(new TranslateTransform(offsetX, 0d)); transform.Children.Add(parentMap.MapProjection.ViewportTransform); Data.Transform = transform; } else { Data.Transform = Transform.Identity; } } private void OnViewportChanged(object sender, ViewportChangedEventArgs e) { var transform = (TransformGroup)Data.Transform; var offset = (TranslateTransform)transform.Children[0]; offset.X = GetLongitudeOffset() * parentMap.MapProjection.TrueScale; if (e.ProjectionChanged) { transform.Children[1] = parentMap.MapProjection.ViewportTransform; } if (e.ProjectionChanged || parentMap.MapProjection.IsAzimuthal) { UpdateData(); } else if (Fill != null) { InvalidateVisual(); // Fill brush may be rendered only partially or not at all } } bool IWeakEventListener.ReceiveWeakEvent(Type managerType, object sender, EventArgs e) { UpdateData(); return true; } protected void DataCollectionPropertyChanged(DependencyPropertyChangedEventArgs e) { INotifyCollectionChanged collection; if ((collection = e.OldValue as INotifyCollectionChanged) != null) { CollectionChangedEventManager.RemoveListener(collection, this); } if ((collection = e.NewValue as INotifyCollectionChanged) != null) { CollectionChangedEventManager.AddListener(collection, this); } UpdateData(); } protected void AddPolylineFigure(PathFigureCollection figures, IEnumerable locations, bool closed) { if (locations != null && locations.Count() >= 2) { var points = locations.Select(loc => LocationToPoint(loc)); var figure = new PathFigure { StartPoint = points.First(), IsClosed = closed, IsFilled = closed }; figure.Segments.Add(new PolyLineSegment(points.Skip(1), true)); figures.Add(figure); } } } }