// XAML Map Control - http://xamlmapcontrol.codeplex.com/ // © 2015 Clemens Fischer // Licensed under the Microsoft Public License (Ms-PL) using System; using System.Globalization; namespace MapControl { /// /// Provides the URI of a map tile. /// public partial class TileSource { public const int TileSize = 256; public const double MetersPerDegree = 6378137d * Math.PI / 180d; // WGS 84 semi major axis private Func getUri; private string uriFormat = string.Empty; public TileSource() { } protected TileSource(string uriFormat) { this.uriFormat = uriFormat; } 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; if (uriFormat.Contains("{x}") && uriFormat.Contains("{y}") && uriFormat.Contains("{z}")) { if (uriFormat.Contains("{c}")) { getUri = GetOpenStreetMapUri; } else if (uriFormat.Contains("{i}")) { getUri = GetGoogleMapsUri; } else if (uriFormat.Contains("{n}")) { getUri = GetMapQuestUri; } else { getUri = GetBasicUri; } } else if (uriFormat.Contains("{q}")) // {i} is optional { getUri = GetQuadKeyUri; } else if (uriFormat.Contains("{W}") && uriFormat.Contains("{S}") && uriFormat.Contains("{E}") && uriFormat.Contains("{N}")) { getUri = GetBoundingBoxUri; } else if (uriFormat.Contains("{w}") && uriFormat.Contains("{s}") && uriFormat.Contains("{e}") && uriFormat.Contains("{n}")) { getUri = GetLatLonBoundingBoxUri; } else if (uriFormat.Contains("{x}") && uriFormat.Contains("{v}") && uriFormat.Contains("{z}")) { getUri = GetTmsUri; } } } public virtual Uri GetUri(int x, int y, int zoomLevel) { return getUri != null ? getUri(x, y, zoomLevel) : null; } private Uri GetBasicUri(int x, int y, int zoomLevel) { return new Uri(uriFormat. Replace("{x}", x.ToString()). Replace("{y}", y.ToString()). Replace("{z}", zoomLevel.ToString())); } private Uri GetOpenStreetMapUri(int x, int y, int zoomLevel) { var hostIndex = (x + y) % 3; return new Uri(uriFormat. Replace("{c}", "abc".Substring(hostIndex, 1)). Replace("{x}", x.ToString()). Replace("{y}", y.ToString()). Replace("{z}", zoomLevel.ToString())); } private Uri GetGoogleMapsUri(int x, int y, int zoomLevel) { var hostIndex = (x + y) % 4; return new Uri(uriFormat. Replace("{i}", hostIndex.ToString()). Replace("{x}", x.ToString()). Replace("{y}", y.ToString()). Replace("{z}", zoomLevel.ToString())); } private Uri GetMapQuestUri(int x, int y, int zoomLevel) { var hostIndex = (x + y) % 4 + 1; return new Uri(uriFormat. Replace("{n}", hostIndex.ToString()). Replace("{x}", x.ToString()). Replace("{y}", y.ToString()). Replace("{z}", zoomLevel.ToString())); } private Uri GetTmsUri(int x, int y, int zoomLevel) { y = (1 << zoomLevel) - 1 - y; return new Uri(uriFormat. Replace("{x}", x.ToString()). Replace("{v}", y.ToString()). Replace("{z}", zoomLevel.ToString())); } private Uri GetQuadKeyUri(int x, int y, int zoomLevel) { if (zoomLevel < 1) { return null; } var quadkey = new char[zoomLevel]; for (var z = zoomLevel - 1; z >= 0; z--, x /= 2, y /= 2) { quadkey[z] = (char)('0' + 2 * (y % 2) + (x % 2)); } return new Uri(uriFormat. Replace("{i}", new string(quadkey[zoomLevel - 1], 1)). Replace("{q}", new string(quadkey))); } private Uri GetBoundingBoxUri(int x, int y, int zoomLevel) { var tileSize = 360d / (double)(1 << zoomLevel); // tile width in degrees var west = MetersPerDegree * ((double)x * tileSize - 180d); var east = MetersPerDegree * ((double)(x + 1) * tileSize - 180d); var south = MetersPerDegree * (180d - (double)(y + 1) * tileSize); var north = MetersPerDegree * (180d - (double)y * tileSize); return new Uri(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))); } private Uri GetLatLonBoundingBoxUri(int x, int y, int zoomLevel) { var tileSize = 360d / (double)(1 << zoomLevel); // tile width in degrees var west = (double)x * tileSize - 180d; var east = (double)(x + 1) * tileSize - 180d; var south = MercatorTransform.YToLatitude(180d - (double)(y + 1) * tileSize); var north = MercatorTransform.YToLatitude(180d - (double)y * tileSize); return new Uri(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))); } } }