// 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 . using QuickLook.Common.Helpers; using QuickLook.Common.Plugin; using QuickLook.Plugin.HtmlViewer; using System; using System.Diagnostics; using System.IO; using System.Linq; using System.Text; using System.Windows; using System.Windows.Threading; using UtfUnknown; namespace QuickLook.Plugin.MarkdownViewer; public class Plugin : IViewer { private WebpagePanel? _panel; private string? _currentHtmlPath; private static readonly string _resourcePath = Path.Combine(SettingHelper.LocalDataPath, "QuickLook.Plugin.MarkdownViewer"); private static readonly string _resourcePrefix = "QuickLook.Plugin.MarkdownViewer.Resources."; private static readonly ResourceManager _resourceManager = new(_resourcePath, _resourcePrefix); public int Priority => 0; public void Init() { // Initialize resources and handle versioning _resourceManager.InitializeResources(); // Clean up any temporary HTML files if QuickLook was forcibly terminated CleanupTempFiles(); } public bool CanHandle(string path) { return !Directory.Exists(path) && new[] { ".md", ".mdown", ".rmd", ".markdown" }.Any(path.ToLower().EndsWith); } public void Prepare(string path, ContextObject context) { context.PreferredSize = new Size(1000, 600); } public void View(string path, ContextObject context) { _panel = new WebpagePanel(); context.ViewerContent = _panel; context.Title = Path.GetFileName(path); var htmlPath = GenerateMarkdownHtml(path); _panel.NavigateToFile(htmlPath, Path.GetDirectoryName(path)); _panel.Dispatcher.Invoke(() => { context.IsBusy = false; }, DispatcherPriority.Loaded); } private string GenerateMarkdownHtml(string path) { var templatePath = Path.Combine(_resourcePath, "md2html.html"); if (!File.Exists(templatePath)) throw new FileNotFoundException($"Required template file md2html.html not found in extracted resources at {templatePath}"); var bytes = File.ReadAllBytes(path); var encoding = CharsetDetector.DetectFromBytes(bytes).Detected?.Encoding ?? Encoding.Default; var content = encoding.GetString(bytes); var template = File.ReadAllText(templatePath); var html = template.Replace("{{content}}", content); // Generate unique filename and ensure it doesn't exist string outputPath; do { var uniqueId = Guid.NewGuid().ToString("N").Substring(0, 8); var outputFileName = $"temp_{uniqueId}.html"; outputPath = Path.Combine(_resourcePath, outputFileName); } while (File.Exists(outputPath)); // Clean up previous file if it exists CleanupTempHtmlFile(); File.WriteAllText(outputPath, html); _currentHtmlPath = outputPath; return outputPath; } #region Cleanup private void CleanupTempHtmlFile() { if (!string.IsNullOrEmpty(_currentHtmlPath) && File.Exists(_currentHtmlPath)) { try { File.Delete(_currentHtmlPath); } catch (IOException) { } // Ignore deletion errors } } private void CleanupTempFiles() { try { var tempFiles = Directory.GetFiles(_resourcePath, "temp_*.html"); foreach (var file in tempFiles) { try { File.Delete(file); } catch (IOException) { } // Ignore deletion errors } } catch (Exception ex) { Debug.WriteLine($"Failed to clean up temporary HTML files: {ex.Message}"); } } public void Cleanup() { GC.SuppressFinalize(this); CleanupTempHtmlFile(); _panel?.Dispose(); _panel = null; } #endregion Cleanup }