mirror of
https://github.com/QL-Win/QuickLook.git
synced 2025-09-11 17:59:17 +00:00
273 lines
8.6 KiB
C#
273 lines
8.6 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.DelayedReRender += 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.DelayedReRender -= 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;
|
|
|
|
BeginLoadThumbnails();
|
|
|
|
if (PdfDocumentWrapper.PdfDocument.PageCount < 2)
|
|
listThumbnails.Visibility = Visibility.Collapsed;
|
|
}
|
|
|
|
public void LoadPdf(MemoryStream stream)
|
|
{
|
|
stream.Position = 0;
|
|
PdfDocumentWrapper = new PdfDocumentWrapper(stream);
|
|
_pdfLoaded = true;
|
|
|
|
BeginLoadThumbnails();
|
|
|
|
if (PdfDocumentWrapper.PdfDocument.PageCount < 2)
|
|
listThumbnails.Visibility = Visibility.Collapsed;
|
|
}
|
|
|
|
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));
|
|
}
|
|
}
|
|
} |