// ==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 data:image/png;base64,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
// @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(/^