mirror of
https://github.com/QL-Win/QuickLook.git
synced 2025-09-12 02:09:05 +00:00
278 lines
8.7 KiB
C#
278 lines
8.7 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.Threading.Tasks;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Media.Imaging;
|
|
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 PdfDocument _pdfHandle;
|
|
private bool _pdfLoaded;
|
|
private Stream _pdfStream;
|
|
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<BitmapSource> PageThumbnails { get; set; } =
|
|
new ObservableCollection<BitmapSource>();
|
|
|
|
public int TotalPages => _pdfHandle.PageCount;
|
|
|
|
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?.Dispose();
|
|
_pdfHandle = null;
|
|
_pdfStream.Close();
|
|
}
|
|
|
|
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.PageSizes[CurrentPage].Height,
|
|
pagePanel.ActualWidth / _pdfHandle.PageSizes[CurrentPage].Width);
|
|
_viewRenderFactor = factor;
|
|
_minZoomFactor = 0.1 * factor;
|
|
_maxZoomFactor = 5 * factor;
|
|
}
|
|
else if (pagePanel.ZoomToFit)
|
|
{
|
|
factor = Math.Min(pagePanel.ActualHeight / _pdfHandle.PageSizes[CurrentPage].Height,
|
|
pagePanel.ActualWidth / _pdfHandle.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 = _pdfHandle.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)
|
|
{
|
|
_pdfStream = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
|
|
_pdfHandle = PdfDocument.Load(_pdfStream);
|
|
_pdfLoaded = true;
|
|
|
|
BeginLoadThumbnails(path);
|
|
|
|
if (_pdfHandle.PageCount < 2)
|
|
listThumbnails.Visibility = Visibility.Collapsed;
|
|
}
|
|
|
|
private void BeginLoadThumbnails(string path, string password = null)
|
|
{
|
|
new Task(() =>
|
|
{
|
|
using (var stream = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
|
|
{
|
|
using (var handle = PdfDocument.Load(stream, password))
|
|
{
|
|
for (var p = 0; p < handle.PageCount; p++)
|
|
{
|
|
var bs = handle.RenderThumbnail(p);
|
|
|
|
Dispatcher.BeginInvoke(new Action(() => PageThumbnails.Add(bs)));
|
|
}
|
|
|
|
handle.Dispose();
|
|
}
|
|
}
|
|
}).Start();
|
|
}
|
|
|
|
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
|
|
{
|
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
|
}
|
|
}
|
|
} |