2014-07-14 19:03:52 +08:00
|
|
|
|
using System;
|
2016-04-22 06:37:40 +08:00
|
|
|
|
using System.Collections.Concurrent;
|
2014-07-14 19:03:52 +08:00
|
|
|
|
using System.IO;
|
2016-04-26 08:20:10 +08:00
|
|
|
|
using System.Linq;
|
2016-05-04 04:18:26 +08:00
|
|
|
|
using System.Threading.Tasks;
|
2014-07-14 19:03:52 +08:00
|
|
|
|
using System.Windows.Media;
|
|
|
|
|
using System.Windows.Media.Imaging;
|
2016-05-03 05:37:01 +08:00
|
|
|
|
using Wox.Infrastructure.Logger;
|
2016-04-21 08:53:21 +08:00
|
|
|
|
using Wox.Infrastructure.Storage;
|
2014-07-14 19:03:52 +08:00
|
|
|
|
|
2016-04-26 08:20:10 +08:00
|
|
|
|
namespace Wox.Infrastructure.Image
|
2014-07-14 19:03:52 +08:00
|
|
|
|
{
|
2016-04-26 09:40:23 +08:00
|
|
|
|
public static class ImageLoader
|
2014-07-14 19:03:52 +08:00
|
|
|
|
{
|
2017-01-13 23:40:32 +08:00
|
|
|
|
private static readonly ImageCache ImageCache = new ImageCache();
|
2017-02-13 00:57:24 +08:00
|
|
|
|
private static BinaryStorage<ConcurrentDictionary<string, int>> _storage;
|
2016-08-20 08:02:47 +08:00
|
|
|
|
|
2014-07-14 19:03:52 +08:00
|
|
|
|
|
2018-03-31 15:19:55 +08:00
|
|
|
|
private static readonly string[] ImageExtensions =
|
2014-07-14 19:03:52 +08:00
|
|
|
|
{
|
|
|
|
|
".png",
|
|
|
|
|
".jpg",
|
|
|
|
|
".jpeg",
|
|
|
|
|
".gif",
|
|
|
|
|
".bmp",
|
|
|
|
|
".tiff",
|
|
|
|
|
".ico"
|
|
|
|
|
};
|
|
|
|
|
|
2016-03-28 10:09:57 +08:00
|
|
|
|
|
2017-02-13 00:57:24 +08:00
|
|
|
|
public static void Initialize()
|
2016-03-28 10:09:57 +08:00
|
|
|
|
{
|
2017-02-13 00:57:24 +08:00
|
|
|
|
_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}>");
|
|
|
|
|
});
|
2016-04-21 08:53:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
2016-04-26 09:40:23 +08:00
|
|
|
|
public static void Save()
|
2016-04-21 08:53:21 +08:00
|
|
|
|
{
|
2017-01-13 23:40:32 +08:00
|
|
|
|
ImageCache.Cleanup();
|
2017-02-13 00:57:24 +08:00
|
|
|
|
_storage.Save(ImageCache.Usage);
|
2016-03-28 10:09:57 +08:00
|
|
|
|
}
|
2018-03-31 15:19:55 +08:00
|
|
|
|
|
|
|
|
|
public static ImageSource Load(string path, bool loadFullImage = false)
|
2014-07-14 19:03:52 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
ImageSource image;
|
2015-01-15 20:47:48 +08:00
|
|
|
|
try
|
2014-07-14 19:03:52 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
if (string.IsNullOrEmpty(path))
|
2015-01-15 20:47:48 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
return ImageCache[Constant.ErrorIcon];
|
2016-05-04 06:21:03 +08:00
|
|
|
|
}
|
2018-03-31 15:19:55 +08:00
|
|
|
|
if (ImageCache.ContainsKey(path))
|
2016-05-04 06:21:03 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
return ImageCache[path];
|
2015-01-15 20:47:48 +08:00
|
|
|
|
}
|
2018-03-31 15:19:55 +08:00
|
|
|
|
|
2016-05-04 04:18:26 +08:00
|
|
|
|
if (path.StartsWith("data:", StringComparison.OrdinalIgnoreCase))
|
2015-02-04 23:16:41 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
return new BitmapImage(new Uri(path));
|
2015-02-04 23:16:41 +08:00
|
|
|
|
}
|
2018-03-31 15:19:55 +08:00
|
|
|
|
|
|
|
|
|
if (!Path.IsPathRooted(path))
|
2015-02-04 23:16:41 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +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
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
if (loadFullImage)
|
2016-05-04 06:21:03 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
image = LoadFullImage(path);
|
2016-05-04 06:21:03 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
/* 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
|
|
|
|
}
|
2016-05-04 04:18:26 +08:00
|
|
|
|
else
|
2015-11-02 08:04:05 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
image = WindowsThumbnailProvider.GetThumbnail(path, Constant.ThumbnailSize,
|
|
|
|
|
Constant.ThumbnailSize, ThumbnailOptions.None);
|
2015-11-02 08:04:05 +08:00
|
|
|
|
}
|
2016-05-04 04:18:26 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
image = ImageCache[Constant.ErrorIcon];
|
|
|
|
|
path = Constant.ErrorIcon;
|
2014-07-14 19:03:52 +08:00
|
|
|
|
}
|
2017-01-13 23:40:32 +08:00
|
|
|
|
ImageCache[path] = image;
|
2016-05-22 12:30:38 +08:00
|
|
|
|
image.Freeze();
|
2016-05-04 04:18:26 +08:00
|
|
|
|
}
|
2018-03-31 15:19:55 +08:00
|
|
|
|
catch (System.Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Exception($"|ImageLoader.Load|Failed to get thumbnail for {path}", e);
|
|
|
|
|
|
|
|
|
|
image = ImageCache[Constant.ErrorIcon];
|
|
|
|
|
ImageCache[path] = image;
|
|
|
|
|
}
|
2016-04-26 06:20:16 +08:00
|
|
|
|
return image;
|
2014-07-14 19:03:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-03-31 15:19:55 +08:00
|
|
|
|
private static BitmapImage LoadFullImage(string path)
|
2016-08-20 20:10:33 +08:00
|
|
|
|
{
|
2018-03-31 15:19:55 +08:00
|
|
|
|
BitmapImage image = new BitmapImage();
|
|
|
|
|
image.BeginInit();
|
|
|
|
|
image.CacheOption = BitmapCacheOption.OnLoad;
|
|
|
|
|
image.UriSource = new Uri(path);
|
|
|
|
|
image.EndInit();
|
|
|
|
|
return image;
|
2016-08-20 20:10:33 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-07-14 19:03:52 +08:00
|
|
|
|
}
|