@hejiayue/x-markdown-test 0.0.1-beta.112 → 0.0.1-beta.115
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/style.css +1 -1
- package/dist/x-markdown.cjs.js +1 -1
- package/dist/x-markdown.cjs10.js +2 -0
- package/dist/x-markdown.cjs10.js.map +1 -0
- package/dist/x-markdown.cjs11.js +2 -0
- package/dist/x-markdown.cjs11.js.map +1 -0
- package/dist/x-markdown.cjs13.js +2 -0
- package/dist/x-markdown.cjs13.js.map +1 -0
- package/dist/x-markdown.cjs14.js +2 -0
- package/dist/x-markdown.cjs14.js.map +1 -0
- package/dist/x-markdown.cjs15.js +2 -0
- package/dist/x-markdown.cjs15.js.map +1 -0
- package/dist/x-markdown.cjs16.js +2 -0
- package/dist/x-markdown.cjs16.js.map +1 -0
- package/dist/x-markdown.cjs17.js +2 -0
- package/dist/x-markdown.cjs17.js.map +1 -0
- package/dist/x-markdown.cjs18.js +2 -0
- package/dist/x-markdown.cjs18.js.map +1 -0
- package/dist/x-markdown.cjs19.js +2 -0
- package/dist/x-markdown.cjs19.js.map +1 -0
- package/dist/x-markdown.cjs2.js +2 -0
- package/dist/x-markdown.cjs2.js.map +1 -0
- package/dist/x-markdown.cjs21.js +2 -0
- package/dist/x-markdown.cjs21.js.map +1 -0
- package/dist/x-markdown.cjs22.js +2 -0
- package/dist/x-markdown.cjs22.js.map +1 -0
- package/dist/x-markdown.cjs24.js +2 -0
- package/dist/x-markdown.cjs24.js.map +1 -0
- package/dist/x-markdown.cjs26.js +2 -0
- package/dist/x-markdown.cjs26.js.map +1 -0
- package/dist/x-markdown.cjs27.js +2 -0
- package/dist/x-markdown.cjs27.js.map +1 -0
- package/dist/x-markdown.cjs28.js +2 -0
- package/dist/x-markdown.cjs28.js.map +1 -0
- package/dist/x-markdown.cjs3.js +2 -0
- package/dist/x-markdown.cjs3.js.map +1 -0
- package/dist/x-markdown.cjs30.js +2 -0
- package/dist/x-markdown.cjs30.js.map +1 -0
- package/dist/x-markdown.cjs31.js +2 -0
- package/dist/x-markdown.cjs31.js.map +1 -0
- package/dist/x-markdown.cjs33.js +2 -0
- package/dist/x-markdown.cjs33.js.map +1 -0
- package/dist/x-markdown.cjs4.js +2 -0
- package/dist/x-markdown.cjs4.js.map +1 -0
- package/dist/x-markdown.cjs5.js +2 -0
- package/dist/x-markdown.cjs5.js.map +1 -0
- package/dist/x-markdown.cjs6.js +2 -0
- package/dist/x-markdown.cjs6.js.map +1 -0
- package/dist/x-markdown.cjs7.js +2 -0
- package/dist/x-markdown.cjs7.js.map +1 -0
- package/dist/x-markdown.cjs8.js +2 -0
- package/dist/x-markdown.cjs8.js.map +1 -0
- package/dist/x-markdown.cjs9.js +2 -0
- package/dist/x-markdown.cjs9.js.map +1 -0
- package/dist/x-markdown.es.js +26 -17
- package/dist/x-markdown.es.js.map +1 -1
- package/dist/x-markdown.es10.js +49 -0
- package/dist/x-markdown.es10.js.map +1 -0
- package/dist/x-markdown.es11.js +54 -0
- package/dist/x-markdown.es11.js.map +1 -0
- package/dist/x-markdown.es13.js +89 -0
- package/dist/x-markdown.es13.js.map +1 -0
- package/dist/x-markdown.es14.js +34 -0
- package/dist/x-markdown.es14.js.map +1 -0
- package/dist/x-markdown.es15.js +5 -0
- package/dist/x-markdown.es15.js.map +1 -0
- package/dist/x-markdown.es16.js +6 -0
- package/dist/x-markdown.es16.js.map +1 -0
- package/dist/x-markdown.es17.js +8 -0
- package/dist/x-markdown.es17.js.map +1 -0
- package/dist/x-markdown.es18.js +8 -0
- package/dist/x-markdown.es18.js.map +1 -0
- package/dist/x-markdown.es19.js +207 -0
- package/dist/x-markdown.es19.js.map +1 -0
- package/dist/x-markdown.es2.js +83 -0
- package/dist/x-markdown.es2.js.map +1 -0
- package/dist/x-markdown.es21.js +11 -0
- package/dist/x-markdown.es21.js.map +1 -0
- package/dist/x-markdown.es22.js +75 -0
- package/dist/x-markdown.es22.js.map +1 -0
- package/dist/{index-SElRorTo.js → x-markdown.es24.js} +7 -139
- package/dist/x-markdown.es24.js.map +1 -0
- package/dist/x-markdown.es26.js +160 -0
- package/dist/x-markdown.es26.js.map +1 -0
- package/dist/x-markdown.es27.js +8 -0
- package/dist/x-markdown.es27.js.map +1 -0
- package/dist/x-markdown.es28.js +142 -0
- package/dist/x-markdown.es28.js.map +1 -0
- package/dist/x-markdown.es3.js +94 -0
- package/dist/x-markdown.es3.js.map +1 -0
- package/dist/x-markdown.es30.js +78 -0
- package/dist/x-markdown.es30.js.map +1 -0
- package/dist/x-markdown.es31.js +125 -0
- package/dist/x-markdown.es31.js.map +1 -0
- package/dist/x-markdown.es33.js +6 -0
- package/dist/x-markdown.es33.js.map +1 -0
- package/dist/x-markdown.es4.js +150 -0
- package/dist/x-markdown.es4.js.map +1 -0
- package/dist/x-markdown.es5.js +40 -0
- package/dist/x-markdown.es5.js.map +1 -0
- package/dist/x-markdown.es6.js +24 -0
- package/dist/x-markdown.es6.js.map +1 -0
- package/dist/x-markdown.es7.js +279 -0
- package/dist/x-markdown.es7.js.map +1 -0
- package/dist/x-markdown.es8.js +27 -0
- package/dist/x-markdown.es8.js.map +1 -0
- package/dist/x-markdown.es9.js +409 -0
- package/dist/x-markdown.es9.js.map +1 -0
- package/package.json +84 -84
- package/LICENSE +0 -21
- package/dist/index-D9u8JldH.cjs +0 -2
- package/dist/index-D9u8JldH.cjs.map +0 -1
- package/dist/index-SElRorTo.js.map +0 -1
- package/dist/index-lQ_FQFr2.js +0 -1882
- package/dist/index-lQ_FQFr2.js.map +0 -1
- package/dist/index-nZ5iH1aN.cjs +0 -2
- package/dist/index-nZ5iH1aN.cjs.map +0 -1
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
import { throttle } from "lodash-es";
|
|
2
|
+
import { computed, ref, watch, onUnmounted } from "vue";
|
|
3
|
+
function downloadSvgAsPng(svg) {
|
|
4
|
+
if (!svg) return;
|
|
5
|
+
try {
|
|
6
|
+
const svgDataUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
|
|
7
|
+
const img = new Image();
|
|
8
|
+
img.onload = () => {
|
|
9
|
+
try {
|
|
10
|
+
const canvas = document.createElement("canvas");
|
|
11
|
+
const ctx = canvas.getContext("2d", { willReadFrequently: false });
|
|
12
|
+
if (!ctx) return;
|
|
13
|
+
const scale = 2;
|
|
14
|
+
canvas.width = img.width * scale;
|
|
15
|
+
canvas.height = img.height * scale;
|
|
16
|
+
ctx.imageSmoothingEnabled = true;
|
|
17
|
+
ctx.imageSmoothingQuality = "high";
|
|
18
|
+
ctx.fillStyle = "#ffffff";
|
|
19
|
+
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
20
|
+
ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
|
|
21
|
+
const timestamp = (/* @__PURE__ */ new Date()).toISOString().slice(0, 19).replace(/:/g, "-");
|
|
22
|
+
try {
|
|
23
|
+
canvas.toBlob(
|
|
24
|
+
(blob) => {
|
|
25
|
+
if (!blob) return;
|
|
26
|
+
const url = URL.createObjectURL(blob);
|
|
27
|
+
const link = document.createElement("a");
|
|
28
|
+
link.href = url;
|
|
29
|
+
link.download = `mermaid-diagram-${timestamp}.png`;
|
|
30
|
+
document.body.appendChild(link);
|
|
31
|
+
link.click();
|
|
32
|
+
document.body.removeChild(link);
|
|
33
|
+
URL.revokeObjectURL(url);
|
|
34
|
+
},
|
|
35
|
+
"image/png",
|
|
36
|
+
0.95
|
|
37
|
+
);
|
|
38
|
+
} catch (toBlobError) {
|
|
39
|
+
console.error("Failed to convert canvas to blob:", toBlobError);
|
|
40
|
+
try {
|
|
41
|
+
const dataUrl = canvas.toDataURL("image/png", 0.95);
|
|
42
|
+
const link = document.createElement("a");
|
|
43
|
+
link.href = dataUrl;
|
|
44
|
+
link.download = `mermaid-diagram-${timestamp}.png`;
|
|
45
|
+
document.body.appendChild(link);
|
|
46
|
+
link.click();
|
|
47
|
+
document.body.removeChild(link);
|
|
48
|
+
} catch (dataUrlError) {
|
|
49
|
+
console.error("Failed to convert canvas to data URL:", dataUrlError);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
} catch (canvasError) {
|
|
53
|
+
console.error("Canvas operation failed:", canvasError);
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
img.onerror = (error) => {
|
|
57
|
+
console.error("Failed to load image:", error);
|
|
58
|
+
};
|
|
59
|
+
img.src = svgDataUrl;
|
|
60
|
+
} catch (error) {
|
|
61
|
+
console.error("Failed to download SVG:", error);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
let mermaidPromise = null;
|
|
65
|
+
let hasShownMermaidHint = false;
|
|
66
|
+
let mermaidAvailableCache = null;
|
|
67
|
+
let mermaidCheckPromise = null;
|
|
68
|
+
async function checkMermaidAvailable() {
|
|
69
|
+
if (mermaidAvailableCache !== null) {
|
|
70
|
+
return mermaidAvailableCache;
|
|
71
|
+
}
|
|
72
|
+
if (mermaidCheckPromise) {
|
|
73
|
+
return mermaidCheckPromise;
|
|
74
|
+
}
|
|
75
|
+
mermaidCheckPromise = (async () => {
|
|
76
|
+
try {
|
|
77
|
+
await import("mermaid");
|
|
78
|
+
mermaidAvailableCache = true;
|
|
79
|
+
return true;
|
|
80
|
+
} catch (error) {
|
|
81
|
+
mermaidAvailableCache = false;
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
})();
|
|
85
|
+
return mermaidCheckPromise;
|
|
86
|
+
}
|
|
87
|
+
const showMermaidHint = () => {
|
|
88
|
+
if (hasShownMermaidHint) return;
|
|
89
|
+
hasShownMermaidHint = true;
|
|
90
|
+
console.log(
|
|
91
|
+
"%c[x-markdown]%c Mermaid 图表功能已降级为代码块显示",
|
|
92
|
+
"font-weight: bold; color: #9333ea;",
|
|
93
|
+
"color: #666;"
|
|
94
|
+
);
|
|
95
|
+
console.log(
|
|
96
|
+
"%c如需 Mermaid 图表渲染功能,请安装:",
|
|
97
|
+
"color: #666; font-weight: bold;"
|
|
98
|
+
);
|
|
99
|
+
console.log(
|
|
100
|
+
"%c pnpm add mermaid",
|
|
101
|
+
"color: #9333ea; font-family: monospace;"
|
|
102
|
+
);
|
|
103
|
+
console.log(
|
|
104
|
+
"%c安装后请重启开发服务器",
|
|
105
|
+
"color: #999; font-size: 12px;"
|
|
106
|
+
);
|
|
107
|
+
};
|
|
108
|
+
async function loadMermaid() {
|
|
109
|
+
if (typeof window === "undefined") return null;
|
|
110
|
+
if (!mermaidPromise) {
|
|
111
|
+
mermaidPromise = (async () => {
|
|
112
|
+
try {
|
|
113
|
+
const mod = await import("mermaid");
|
|
114
|
+
return mod.default;
|
|
115
|
+
} catch {
|
|
116
|
+
showMermaidHint();
|
|
117
|
+
return null;
|
|
118
|
+
}
|
|
119
|
+
})();
|
|
120
|
+
}
|
|
121
|
+
return mermaidPromise;
|
|
122
|
+
}
|
|
123
|
+
const renderQueue = [];
|
|
124
|
+
let isProcessingQueue = false;
|
|
125
|
+
async function processRenderQueue() {
|
|
126
|
+
if (isProcessingQueue) return;
|
|
127
|
+
isProcessingQueue = true;
|
|
128
|
+
while (renderQueue.length > 0) {
|
|
129
|
+
const task = renderQueue.shift();
|
|
130
|
+
if (task) {
|
|
131
|
+
try {
|
|
132
|
+
await task();
|
|
133
|
+
} catch (err) {
|
|
134
|
+
console.error("Mermaid render queue error:", err);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
isProcessingQueue = false;
|
|
139
|
+
}
|
|
140
|
+
function addToRenderQueue(task) {
|
|
141
|
+
renderQueue.push(task);
|
|
142
|
+
processRenderQueue();
|
|
143
|
+
}
|
|
144
|
+
function useMermaid(content, options = {}) {
|
|
145
|
+
const optionsRef = computed(() => typeof options === "object" && "value" in options ? options.value : options);
|
|
146
|
+
const mermaidConfig = computed(() => ({
|
|
147
|
+
suppressErrorRendering: true,
|
|
148
|
+
startOnLoad: false,
|
|
149
|
+
securityLevel: "loose",
|
|
150
|
+
theme: optionsRef.value.theme || "default",
|
|
151
|
+
...optionsRef.value.config || {}
|
|
152
|
+
}));
|
|
153
|
+
const data = ref("");
|
|
154
|
+
const error = ref(null);
|
|
155
|
+
const isLoading = ref(false);
|
|
156
|
+
let isUnmounted = false;
|
|
157
|
+
const getRenderContainer = () => {
|
|
158
|
+
const containerOption = optionsRef.value.container;
|
|
159
|
+
if (containerOption) {
|
|
160
|
+
return typeof containerOption === "object" && "value" in containerOption ? containerOption.value : containerOption;
|
|
161
|
+
}
|
|
162
|
+
return null;
|
|
163
|
+
};
|
|
164
|
+
const throttledRender = throttle(
|
|
165
|
+
() => {
|
|
166
|
+
const contentValue = typeof content === "string" ? content : content.value;
|
|
167
|
+
if (!contentValue?.trim()) {
|
|
168
|
+
data.value = "";
|
|
169
|
+
error.value = null;
|
|
170
|
+
isLoading.value = false;
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
isLoading.value = true;
|
|
174
|
+
addToRenderQueue(async () => {
|
|
175
|
+
if (isUnmounted) return;
|
|
176
|
+
try {
|
|
177
|
+
const mermaidInstance = await loadMermaid();
|
|
178
|
+
if (!mermaidInstance) {
|
|
179
|
+
data.value = contentValue;
|
|
180
|
+
error.value = null;
|
|
181
|
+
isLoading.value = false;
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
mermaidInstance.initialize(mermaidConfig.value);
|
|
185
|
+
const isValid = await mermaidInstance.parse(contentValue.trim());
|
|
186
|
+
if (!isValid) {
|
|
187
|
+
data.value = "";
|
|
188
|
+
error.value = new Error("Mermaid parse error: Invalid syntax");
|
|
189
|
+
isLoading.value = false;
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
const renderId = `${optionsRef.value.id || "mermaid"}-${Math.random().toString(36).substring(2, 11)}`;
|
|
193
|
+
const container = getRenderContainer();
|
|
194
|
+
if (!container) {
|
|
195
|
+
isLoading.value = false;
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
const { svg } = await mermaidInstance.render(renderId, contentValue, container);
|
|
199
|
+
data.value = svg;
|
|
200
|
+
error.value = null;
|
|
201
|
+
isLoading.value = false;
|
|
202
|
+
} catch (err) {
|
|
203
|
+
data.value = "";
|
|
204
|
+
error.value = err;
|
|
205
|
+
isLoading.value = false;
|
|
206
|
+
}
|
|
207
|
+
});
|
|
208
|
+
},
|
|
209
|
+
100,
|
|
210
|
+
{ leading: false, trailing: true }
|
|
211
|
+
);
|
|
212
|
+
watch(
|
|
213
|
+
[() => typeof content === "string" ? content : content.value, () => mermaidConfig.value],
|
|
214
|
+
() => {
|
|
215
|
+
throttledRender();
|
|
216
|
+
},
|
|
217
|
+
{ immediate: true }
|
|
218
|
+
);
|
|
219
|
+
onUnmounted(() => {
|
|
220
|
+
isUnmounted = true;
|
|
221
|
+
});
|
|
222
|
+
return {
|
|
223
|
+
data,
|
|
224
|
+
error,
|
|
225
|
+
isLoading
|
|
226
|
+
};
|
|
227
|
+
}
|
|
228
|
+
function useMermaidZoom(options) {
|
|
229
|
+
const { container } = options;
|
|
230
|
+
const scale = ref(1);
|
|
231
|
+
const posX = ref(0);
|
|
232
|
+
const posY = ref(0);
|
|
233
|
+
const isDragging = ref(false);
|
|
234
|
+
let removeEvents = null;
|
|
235
|
+
const getSvg = () => container.value?.querySelector(".syntax-mermaid__content svg");
|
|
236
|
+
const updateTransform = (svg) => {
|
|
237
|
+
svg.style.transformOrigin = "center center";
|
|
238
|
+
svg.style.transform = `translate(${posX.value}px, ${posY.value}px) scale(${scale.value})`;
|
|
239
|
+
};
|
|
240
|
+
const resetState = () => {
|
|
241
|
+
scale.value = 1;
|
|
242
|
+
posX.value = 0;
|
|
243
|
+
posY.value = 0;
|
|
244
|
+
isDragging.value = false;
|
|
245
|
+
};
|
|
246
|
+
const addInteractionEvents = (containerEl) => {
|
|
247
|
+
let startX = 0;
|
|
248
|
+
let startY = 0;
|
|
249
|
+
let isInteractingWithMermaid = false;
|
|
250
|
+
const onStart = (clientX, clientY) => {
|
|
251
|
+
isDragging.value = true;
|
|
252
|
+
startX = clientX - posX.value;
|
|
253
|
+
startY = clientY - posY.value;
|
|
254
|
+
document.body.style.userSelect = "none";
|
|
255
|
+
};
|
|
256
|
+
const onMove = (clientX, clientY) => {
|
|
257
|
+
if (isDragging.value && isInteractingWithMermaid) {
|
|
258
|
+
posX.value = clientX - startX;
|
|
259
|
+
posY.value = clientY - startY;
|
|
260
|
+
const svg = getSvg();
|
|
261
|
+
if (svg) {
|
|
262
|
+
updateTransform(svg);
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
};
|
|
266
|
+
const onEnd = () => {
|
|
267
|
+
isDragging.value = false;
|
|
268
|
+
isInteractingWithMermaid = false;
|
|
269
|
+
document.body.style.userSelect = "";
|
|
270
|
+
};
|
|
271
|
+
const onMouseDown = (e) => {
|
|
272
|
+
if (e.button !== 0) return;
|
|
273
|
+
if (e.target === containerEl || containerEl.contains(e.target)) {
|
|
274
|
+
e.preventDefault();
|
|
275
|
+
isInteractingWithMermaid = true;
|
|
276
|
+
onStart(e.clientX, e.clientY);
|
|
277
|
+
}
|
|
278
|
+
};
|
|
279
|
+
const onMouseMove = (e) => {
|
|
280
|
+
if (isInteractingWithMermaid) {
|
|
281
|
+
onMove(e.clientX, e.clientY);
|
|
282
|
+
}
|
|
283
|
+
};
|
|
284
|
+
const handleWheelZoom = (e) => {
|
|
285
|
+
const svg = getSvg();
|
|
286
|
+
if (!svg) return;
|
|
287
|
+
const containerRect = containerEl.getBoundingClientRect();
|
|
288
|
+
const svgRect = svg.getBoundingClientRect();
|
|
289
|
+
const mouseX = e.clientX - containerRect.left;
|
|
290
|
+
const mouseY = e.clientY - containerRect.top;
|
|
291
|
+
const svgCenterX = svgRect.left - containerRect.left + svgRect.width / 2;
|
|
292
|
+
const svgCenterY = svgRect.top - containerRect.top + svgRect.height / 2;
|
|
293
|
+
const offsetX = (mouseX - svgCenterX - posX.value) / scale.value;
|
|
294
|
+
const offsetY = (mouseY - svgCenterY - posY.value) / scale.value;
|
|
295
|
+
const delta = e.deltaY > 0 ? -0.05 : 0.05;
|
|
296
|
+
const newScale = Math.min(Math.max(scale.value + delta, 0.1), 10);
|
|
297
|
+
if (newScale === scale.value) return;
|
|
298
|
+
scale.value = newScale;
|
|
299
|
+
posX.value = mouseX - svgCenterX - offsetX * scale.value;
|
|
300
|
+
posY.value = mouseY - svgCenterY - offsetY * scale.value;
|
|
301
|
+
updateTransform(svg);
|
|
302
|
+
};
|
|
303
|
+
const throttledWheelZoom = throttle(handleWheelZoom, 20, { leading: true, trailing: true });
|
|
304
|
+
const onWheel = (e) => {
|
|
305
|
+
if (e.target === containerEl || containerEl.contains(e.target)) {
|
|
306
|
+
e.preventDefault();
|
|
307
|
+
throttledWheelZoom(e);
|
|
308
|
+
}
|
|
309
|
+
};
|
|
310
|
+
const onTouchStart = (e) => {
|
|
311
|
+
if (e.target === containerEl || containerEl.contains(e.target)) {
|
|
312
|
+
if (e.touches.length === 1) {
|
|
313
|
+
e.preventDefault();
|
|
314
|
+
isInteractingWithMermaid = true;
|
|
315
|
+
onStart(e.touches[0].clientX, e.touches[0].clientY);
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
};
|
|
319
|
+
const onTouchMove = (e) => {
|
|
320
|
+
if (isInteractingWithMermaid) {
|
|
321
|
+
e.preventDefault();
|
|
322
|
+
onMove(e.touches[0].clientX, e.touches[0].clientY);
|
|
323
|
+
}
|
|
324
|
+
};
|
|
325
|
+
containerEl.addEventListener("mousedown", onMouseDown);
|
|
326
|
+
document.addEventListener("mousemove", onMouseMove);
|
|
327
|
+
document.addEventListener("mouseup", onEnd);
|
|
328
|
+
containerEl.addEventListener("wheel", onWheel, { passive: false });
|
|
329
|
+
containerEl.addEventListener("touchstart", onTouchStart, { passive: false });
|
|
330
|
+
containerEl.addEventListener("touchmove", onTouchMove, { passive: false });
|
|
331
|
+
document.addEventListener("touchend", onEnd);
|
|
332
|
+
return () => {
|
|
333
|
+
containerEl.removeEventListener("mousedown", onMouseDown);
|
|
334
|
+
document.removeEventListener("mousemove", onMouseMove);
|
|
335
|
+
document.removeEventListener("mouseup", onEnd);
|
|
336
|
+
containerEl.removeEventListener("wheel", onWheel);
|
|
337
|
+
containerEl.removeEventListener("touchstart", onTouchStart);
|
|
338
|
+
containerEl.removeEventListener("touchmove", onTouchMove);
|
|
339
|
+
document.removeEventListener("touchend", onEnd);
|
|
340
|
+
document.body.style.userSelect = "";
|
|
341
|
+
};
|
|
342
|
+
};
|
|
343
|
+
const zoomIn = () => {
|
|
344
|
+
const svg = getSvg();
|
|
345
|
+
if (svg) {
|
|
346
|
+
scale.value = Math.min(scale.value + 0.2, 10);
|
|
347
|
+
updateTransform(svg);
|
|
348
|
+
}
|
|
349
|
+
};
|
|
350
|
+
const zoomOut = () => {
|
|
351
|
+
const svg = getSvg();
|
|
352
|
+
if (svg) {
|
|
353
|
+
scale.value = Math.max(scale.value - 0.2, 0.1);
|
|
354
|
+
updateTransform(svg);
|
|
355
|
+
}
|
|
356
|
+
};
|
|
357
|
+
const reset = () => {
|
|
358
|
+
const svg = getSvg();
|
|
359
|
+
if (svg) {
|
|
360
|
+
resetState();
|
|
361
|
+
updateTransform(svg);
|
|
362
|
+
}
|
|
363
|
+
};
|
|
364
|
+
const fullscreen = () => {
|
|
365
|
+
if (!container.value) return;
|
|
366
|
+
if (document.fullscreenElement) {
|
|
367
|
+
document.exitFullscreen();
|
|
368
|
+
} else {
|
|
369
|
+
container.value.requestFullscreen?.();
|
|
370
|
+
}
|
|
371
|
+
};
|
|
372
|
+
const initialize = () => {
|
|
373
|
+
if (!container.value) return;
|
|
374
|
+
resetState();
|
|
375
|
+
removeEvents = addInteractionEvents(container.value);
|
|
376
|
+
const svg = getSvg();
|
|
377
|
+
if (svg) {
|
|
378
|
+
updateTransform(svg);
|
|
379
|
+
}
|
|
380
|
+
};
|
|
381
|
+
const destroy = () => {
|
|
382
|
+
removeEvents?.();
|
|
383
|
+
removeEvents = null;
|
|
384
|
+
resetState();
|
|
385
|
+
};
|
|
386
|
+
watch(
|
|
387
|
+
() => container.value,
|
|
388
|
+
() => {
|
|
389
|
+
destroy();
|
|
390
|
+
resetState();
|
|
391
|
+
}
|
|
392
|
+
);
|
|
393
|
+
onUnmounted(destroy);
|
|
394
|
+
return {
|
|
395
|
+
zoomIn,
|
|
396
|
+
zoomOut,
|
|
397
|
+
reset,
|
|
398
|
+
fullscreen,
|
|
399
|
+
destroy,
|
|
400
|
+
initialize
|
|
401
|
+
};
|
|
402
|
+
}
|
|
403
|
+
export {
|
|
404
|
+
checkMermaidAvailable,
|
|
405
|
+
downloadSvgAsPng,
|
|
406
|
+
useMermaid,
|
|
407
|
+
useMermaidZoom
|
|
408
|
+
};
|
|
409
|
+
//# sourceMappingURL=x-markdown.es9.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"x-markdown.es9.js","sources":["../src/hooks/useMermaid.ts"],"sourcesContent":["import type { Ref } from 'vue'\r\nimport { throttle } from 'lodash-es'\r\nimport { computed, ref, watch, onUnmounted } from 'vue'\r\nimport type { MermaidZoomControls, UseMermaidZoomOptions, UseMermaidResult } from '../components/Mermaid/types'\r\n\r\nexport function downloadSvgAsPng(svg: string): void {\r\n if (!svg) return\r\n\r\n try {\r\n const svgDataUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`\r\n const img = new Image()\r\n\r\n img.onload = () => {\r\n try {\r\n const canvas = document.createElement('canvas')\r\n const ctx = canvas.getContext('2d', { willReadFrequently: false })\r\n if (!ctx) return\r\n\r\n const scale = 2\r\n canvas.width = img.width * scale\r\n canvas.height = img.height * scale\r\n ctx.imageSmoothingEnabled = true\r\n ctx.imageSmoothingQuality = 'high'\r\n\r\n ctx.fillStyle = '#ffffff'\r\n ctx.fillRect(0, 0, canvas.width, canvas.height)\r\n ctx.drawImage(img, 0, 0, canvas.width, canvas.height)\r\n\r\n const timestamp = new Date().toISOString().slice(0, 19).replace(/:/g, '-')\r\n\r\n try {\r\n canvas.toBlob(\r\n (blob) => {\r\n if (!blob) return\r\n const url = URL.createObjectURL(blob)\r\n const link = document.createElement('a')\r\n link.href = url\r\n link.download = `mermaid-diagram-${timestamp}.png`\r\n document.body.appendChild(link)\r\n link.click()\r\n document.body.removeChild(link)\r\n URL.revokeObjectURL(url)\r\n },\r\n 'image/png',\r\n 0.95,\r\n )\r\n } catch (toBlobError) {\r\n console.error('Failed to convert canvas to blob:', toBlobError)\r\n try {\r\n const dataUrl = canvas.toDataURL('image/png', 0.95)\r\n const link = document.createElement('a')\r\n link.href = dataUrl\r\n link.download = `mermaid-diagram-${timestamp}.png`\r\n document.body.appendChild(link)\r\n link.click()\r\n document.body.removeChild(link)\r\n } catch (dataUrlError) {\r\n console.error('Failed to convert canvas to data URL:', dataUrlError)\r\n }\r\n }\r\n } catch (canvasError) {\r\n console.error('Canvas operation failed:', canvasError)\r\n }\r\n }\r\n\r\n img.onerror = (error) => {\r\n console.error('Failed to load image:', error)\r\n }\r\n\r\n img.src = svgDataUrl\r\n } catch (error) {\r\n console.error('Failed to download SVG:', error)\r\n }\r\n}\r\n\r\ninterface UseMermaidOptions {\r\n id?: string\r\n theme?: 'default' | 'dark' | 'forest' | 'neutral' | string\r\n config?: any\r\n container?: HTMLElement | Ref<HTMLElement | null> | null\r\n}\r\n\r\ntype UseMermaidOptionsInput = UseMermaidOptions | Ref<UseMermaidOptions>\r\n\r\nlet mermaidPromise: Promise<any> | null = null\r\nlet hasShownMermaidHint = false\r\nlet mermaidAvailableCache: boolean | null = null\r\nlet mermaidCheckPromise: Promise<boolean> | null = null\r\n\r\n/**\r\n * 同步检查缓存状态(不触发检测)\r\n * @returns 缓存状态,null 表示未检测\r\n */\r\nexport function getMermaidAvailableCache(): boolean | null {\r\n return mermaidAvailableCache\r\n}\r\n\r\n/**\r\n * 检测 mermaid 是否可用(全局缓存,只检测一次)\r\n */\r\nexport async function checkMermaidAvailable(): Promise<boolean> {\r\n // 如果已经有缓存结果,直接返回\r\n if (mermaidAvailableCache !== null) {\r\n return mermaidAvailableCache\r\n }\r\n\r\n // 如果正在检测,返回检测 Promise\r\n if (mermaidCheckPromise) {\r\n return mermaidCheckPromise\r\n }\r\n\r\n // 开始检测\r\n mermaidCheckPromise = (async () => {\r\n try {\r\n // 尝试静态导入 mermaid\r\n await import('mermaid')\r\n mermaidAvailableCache = true\r\n return true\r\n } catch (error) {\r\n mermaidAvailableCache = false\r\n return false\r\n }\r\n })()\r\n\r\n return mermaidCheckPromise\r\n}\r\n\r\nconst showMermaidHint = () => {\r\n if (hasShownMermaidHint) return\r\n hasShownMermaidHint = true\r\n\r\n console.log(\r\n '%c[x-markdown]%c Mermaid 图表功能已降级为代码块显示',\r\n 'font-weight: bold; color: #9333ea;',\r\n 'color: #666;'\r\n )\r\n console.log(\r\n '%c如需 Mermaid 图表渲染功能,请安装:',\r\n 'color: #666; font-weight: bold;'\r\n )\r\n console.log(\r\n '%c pnpm add mermaid',\r\n 'color: #9333ea; font-family: monospace;'\r\n )\r\n console.log(\r\n '%c安装后请重启开发服务器',\r\n 'color: #999; font-size: 12px;'\r\n )\r\n}\r\n\r\nasync function loadMermaid() {\r\n if (typeof window === 'undefined') return null\r\n if (!mermaidPromise) {\r\n mermaidPromise = (async () => {\r\n try {\r\n // 直接静态导入,让 Vite/Rollup 在构建时处理\r\n const mod = await import('mermaid')\r\n return (mod as any).default\r\n } catch {\r\n showMermaidHint()\r\n return null\r\n }\r\n })()\r\n }\r\n return mermaidPromise\r\n}\r\n\r\ntype RenderTask = () => Promise<void>\r\nconst renderQueue: RenderTask[] = []\r\nlet isProcessingQueue = false\r\n\r\nasync function processRenderQueue() {\r\n if (isProcessingQueue) return\r\n isProcessingQueue = true\r\n\r\n while (renderQueue.length > 0) {\r\n const task = renderQueue.shift()\r\n if (task) {\r\n try {\r\n await task()\r\n } catch (err) {\r\n console.error('Mermaid render queue error:', err)\r\n }\r\n }\r\n }\r\n\r\n isProcessingQueue = false\r\n}\r\n\r\nfunction addToRenderQueue(task: RenderTask) {\r\n renderQueue.push(task)\r\n processRenderQueue()\r\n}\r\n\r\nexport function useMermaid(content: string | Ref<string>, options: UseMermaidOptionsInput = {}): UseMermaidResult {\r\n const optionsRef = computed(() => (typeof options === 'object' && 'value' in options ? options.value : options))\r\n const mermaidConfig = computed(() => ({\r\n suppressErrorRendering: true,\r\n startOnLoad: false,\r\n securityLevel: 'loose',\r\n theme: optionsRef.value.theme || 'default',\r\n ...(optionsRef.value.config || {}),\r\n }))\r\n const data = ref('')\r\n const error = ref<unknown>(null)\r\n const isLoading = ref(false)\r\n\r\n let isUnmounted = false\r\n\r\n const getRenderContainer = () => {\r\n const containerOption = optionsRef.value.container\r\n if (containerOption) {\r\n return typeof containerOption === 'object' && 'value' in containerOption ? containerOption.value : containerOption\r\n }\r\n return null\r\n }\r\n\r\n const throttledRender = throttle(\r\n () => {\r\n const contentValue = typeof content === 'string' ? content : content.value\r\n if (!contentValue?.trim()) {\r\n data.value = ''\r\n error.value = null\r\n isLoading.value = false\r\n return\r\n }\r\n\r\n isLoading.value = true\r\n\r\n addToRenderQueue(async () => {\r\n if (isUnmounted) return\r\n\r\n try {\r\n const mermaidInstance = await loadMermaid()\r\n if (!mermaidInstance) {\r\n data.value = contentValue\r\n error.value = null\r\n isLoading.value = false\r\n return\r\n }\r\n\r\n mermaidInstance.initialize(mermaidConfig.value)\r\n\r\n const isValid = await mermaidInstance.parse(contentValue.trim())\r\n if (!isValid) {\r\n data.value = ''\r\n error.value = new Error('Mermaid parse error: Invalid syntax')\r\n isLoading.value = false\r\n return\r\n }\r\n\r\n const renderId = `${optionsRef.value.id || 'mermaid'}-${Math.random().toString(36).substring(2, 11)}`\r\n const container = getRenderContainer()\r\n if (!container) {\r\n isLoading.value = false\r\n return\r\n }\r\n\r\n const { svg } = await mermaidInstance.render(renderId, contentValue, container)\r\n data.value = svg\r\n error.value = null\r\n isLoading.value = false\r\n } catch (err) {\r\n // Mermaid render error\r\n data.value = ''\r\n error.value = err\r\n isLoading.value = false\r\n }\r\n })\r\n },\r\n 100,\r\n { leading: false, trailing: true },\r\n )\r\n\r\n watch(\r\n [() => (typeof content === 'string' ? content : content.value), () => mermaidConfig.value],\r\n () => {\r\n throttledRender()\r\n },\r\n { immediate: true },\r\n )\r\n\r\n onUnmounted(() => {\r\n isUnmounted = true\r\n })\r\n\r\n return {\r\n data,\r\n error,\r\n isLoading,\r\n }\r\n}\r\n\r\nexport function useMermaidZoom(options: UseMermaidZoomOptions): MermaidZoomControls {\r\n const { container } = options\r\n\r\n const scale = ref(1)\r\n const posX = ref(0)\r\n const posY = ref(0)\r\n const isDragging = ref(false)\r\n\r\n let removeEvents: (() => void) | null = null\r\n\r\n const getSvg = () => container.value?.querySelector('.syntax-mermaid__content svg') as HTMLElement\r\n\r\n const updateTransform = (svg: HTMLElement) => {\r\n svg.style.transformOrigin = 'center center'\r\n svg.style.transform = `translate(${posX.value}px, ${posY.value}px) scale(${scale.value})`\r\n }\r\n\r\n const resetState = () => {\r\n scale.value = 1\r\n posX.value = 0\r\n posY.value = 0\r\n isDragging.value = false\r\n }\r\n\r\n const addInteractionEvents = (containerEl: HTMLElement) => {\r\n let startX = 0\r\n let startY = 0\r\n let isInteractingWithMermaid = false\r\n\r\n const onStart = (clientX: number, clientY: number) => {\r\n isDragging.value = true\r\n startX = clientX - posX.value\r\n startY = clientY - posY.value\r\n document.body.style.userSelect = 'none'\r\n }\r\n\r\n const onMove = (clientX: number, clientY: number) => {\r\n if (isDragging.value && isInteractingWithMermaid) {\r\n posX.value = clientX - startX\r\n posY.value = clientY - startY\r\n const svg = getSvg()\r\n if (svg) {\r\n updateTransform(svg)\r\n }\r\n }\r\n }\r\n\r\n const onEnd = () => {\r\n isDragging.value = false\r\n isInteractingWithMermaid = false\r\n document.body.style.userSelect = ''\r\n }\r\n\r\n const onMouseDown = (e: MouseEvent) => {\r\n if (e.button !== 0) return\r\n if (e.target === containerEl || containerEl.contains(e.target as Node)) {\r\n e.preventDefault()\r\n isInteractingWithMermaid = true\r\n onStart(e.clientX, e.clientY)\r\n }\r\n }\r\n\r\n const onMouseMove = (e: MouseEvent) => {\r\n if (isInteractingWithMermaid) {\r\n onMove(e.clientX, e.clientY)\r\n }\r\n }\r\n\r\n const handleWheelZoom = (e: WheelEvent) => {\r\n const svg = getSvg()\r\n if (!svg) return\r\n\r\n const containerRect = containerEl.getBoundingClientRect()\r\n const svgRect = svg.getBoundingClientRect()\r\n\r\n const mouseX = e.clientX - containerRect.left\r\n const mouseY = e.clientY - containerRect.top\r\n\r\n const svgCenterX = svgRect.left - containerRect.left + svgRect.width / 2\r\n const svgCenterY = svgRect.top - containerRect.top + svgRect.height / 2\r\n\r\n const offsetX = (mouseX - svgCenterX - posX.value) / scale.value\r\n const offsetY = (mouseY - svgCenterY - posY.value) / scale.value\r\n\r\n const delta = e.deltaY > 0 ? -0.05 : 0.05\r\n const newScale = Math.min(Math.max(scale.value + delta, 0.1), 10)\r\n\r\n if (newScale === scale.value) return\r\n\r\n scale.value = newScale\r\n\r\n posX.value = mouseX - svgCenterX - offsetX * scale.value\r\n posY.value = mouseY - svgCenterY - offsetY * scale.value\r\n\r\n updateTransform(svg)\r\n }\r\n\r\n const throttledWheelZoom = throttle(handleWheelZoom, 20, { leading: true, trailing: true })\r\n\r\n const onWheel = (e: WheelEvent) => {\r\n if (e.target === containerEl || containerEl.contains(e.target as Node)) {\r\n e.preventDefault()\r\n throttledWheelZoom(e)\r\n }\r\n }\r\n\r\n const onTouchStart = (e: TouchEvent) => {\r\n if (e.target === containerEl || containerEl.contains(e.target as Node)) {\r\n if (e.touches.length === 1) {\r\n e.preventDefault()\r\n isInteractingWithMermaid = true\r\n onStart(e.touches[0].clientX, e.touches[0].clientY)\r\n }\r\n }\r\n }\r\n\r\n const onTouchMove = (e: TouchEvent) => {\r\n if (isInteractingWithMermaid) {\r\n e.preventDefault()\r\n onMove(e.touches[0].clientX, e.touches[0].clientY)\r\n }\r\n }\r\n\r\n containerEl.addEventListener('mousedown', onMouseDown)\r\n document.addEventListener('mousemove', onMouseMove)\r\n document.addEventListener('mouseup', onEnd)\r\n containerEl.addEventListener('wheel', onWheel, { passive: false })\r\n containerEl.addEventListener('touchstart', onTouchStart, { passive: false })\r\n containerEl.addEventListener('touchmove', onTouchMove, { passive: false })\r\n document.addEventListener('touchend', onEnd)\r\n\r\n return () => {\r\n containerEl.removeEventListener('mousedown', onMouseDown)\r\n document.removeEventListener('mousemove', onMouseMove)\r\n document.removeEventListener('mouseup', onEnd)\r\n containerEl.removeEventListener('wheel', onWheel)\r\n containerEl.removeEventListener('touchstart', onTouchStart)\r\n containerEl.removeEventListener('touchmove', onTouchMove)\r\n document.removeEventListener('touchend', onEnd)\r\n document.body.style.userSelect = ''\r\n }\r\n }\r\n\r\n const zoomIn = () => {\r\n const svg = getSvg()\r\n if (svg) {\r\n scale.value = Math.min(scale.value + 0.2, 10)\r\n updateTransform(svg)\r\n }\r\n }\r\n\r\n const zoomOut = () => {\r\n const svg = getSvg()\r\n if (svg) {\r\n scale.value = Math.max(scale.value - 0.2, 0.1)\r\n updateTransform(svg)\r\n }\r\n }\r\n\r\n const reset = () => {\r\n const svg = getSvg()\r\n if (svg) {\r\n resetState()\r\n updateTransform(svg)\r\n }\r\n }\r\n\r\n const fullscreen = () => {\r\n if (!container.value) return\r\n\r\n if (document.fullscreenElement) {\r\n document.exitFullscreen()\r\n } else {\r\n container.value.requestFullscreen?.()\r\n }\r\n }\r\n\r\n const initialize = () => {\r\n if (!container.value) return\r\n\r\n resetState()\r\n\r\n removeEvents = addInteractionEvents(container.value)\r\n\r\n const svg = getSvg()\r\n if (svg) {\r\n updateTransform(svg)\r\n }\r\n }\r\n\r\n const destroy = () => {\r\n removeEvents?.()\r\n removeEvents = null\r\n resetState()\r\n }\r\n\r\n watch(\r\n () => container.value,\r\n () => {\r\n destroy()\r\n resetState()\r\n },\r\n )\r\n\r\n onUnmounted(destroy)\r\n\r\n return {\r\n zoomIn,\r\n zoomOut,\r\n reset,\r\n fullscreen,\r\n destroy,\r\n initialize,\r\n }\r\n}\r\n"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,84 +1,84 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "@hejiayue/x-markdown-test",
|
|
3
|
-
"version": "0.0.1-beta.
|
|
4
|
-
"description": "A markdown component library for Vue",
|
|
5
|
-
"type": "module",
|
|
6
|
-
"main": "./dist/
|
|
7
|
-
"module": "./dist/
|
|
8
|
-
"types": "./dist/types/index.d.ts",
|
|
9
|
-
"exports": {
|
|
10
|
-
".": {
|
|
11
|
-
"import": {
|
|
12
|
-
"types": "./dist/types/index.d.ts",
|
|
13
|
-
"default": "./dist/
|
|
14
|
-
},
|
|
15
|
-
"require": {
|
|
16
|
-
"types": "./dist/types/index.d.ts",
|
|
17
|
-
"default": "./dist/
|
|
18
|
-
}
|
|
19
|
-
},
|
|
20
|
-
"./style": "./dist/style.css"
|
|
21
|
-
},
|
|
22
|
-
"files": [
|
|
23
|
-
"dist"
|
|
24
|
-
],
|
|
25
|
-
"sideEffects": [
|
|
26
|
-
"*.css",
|
|
27
|
-
"*.scss",
|
|
28
|
-
"*.vue"
|
|
29
|
-
],
|
|
30
|
-
"
|
|
31
|
-
"
|
|
32
|
-
"
|
|
33
|
-
"
|
|
34
|
-
"
|
|
35
|
-
|
|
36
|
-
"
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
"
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
"
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
"
|
|
49
|
-
"
|
|
50
|
-
"
|
|
51
|
-
"
|
|
52
|
-
"
|
|
53
|
-
"
|
|
54
|
-
"
|
|
55
|
-
"
|
|
56
|
-
"
|
|
57
|
-
"
|
|
58
|
-
"
|
|
59
|
-
"
|
|
60
|
-
"
|
|
61
|
-
"
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
"
|
|
65
|
-
"
|
|
66
|
-
"
|
|
67
|
-
"
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
"@
|
|
71
|
-
"
|
|
72
|
-
"
|
|
73
|
-
"
|
|
74
|
-
"
|
|
75
|
-
"vue": "^
|
|
76
|
-
"
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
"
|
|
80
|
-
"
|
|
81
|
-
"
|
|
82
|
-
"
|
|
83
|
-
}
|
|
84
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"name": "@hejiayue/x-markdown-test",
|
|
3
|
+
"version": "0.0.1-beta.115",
|
|
4
|
+
"description": "A markdown component library for Vue",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"main": "./dist/index.cjs",
|
|
7
|
+
"module": "./dist/index.js",
|
|
8
|
+
"types": "./dist/types/index.d.ts",
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": {
|
|
12
|
+
"types": "./dist/types/index.d.ts",
|
|
13
|
+
"default": "./dist/index.js"
|
|
14
|
+
},
|
|
15
|
+
"require": {
|
|
16
|
+
"types": "./dist/types/index.d.ts",
|
|
17
|
+
"default": "./dist/index.cjs"
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
"./style": "./dist/style.css"
|
|
21
|
+
},
|
|
22
|
+
"files": [
|
|
23
|
+
"dist"
|
|
24
|
+
],
|
|
25
|
+
"sideEffects": [
|
|
26
|
+
"*.css",
|
|
27
|
+
"*.scss",
|
|
28
|
+
"*.vue"
|
|
29
|
+
],
|
|
30
|
+
"scripts": {
|
|
31
|
+
"dev": "vite",
|
|
32
|
+
"build": "vite build && tsc --project tsconfig.build.json",
|
|
33
|
+
"lint": "eslint src --ext .ts,.tsx,.vue",
|
|
34
|
+
"clean": "rm -rf dist node_modules"
|
|
35
|
+
},
|
|
36
|
+
"keywords": [
|
|
37
|
+
"vue",
|
|
38
|
+
"markdown",
|
|
39
|
+
"component",
|
|
40
|
+
"vue3"
|
|
41
|
+
],
|
|
42
|
+
"author": "",
|
|
43
|
+
"license": "MIT",
|
|
44
|
+
"peerDependencies": {
|
|
45
|
+
"vue": "^3.3.0"
|
|
46
|
+
},
|
|
47
|
+
"dependencies": {
|
|
48
|
+
"deepmerge": "^4.3.1",
|
|
49
|
+
"dompurify": "^3.0.0",
|
|
50
|
+
"element-plus": "^2.5.0",
|
|
51
|
+
"hast": "^1.0.0",
|
|
52
|
+
"lodash-es": "^4.17.21",
|
|
53
|
+
"mdast": "^3.0.0",
|
|
54
|
+
"mdast-util-to-hast": "^13.2.1",
|
|
55
|
+
"property-information": "^7.1.0",
|
|
56
|
+
"rehype": "^13.0.0",
|
|
57
|
+
"rehype-katex": "^7.0.0",
|
|
58
|
+
"rehype-raw": "^7.0.0",
|
|
59
|
+
"rehype-sanitize": "^6.0.0",
|
|
60
|
+
"remark": "^15.0.0",
|
|
61
|
+
"remark-breaks": "^4.0.0",
|
|
62
|
+
"remark-gfm": "^4.0.0",
|
|
63
|
+
"remark-math": "^6.0.0",
|
|
64
|
+
"remark-parse": "^11.0.0",
|
|
65
|
+
"remark-rehype": "^11.0.0",
|
|
66
|
+
"unified": "^11.0.0",
|
|
67
|
+
"unist-util-visit": "^5.0.0"
|
|
68
|
+
},
|
|
69
|
+
"devDependencies": {
|
|
70
|
+
"@types/dompurify": "^3.0.0",
|
|
71
|
+
"@types/hast": "^3.0.4",
|
|
72
|
+
"@types/lodash-es": "^4.17.12",
|
|
73
|
+
"@types/mdast": "^4.0.4",
|
|
74
|
+
"@types/node": "^20.0.0",
|
|
75
|
+
"@vitejs/plugin-vue": "^5.0.0",
|
|
76
|
+
"@vueuse/core": "^14.1.0",
|
|
77
|
+
"sass": "^1.70.0",
|
|
78
|
+
"terser": "^5.44.1",
|
|
79
|
+
"typescript": "^5.3.0",
|
|
80
|
+
"vite": "^5.0.0",
|
|
81
|
+
"vue": "^3.3.0",
|
|
82
|
+
"vue-tsc": "^2.0.0"
|
|
83
|
+
}
|
|
84
|
+
}
|
package/LICENSE
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
MIT License
|
|
2
|
-
|
|
3
|
-
Copyright (c) 2025 element-plus-x
|
|
4
|
-
|
|
5
|
-
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
-
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
-
in the Software without restriction, including without limitation the rights
|
|
8
|
-
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
-
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
-
furnished to do so, subject to the following conditions:
|
|
11
|
-
|
|
12
|
-
The above copyright notice and this permission notice shall be included in all
|
|
13
|
-
copies or substantial portions of the Software.
|
|
14
|
-
|
|
15
|
-
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
-
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
-
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
-
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
-
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
-
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
-
SOFTWARE.
|