Files
QuickLook/QuickLook.Plugin/QuickLook.Plugin.PDFViewer/PdfViewerControl.xaml.cs
2020-08-23 17:14:02 +02:00

273 lines
8.5 KiB
C#

// 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 <http://www.gnu.org/licenses/>.
using System;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Diagnostics;
using System.IO;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Windows;
using System.Windows.Controls;
using PdfiumViewer;
using QuickLook.Common.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();
// remove theme used in designer
Resources.MergedDictionaries.RemoveAt(0);
listThumbnails.SelectionChanged += UpdatePageViewWhenSelectionChanged;
pagePanel.ZoomChanged += ReRenderCurrentPageDelayed;
pagePanel.ImageScrolled += NavigatePage;
}
public ObservableCollection<int> PageThumbnails { get; set; } = new ObservableCollection<int>();
public int TotalPages => PdfDocumentWrapper.PdfDocument.PageCount;
public int CurrentPage
{
get => listThumbnails.SelectedIndex;
set
{
listThumbnails.SelectedIndex = value;
listThumbnails.ScrollIntoView(listThumbnails.SelectedItem);
CurrentPageChanged?.Invoke(this, new EventArgs());
OnPropertyChanged();
}
}
public PdfDocumentWrapper PdfDocumentWrapper { get; private set; }
public void Dispose()
{
GC.SuppressFinalize(this);
if (pagePanel != null)
{
pagePanel.ZoomChanged -= ReRenderCurrentPageDelayed;
pagePanel.ImageScrolled -= NavigatePage;
}
_pdfLoaded = false;
PdfDocumentWrapper?.Dispose();
PdfDocumentWrapper = 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;
if (CurrentPage < 0 || CurrentPage >= TotalPages)
return;
Debug.WriteLine($"Re-rendering 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 / PdfDocumentWrapper.PdfDocument.PageSizes[CurrentPage].Height,
pagePanel.ActualWidth / PdfDocumentWrapper.PdfDocument.PageSizes[CurrentPage].Width);
_viewRenderFactor = factor;
_minZoomFactor = 0.1 * factor;
_maxZoomFactor = 5 * factor;
}
else if (pagePanel.ZoomToFit)
{
factor = Math.Min(pagePanel.ActualHeight / PdfDocumentWrapper.PdfDocument.PageSizes[CurrentPage].Height,
pagePanel.ActualWidth / PdfDocumentWrapper.PdfDocument.PageSizes[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 = PdfDocumentWrapper.Render(CurrentPage, 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 s = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
{
using (var tempHandle = PdfDocument.Load(s))
{
size = new Size(0, 0);
tempHandle.PageSizes.Take(5).ForEach(p =>
{
size.Width = Math.Max(size.Width, p.Width);
size.Height = Math.Max(size.Height, p.Height);
});
if (tempHandle.PageCount > 1)
size.Width += /*listThumbnails.ActualWidth*/ 150;
}
}
return new Size(size.Width * 3, size.Height * 3);
}
public void LoadPdf(string path)
{
var stream = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
PdfDocumentWrapper = new PdfDocumentWrapper(stream);
_pdfLoaded = true;
if (PdfDocumentWrapper.PdfDocument.PageCount < 2)
listThumbnails.Visibility = Visibility.Collapsed;
BeginLoadThumbnails();
}
public void LoadPdf(MemoryStream stream)
{
stream.Position = 0;
PdfDocumentWrapper = new PdfDocumentWrapper(stream);
_pdfLoaded = true;
if (PdfDocumentWrapper.PdfDocument.PageCount < 2)
listThumbnails.Visibility = Visibility.Collapsed;
BeginLoadThumbnails();
}
private void BeginLoadThumbnails()
{
Enumerable.Range(0, PdfDocumentWrapper.PdfDocument.PageCount).ForEach(PageThumbnails.Add);
}
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
}
}