// Copyright © 2018 Paddy Xu // // This file is part of QuickLook program. // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . using System; using System.Threading.Tasks; using System.Windows; using System.Windows.Media; using System.Windows.Media.Animation; using System.Windows.Media.Imaging; using QuickLook.Common.Helpers; namespace QuickLook.Plugin.ImageViewer.AnimatedImage { internal class NativeImageProvider : AnimationProvider { public NativeImageProvider(string path, NConvert meta) : base(path, meta) { Animator = new Int32AnimationUsingKeyFrames(); Animator.KeyFrames.Add(new DiscreteInt32KeyFrame(0, KeyTime.FromTimeSpan(TimeSpan.Zero))); } public override Task GetThumbnail(Size size, Size fullSize) { var decodeWidth = (int) Math.Round(fullSize.Width * Math.Min(size.Width / 2 / fullSize.Width, size.Height / 2 / fullSize.Height)); var decodeHeight = (int) Math.Round(fullSize.Height / fullSize.Width * decodeWidth); var orientation = Meta.GetOrientation(); var rotate = ShouldRotate(orientation); return new Task(() => { try { var img = new BitmapImage(); img.BeginInit(); img.UriSource = new Uri(Path); img.CacheOption = BitmapCacheOption.OnLoad; img.DecodePixelWidth = rotate ? decodeHeight : decodeWidth; img.DecodePixelHeight = rotate ? decodeWidth : decodeHeight; // specific size to avoid .net's double to int conversion img.EndInit(); var scaled = rotate ? new TransformedBitmap(img, new ScaleTransform(fullSize.Height / img.PixelWidth, fullSize.Width / img.PixelHeight)) : new TransformedBitmap(img, new ScaleTransform(fullSize.Width / img.PixelWidth, fullSize.Height / img.PixelHeight)); var rotated = ApplyTransformFromExif(scaled, orientation); rotated.Freeze(); return rotated; } catch (Exception e) { ProcessHelper.WriteLog(e.ToString()); return null; } }); } public override Task GetRenderedFrame(int index) { return new Task(() => { try { var img = new BitmapImage(); img.BeginInit(); img.UriSource = new Uri(Path); img.CacheOption = BitmapCacheOption.OnLoad; img.EndInit(); var img2 = ApplyTransformFromExif(img, Meta.GetOrientation()); img2.Freeze(); return img2; } catch (Exception e) { ProcessHelper.WriteLog(e.ToString()); return null; } }); } private static bool ShouldRotate(Orientation orientation) { bool rotate = false; switch (orientation) { case Orientation.LeftTop: case Orientation.RightTop: case Orientation.RightBottom: case Orientation.Leftbottom: rotate = true; break; } return rotate; } private static BitmapSource ApplyTransformFromExif(BitmapSource image, Orientation orientation) { switch (orientation) { case Orientation.Undefined: case Orientation.TopLeft: return image; case Orientation.TopRight: return new TransformedBitmap(image, new ScaleTransform(-1, 1, 0, 0)); case Orientation.BottomRight: return new TransformedBitmap(image, new RotateTransform(180)); case Orientation.BottomLeft: return new TransformedBitmap(image, new ScaleTransform(1, 1, 0, 0)); case Orientation.LeftTop: return new TransformedBitmap( new TransformedBitmap(image, new RotateTransform(90)), new ScaleTransform(-1, 1, 0, 0)); case Orientation.RightTop: return new TransformedBitmap(image, new RotateTransform(90)); case Orientation.RightBottom: return new TransformedBitmap( new TransformedBitmap(image, new RotateTransform(270)), new ScaleTransform(-1, 1, 0, 0)); case Orientation.Leftbottom: return new TransformedBitmap(image, new RotateTransform(270)); } return image; } public override void Dispose() { } } }