221 lines
4.8 KiB
Vue
221 lines
4.8 KiB
Vue
<template>
|
|
<div class="card mermaid-card">
|
|
<div class="panel-title">
|
|
<h3>{{ title }}</h3>
|
|
<div class="actions">
|
|
<button class="pill mono action" type="button" @click="open">Full screen</button>
|
|
<span class="pill mono">Mermaid</span>
|
|
</div>
|
|
</div>
|
|
<p class="description">{{ description }}</p>
|
|
<div class="diagram" role="button" tabindex="0" @click="open" v-html="svgContent"></div>
|
|
<div v-if="isOpen" class="overlay" @click.self="close">
|
|
<div class="modal card glass">
|
|
<div class="modal-head">
|
|
<div>
|
|
<div class="modal-title">{{ title }}</div>
|
|
<div class="modal-sub mono">Click outside or press Esc to close</div>
|
|
</div>
|
|
<button class="close" type="button" @click="close">Close</button>
|
|
</div>
|
|
<div class="modal-body" v-html="svgContent"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { onMounted, onUnmounted, ref, watch } from "vue";
|
|
import mermaid from "mermaid";
|
|
|
|
const props = defineProps({
|
|
title: String,
|
|
description: String,
|
|
diagram: String,
|
|
cardId: String,
|
|
});
|
|
|
|
const svgContent = ref("");
|
|
const renderKey = ref(props.cardId || `mermaid-${Math.random().toString(36).slice(2)}`);
|
|
const isOpen = ref(false);
|
|
let initialized = false;
|
|
let scheduledHandle = null;
|
|
let scheduledKind = "";
|
|
|
|
const renderDiagram = async () => {
|
|
if (!props.diagram) return;
|
|
if (!initialized) {
|
|
mermaid.initialize({
|
|
startOnLoad: false,
|
|
theme: "dark",
|
|
securityLevel: "loose",
|
|
themeVariables: {
|
|
primaryColor: "#0b1228",
|
|
primaryTextColor: "#e8f3ff",
|
|
primaryBorderColor: "#00e5c5",
|
|
lineColor: "#7f7cff",
|
|
nodeTextColor: "#e8f3ff",
|
|
},
|
|
});
|
|
initialized = true;
|
|
}
|
|
try {
|
|
const { svg } = await mermaid.render(`${renderKey.value}-${Date.now()}`, props.diagram);
|
|
svgContent.value = svg;
|
|
} catch (err) {
|
|
svgContent.value = `<pre class="mono" style="color:#ff4f93">Mermaid render error: ${err}</pre>`;
|
|
}
|
|
};
|
|
|
|
function cancelScheduledRender() {
|
|
if (!scheduledHandle) return;
|
|
if (scheduledKind === "idle" && window.cancelIdleCallback) {
|
|
window.cancelIdleCallback(scheduledHandle);
|
|
} else {
|
|
window.clearTimeout(scheduledHandle);
|
|
}
|
|
scheduledHandle = null;
|
|
scheduledKind = "";
|
|
}
|
|
|
|
function scheduleRenderDiagram() {
|
|
cancelScheduledRender();
|
|
if (!props.diagram) return;
|
|
const runner = () => {
|
|
scheduledHandle = null;
|
|
scheduledKind = "";
|
|
renderDiagram();
|
|
};
|
|
if (window.requestIdleCallback) {
|
|
scheduledKind = "idle";
|
|
scheduledHandle = window.requestIdleCallback(runner, { timeout: 1500 });
|
|
} else {
|
|
scheduledKind = "timeout";
|
|
scheduledHandle = window.setTimeout(runner, 0);
|
|
}
|
|
}
|
|
|
|
onMounted(scheduleRenderDiagram);
|
|
watch(
|
|
() => props.diagram,
|
|
() => scheduleRenderDiagram()
|
|
);
|
|
|
|
const onKeyDown = (event) => {
|
|
if (event.key === "Escape") close();
|
|
};
|
|
|
|
const open = () => {
|
|
if (!svgContent.value) scheduleRenderDiagram();
|
|
isOpen.value = true;
|
|
};
|
|
|
|
const close = () => {
|
|
isOpen.value = false;
|
|
};
|
|
|
|
watch(isOpen, (value) => {
|
|
if (value) {
|
|
document.body.style.overflow = "hidden";
|
|
window.addEventListener("keydown", onKeyDown);
|
|
} else {
|
|
document.body.style.overflow = "";
|
|
window.removeEventListener("keydown", onKeyDown);
|
|
}
|
|
});
|
|
|
|
onUnmounted(() => {
|
|
cancelScheduledRender();
|
|
document.body.style.overflow = "";
|
|
window.removeEventListener("keydown", onKeyDown);
|
|
});
|
|
</script>
|
|
|
|
<style scoped>
|
|
.mermaid-card {
|
|
min-height: 320px;
|
|
}
|
|
|
|
.actions {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 10px;
|
|
}
|
|
|
|
.action {
|
|
cursor: pointer;
|
|
}
|
|
|
|
.description {
|
|
color: var(--text-muted);
|
|
margin-top: 0;
|
|
}
|
|
|
|
.diagram {
|
|
margin-top: 12px;
|
|
padding: 10px;
|
|
border-radius: var(--radius-sm);
|
|
border: 1px dashed rgba(255, 255, 255, 0.08);
|
|
background: rgba(255, 255, 255, 0.02);
|
|
overflow-x: auto;
|
|
cursor: zoom-in;
|
|
}
|
|
|
|
.overlay {
|
|
position: fixed;
|
|
inset: 0;
|
|
background: rgba(0, 0, 0, 0.72);
|
|
display: grid;
|
|
place-items: center;
|
|
padding: 18px;
|
|
z-index: 50;
|
|
}
|
|
|
|
.modal {
|
|
width: min(1200px, 96vw);
|
|
max-height: 92vh;
|
|
overflow: hidden;
|
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
}
|
|
|
|
.modal-head {
|
|
display: flex;
|
|
align-items: flex-start;
|
|
justify-content: space-between;
|
|
gap: 12px;
|
|
padding: 14px 16px;
|
|
border-bottom: 1px solid rgba(255, 255, 255, 0.08);
|
|
}
|
|
|
|
.modal-title {
|
|
font-weight: 800;
|
|
color: var(--text-strong);
|
|
}
|
|
|
|
.modal-sub {
|
|
color: var(--text-muted);
|
|
font-size: 13px;
|
|
margin-top: 4px;
|
|
}
|
|
|
|
.close {
|
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
|
background: rgba(255, 255, 255, 0.04);
|
|
color: var(--text-strong);
|
|
padding: 8px 12px;
|
|
border-radius: 10px;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.modal-body {
|
|
padding: 12px;
|
|
overflow: auto;
|
|
max-height: calc(92vh - 64px);
|
|
}
|
|
|
|
.modal-body :deep(svg) {
|
|
width: 100%;
|
|
height: auto;
|
|
}
|
|
</style>
|