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

196 lines
6.5 KiB
C#
Raw Normal View History

// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
// © 2018 Clemens Fischer
2012-05-04 12:52:20 +02:00
// Licensed under the Microsoft Public License (Ms-PL)
using System;
2012-04-25 22:02:53 +02:00
using System.Globalization;
2017-10-08 17:35:07 +02:00
using System.Threading.Tasks;
#if WINDOWS_UWP
using Windows.UI.Xaml.Media;
#else
using System.ComponentModel;
2017-10-08 17:35:07 +02:00
using System.Windows.Media;
#endif
2012-04-25 22:02:53 +02:00
namespace MapControl
{
2012-05-04 12:52:20 +02:00
/// <summary>
2017-08-04 21:38:58 +02:00
/// Provides the download Uri or ImageSource of map tiles.
2012-05-04 12:52:20 +02:00
/// </summary>
#if !WINDOWS_UWP
[TypeConverter(typeof(TileSourceConverter))]
#endif
public class TileSource
2012-04-25 22:02:53 +02:00
{
2017-10-09 19:17:04 +02:00
private Func<int, int, int, string> getUri;
private string uriFormat;
private int subdomainIndex = -1;
public TileSource()
{
}
protected TileSource(string uriFormat)
{
this.uriFormat = uriFormat;
}
2017-10-09 19:17:04 +02:00
public string[] Subdomains { get; set; }
2017-08-04 21:38:58 +02:00
/// <summary>
/// Gets or sets the format string to produce tile Uris.
/// </summary>
public string UriFormat
{
get { return uriFormat; }
set
{
if (string.IsNullOrEmpty(value))
{
throw new ArgumentException("The value of the UriFormat property must not be null or empty.");
}
uriFormat = value;
2017-10-09 19:17:04 +02:00
if (uriFormat.Contains("{x}") && uriFormat.Contains("{z}"))
{
2017-10-09 19:17:04 +02:00
if (uriFormat.Contains("{y}"))
2013-01-29 17:55:53 +01:00
{
2017-10-09 19:17:04 +02:00
getUri = GetDefaultUri;
2013-01-29 17:55:53 +01:00
}
2017-10-09 19:17:04 +02:00
else if (uriFormat.Contains("{v}"))
{
2017-10-09 19:17:04 +02:00
getUri = GetTmsUri;
}
}
2013-01-29 17:55:53 +01:00
else if (uriFormat.Contains("{q}")) // {i} is optional
{
2013-01-29 17:55:53 +01:00
getUri = GetQuadKeyUri;
}
else if (uriFormat.Contains("{W}") && uriFormat.Contains("{S}") && uriFormat.Contains("{E}") && uriFormat.Contains("{N}"))
{
2013-01-29 17:55:53 +01:00
getUri = GetBoundingBoxUri;
}
else if (uriFormat.Contains("{w}") && uriFormat.Contains("{s}") && uriFormat.Contains("{e}") && uriFormat.Contains("{n}"))
{
getUri = GetLatLonBoundingBoxUri;
}
2017-10-09 19:17:04 +02:00
if (Subdomains == null && uriFormat.Contains("{c}"))
{
Subdomains = new string[] { "a", "b", "c" };
}
}
2013-01-29 17:55:53 +01:00
}
2017-08-04 21:38:58 +02:00
/// <summary>
2017-10-09 19:17:04 +02:00
/// Gets the image Uri for the specified tile indices and zoom level.
2017-08-04 21:38:58 +02:00
/// </summary>
2013-01-29 17:55:53 +01:00
public virtual Uri GetUri(int x, int y, int zoomLevel)
{
2017-10-09 19:17:04 +02:00
if (getUri == null)
{
return null;
}
var uri = getUri(x, y, zoomLevel);
if (Subdomains != null && Subdomains.Length > 0)
{
subdomainIndex = (subdomainIndex + 1) % Subdomains.Length;
uri = uri.Replace("{c}", Subdomains[subdomainIndex]);
}
return new Uri(uri, UriKind.RelativeOrAbsolute);
}
2017-10-08 17:35:07 +02:00
/// <summary>
2017-10-09 19:17:04 +02:00
/// Loads a tile ImageSource asynchronously from GetUri(x, y, zoomLevel).
2017-10-08 17:35:07 +02:00
/// </summary>
public virtual async Task<ImageSource> LoadImageAsync(int x, int y, int zoomLevel)
{
ImageSource imageSource = null;
var uri = GetUri(x, y, zoomLevel);
if (uri != null)
{
imageSource = await ImageLoader.LoadImageAsync(uri);
2017-10-08 17:35:07 +02:00
}
return imageSource;
}
2017-10-09 19:17:04 +02:00
private string GetDefaultUri(int x, int y, int zoomLevel)
2012-04-25 22:02:53 +02:00
{
2017-10-09 19:17:04 +02:00
return uriFormat
.Replace("{x}", x.ToString())
.Replace("{y}", y.ToString())
2017-10-09 19:17:04 +02:00
.Replace("{z}", zoomLevel.ToString());
2012-04-25 22:02:53 +02:00
}
2017-10-09 19:17:04 +02:00
private string GetTmsUri(int x, int y, int zoomLevel)
{
y = (1 << zoomLevel) - 1 - y;
2017-10-09 19:17:04 +02:00
return uriFormat
.Replace("{x}", x.ToString())
.Replace("{v}", y.ToString())
2017-10-09 19:17:04 +02:00
.Replace("{z}", zoomLevel.ToString());
}
2017-10-09 19:17:04 +02:00
private string GetQuadKeyUri(int x, int y, int zoomLevel)
2012-04-25 22:02:53 +02:00
{
if (zoomLevel < 1)
{
return null;
}
var quadkey = new char[zoomLevel];
2012-04-25 22:02:53 +02:00
for (var z = zoomLevel - 1; z >= 0; z--, x /= 2, y /= 2)
2012-04-25 22:02:53 +02:00
{
quadkey[z] = (char)('0' + 2 * (y % 2) + (x % 2));
2012-04-25 22:02:53 +02:00
}
2017-10-09 19:17:04 +02:00
return uriFormat
.Replace("{i}", new string(quadkey, zoomLevel - 1, 1))
2017-10-09 19:17:04 +02:00
.Replace("{q}", new string(quadkey));
2012-04-25 22:02:53 +02:00
}
2017-10-09 19:17:04 +02:00
private string GetBoundingBoxUri(int x, int y, int zoomLevel)
2012-04-25 22:02:53 +02:00
{
var tileSize = 360d / (1 << zoomLevel); // tile width in degrees
var west = MapProjection.MetersPerDegree * (x * tileSize - 180d);
var east = MapProjection.MetersPerDegree * ((x + 1) * tileSize - 180d);
var south = MapProjection.MetersPerDegree * (180d - (y + 1) * tileSize);
var north = MapProjection.MetersPerDegree * (180d - y * tileSize);
2017-10-09 19:17:04 +02:00
return uriFormat
.Replace("{W}", west.ToString(CultureInfo.InvariantCulture))
.Replace("{S}", south.ToString(CultureInfo.InvariantCulture))
.Replace("{E}", east.ToString(CultureInfo.InvariantCulture))
.Replace("{N}", north.ToString(CultureInfo.InvariantCulture))
2017-08-04 21:38:58 +02:00
.Replace("{X}", MapProjection.TileSize.ToString())
2017-10-09 19:17:04 +02:00
.Replace("{Y}", MapProjection.TileSize.ToString());
}
2017-10-09 19:17:04 +02:00
private string GetLatLonBoundingBoxUri(int x, int y, int zoomLevel)
{
var tileSize = 360d / (1 << zoomLevel); // tile width in degrees
var west = x * tileSize - 180d;
var east = (x + 1) * tileSize - 180d;
var south = WebMercatorProjection.YToLatitude(180d - (y + 1) * tileSize);
var north = WebMercatorProjection.YToLatitude(180d - y * tileSize);
2017-10-09 19:17:04 +02:00
return uriFormat
.Replace("{w}", west.ToString(CultureInfo.InvariantCulture))
.Replace("{s}", south.ToString(CultureInfo.InvariantCulture))
.Replace("{e}", east.ToString(CultureInfo.InvariantCulture))
.Replace("{n}", north.ToString(CultureInfo.InvariantCulture))
2017-08-04 21:38:58 +02:00
.Replace("{X}", MapProjection.TileSize.ToString())
2017-10-09 19:17:04 +02:00
.Replace("{Y}", MapProjection.TileSize.ToString());
2012-04-25 22:02:53 +02:00
}
}
}