PowerToys/Wox.Infrastructure/Image/ImageLoader.cs

155 lines
5.5 KiB
C#
Raw Normal View History

2014-07-14 19:03:52 +08:00
using System;
using System.Collections.Concurrent;
2014-07-14 19:03:52 +08:00
using System.IO;
using System.Linq;
using System.Threading.Tasks;
2014-07-14 19:03:52 +08:00
using System.Windows.Media;
using System.Windows.Media.Imaging;
using Wox.Infrastructure.Logger;
using Wox.Infrastructure.Storage;
2014-07-14 19:03:52 +08:00
namespace Wox.Infrastructure.Image
2014-07-14 19:03:52 +08:00
{
public static class ImageLoader
2014-07-14 19:03:52 +08:00
{
private static readonly ImageCache ImageCache = new ImageCache();
private static BinaryStorage<ConcurrentDictionary<string, int>> _storage;
2016-08-20 08:02:47 +08:00
2014-07-14 19:03:52 +08:00
private static readonly string[] ImageExtensions =
2014-07-14 19:03:52 +08:00
{
".png",
".jpg",
".jpeg",
".gif",
".bmp",
".tiff",
".ico"
};
public static void Initialize()
{
_storage = new BinaryStorage<ConcurrentDictionary<string, int>> ("Image");
ImageCache.Usage = _storage.TryLoad(new ConcurrentDictionary<string, int>());
foreach (var icon in new[] { Constant.DefaultIcon, Constant.ErrorIcon })
{
ImageSource img = new BitmapImage(new Uri(icon));
img.Freeze();
ImageCache[icon] = img;
}
Task.Run(() =>
{
Stopwatch.Normal("|ImageLoader.Initialize|Preload images cost", () =>
{
ImageCache.Usage.AsParallel().Where(i => !ImageCache.ContainsKey(i.Key)).ForAll(i =>
{
var img = Load(i.Key);
if (img != null)
{
ImageCache[i.Key] = img;
}
});
});
Log.Info($"|ImageLoader.Initialize|Number of preload images is <{ImageCache.Usage.Count}>");
});
}
public static void Save()
{
ImageCache.Cleanup();
_storage.Save(ImageCache.Usage);
}
public static ImageSource Load(string path, bool loadFullImage = false)
2014-07-14 19:03:52 +08:00
{
ImageSource image;
2015-01-15 20:47:48 +08:00
try
2014-07-14 19:03:52 +08:00
{
if (string.IsNullOrEmpty(path))
2015-01-15 20:47:48 +08:00
{
return ImageCache[Constant.ErrorIcon];
2016-05-04 06:21:03 +08:00
}
if (ImageCache.ContainsKey(path))
2016-05-04 06:21:03 +08:00
{
return ImageCache[path];
2015-01-15 20:47:48 +08:00
}
if (path.StartsWith("data:", StringComparison.OrdinalIgnoreCase))
2015-02-04 23:16:41 +08:00
{
return new BitmapImage(new Uri(path));
2015-02-04 23:16:41 +08:00
}
if (!Path.IsPathRooted(path))
2015-02-04 23:16:41 +08:00
{
path = Path.Combine(Constant.ProgramDirectory, "Images", Path.GetFileName(path));
}
if (Directory.Exists(path))
{
/* Directories can also have thumbnails instead of shell icons.
* Generating thumbnails for a bunch of folders while scrolling through
* results from Everything makes a big impact on performance and
* Wox responsibility.
* - Solution: just load the icon
*/
image = WindowsThumbnailProvider.GetThumbnail(path, Constant.ThumbnailSize,
Constant.ThumbnailSize, ThumbnailOptions.IconOnly);
}
else if (File.Exists(path))
{
var extension = Path.GetExtension(path).ToLower();
if (ImageExtensions.Contains(extension))
2015-11-02 08:04:05 +08:00
{
if (loadFullImage)
2016-05-04 06:21:03 +08:00
{
image = LoadFullImage(path);
2016-05-04 06:21:03 +08:00
}
else
{
/* Although the documentation for GetImage on MSDN indicates that
* if a thumbnail is available it will return one, this has proved to not
* be the case in many situations while testing.
* - Solution: explicitly pass the ThumbnailOnly flag
*/
image = WindowsThumbnailProvider.GetThumbnail(path, Constant.ThumbnailSize,
Constant.ThumbnailSize, ThumbnailOptions.ThumbnailOnly);
2016-05-04 06:21:03 +08:00
}
2015-11-02 08:04:05 +08:00
}
else
2015-11-02 08:04:05 +08:00
{
image = WindowsThumbnailProvider.GetThumbnail(path, Constant.ThumbnailSize,
Constant.ThumbnailSize, ThumbnailOptions.None);
2015-11-02 08:04:05 +08:00
}
}
else
{
image = ImageCache[Constant.ErrorIcon];
path = Constant.ErrorIcon;
2014-07-14 19:03:52 +08:00
}
ImageCache[path] = image;
image.Freeze();
}
catch (System.Exception e)
{
Log.Exception($"|ImageLoader.Load|Failed to get thumbnail for {path}", e);
image = ImageCache[Constant.ErrorIcon];
ImageCache[path] = image;
}
return image;
2014-07-14 19:03:52 +08:00
}
private static BitmapImage LoadFullImage(string path)
{
BitmapImage image = new BitmapImage();
image.BeginInit();
image.CacheOption = BitmapCacheOption.OnLoad;
image.UriSource = new Uri(path);
image.EndInit();
return image;
}
}
2014-07-14 19:03:52 +08:00
}