// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control // © 2018 Clemens Fischer // Licensed under the Microsoft Public License (Ms-PL) using System; #if !WINDOWS_UWP using System.Windows; #endif namespace MapControl.Projections { /// /// Elliptical Polar Stereographic Projection with a given scale factor at the pole and /// optional false easting and northing, as used by the UPS North and UPS South projections. /// See "Map Projections - A Working Manual" (https://pubs.usgs.gov/pp/1395/report.pdf), p.160-162. /// public class PolarStereographicProjection : MapProjection { public static double ConvergenceTolerance = 1e-6; public static int MaxIterations = 10; private readonly bool north; private readonly double scaleFactor; private readonly double falseEasting; private readonly double falseNorthing; public PolarStereographicProjection(string crsId, bool north, double scaleFactor = 1d, double falseEasting = 0d, double falseNorthing = 0d) { CrsId = crsId; TrueScale = scaleFactor * MetersPerDegree; this.north = north; this.scaleFactor = scaleFactor; this.falseEasting = falseEasting; this.falseNorthing = falseNorthing; } public override Vector GetMapScale(Location location) { var lat = (north ? location.Latitude : -location.Latitude) * Math.PI / 180d; var a = Wgs84EquatorialRadius; var e = Wgs84Eccentricity; var s = Math.Sqrt(Math.Pow(1 + e, 1 + e) * Math.Pow(1 - e, 1 - e)); var t = Math.Tan(Math.PI / 4d - lat / 2d) / ConformalFactor(lat); var rho = 2d * a * scaleFactor * t / s; var eSinLat = e * Math.Sin(lat); var m = Math.Cos(lat) / Math.Sqrt(1d - eSinLat * eSinLat); var k = rho / (a * m); return new Vector(ViewportScale * k, ViewportScale * k); } public override Point LocationToPoint(Location location) { var lat = location.Latitude * Math.PI / 180d; var lon = location.Longitude * Math.PI / 180d; if (north) { lon = Math.PI - lon; } else { lat = -lat; } var a = Wgs84EquatorialRadius; var e = Wgs84Eccentricity; var s = Math.Sqrt(Math.Pow(1 + e, 1 + e) * Math.Pow(1 - e, 1 - e)); var t = Math.Tan(Math.PI / 4d - lat / 2d) / ConformalFactor(lat); var rho = 2d * a * scaleFactor * t / s; return new Point(rho * Math.Sin(lon) + falseEasting, rho * Math.Cos(lon) + falseNorthing); } public override Location PointToLocation(Point point) { point.X -= falseEasting; point.Y -= falseNorthing; var lon = Math.Atan2(point.X, point.Y); var rho = Math.Sqrt(point.X * point.X + point.Y * point.Y); var a = Wgs84EquatorialRadius; var e = Wgs84Eccentricity; var s = Math.Sqrt(Math.Pow(1 + e, 1 + e) * Math.Pow(1 - e, 1 - e)); var t = rho * s / (2d * a * scaleFactor); var lat = Math.PI / 2d - 2d * Math.Atan(t); var relChange = 1d; for (int i = 0; i < MaxIterations && relChange > ConvergenceTolerance; i++) { var newLat = Math.PI / 2d - 2d * Math.Atan(t * ConformalFactor(lat)); relChange = Math.Abs(1d - newLat / lat); lat = newLat; } if (north) { lon = Math.PI - lon; } else { lat = -lat; } return new Location(lat * 180d / Math.PI, lon * 180d / Math.PI); } private static double ConformalFactor(double lat) { var eSinLat = Wgs84Eccentricity * Math.Sin(lat); return Math.Pow((1d - eSinLat) / (1d + eSinLat), Wgs84Eccentricity / 2d); } } public class UpsNorthProjection : PolarStereographicProjection { public UpsNorthProjection() : base("EPSG:32661", true, 0.994, 2e6, 2e6) { } } public class UpsSouthProjection : PolarStereographicProjection { public UpsSouthProjection() : base("EPSG:32761", false, 0.994, 2e6, 2e6) { } } }