2017-08-04 21:38:58 +02:00
|
|
|
|
// XAML Map Control - https://github.com/ClemensFischer/XAML-Map-Control
|
2019-03-27 18:39:59 +01:00
|
|
|
|
// © 2019 Clemens Fischer
|
2017-08-04 21:38:58 +02:00
|
|
|
|
// Licensed under the Microsoft Public License (Ms-PL)
|
|
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
|
using System.IO;
|
|
|
|
|
|
using System.Runtime.Caching;
|
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
using System.Threading.Tasks;
|
2018-08-21 23:56:54 +02:00
|
|
|
|
using System.Windows.Media;
|
2017-08-04 21:38:58 +02:00
|
|
|
|
|
|
|
|
|
|
namespace MapControl
|
|
|
|
|
|
{
|
2018-08-08 23:31:52 +02:00
|
|
|
|
public partial class TileImageLoader
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
2017-09-06 20:43:46 +02:00
|
|
|
|
/// Default folder path where an ObjectCache instance may save cached data,
|
|
|
|
|
|
/// i.e. C:\ProgramData\MapControl\TileCache
|
2017-08-04 21:38:58 +02:00
|
|
|
|
/// </summary>
|
2017-09-06 20:43:46 +02:00
|
|
|
|
public static string DefaultCacheFolder
|
|
|
|
|
|
{
|
|
|
|
|
|
get { return Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData), "MapControl", "TileCache"); }
|
|
|
|
|
|
}
|
2017-08-04 21:38:58 +02:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// The ObjectCache used to cache tile images. The default is MemoryCache.Default.
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public static ObjectCache Cache { get; set; } = MemoryCache.Default;
|
|
|
|
|
|
|
2019-06-10 19:30:47 +02:00
|
|
|
|
private async Task LoadCachedTileImageAsync(Tile tile, Uri uri, string cacheKey)
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
|
|
|
|
|
DateTime expiration;
|
2018-08-09 18:21:16 +02:00
|
|
|
|
var cacheBuffer = GetCachedImage(cacheKey, out expiration);
|
2017-08-04 21:38:58 +02:00
|
|
|
|
|
2018-08-09 18:21:16 +02:00
|
|
|
|
if (cacheBuffer == null || expiration < DateTime.UtcNow)
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
2019-06-13 21:38:01 +02:00
|
|
|
|
var response = await ImageLoader.LoadHttpStreamAsync(uri).ConfigureAwait(false);
|
2018-08-09 18:21:16 +02:00
|
|
|
|
|
2019-06-13 21:38:01 +02:00
|
|
|
|
if (response != null) // download succeeded
|
2017-10-27 17:15:18 +02:00
|
|
|
|
{
|
2018-08-09 18:21:16 +02:00
|
|
|
|
cacheBuffer = null; // discard cached image
|
|
|
|
|
|
|
2019-06-13 21:38:01 +02:00
|
|
|
|
if (response.Stream != null) // tile image available
|
2018-08-09 18:21:16 +02:00
|
|
|
|
{
|
2019-06-13 21:38:01 +02:00
|
|
|
|
using (var stream = response.Stream)
|
2018-08-08 23:31:52 +02:00
|
|
|
|
{
|
2018-08-21 23:56:54 +02:00
|
|
|
|
LoadTileImage(tile, stream);
|
2019-06-13 21:38:01 +02:00
|
|
|
|
SetCachedImage(cacheKey, stream, GetExpiration(response.MaxAge));
|
2018-08-09 18:21:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-08-04 21:38:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2019-06-08 23:18:33 +02:00
|
|
|
|
if (cacheBuffer != null) // cached image not expired or download failed
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
2018-08-09 18:21:16 +02:00
|
|
|
|
using (var stream = new MemoryStream(cacheBuffer))
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
2018-08-21 23:56:54 +02:00
|
|
|
|
LoadTileImage(tile, stream);
|
2017-08-04 21:38:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-08-21 23:56:54 +02:00
|
|
|
|
private async Task LoadTileImageAsync(Tile tile, TileSource tileSource)
|
|
|
|
|
|
{
|
2019-06-13 21:38:01 +02:00
|
|
|
|
SetTileImage(tile, await tileSource.LoadImageAsync(tile.XIndex, tile.Y, tile.ZoomLevel).ConfigureAwait(false));
|
2018-08-21 23:56:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void LoadTileImage(Tile tile, Stream stream)
|
2017-08-04 21:38:58 +02:00
|
|
|
|
{
|
2018-08-21 23:56:54 +02:00
|
|
|
|
SetTileImage(tile, ImageLoader.LoadImage(stream));
|
|
|
|
|
|
}
|
2017-08-04 21:38:58 +02:00
|
|
|
|
|
2018-08-21 23:56:54 +02:00
|
|
|
|
private void SetTileImage(Tile tile, ImageSource imageSource)
|
|
|
|
|
|
{
|
2018-08-09 18:21:16 +02:00
|
|
|
|
tile.Image.Dispatcher.InvokeAsync(() => tile.SetImage(imageSource));
|
2017-08-04 21:38:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private static byte[] GetCachedImage(string cacheKey, out DateTime expiration)
|
|
|
|
|
|
{
|
|
|
|
|
|
var buffer = Cache.Get(cacheKey) as byte[];
|
|
|
|
|
|
|
|
|
|
|
|
if (buffer != null && buffer.Length >= 16 &&
|
|
|
|
|
|
Encoding.ASCII.GetString(buffer, buffer.Length - 16, 8) == "EXPIRES:")
|
|
|
|
|
|
{
|
|
|
|
|
|
expiration = new DateTime(BitConverter.ToInt64(buffer, buffer.Length - 8), DateTimeKind.Utc);
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
expiration = DateTime.MinValue;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return buffer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private static void SetCachedImage(string cacheKey, MemoryStream stream, DateTime expiration)
|
|
|
|
|
|
{
|
|
|
|
|
|
stream.Seek(0, SeekOrigin.End);
|
|
|
|
|
|
stream.Write(Encoding.ASCII.GetBytes("EXPIRES:"), 0, 8);
|
|
|
|
|
|
stream.Write(BitConverter.GetBytes(expiration.Ticks), 0, 8);
|
|
|
|
|
|
|
|
|
|
|
|
Cache.Set(cacheKey, stream.ToArray(), new CacheItemPolicy { AbsoluteExpiration = expiration });
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|