// ==UserScript== // @name claude-mermaid-viewer // @namespace https://github.com/sansan0/useful-userscripts // @version 1.5 // @description 在 Claude 聊天界面中渲染和查看 Mermaid 图表的工具 // @author sansan // @match https://claude.ai/* // @grant GM.xmlHttpRequest // @grant unsafeWindow // @license GPL-3.0 License // @icon  // @downloadURL https://update.greasyfork.icu/scripts/535980/claude-mermaid-viewer.user.js // @updateURL https://update.greasyfork.icu/scripts/535980/claude-mermaid-viewer.meta.js // ==/UserScript== (function () { "use strict"; let initialScale = 1; const styleSheet = document.createElement("style"); styleSheet.textContent = ` .mermaid-content-wrapper { width: 100%; height: calc(100% - 50px); overflow: scroll; padding: 20px; box-sizing: border-box; position: absolute; top: 50px; left: 0; right: 0; bottom: 0; } .mermaid-content-wrapper::-webkit-scrollbar { width: 8px; height: 8px; display: block; } .mermaid-content-wrapper::-webkit-scrollbar-track { background: #f1f1f1; border-radius: 4px; } .mermaid-content-wrapper::-webkit-scrollbar-thumb { background: #888; border-radius: 4px; min-height: 40px; } .mermaid-content-wrapper::-webkit-scrollbar-thumb:hover { background: #555; } .control-button { width: 36px; height: 36px; padding: 6px; border: none; border-radius: 6px; background-color: transparent; display: flex; align-items: center; justify-content: center; cursor: pointer; transition: all 0.2s ease; } .control-button:hover { background-color: #f3f4f6; } .control-button svg { width: 24px; height: 24px; } .control-button.active { background-color: #4b5563; } .control-button.active svg { stroke: #ffffff; } div.text-text-300.absolute.pl-3.pt-2\\.5.text-xs.mermaid-toggle { opacity: 1 !important; visibility: visible !important; pointer-events: auto !important; background-color: #334155 !important; color: #ffffff !important; display: flex !important; align-items: center !important; gap: 4px !important; padding: 4px 8px !important; border-radius: 4px !important; font-size: 12px !important; font-weight: 500 !important; border: 1px solid transparent !important; transition: all 0.2s ease-in-out !important; cursor: pointer !important; } .mermaid-toggle svg { width: 14px; height: 14px; stroke: currentColor; stroke-width: 2; } div.text-text-500.text-xs.p-3\\.5.pb-0.mermaid-toggle { display: flex !important; align-items: center !important; gap: 5px !important; cursor: pointer !important; } div.mermaid-toggle span { display: flex !important; align-items: center !important; justify-content: center !important; } div.mermaid-toggle svg { width: 14px !important; height: 14px !important; stroke: currentColor !important; stroke-width: 2 !important; margin-right: 3px !important; } `; document.head.appendChild(styleSheet); /** * 加载 Mermaid 库 * @param {function} callback - 加载完成后的回调函数 */ function loadMermaidLibrary(callback) { GM.xmlHttpRequest({ method: "GET", url: "https://cdn.jsdelivr.net/npm/mermaid@11.4.1/dist/mermaid.min.js", onload: function (response) { const script = document.createElement("script"); script.textContent = response.responseText; document.head.appendChild(script); unsafeWindow.mermaid.initialize({ startOnLoad: false, flowchart: { htmlLabels: true, wrappingWidth: 300, padding: 20, }, class: { wrappingWidth: 300, }, state: { wrappingWidth: 300, }, er: { wrappingWidth: 300, }, }); callback(); }, }); } /** * 在模态框中渲染 Mermaid 图 * @param {string} mermaidContent - Mermaid 图的内容 */ function renderMermaidInModal(mermaidContent) { const modal = document.createElement("div"); modal.style.position = "fixed"; modal.style.top = "0"; modal.style.left = "0"; modal.style.width = "100%"; modal.style.height = "100%"; modal.style.backgroundColor = "rgba(0, 0, 0, 0.5)"; modal.style.zIndex = "9999"; modal.style.display = "flex"; modal.style.justifyContent = "center"; modal.style.alignItems = "center"; const container = document.createElement("div"); container.style.width = "90%"; container.style.height = "90%"; container.style.backgroundColor = "white"; container.style.boxSizing = "border-box"; container.style.position = "relative"; container.style.borderRadius = "8px"; container.style.boxShadow = "0 4px 6px rgba(0, 0, 0, 0.1)"; const contentWrapper = document.createElement("div"); contentWrapper.classList.add("mermaid-content-wrapper"); const buttonContainer = document.createElement("div"); buttonContainer.style.position = "absolute"; buttonContainer.style.top = "10px"; buttonContainer.style.right = "10px"; buttonContainer.style.height = "40px"; buttonContainer.style.display = "flex"; buttonContainer.style.gap = "4px"; buttonContainer.style.zIndex = "1"; const panButton = createControlButton( "pan", ` ` ); const zoomOutButton = createControlButton( "zoom-out", ` ` ); const resetButton = createControlButton("reset", "Reset", true); const zoomInButton = createControlButton( "zoom-in", ` ` ); const downloadButton = createControlButton( "download", ` ` ); const closeButton = createControlButton( "close", ` ` ); closeButton.style.marginLeft = "8px"; let currentScale = 1; const scaleStep = 0.2; let isPanning = false; let isEnabled = false; panButton.addEventListener("click", togglePanMode); togglePanMode(); let startX, startY, scrollLeft, scrollTop; contentWrapper.addEventListener("mousedown", handleMouseDown); contentWrapper.addEventListener("mousemove", handleMouseMove); document.addEventListener("mouseup", handleMouseUp); contentWrapper.addEventListener("selectstart", handleSelectStart); zoomInButton.addEventListener("click", zoomIn); zoomOutButton.addEventListener("click", zoomOut); resetButton.addEventListener("click", resetZoom); closeButton.addEventListener("click", closeModal); downloadButton.addEventListener("click", downloadAsPng); buttonContainer.appendChild(panButton); buttonContainer.appendChild(zoomOutButton); buttonContainer.appendChild(resetButton); buttonContainer.appendChild(zoomInButton); buttonContainer.appendChild(downloadButton); buttonContainer.appendChild(closeButton); const mermaidElement = document.createElement("pre"); mermaidElement.classList.add("mermaid"); mermaidElement.textContent = mermaidContent; mermaidElement.style.margin = "0"; mermaidElement.style.display = "inline-block"; mermaidElement.style.position = "relative"; mermaidElement.style.minWidth = "100%"; contentWrapper.appendChild(mermaidElement); container.appendChild(buttonContainer); container.appendChild(contentWrapper); modal.appendChild(container); document.body.appendChild(modal); const observer = new MutationObserver(handleMutations); observer.observe(mermaidElement, { childList: true }); unsafeWindow.mermaid.init(undefined, mermaidElement).then(() => { resetZoom(); }); modal.addEventListener("click", handleModalClick); function downloadAsPng() { const contentWrapper = mermaidElement.closest(".mermaid-content-wrapper"); const scrollLeft = contentWrapper.scrollLeft; const scrollTop = contentWrapper.scrollTop; const svg = mermaidElement.querySelector("svg"); if (!svg) return; const originalTransform = svg.style.transform; const originalTransformOrigin = svg.style.transformOrigin; svg.style.transform = "scale(1)"; const svgClone = svg.cloneNode(true); if ( !svgClone.hasAttribute("viewBox") && svgClone.hasAttribute("width") && svgClone.hasAttribute("height") ) { svgClone.setAttribute( "viewBox", `0 0 ${svgClone.getAttribute("width")} ${svgClone.getAttribute( "height" )}` ); } const padding = 20; const bbox = svg.getBBox(); const width = Math.ceil(bbox.width + padding * 2); const height = Math.ceil(bbox.height + padding * 2); svgClone.setAttribute("width", width); svgClone.setAttribute("height", height); svgClone.setAttribute( "viewBox", `${bbox.x - padding} ${bbox.y - padding} ${width} ${height}` ); const serializer = new XMLSerializer(); let source = serializer.serializeToString(svgClone); if ( !source.match(/^]+xmlns="http\:\/\/www\.w3\.org\/2000\/svg"/) ) { source = source.replace( /^${content}`; } return button; } /** * 切换平移模式 */ function togglePanMode() { isEnabled = !isEnabled; panButton.classList.toggle("active", isEnabled); contentWrapper.style.cursor = isEnabled ? "grab" : "default"; } /** * 处理鼠标按下事件 * @param {MouseEvent} e - 鼠标事件对象 */ function handleMouseDown(e) { if (!isEnabled) return; isPanning = true; contentWrapper.style.cursor = "grabbing"; startX = e.pageX - contentWrapper.offsetLeft; startY = e.pageY - contentWrapper.offsetTop; scrollLeft = contentWrapper.scrollLeft; scrollTop = contentWrapper.scrollTop; } /** * 处理鼠标移动事件 * @param {MouseEvent} e - 鼠标事件对象 */ function handleMouseMove(e) { if (!isPanning) return; e.preventDefault(); const x = e.pageX - contentWrapper.offsetLeft; const y = e.pageY - contentWrapper.offsetTop; const walkX = (x - startX) * 1.5; const walkY = (y - startY) * 1.5; contentWrapper.scrollLeft = scrollLeft - walkX; contentWrapper.scrollTop = scrollTop - walkY; } /** * 处理鼠标松开事件 */ function handleMouseUp() { isPanning = false; if (isEnabled) { contentWrapper.style.cursor = "grab"; } } /** * 处理选择开始事件 * @param {Event} e - 事件对象 */ function handleSelectStart(e) { if (isEnabled) { e.preventDefault(); } } /** * 更新 SVG 缩放 */ function updateSvgScale() { const svg = mermaidElement.querySelector("svg"); if (svg) { svg.style.transform = `scale(${currentScale})`; svg.style.transformOrigin = "top left"; const boundingRect = svg.getBoundingClientRect(); const scaledWidth = boundingRect.width; const scaledHeight = boundingRect.height; mermaidElement.style.width = `${scaledWidth}px`; mermaidElement.style.height = `${scaledHeight}px`; mermaidElement.style.margin = "20px"; const wrapperHeight = contentWrapper.clientHeight; const centerY = (wrapperHeight - scaledHeight) / 2; contentWrapper.scrollTop = centerY; } } /** * 放大 */ function zoomIn() { currentScale += scaleStep; updateSvgScale(); } /** * 缩小 */ function zoomOut() { currentScale = Math.max(0.2, currentScale - scaleStep); updateSvgScale(); } /** * 重置缩放 */ function resetZoom() { currentScale = initialScale; const svg = mermaidElement.querySelector("svg"); svg.style.transform = ""; mermaidElement.style.width = ""; mermaidElement.style.height = ""; setTimeout(() => { updateSvgScale(); setTimeout(() => { const centerX = (contentWrapper.scrollWidth - contentWrapper.clientWidth) / 2; const centerY = (contentWrapper.scrollHeight - contentWrapper.clientHeight) / 2; contentWrapper.scrollLeft = centerX; contentWrapper.scrollTop = centerY; }, 0); }, 0); } /** * 关闭模态框 */ function closeModal() { modal.remove(); } /** * 处理 Mutation * @param {MutationRecord[]} mutations - Mutation 记录数组 */ function handleMutations(mutations) { mutations.forEach((mutation) => { if (mutation.type === "childList") { const addedNodes = mutation.addedNodes; for (const node of addedNodes) { if (node.nodeName === "svg") { const svg = node; const boundingRect = svg.getBoundingClientRect(); const svgWidth = boundingRect.width; const svgHeight = boundingRect.height; const wrapperWidth = contentWrapper.clientWidth; const wrapperHeight = contentWrapper.clientHeight; const scaleX = wrapperWidth / svgWidth; const scaleY = wrapperHeight / svgHeight; initialScale = Math.min(scaleX, scaleY) * 0.95; updateSvgScale(); const centerX = (contentWrapper.scrollWidth - contentWrapper.clientWidth) / 2; const centerY = (contentWrapper.scrollHeight - contentWrapper.clientHeight) / 2; contentWrapper.scrollLeft = centerX; contentWrapper.scrollTop = centerY; observer.disconnect(); break; } } } }); } /** * 处理模态框点击事件 * @param {MouseEvent} event - 鼠标事件对象 */ function handleModalClick(event) { if (event.target === modal) { modal.remove(); } } } /** * 获取 Mermaid 内容 * @param {HTMLElement} element - 元素 * @returns {string} - Mermaid 内容 */ function getMermaidContent(element) { const codeElement = element .closest("div") .nextElementSibling.querySelector("code.language-mermaid"); if (!codeElement) { console.error("Mermaid code element not found"); return ""; } return extractMermaidSyntax(codeElement); } /** * 提取 Mermaid 语法 * @param {HTMLElement} codeElement - 代码元素 * @returns {string} - Mermaid 语法 */ function extractMermaidSyntax(codeElement) { if (!codeElement) { console.error("Code element not found"); return ""; } try { let rawText = ""; rawText = codeElement.textContent; const lines = rawText.split("\n"); // 处理每一行,移除多余空白字符和特殊Unicode字符 const cleanedLines = lines .map( (line) => line .replace(/[\u200B-\u200D\uFEFF]/g, "") // 移除零宽字符 .replace(/\t/g, " ") // 制表符替换为空格 .trimEnd() // 移除行尾空格 ) .filter((line) => line.trim() !== ""); // 过滤空行 const mermaidSyntax = cleanedLines.join("\n"); console.log("Extracted Mermaid Syntax:"); console.log(mermaidSyntax); return mermaidSyntax; } catch (error) { console.error("Error extracting Mermaid syntax:", error); return ""; } } /** * 处理元素 */ function processElements() { const observer = new MutationObserver((mutations) => { mutations.forEach(() => { const elements = document.querySelectorAll( "div.text-text-500.text-xs.p-3\\.5.pb-0[style*='margin-bottom: 20px;']" ); elements.forEach((element) => { if ( element.textContent.trim() === "mermaid" && !element.classList.contains("mermaid-toggle") ) { element.classList.add("mermaid-toggle"); const icon = document.createElement("span"); icon.style.display = "inline-flex"; icon.style.alignItems = "center"; icon.innerHTML = ` `; element.innerHTML = ""; element.appendChild(icon); element.appendChild(document.createTextNode("mermaid")); element.style.cursor = "pointer"; element.style.transition = "opacity 0.2s ease"; element.addEventListener("mouseenter", () => { element.style.opacity = "0.8"; }); element.addEventListener("mouseleave", () => { element.style.opacity = "1"; }); element.addEventListener("click", () => { const mermaidContent = getMermaidContent(element); renderMermaidInModal(mermaidContent); }); } }); }); }); observer.observe(document.body, { childList: true, subtree: true, }); } loadMermaidLibrary(() => { processElements(); window.addEventListener("load", processElements); }); })();