// ==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(/^