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

142 lines
5.3 KiB
C#
Raw Normal View History

// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
2024-02-03 21:01:53 +01:00
// Copyright © 2024 Clemens Fischer
// Licensed under the Microsoft Public License (Ms-PL)
using System;
using System.Globalization;
2024-05-22 11:25:32 +02:00
#if WPF
using System.Windows;
#endif
namespace MapControl
{
2022-03-05 18:40:57 +01:00
public enum MapProjectionType
{
WebMercator, // normal cylindrical projection compatible with MapTileLayer
NormalCylindrical,
TransverseCylindrical,
Azimuthal,
Other
}
/// <summary>
/// Defines a map projection between geographic coordinates and cartesian map coordinates.
/// </summary>
2017-08-04 21:38:58 +02:00
public abstract class MapProjection
{
public const double Wgs84EquatorialRadius = 6378137d;
2022-03-02 22:03:18 +01:00
public const double Wgs84MeterPerDegree = Wgs84EquatorialRadius * Math.PI / 180d;
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);
/// <summary>
2022-03-05 18:40:57 +01:00
/// Gets the type of the projection.
/// </summary>
2022-03-05 18:40:57 +01:00
public MapProjectionType Type { get; protected set; } = MapProjectionType.Other;
/// <summary>
2022-03-05 18:40:57 +01:00
/// Gets the WMS 1.3.0 CRS identifier.
/// </summary>
2024-07-12 18:42:18 +02:00
public string CrsId { get; protected set; } = "";
/// <summary>
2022-03-06 17:28:27 +01:00
/// Gets or sets an optional projection center.
/// </summary>
2024-07-12 13:57:27 +02:00
public virtual Location Center { get; protected internal set; } = new Location();
/// <summary>
/// Gets the relative map scale at the specified Location.
/// </summary>
public virtual Point GetRelativeScale(Location location) => new Point(1d, 1d);
/// <summary>
2022-11-05 17:32:29 +01:00
/// Transforms a Location in geographic coordinates to a Point in projected map coordinates.
/// Returns null when the Location can not be transformed.
/// </summary>
public abstract Point? LocationToMap(Location location);
/// <summary>
2022-11-05 17:32:29 +01:00
/// Transforms a Point in projected map coordinates to a Location in geographic coordinates.
2022-03-04 22:28:18 +01:00
/// Returns null when the Point can not be transformed.
/// </summary>
public abstract Location MapToLocation(Point point);
/// <summary>
/// Transforms a BoundingBox in geographic coordinates to a Rect in projected map coordinates.
/// Returns null when the BoundingBox can not be transformed.
/// </summary>
public virtual Rect? BoundingBoxToMap(BoundingBox boundingBox)
{
Rect? rect = null;
2022-12-08 10:37:56 +01:00
2024-04-11 14:57:54 +02:00
if (boundingBox.South < boundingBox.North && boundingBox.West < boundingBox.East)
2022-12-07 23:34:42 +01:00
{
var p1 = LocationToMap(new Location(boundingBox.South, boundingBox.West));
var p2 = LocationToMap(new Location(boundingBox.North, boundingBox.East));
if (p1.HasValue && p2.HasValue)
{
rect = new Rect(p1.Value, p2.Value);
2022-12-07 23:34:42 +01:00
}
}
else if (boundingBox.Center != null)
{
2022-12-08 15:05:19 +01:00
// boundingBox is a CenteredBoundingBox
//
2022-12-07 23:34:42 +01:00
var center = LocationToMap(boundingBox.Center);
if (center.HasValue)
{
var width = boundingBox.Width * Wgs84MeterPerDegree;
var height = boundingBox.Height * Wgs84MeterPerDegree;
var x = center.Value.X - width / 2d;
var y = center.Value.Y - height / 2d;
rect = new Rect(x, y, width, height);
2022-12-07 23:34:42 +01:00
}
}
return rect;
}
/// <summary>
/// Transforms a Rect in projected map coordinates to a BoundingBox in geographic coordinates.
/// Returns null when the MapRect can not be transformed.
/// </summary>
public virtual BoundingBox MapToBoundingBox(Rect rect)
{
var southWest = MapToLocation(new Point(rect.X, rect.Y));
var northEast = MapToLocation(new Point(rect.X + rect.Width, rect.Y + rect.Height));
2022-12-07 23:34:42 +01:00
return southWest != null && northEast != null
? new BoundingBox(southWest, northEast)
: null;
}
/// <summary>
/// Gets the CRS parameter value for a WMS GetMap request.
/// </summary>
public virtual string GetCrsValue()
{
2024-07-12 13:57:27 +02:00
return CrsId.StartsWith("AUTO2:") || CrsId.StartsWith("AUTO:")
? string.Format(CultureInfo.InvariantCulture, "{0},1,{1},{2}", CrsId, Center.Longitude, Center.Latitude)
: CrsId;
}
/// <summary>
/// Gets the BBOX parameter value for a WMS GetMap request.
/// </summary>
public virtual string GetBboxValue(Rect rect)
{
2024-04-17 20:10:45 +02:00
// Truncate values for seamless stitching of two WMS images at 180° longitude,
// as done in WmsImageLayer.GetImageAsync.
2022-12-07 17:00:25 +01:00
//
2022-12-14 18:02:19 +01:00
return string.Format(CultureInfo.InvariantCulture, "{0:F2},{1:F2},{2:F2},{3:F2}",
0.01 * Math.Ceiling(100d * rect.X),
0.01 * Math.Ceiling(100d * rect.Y),
0.01 * Math.Floor(100d * (rect.X + rect.Width)),
0.01 * Math.Floor(100d * (rect.Y + rect.Height)));
}
}
}