mirror of
https://github.com/ClemensFischer/XAML-Map-Control.git
synced 2025-12-06 07:12:04 +01:00
134 lines
4.5 KiB
C#
134 lines
4.5 KiB
C#
// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
|
|
// © 2020 Clemens Fischer
|
|
// Licensed under the Microsoft Public License (Ms-PL)
|
|
|
|
using System;
|
|
#if !WINDOWS_UWP
|
|
using System.Windows;
|
|
#endif
|
|
|
|
namespace MapControl.Projections
|
|
{
|
|
/// <summary>
|
|
/// 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.
|
|
/// </summary>
|
|
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;
|
|
this.north = north;
|
|
this.scaleFactor = scaleFactor;
|
|
this.falseEasting = falseEasting;
|
|
this.falseNorthing = falseNorthing;
|
|
}
|
|
|
|
public override double UnitsPerDegree
|
|
{
|
|
get { return scaleFactor * Wgs84MetersPerDegree; }
|
|
}
|
|
|
|
public override Vector GetRelativeScale(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(k, k);
|
|
}
|
|
|
|
public override Point LocationToMap(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 MapToLocation(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)
|
|
{
|
|
}
|
|
}
|
|
}
|