Files
QuickLook/QuickLook.Plugin/QuickLook.Plugin.PDFViewer/PdfViewerControl.xaml.cs
2018-01-24 19:11:49 +02:00

266 lines
8.0 KiB
C#

// Copyright © 2017 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 <http://www.gnu.org/licenses/>.
using System;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Diagnostics;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Media.Imaging;
using PDFiumSharp;
using QuickLook.ExtensionMethods;
namespace QuickLook.Plugin.PDFViewer
{
/// <summary>
/// Interaction logic for PdfViewer.xaml
/// </summary>
public partial class PdfViewerControl : UserControl, INotifyPropertyChanged, IDisposable
{
private int _changePageDeltaSum;
private bool _initPage = true;
private double _maxZoomFactor = double.NaN;
private double _minZoomFactor = double.NaN;
private bool _pdfLoaded;
private double _viewRenderFactor = double.NaN;
public PdfViewerControl()
{
InitializeComponent();
listThumbnails.SelectionChanged += UpdatePageViewWhenSelectionChanged;
pagePanel.DelayedReRender += ReRenderCurrentPageDelayed;
pagePanel.ImageScrolled += NavigatePage;
}
public ObservableCollection<BitmapSource> PageThumbnails { get; set; } =
new ObservableCollection<BitmapSource>();
public PdfDocument PdfHandle { get; private set; }
public int TotalPages => PdfHandle.Pages.Count;
public int CurrentPage
{
get => listThumbnails.SelectedIndex;
set
{
listThumbnails.SelectedIndex = value;
listThumbnails.ScrollIntoView(listThumbnails.SelectedItem);
CurrentPageChanged?.Invoke(this, new EventArgs());
OnPropertyChanged();
}
}
public void Dispose()
{
GC.SuppressFinalize(this);
if (pagePanel != null)
{
pagePanel.DelayedReRender -= ReRenderCurrentPageDelayed;
pagePanel.ImageScrolled -= NavigatePage;
}
_pdfLoaded = false;
PdfHandle?.Close();
PdfHandle = null;
}
public event PropertyChangedEventHandler PropertyChanged;
private void ReRenderCurrentPageDelayed(object sender, EventArgs e)
{
ReRenderCurrentPage();
}
public event EventHandler CurrentPageChanged;
private void NavigatePage(object sender, int delta)
{
if (!_pdfLoaded)
return;
var pos = pagePanel.GetScrollPosition();
var size = pagePanel.GetScrollSize();
const double tolerance = 0.0001d;
if (Math.Abs(pos.Y) < tolerance && delta > 0)
{
_changePageDeltaSum += delta;
if (Math.Abs(_changePageDeltaSum) < 20)
return;
PrevPage();
_changePageDeltaSum = 0;
}
else if (Math.Abs(pos.Y - size.Height) < tolerance && delta < -0)
{
_changePageDeltaSum += delta;
if (Math.Abs(_changePageDeltaSum) < 20)
return;
NextPage();
_changePageDeltaSum = 0;
}
}
private void NextPage()
{
if (CurrentPage < TotalPages - 1)
{
CurrentPage++;
pagePanel.ScrollToTop();
}
}
private void PrevPage()
{
if (CurrentPage > 0)
{
CurrentPage--;
pagePanel.ScrollToBottom();
}
}
private void ReRenderCurrentPage()
{
if (!_pdfLoaded)
return;
Debug.WriteLine($"Renrendering page {CurrentPage}");
var pos = pagePanel.GetScrollPosition();
double factor;
// First time showing. Set thresholds here.
if (double.IsNaN(_minZoomFactor) || double.IsNaN(_maxZoomFactor))
{
factor = Math.Min(pagePanel.ActualHeight / PdfHandle.Pages[CurrentPage].Height,
pagePanel.ActualWidth / PdfHandle.Pages[CurrentPage].Width);
_viewRenderFactor = factor;
_minZoomFactor = 0.1 * factor;
_maxZoomFactor = 5 * factor;
}
else if (pagePanel.ZoomToFit)
{
factor = Math.Min(pagePanel.ActualHeight / PdfHandle.Pages[CurrentPage].Height,
pagePanel.ActualWidth / PdfHandle.Pages[CurrentPage].Width);
}
else
{
factor = pagePanel.ZoomFactor * _viewRenderFactor;
factor = Math.Max(factor, _minZoomFactor);
factor = Math.Min(factor, _maxZoomFactor);
pagePanel.MinZoomFactor = _minZoomFactor / factor;
pagePanel.MaxZoomFactor = _maxZoomFactor / factor;
}
var image = PdfHandle.Pages[CurrentPage].Render(factor);
pagePanel.Source = image;
pagePanel.ResetZoom();
_viewRenderFactor = factor;
pagePanel.SetScrollPosition(pos);
Dispatcher.Delay(500, t => GC.Collect());
}
private void UpdatePageViewWhenSelectionChanged(object sender, SelectionChangedEventArgs e)
{
if (!_pdfLoaded)
return;
if (CurrentPage == -1)
return;
CurrentPageChanged?.Invoke(this, new EventArgs());
ReRenderCurrentPage();
if (_initPage)
{
_initPage = false;
pagePanel.DoZoomToFit();
}
}
public static Size GetDesiredControlSizeByFirstPage(string path)
{
Size size;
using (var tempHandle = new PdfDocument(path))
{
size = new Size(0, 0);
tempHandle.Pages.Take(5).ForEach(p =>
{
size.Width = Math.Max(size.Width, p.Width);
size.Height = Math.Max(size.Height, p.Height);
});
if (tempHandle.Pages.Count > 1)
size.Width += /*listThumbnails.ActualWidth*/ 150;
tempHandle.Close();
}
return new Size(size.Width * 3, size.Height * 3);
}
public void LoadPdf(string path)
{
PdfHandle = new PdfDocument(path);
_pdfLoaded = true;
BeginLoadThumbnails(path);
if (PdfHandle.Pages.Count < 2)
listThumbnails.Visibility = Visibility.Collapsed;
}
private void BeginLoadThumbnails(string path, string password = null)
{
new Task(() =>
{
using (var handle = new PdfDocument(path, password))
{
handle.Pages.ForEach(p =>
{
var bs = p.RenderThumbnail();
Dispatcher.BeginInvoke(new Action(() => PageThumbnails.Add(bs)));
});
handle.Close();
}
}).Start();
}
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
}
}