mirror of
https://github.com/QL-Win/QuickLook.git
synced 2025-09-04 03:36:04 +00:00
Compare commits
4 Commits
417876edd2
...
5cedcff912
Author | SHA1 | Date | |
---|---|---|---|
![]() |
5cedcff912 | ||
![]() |
9fe37520d3 | ||
![]() |
67b5dbf310 | ||
![]() |
efe47ff43b |
@@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
using LibAPNG;
|
using LibAPNG;
|
||||||
using QuickLook.Common.ExtensionMethods;
|
using QuickLook.Common.ExtensionMethods;
|
||||||
|
using QuickLook.Common.Helpers;
|
||||||
using QuickLook.Common.Plugin;
|
using QuickLook.Common.Plugin;
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
@@ -30,20 +31,28 @@ using System.Windows.Media.Imaging;
|
|||||||
|
|
||||||
namespace QuickLook.Plugin.ImageViewer.AnimatedImage.Providers;
|
namespace QuickLook.Plugin.ImageViewer.AnimatedImage.Providers;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This provider is only for Animated PNG.
|
||||||
|
/// The others will fall back to another provider.
|
||||||
|
/// </summary>
|
||||||
internal class APngProvider : AnimationProvider
|
internal class APngProvider : AnimationProvider
|
||||||
{
|
{
|
||||||
private readonly Frame _baseFrame;
|
private readonly Frame _baseFrame;
|
||||||
private readonly List<FrameInfo> _frames;
|
private readonly List<FrameInfo> _frames;
|
||||||
private readonly List<BitmapSource> _renderedFrames;
|
private readonly List<BitmapSource> _renderedFrames;
|
||||||
private int _lastEffectivePreviousPreviousFrameIndex;
|
private int _lastEffectivePreviousPreviousFrameIndex;
|
||||||
private NativeProvider _nativeImageProvider;
|
private AnimationProvider _fallbackImageProvider;
|
||||||
|
|
||||||
public APngProvider(Uri path, MetaProvider meta, ContextObject contextObject) : base(path, meta, contextObject)
|
public APngProvider(Uri path, MetaProvider meta, ContextObject contextObject) : base(path, meta, contextObject)
|
||||||
{
|
{
|
||||||
if (!IsAnimatedPng(path.LocalPath))
|
if (!IsAnimatedPng(path.LocalPath))
|
||||||
{
|
{
|
||||||
_nativeImageProvider = new NativeProvider(path, meta, contextObject);
|
var useNativeProvider = SettingHelper.Get("UseNativeProvider", true, "QuickLook.Plugin.ImageViewer");
|
||||||
Animator = _nativeImageProvider.Animator;
|
|
||||||
|
_fallbackImageProvider = useNativeProvider ?
|
||||||
|
new NativeProvider(path, meta, contextObject) :
|
||||||
|
new ImageMagickProvider(path, meta, contextObject);
|
||||||
|
Animator = _fallbackImageProvider.Animator;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -71,8 +80,8 @@ internal class APngProvider : AnimationProvider
|
|||||||
|
|
||||||
public override Task<BitmapSource> GetThumbnail(Size renderSize)
|
public override Task<BitmapSource> GetThumbnail(Size renderSize)
|
||||||
{
|
{
|
||||||
if (_nativeImageProvider != null)
|
if (_fallbackImageProvider != null)
|
||||||
return _nativeImageProvider.GetThumbnail(renderSize);
|
return _fallbackImageProvider.GetThumbnail(renderSize);
|
||||||
|
|
||||||
return new Task<BitmapSource>(() =>
|
return new Task<BitmapSource>(() =>
|
||||||
{
|
{
|
||||||
@@ -85,8 +94,8 @@ internal class APngProvider : AnimationProvider
|
|||||||
|
|
||||||
public override Task<BitmapSource> GetRenderedFrame(int index)
|
public override Task<BitmapSource> GetRenderedFrame(int index)
|
||||||
{
|
{
|
||||||
if (_nativeImageProvider != null)
|
if (_fallbackImageProvider != null)
|
||||||
return _nativeImageProvider.GetRenderedFrame(index);
|
return _fallbackImageProvider.GetRenderedFrame(index);
|
||||||
|
|
||||||
if (_renderedFrames[index] != null)
|
if (_renderedFrames[index] != null)
|
||||||
return new Task<BitmapSource>(() => _renderedFrames[index]);
|
return new Task<BitmapSource>(() => _renderedFrames[index]);
|
||||||
@@ -102,10 +111,10 @@ internal class APngProvider : AnimationProvider
|
|||||||
|
|
||||||
public override void Dispose()
|
public override void Dispose()
|
||||||
{
|
{
|
||||||
if (_nativeImageProvider != null)
|
if (_fallbackImageProvider != null)
|
||||||
{
|
{
|
||||||
_nativeImageProvider.Dispose();
|
_fallbackImageProvider.Dispose();
|
||||||
_nativeImageProvider = null;
|
_fallbackImageProvider = null;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -214,7 +223,7 @@ internal class APngProvider : AnimationProvider
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint ToUInt32BE(byte[] data)
|
static uint ToUInt32BE(byte[] data)
|
||||||
{
|
{
|
||||||
Array.Reverse(data);
|
Array.Reverse(data);
|
||||||
return BitConverter.ToUInt32(data, 0);
|
return BitConverter.ToUInt32(data, 0);
|
||||||
|
@@ -40,11 +40,6 @@ internal class NativeProvider : AnimationProvider
|
|||||||
public override Task<BitmapSource> GetThumbnail(Size renderSize)
|
public override Task<BitmapSource> GetThumbnail(Size renderSize)
|
||||||
{
|
{
|
||||||
var fullSize = Meta.GetSize();
|
var fullSize = Meta.GetSize();
|
||||||
|
|
||||||
//var decodeWidth = (int) Math.Round(fullSize.Width *
|
|
||||||
// Math.Min(renderSize.Width / 2 / fullSize.Width,
|
|
||||||
// renderSize.Height / 2 / fullSize.Height));
|
|
||||||
//var decodeHeight = (int) Math.Round(fullSize.Height / fullSize.Width * decodeWidth);
|
|
||||||
var decodeWidth =
|
var decodeWidth =
|
||||||
(int)Math.Round(Math.Min(Meta.GetSize().Width, Math.Max(1d, Math.Floor(renderSize.Width))));
|
(int)Math.Round(Math.Min(Meta.GetSize().Width, Math.Max(1d, Math.Floor(renderSize.Width))));
|
||||||
var decodeHeight =
|
var decodeHeight =
|
||||||
|
@@ -1,104 +0,0 @@
|
|||||||
// Copyright © 2017-2025 QL-Win Contributors
|
|
||||||
//
|
|
||||||
// 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 QuickLook.Common.Helpers;
|
|
||||||
using QuickLook.Common.Plugin;
|
|
||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using System.Windows;
|
|
||||||
using System.Windows.Media.Animation;
|
|
||||||
using System.Windows.Media.Imaging;
|
|
||||||
using System.Xml.Linq;
|
|
||||||
|
|
||||||
namespace QuickLook.Plugin.ImageViewer.AnimatedImage.Providers;
|
|
||||||
|
|
||||||
internal class PdnProvider : AnimationProvider
|
|
||||||
{
|
|
||||||
public PdnProvider(Uri path, MetaProvider meta, ContextObject contextObject) : base(path, meta, contextObject)
|
|
||||||
{
|
|
||||||
Animator = new Int32AnimationUsingKeyFrames();
|
|
||||||
Animator.KeyFrames.Add(new DiscreteInt32KeyFrame(0,
|
|
||||||
KeyTime.FromTimeSpan(TimeSpan.Zero)));
|
|
||||||
}
|
|
||||||
|
|
||||||
public override Task<BitmapSource> GetThumbnail(Size renderSize)
|
|
||||||
{
|
|
||||||
// Skip thumbnail
|
|
||||||
return new Task<BitmapSource>(() => null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override Task<BitmapSource> GetRenderedFrame(int index)
|
|
||||||
{
|
|
||||||
return new Task<BitmapSource>(() =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
using TextReader reader = new StreamReader(Path.LocalPath, Encoding.UTF8);
|
|
||||||
string line = reader.ReadLine();
|
|
||||||
|
|
||||||
if (!line.StartsWith("PDN"))
|
|
||||||
return null;
|
|
||||||
|
|
||||||
int indexOfStart = line.IndexOf("<");
|
|
||||||
int indexOfEnd = line.LastIndexOf(">");
|
|
||||||
|
|
||||||
if (indexOfStart < 0 || indexOfEnd < 0)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
string xml = line.Substring(indexOfStart, indexOfEnd - indexOfStart + 1);
|
|
||||||
|
|
||||||
// <pdnImage>
|
|
||||||
// <custom>
|
|
||||||
// <thumb png="..." />
|
|
||||||
// </custom>
|
|
||||||
// </pdnImage>
|
|
||||||
XDocument doc = XDocument.Parse(xml);
|
|
||||||
var pngBase64 = doc.Root
|
|
||||||
?.Element("custom")
|
|
||||||
?.Element("thumb")
|
|
||||||
?.Attribute("png")
|
|
||||||
?.Value;
|
|
||||||
|
|
||||||
if (pngBase64 != null)
|
|
||||||
{
|
|
||||||
byte[] imageBytes = Convert.FromBase64String(pngBase64);
|
|
||||||
MemoryStream ms = new(imageBytes);
|
|
||||||
BitmapImage bitmap = new();
|
|
||||||
bitmap.BeginInit();
|
|
||||||
bitmap.CacheOption = BitmapCacheOption.OnLoad;
|
|
||||||
bitmap.StreamSource = ms;
|
|
||||||
bitmap.EndInit();
|
|
||||||
bitmap.Freeze();
|
|
||||||
return bitmap;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
ProcessHelper.WriteLog(e.ToString());
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void Dispose()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@@ -65,8 +65,16 @@ public class Plugin : IViewer
|
|||||||
|
|
||||||
public void Init()
|
public void Init()
|
||||||
{
|
{
|
||||||
|
// Option of UseColorProfile:
|
||||||
|
// Default is False (disable color profile conversion)
|
||||||
|
// Note that enabling this feature will slow down image previewing, especially on large images.
|
||||||
var useColorProfile = SettingHelper.Get("UseColorProfile", false, "QuickLook.Plugin.ImageViewer");
|
var useColorProfile = SettingHelper.Get("UseColorProfile", false, "QuickLook.Plugin.ImageViewer");
|
||||||
|
|
||||||
|
// Option of UseNativeProvider:
|
||||||
|
// Default is True (disable precise colors and choose faster response)
|
||||||
|
// Note that disabling this feature may slightly slow down image previewing but you can get precise colors.
|
||||||
|
var useNativeProvider = SettingHelper.Get("UseNativeProvider", true, "QuickLook.Plugin.ImageViewer");
|
||||||
|
|
||||||
AnimatedImage.AnimatedImage.Providers.Add(
|
AnimatedImage.AnimatedImage.Providers.Add(
|
||||||
new KeyValuePair<string[], Type>(
|
new KeyValuePair<string[], Type>(
|
||||||
useColorProfile ? [".apng"] : [".apng", ".png"],
|
useColorProfile ? [".apng"] : [".apng", ".png"],
|
||||||
@@ -76,7 +84,7 @@ public class Plugin : IViewer
|
|||||||
typeof(GifProvider)));
|
typeof(GifProvider)));
|
||||||
AnimatedImage.AnimatedImage.Providers.Add(
|
AnimatedImage.AnimatedImage.Providers.Add(
|
||||||
new KeyValuePair<string[], Type>(
|
new KeyValuePair<string[], Type>(
|
||||||
useColorProfile ? [] : [".bmp", ".jpg", ".jpeg", ".jfif", ".tif", ".tiff"],
|
useColorProfile ? [] : (useNativeProvider ? [".bmp", ".jpg", ".jpeg", ".jfif", ".tif", ".tiff"] : []),
|
||||||
typeof(NativeProvider)));
|
typeof(NativeProvider)));
|
||||||
AnimatedImage.AnimatedImage.Providers.Add(
|
AnimatedImage.AnimatedImage.Providers.Add(
|
||||||
new KeyValuePair<string[], Type>([".jxr"],
|
new KeyValuePair<string[], Type>([".jxr"],
|
||||||
|
2314
QuickLook.Plugin/QuickLook.Plugin.MarkdownViewer/Resources/js/mermaid.min.js
vendored
Normal file
2314
QuickLook.Plugin/QuickLook.Plugin.MarkdownViewer/Resources/js/mermaid.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
@@ -19,6 +19,7 @@
|
|||||||
<script src="highlight.js/highlight.min.js"></script>
|
<script src="highlight.js/highlight.min.js"></script>
|
||||||
|
|
||||||
<script src="js/markdownItAnchor.umd.js"></script>
|
<script src="js/markdownItAnchor.umd.js"></script>
|
||||||
|
<script src="js/mermaid.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body class="markdown-body">
|
<body class="markdown-body">
|
||||||
<link rel="stylesheet" href="css/github-markdown.css" />
|
<link rel="stylesheet" href="css/github-markdown.css" />
|
||||||
@@ -172,6 +173,33 @@
|
|||||||
padding-right: calc(1em - 2px);
|
padding-right: calc(1em - 2px);
|
||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Mermaid diagram styles */
|
||||||
|
.mermaid {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
margin: 1em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mermaid svg {
|
||||||
|
max-width: 100%;
|
||||||
|
height: auto;
|
||||||
|
border-radius: 6px;
|
||||||
|
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.mermaid-placeholder {
|
||||||
|
border: 2px dashed var(--borderColor-default);
|
||||||
|
background: var(--bgColor-muted);
|
||||||
|
color: var(--fgColor-muted);
|
||||||
|
border-radius: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mermaid-placeholder pre {
|
||||||
|
background: var(--bgColor-default);
|
||||||
|
border: 1px solid var(--borderColor-default);
|
||||||
|
color: var(--fgColor-default);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<textarea id="text-input" style="display: none">{{content}}</textarea>
|
<textarea id="text-input" style="display: none">{{content}}</textarea>
|
||||||
@@ -188,6 +216,9 @@
|
|||||||
typographer: false,
|
typographer: false,
|
||||||
quotes: "“”‘’",
|
quotes: "“”‘’",
|
||||||
highlight: function (str, lang) {
|
highlight: function (str, lang) {
|
||||||
|
if (lang === 'mermaid') {
|
||||||
|
return '<div class="mermaid">' + str + '</div>';
|
||||||
|
}
|
||||||
if (lang && hljs.getLanguage(lang)) {
|
if (lang && hljs.getLanguage(lang)) {
|
||||||
try {
|
try {
|
||||||
return (
|
return (
|
||||||
@@ -218,6 +249,62 @@
|
|||||||
document.getElementById("text-input").value
|
document.getElementById("text-input").value
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Initialize Mermaid after markdown rendering
|
||||||
|
if (window.mermaid) {
|
||||||
|
// Determine theme based on system preference
|
||||||
|
const isDarkMode = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
||||||
|
|
||||||
|
mermaid.initialize({
|
||||||
|
startOnLoad: false,
|
||||||
|
theme: isDarkMode ? 'dark' : 'default',
|
||||||
|
securityLevel: 'loose', // Allow HTML in diagrams for enhanced features
|
||||||
|
fontFamily: 'Segoe UI, Helvetica, Arial, sans-serif',
|
||||||
|
themeVariables: {
|
||||||
|
primaryColor: isDarkMode ? '#58a6ff' : '#0969da',
|
||||||
|
primaryTextColor: isDarkMode ? '#f0f6fc' : '#24292f',
|
||||||
|
primaryBorderColor: isDarkMode ? '#30363d' : '#d0d7de',
|
||||||
|
lineColor: isDarkMode ? '#484f58' : '#656d76',
|
||||||
|
sectionBkgColor: isDarkMode ? '#21262d' : '#f6f8fa',
|
||||||
|
altSectionBkgColor: isDarkMode ? '#161b22' : '#ffffff',
|
||||||
|
gridColor: isDarkMode ? '#21262d' : '#f6f8fa',
|
||||||
|
cScale0: isDarkMode ? '#58a6ff' : '#0969da',
|
||||||
|
cScale1: isDarkMode ? '#a5f3fc' : '#54aeff',
|
||||||
|
cScale2: isDarkMode ? '#ff7b72' : '#d1242f'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Render Mermaid diagrams with error handling
|
||||||
|
setTimeout(() => {
|
||||||
|
try {
|
||||||
|
mermaid.init(undefined, document.querySelectorAll('.mermaid'));
|
||||||
|
} catch (error) {
|
||||||
|
console.warn('Mermaid rendering error:', error);
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
|
||||||
|
// Listen for theme changes and re-render diagrams
|
||||||
|
window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', (e) => {
|
||||||
|
if (window.mermaid) {
|
||||||
|
const newTheme = e.matches ? 'dark' : 'default';
|
||||||
|
mermaid.initialize({
|
||||||
|
startOnLoad: false,
|
||||||
|
theme: newTheme,
|
||||||
|
securityLevel: 'loose',
|
||||||
|
fontFamily: 'Segoe UI, Helvetica, Arial, sans-serif'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Re-render all Mermaid diagrams
|
||||||
|
setTimeout(() => {
|
||||||
|
try {
|
||||||
|
mermaid.init(undefined, document.querySelectorAll('.mermaid'));
|
||||||
|
} catch (error) {
|
||||||
|
console.warn('Mermaid re-rendering error:', error);
|
||||||
|
}
|
||||||
|
}, 100);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/* codes below are adopted from https://codepen.io/jtojnar/full/Juiop */
|
/* codes below are adopted from https://codepen.io/jtojnar/full/Juiop */
|
||||||
var ToC = `<nav role='navigation' class='table-of-contents'>
|
var ToC = `<nav role='navigation' class='table-of-contents'>
|
||||||
<h2>Contents</h2>
|
<h2>Contents</h2>
|
||||||
|
@@ -21,6 +21,7 @@ using PureSharpCompress.Readers;
|
|||||||
using QuickLook.Common.Helpers;
|
using QuickLook.Common.Helpers;
|
||||||
using QuickLook.Common.Plugin;
|
using QuickLook.Common.Plugin;
|
||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Windows;
|
using System.Windows;
|
||||||
@@ -30,8 +31,18 @@ namespace QuickLook.Plugin.ThumbnailViewer;
|
|||||||
|
|
||||||
internal static class Handler
|
internal static class Handler
|
||||||
{
|
{
|
||||||
|
// List<Pair<formats, type>>
|
||||||
|
public static List<KeyValuePair<string[], Type>> Providers = [];
|
||||||
|
|
||||||
public static void Prepare(string path, ContextObject context)
|
public static void Prepare(string path, ContextObject context)
|
||||||
{
|
{
|
||||||
|
// Temporary codes
|
||||||
|
if (path.EndsWith(".pdn", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
new PdnProvider().Prepare(path, context);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
using Stream imageData = ViewImage(path);
|
using Stream imageData = ViewImage(path);
|
||||||
@@ -47,6 +58,12 @@ internal static class Handler
|
|||||||
|
|
||||||
public static Stream ViewImage(string path)
|
public static Stream ViewImage(string path)
|
||||||
{
|
{
|
||||||
|
// Temporary codes
|
||||||
|
if (path.EndsWith(".pdn", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return new PdnProvider().ViewImage(path);
|
||||||
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
using ZipArchive archive = ZipArchive.Open(path, new());
|
using ZipArchive archive = ZipArchive.Open(path, new());
|
||||||
|
@@ -34,6 +34,7 @@ public class Plugin : IViewer
|
|||||||
".cdr", // CorelDraw
|
".cdr", // CorelDraw
|
||||||
".fig", // Figma
|
".fig", // Figma
|
||||||
".kra", // Krita
|
".kra", // Krita
|
||||||
|
".pdn", // Paint.NET
|
||||||
".pip", ".pix", // Pixso
|
".pip", ".pix", // Pixso
|
||||||
".sketch", // Sketch
|
".sketch", // Sketch
|
||||||
".xd", // AdobeXD
|
".xd", // AdobeXD
|
||||||
@@ -46,6 +47,8 @@ public class Plugin : IViewer
|
|||||||
|
|
||||||
public void Init()
|
public void Init()
|
||||||
{
|
{
|
||||||
|
Handler.Providers.Add(new KeyValuePair<string[], Type>(
|
||||||
|
[".pdn"], typeof(PdnProvider)));
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool CanHandle(string path)
|
public bool CanHandle(string path)
|
||||||
|
@@ -0,0 +1,93 @@
|
|||||||
|
// Copyright © 2017-2025 QL-Win Contributors
|
||||||
|
//
|
||||||
|
// 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 QuickLook.Common.Helpers;
|
||||||
|
using QuickLook.Common.Plugin;
|
||||||
|
using QuickLook.Plugin.ImageViewer;
|
||||||
|
using System;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.IO;
|
||||||
|
using System.Text;
|
||||||
|
using System.Windows;
|
||||||
|
using System.Windows.Media.Imaging;
|
||||||
|
using System.Xml.Linq;
|
||||||
|
|
||||||
|
namespace QuickLook.Plugin.ThumbnailViewer;
|
||||||
|
|
||||||
|
internal class PdnProvider
|
||||||
|
{
|
||||||
|
public void Prepare(string path, ContextObject context)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using Stream imageData = ViewImage(path);
|
||||||
|
BitmapImage bitmap = imageData.ReadAsBitmapImage();
|
||||||
|
context.SetPreferredSizeFit(new Size(bitmap.PixelWidth, bitmap.PixelHeight), 0.8d);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Debug.WriteLine($"Error reading thumbnail from {path}: {ex.Message}");
|
||||||
|
context.PreferredSize = new Size { Width = 800, Height = 600 };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Stream ViewImage(string path)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using TextReader reader = new StreamReader(path, Encoding.UTF8);
|
||||||
|
string line = reader.ReadLine();
|
||||||
|
|
||||||
|
if (!line.StartsWith("PDN"))
|
||||||
|
return null;
|
||||||
|
|
||||||
|
int indexOfStart = line.IndexOf("<");
|
||||||
|
int indexOfEnd = line.LastIndexOf(">");
|
||||||
|
|
||||||
|
if (indexOfStart < 0 || indexOfEnd < 0)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
string xml = line.Substring(indexOfStart, indexOfEnd - indexOfStart + 1);
|
||||||
|
|
||||||
|
// <pdnImage>
|
||||||
|
// <custom>
|
||||||
|
// <thumb png="..." />
|
||||||
|
// </custom>
|
||||||
|
// </pdnImage>
|
||||||
|
XDocument doc = XDocument.Parse(xml);
|
||||||
|
var pngBase64 = doc.Root
|
||||||
|
?.Element("custom")
|
||||||
|
?.Element("thumb")
|
||||||
|
?.Attribute("png")
|
||||||
|
?.Value;
|
||||||
|
|
||||||
|
if (pngBase64 != null)
|
||||||
|
{
|
||||||
|
byte[] imageBytes = Convert.FromBase64String(pngBase64);
|
||||||
|
MemoryStream ms = new(imageBytes);
|
||||||
|
return ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
ProcessHelper.WriteLog(e.ToString());
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -418,9 +418,9 @@
|
|||||||
<MW_BrowseFolder>Przeglądaj {0}</MW_BrowseFolder>
|
<MW_BrowseFolder>Przeglądaj {0}</MW_BrowseFolder>
|
||||||
<MW_StayTop>Przypnij do ekranu</MW_StayTop>
|
<MW_StayTop>Przypnij do ekranu</MW_StayTop>
|
||||||
<MW_PreventClosing>Zablokuj zamykanie</MW_PreventClosing>
|
<MW_PreventClosing>Zablokuj zamykanie</MW_PreventClosing>
|
||||||
<MW_BrowseFolder>Przeglądaj {0}</MW_BrowseFolder>
|
|
||||||
<MW_Open>Otwórz {0}</MW_Open>
|
<MW_Open>Otwórz {0}</MW_Open>
|
||||||
<MW_OpenWith>Otwórz w {0}</MW_OpenWith>
|
<MW_OpenWith>Otwórz w {0}</MW_OpenWith>
|
||||||
|
<MW_OpenWithMenu>Otwórz za pomocą...</MW_OpenWithMenu>
|
||||||
<MW_Run>Uruchom {0}</MW_Run>
|
<MW_Run>Uruchom {0}</MW_Run>
|
||||||
<MW_Share>Udostępnij</MW_Share>
|
<MW_Share>Udostępnij</MW_Share>
|
||||||
<MW_Reload>Przeładuj</MW_Reload>
|
<MW_Reload>Przeładuj</MW_Reload>
|
||||||
@@ -441,6 +441,7 @@
|
|||||||
<InfoPanel_File>{0} plik</InfoPanel_File>
|
<InfoPanel_File>{0} plik</InfoPanel_File>
|
||||||
<InfoPanel_Files>{0} plików</InfoPanel_Files>
|
<InfoPanel_Files>{0} plików</InfoPanel_Files>
|
||||||
<InfoPanel_FolderAndFile>({0} i {1})</InfoPanel_FolderAndFile>
|
<InfoPanel_FolderAndFile>({0} i {1})</InfoPanel_FolderAndFile>
|
||||||
|
<InfoPanel_CantPreventClosing>Anulowanie dla funkcji blokady zamykania okna nie jest wspierane</InfoPanel_CantPreventClosing>
|
||||||
</pl>
|
</pl>
|
||||||
<pt-BR>
|
<pt-BR>
|
||||||
<UI_FontFamily>Segoe UI</UI_FontFamily>
|
<UI_FontFamily>Segoe UI</UI_FontFamily>
|
||||||
|
Reference in New Issue
Block a user