XAML-Map-Control/MapControl/Shared/MapProjection.cs

212 lines
8.2 KiB
C#
Raw Normal View History

// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
// © 2019 Clemens Fischer
// Licensed under the Microsoft Public License (Ms-PL)
using System;
using System.Globalization;
2017-08-04 21:38:58 +02:00
#if WINDOWS_UWP
using Windows.Foundation;
#else
using System.Windows;
using System.Windows.Media;
#endif
namespace MapControl
{
/// <summary>
2017-08-04 21:38:58 +02:00
/// Defines a map projection between geographic coordinates, cartesian map coordinates and viewport coordinates.
/// </summary>
2017-08-04 21:38:58 +02:00
public abstract class MapProjection
{
public const int TileSize = 256;
public const double Wgs84EquatorialRadius = 6378137d;
2018-12-20 21:55:12 +01:00
public const double Wgs84Flattening = 1d / 298.257223563;
public static readonly double Wgs84Eccentricity = Math.Sqrt((2d - Wgs84Flattening) * Wgs84Flattening);
public const double Wgs84MetersPerDegree = Wgs84EquatorialRadius * Math.PI / 180d;
2017-08-04 21:38:58 +02:00
/// <summary>
2019-10-28 17:44:22 +01:00
/// Gets or sets the WMS 1.3.0 CRS identifier.
/// </summary>
public string CrsId { get; set; }
/// <summary>
/// Indicates if this is a normal cylindrical projection.
/// </summary>
public virtual bool IsNormalCylindrical
{
get { return true; }
}
/// <summary>
/// Indicates if this is a web mercator projection, i.e. compatible with MapTileLayer.
/// </summary>
public virtual bool IsWebMercator
{
get { return false; }
}
/// <summary>
/// Gets the absolute value of the minimum and maximum latitude that can be transformed.
/// </summary>
public virtual double MaxLatitude
{
get { return 90d; }
}
/// <summary>
/// Gets the scale factor from geographic to cartesian coordinates, on the line of true scale of a
/// cylindrical projection (usually the equator), or at the projection center of an azimuthal projection.
/// </summary>
public virtual double TrueScale
{
get { return Wgs84MetersPerDegree; }
}
/// <summary>
/// Gets the projection center. Only relevant for azimuthal pprojections.
/// </summary>
public Location ProjectionCenter { get; private set; } = new Location();
/// <summary>
/// Gets the transform matrix from cartesian map coordinates to viewport coordinates (pixels).
/// </summary>
public Matrix ViewportTransform { get; private set; }
2018-12-02 17:13:55 +01:00
/// <summary>
2018-12-20 21:55:12 +01:00
/// Gets the transform matrix from viewport coordinates (pixels) to cartesian map coordinates.
2018-12-02 17:13:55 +01:00
/// </summary>
public Matrix InverseViewportTransform { get; private set; }
/// <summary>
2018-12-20 21:55:12 +01:00
/// Gets the scaling factor from cartesian map coordinates to viewport coordinates (pixels)
/// at the projection's point of true scale.
/// </summary>
public double ViewportScale { get; private set; }
/// <summary>
/// Gets the map scale at the specified Location as viewport coordinate units per meter (px/m).
/// </summary>
public virtual Vector GetMapScale(Location location)
{
return new Vector(ViewportScale, ViewportScale);
}
/// <summary>
/// Transforms a Location in geographic coordinates to a Point in cartesian map coordinates.
/// </summary>
public abstract Point LocationToPoint(Location location);
/// <summary>
/// Transforms a Point in cartesian map coordinates to a Location in geographic coordinates.
/// </summary>
public abstract Location PointToLocation(Point point);
/// <summary>
/// Transforms a BoundingBox in geographic coordinates to a Rect in cartesian map coordinates.
/// </summary>
public virtual Rect BoundingBoxToRect(BoundingBox boundingBox)
{
return new Rect(
LocationToPoint(new Location(boundingBox.South, boundingBox.West)),
LocationToPoint(new Location(boundingBox.North, boundingBox.East)));
}
/// <summary>
/// Transforms a Rect in cartesian map coordinates to a BoundingBox in geographic coordinates.
/// </summary>
public virtual BoundingBox RectToBoundingBox(Rect rect)
{
var sw = PointToLocation(new Point(rect.X, rect.Y));
var ne = PointToLocation(new Point(rect.X + rect.Width, rect.Y + rect.Height));
return new BoundingBox(sw.Latitude, sw.Longitude, ne.Latitude, ne.Longitude);
}
/// <summary>
/// Transforms a Location in geographic coordinates to a Point in viewport coordinates.
/// </summary>
public Point LocationToViewportPoint(Location location)
{
return ViewportTransform.Transform(LocationToPoint(location));
}
/// <summary>
/// Transforms a Point in viewport coordinates to a Location in geographic coordinates.
/// </summary>
public Location ViewportPointToLocation(Point point)
{
2018-12-02 17:13:55 +01:00
return PointToLocation(InverseViewportTransform.Transform(point));
}
/// <summary>
/// Transforms a Rect in viewport coordinates to a BoundingBox in geographic coordinates.
/// </summary>
public BoundingBox ViewportRectToBoundingBox(Rect rect)
{
2019-10-25 19:56:23 +02:00
var p1 = InverseViewportTransform.Transform(new Point(rect.X, rect.Y));
var p2 = InverseViewportTransform.Transform(new Point(rect.X, rect.Y + rect.Height));
var p3 = InverseViewportTransform.Transform(new Point(rect.X + rect.Width, rect.Y));
var p4 = InverseViewportTransform.Transform(new Point(rect.X + rect.Width, rect.Y + rect.Height));
2019-10-25 19:56:23 +02:00
rect.X = Math.Min(p1.X, Math.Min(p2.X, Math.Min(p3.X, p4.X)));
rect.Y = Math.Min(p1.Y, Math.Min(p2.Y, Math.Min(p3.Y, p4.Y)));
rect.Width = Math.Max(p1.X, Math.Max(p2.X, Math.Max(p3.X, p4.X))) - rect.X;
rect.Height = Math.Max(p1.Y, Math.Max(p2.Y, Math.Max(p3.Y, p4.Y))) - rect.Y;
return RectToBoundingBox(rect);
}
/// <summary>
/// Gets the CRS parameter value for a WMS GetMap request.
/// </summary>
public virtual string GetCrsValue()
{
return CrsId.StartsWith("AUTO:") || CrsId.StartsWith("AUTO2:")
? string.Format(CultureInfo.InvariantCulture, "{0},1,{1},{2}", CrsId, ProjectionCenter.Longitude, ProjectionCenter.Latitude)
: CrsId;
}
/// <summary>
/// Gets the BBOX parameter value for a WMS GetMap request.
/// </summary>
public virtual string GetBboxValue(Rect rect)
{
return string.Format(CultureInfo.InvariantCulture,
"{0},{1},{2},{3}", rect.X, rect.Y, (rect.X + rect.Width), (rect.Y + rect.Height));
}
/// <summary>
/// Sets ProjectionCenter, ViewportScale, ViewportTransform and InverseViewportTransform.
/// </summary>
public void SetViewportTransform(Location projectionCenter, Location mapCenter, Point viewportCenter, double zoomLevel, double heading)
{
ProjectionCenter = projectionCenter;
ViewportScale = Math.Pow(2d, zoomLevel) * TileSize / (360d * TrueScale);
var center = LocationToPoint(mapCenter);
2018-03-07 22:19:16 +01:00
var matrix = CreateTransformMatrix(center, ViewportScale, -ViewportScale, heading, viewportCenter);
ViewportTransform = matrix;
2018-03-07 22:19:16 +01:00
matrix.Invert();
2018-12-02 17:13:55 +01:00
InverseViewportTransform = matrix;
}
2018-03-07 22:24:57 +01:00
internal static Matrix CreateTransformMatrix(
Point translation1, double scale, double rotation, Point translation2)
{
return CreateTransformMatrix(translation1, scale, scale, rotation, translation2);
}
2018-03-07 22:19:16 +01:00
internal static Matrix CreateTransformMatrix(
Point translation1, double scaleX, double scaleY, double rotation, Point translation2)
{
2018-03-07 22:19:16 +01:00
var matrix = new Matrix(scaleX, 0d, 0d, scaleY, -translation1.X * scaleX, -translation1.Y * scaleY);
matrix.Rotate(rotation);
matrix.Translate(translation2.X, translation2.Y);
return matrix;
}
}
}