@matechat/ng 0.0.1-alpha.0 → 20.0.1-alpha.0
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/Base/base.component.d.ts +5 -5
- package/Bubble/bubble.component.d.ts +7 -5
- package/Input/button/button.component.d.ts +3 -2
- package/Input/input.component.d.ts +13 -4
- package/MarkdownCard/code-block.component.d.ts +49 -0
- package/MarkdownCard/index.d.ts +3 -0
- package/MarkdownCard/markdown-card.component.d.ts +200 -0
- package/MarkdownCard/markdown-card.module.d.ts +13 -0
- package/README.md +117 -13
- package/components-common/Base/foundation.d.ts +2 -0
- package/components-common/MarkdownCard/codeblock-foundation.d.ts +21 -0
- package/components-common/MarkdownCard/common/MDCardService.d.ts +14 -0
- package/components-common/MarkdownCard/common/MermaidService.d.ts +23 -0
- package/components-common/MarkdownCard/common/mdCard.types.d.ts +56 -0
- package/components-common/MarkdownCard/common/parser.d.ts +150 -0
- package/components-common/MarkdownCard/foundation.d.ts +38 -0
- package/esm2022/Base/base.component.mjs +2 -2
- package/esm2022/Bubble/bubble.component.mjs +15 -8
- package/esm2022/Input/button/button.component.mjs +1 -1
- package/esm2022/Input/input.component.mjs +107 -3
- package/esm2022/Locale/locale.service.mjs +5 -5
- package/esm2022/MarkdownCard/code-block.component.mjs +175 -0
- package/esm2022/MarkdownCard/index.mjs +4 -0
- package/esm2022/MarkdownCard/markdown-card.component.mjs +436 -0
- package/esm2022/MarkdownCard/markdown-card.module.mjs +44 -0
- package/esm2022/components-common/Base/foundation.mjs +4 -1
- package/esm2022/components-common/Input/foundation.mjs +1 -2
- package/esm2022/components-common/MarkdownCard/codeblock-foundation.mjs +132 -0
- package/esm2022/components-common/MarkdownCard/common/MDCardService.mjs +69 -0
- package/esm2022/components-common/MarkdownCard/common/MermaidService.mjs +222 -0
- package/esm2022/components-common/MarkdownCard/common/mdCard.types.mjs +6 -0
- package/esm2022/components-common/MarkdownCard/common/parser.mjs +194 -0
- package/esm2022/components-common/MarkdownCard/foundation.mjs +84 -0
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/matechat-ng.mjs +1514 -17
- package/fesm2022/matechat-ng.mjs.map +1 -1
- package/package.json +12 -7
- package/public-api.d.ts +1 -0
- package/fesm2022/matechat-ng-en-us-DsYnUbZd.mjs +0 -28
- package/fesm2022/matechat-ng-en-us-DsYnUbZd.mjs.map +0 -1
- package/fesm2022/matechat-ng-zh-cn--_YVZHnW.mjs +0 -28
- package/fesm2022/matechat-ng-zh-cn--_YVZHnW.mjs.map +0 -1
|
@@ -42,7 +42,6 @@ export class InputFoundation extends BaseFoundation {
|
|
|
42
42
|
this._adapter.emitChange();
|
|
43
43
|
}
|
|
44
44
|
emitChange() {
|
|
45
|
-
console.log('emitChange---2');
|
|
46
45
|
this._adapter.emitChange();
|
|
47
46
|
}
|
|
48
47
|
submit(inputValue) {
|
|
@@ -69,4 +68,4 @@ export class InputFoundation extends BaseFoundation {
|
|
|
69
68
|
}
|
|
70
69
|
}
|
|
71
70
|
}
|
|
72
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm91bmRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMtbmcvc3JjL2NvbXBvbmVudHMtY29tbW9uL0lucHV0L2ZvdW5kYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxjQUFrQyxNQUFNLG9CQUFvQixDQUFDO0FBQ3BFLE9BQU8sRUFDTCxXQUFXLEVBQ1gsWUFBWSxFQUVaLGNBQWMsR0FDZixNQUFNLGdCQUFnQixDQUFDO0FBUXhCLE1BQU0sT0FBTyxlQUFnQixTQUFRLGNBQTRCO0lBQy9ELFlBQVksT0FBcUI7UUFDL0IsS0FBSyxDQUFDLEVBQUUsR0FBRyxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQ3hCLENBQUM7SUFFRCxlQUFlO1FBQ2IsTUFBTSxFQUFFLFFBQVEsRUFBRSxXQUFXLEVBQUUsT0FBTyxFQUFFLEdBQUcsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQzNELE9BQU87WUFDTCxVQUFVLEVBQUUsSUFBSTtZQUNoQixtQkFBbUIsRUFBRSxRQUFRO1lBQzdCLGlCQUFpQixFQUFFLFdBQVcsS0FBSyxXQUFXLENBQUMsTUFBTTtZQUNyRCxxQkFBcUIsRUFBRSxPQUFPLEtBQUssWUFBWSxDQUFDLFVBQVU7U0FDM0QsQ0FBQztJQUNKLENBQUM7SUFFRCxrQkFBa0I7UUFDaEIsTUFBTSxFQUFFLFdBQVcsRUFBRSxjQUFjLEVBQUUsR0FBRyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDeEQsSUFBSSxXQUFXLEVBQUUsQ0FBQztZQUNoQixPQUFPLFdBQVcsQ0FBQztRQUNyQixDQUFDO1FBRUQsSUFBSSxRQUFRLEdBQUcsRUFBRSxDQUFDO1FBQ2xCLElBQUksYUFBYSxHQUFHLEVBQUUsQ0FBQztRQUV2QixJQUFJLGNBQWMsS0FBSyxjQUFjLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDNUMsUUFBUSxHQUFHLE9BQU8sQ0FBQztZQUNuQixhQUFhLEdBQUcsZUFBZSxDQUFDO1FBQ2xDLENBQUM7YUFBTSxJQUFJLGNBQWMsS0FBSyxjQUFjLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDeEQsUUFBUSxHQUFHLGVBQWUsQ0FBQztZQUMzQixhQUFhLEdBQUcsT0FBTyxDQUFDO1FBQzFCLENBQUM7UUFFRCxPQUFPLFFBQVE7WUFDYixDQUFDLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsOEJBQThCLEVBQUU7Z0JBQ25ELFFBQVE7Z0JBQ1IsYUFBYTthQUNkLENBQUM7WUFDSixDQUFDLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsbUJBQW1CLENBQUMsQ0FBQztJQUNoRCxDQUFDO0lBRUQsVUFBVTtRQUNSLElBQUksQ0FBQyxRQUFRLENBQUM7WUFDWixVQUFVLEVBQUUsRUFBRTtTQUNmLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxRQUFRLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDN0IsQ0FBQztJQUVELFVBQVU7UUFDUixJQUFJLENBQUMsUUFBUSxDQUFDLFVBQVUsRUFBRSxDQUFDO0lBQzdCLENBQUM7SUFFRCxNQUFNLENBQUMsVUFBa0I7UUFDdkIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVELFNBQVMsQ0FBQyxLQUFvQjtRQUM1QixNQUFNLEVBQUUsY0FBYyxFQUFFLElBQUksRUFBRSxHQUFHLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUNqRCxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQy9DLElBQUksY0FBYyxLQUFLLElBQUksRUFBRSxDQUFDO1lBQzVCLE9BQU87UUFDVCxDQUFDO1FBRUQsTUFBTSxRQUFRLEdBQ1osY0FBYyxLQUFLLGNBQWMsQ0FBQyxLQUFLO1lBQ3JDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxRQUFRO1lBQ2pCLENBQUMsQ0FBQyxjQUFjLEtBQUssY0FBYyxDQUFDLFVBQVU7Z0JBQzlDLENBQUMsQ0FBQyxLQUFLLENBQUMsUUFBUTtnQkFDaEIsQ0FBQyxDQUFDLEtBQUssQ0FBQztRQUVaLElBQUksUUFBUSxJQUFJLEtBQUssQ0FBQyxHQUFHLEtBQUssT0FBTyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDL0MsS0FBSyxDQUFDLGNBQWMsRUFBRSxDQUFDO1lBQ3ZCLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7WUFDeEIsSUFBSSxDQUFDLFFBQVEsQ0FBQztnQkFDWixVQUFVLEVBQUUsRUFBRTthQUNmLENBQUMsQ0FBQztZQUNILElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUNwQixDQUFDO0lBQ0gsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJhc2VGb3VuZGF0aW9uLCB7IERlZmF1bHRBZGFwdGVyIH0gZnJvbSAnLi4vQmFzZS9mb3VuZGF0aW9uJztcbmltcG9ydCB7XG4gIERpc3BsYXlUeXBlLFxuICBJbnB1dFZhcmlhbnQsXG4gIFNlbmRCdG5WYXJpYW50LFxuICBTdWJtaXRTaG9ydEtleSxcbn0gZnJvbSAnLi9jb21tb24vdHlwZXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIElucHV0QWRhcHRlciBleHRlbmRzIERlZmF1bHRBZGFwdGVyIHtcbiAgbG9jYWxlKGtleTogc3RyaW5nLCBwYXJhbXM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+KTogc3RyaW5nO1xuICBlbWl0Q2hhbmdlKCk6IHZvaWQ7XG4gIHN1Ym1pdChpbnB1dFZhbHVlOiBzdHJpbmcpOiB2b2lkO1xufVxuXG5leHBvcnQgY2xhc3MgSW5wdXRGb3VuZGF0aW9uIGV4dGVuZHMgQmFzZUZvdW5kYXRpb248SW5wdXRBZGFwdGVyPiB7XG4gIGNvbnN0cnVjdG9yKGFkYXB0ZXI6IElucHV0QWRhcHRlcikge1xuICAgIHN1cGVyKHsgLi4uYWRhcHRlciB9KTtcbiAgfVxuXG4gIGdldElucHV0Q2xhc3NlcygpIHtcbiAgICBjb25zdCB7IGRpc2FibGVkLCBkaXNwbGF5VHlwZSwgdmFyaWFudCB9ID0gdGhpcy5nZXRQcm9wcygpO1xuICAgIHJldHVybiB7XG4gICAgICAnbWMtaW5wdXQnOiB0cnVlLFxuICAgICAgJ21jLWlucHV0LWRpc2FibGVkJzogZGlzYWJsZWQsXG4gICAgICAnbWMtaW5wdXQtc2ltcGxlJzogZGlzcGxheVR5cGUgPT09IERpc3BsYXlUeXBlLlNpbXBsZSxcbiAgICAgICdtYy1pbnB1dC1ib3JkZXJsZXNzJzogdmFyaWFudCA9PT0gSW5wdXRWYXJpYW50LkJvcmRlckxlc3MsXG4gICAgfTtcbiAgfVxuXG4gIGdldFBsYWNlaG9sZGVyVGV4dCgpIHtcbiAgICBjb25zdCB7IHBsYWNlaG9sZGVyLCBzdWJtaXRTaG9ydEtleSB9ID0gdGhpcy5nZXRQcm9wcygpO1xuICAgIGlmIChwbGFjZWhvbGRlcikge1xuICAgICAgcmV0dXJuIHBsYWNlaG9sZGVyO1xuICAgIH1cblxuICAgIGxldCBlbnRlcktleSA9ICcnO1xuICAgIGxldCBzaGlmdEVudGVyS2V5ID0gJyc7XG5cbiAgICBpZiAoc3VibWl0U2hvcnRLZXkgPT09IFN1Ym1pdFNob3J0S2V5LkVudGVyKSB7XG4gICAgICBlbnRlcktleSA9ICdFbnRlcic7XG4gICAgICBzaGlmdEVudGVyS2V5ID0gJ1NoaWZ0ICsgRW50ZXInO1xuICAgIH0gZWxzZSBpZiAoc3VibWl0U2hvcnRLZXkgPT09IFN1Ym1pdFNob3J0S2V5LlNoaWZ0RW50ZXIpIHtcbiAgICAgIGVudGVyS2V5ID0gJ1NoaWZ0ICsgRW50ZXInO1xuICAgICAgc2hpZnRFbnRlcktleSA9ICdFbnRlcic7XG4gICAgfVxuXG4gICAgcmV0dXJuIGVudGVyS2V5XG4gICAgICA/IHRoaXMuX2FkYXB0ZXIubG9jYWxlKCdJbnB1dC5wbGVhc2VFbnRlclBsYWNlaG9sZGVyJywge1xuICAgICAgICAgIGVudGVyS2V5LFxuICAgICAgICAgIHNoaWZ0RW50ZXJLZXksXG4gICAgICAgIH0pXG4gICAgICA6IHRoaXMuX2FkYXB0ZXIubG9jYWxlKCdJbnB1dC5wbGVhc2VFbnRlcicpO1xuICB9XG5cbiAgY2xlYXJJbnB1dCgpOiB2b2lkIHtcbiAgICB0aGlzLnNldFN0YXRlKHtcbiAgICAgIGlucHV0VmFsdWU6ICcnLFxuICAgIH0pO1xuICAgIHRoaXMuX2FkYXB0ZXIuZW1pdENoYW5nZSgpO1xuICB9XG5cbiAgZW1pdENoYW5nZSgpOiB2b2lkIHtcbiAgICB0aGlzLl9hZGFwdGVyLmVtaXRDaGFuZ2UoKTtcbiAgfVxuXG4gIHN1Ym1pdChpbnB1dFZhbHVlOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLl9hZGFwdGVyLnN1Ym1pdChpbnB1dFZhbHVlKTtcbiAgfVxuXG4gIG9uS2V5ZG93bihldmVudDogS2V5Ym9hcmRFdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHsgc3VibWl0U2hvcnRLZXksIGxvY2sgfSA9IHRoaXMuZ2V0UHJvcHMoKTtcbiAgICBjb25zdCBpbnB1dFZhbHVlID0gdGhpcy5nZXRTdGF0ZSgnaW5wdXRWYWx1ZScpO1xuICAgIGlmIChzdWJtaXRTaG9ydEtleSA9PT0gbnVsbCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnN0IHNoaWZ0S2V5ID1cbiAgICAgIHN1Ym1pdFNob3J0S2V5ID09PSBTdWJtaXRTaG9ydEtleS5FbnRlclxuICAgICAgICA/ICFldmVudC5zaGlmdEtleVxuICAgICAgICA6IHN1Ym1pdFNob3J0S2V5ID09PSBTdWJtaXRTaG9ydEtleS5TaGlmdEVudGVyXG4gICAgICAgID8gZXZlbnQuc2hpZnRLZXlcbiAgICAgICAgOiBmYWxzZTtcblxuICAgIGlmIChzaGlmdEtleSAmJiBldmVudC5rZXkgPT09ICdFbnRlcicgJiYgIWxvY2spIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgICB0aGlzLnN1Ym1pdChpbnB1dFZhbHVlKTtcbiAgICAgIHRoaXMuc2V0U3RhdGUoe1xuICAgICAgICBpbnB1dFZhbHVlOiAnJyxcbiAgICAgIH0pO1xuICAgICAgdGhpcy5lbWl0Q2hhbmdlKCk7XG4gICAgfVxuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
import BaseFoundation from '../Base/foundation';
|
|
2
|
+
import { MermaidService } from './common/MermaidService';
|
|
3
|
+
import hljs from 'highlight.js';
|
|
4
|
+
import { MDCardService } from './common/MDCardService';
|
|
5
|
+
export class CodeBlockFoundation extends BaseFoundation {
|
|
6
|
+
constructor(adapter) {
|
|
7
|
+
super({ ...adapter });
|
|
8
|
+
this.toggleExpand = () => {
|
|
9
|
+
this.setState({ expanded: !this.getStates().expanded });
|
|
10
|
+
};
|
|
11
|
+
this.zoomOut = () => {
|
|
12
|
+
const container = this._adapter.getContainer();
|
|
13
|
+
if (container && this.mermaidService) {
|
|
14
|
+
this.mermaidService.zoomOut(container);
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
this.zoomIn = () => {
|
|
18
|
+
const container = this._adapter.getContainer();
|
|
19
|
+
if (container && this.mermaidService) {
|
|
20
|
+
this.mermaidService.zoomIn(container);
|
|
21
|
+
}
|
|
22
|
+
};
|
|
23
|
+
this.checkIsMermaid = () => {
|
|
24
|
+
const { enableMermaid, language } = this.getProps();
|
|
25
|
+
return enableMermaid && language?.toLowerCase() === 'mermaid';
|
|
26
|
+
};
|
|
27
|
+
this.download = () => {
|
|
28
|
+
const container = this._adapter.getContainer();
|
|
29
|
+
if (container && this.mermaidService) {
|
|
30
|
+
this.mermaidService.download(container);
|
|
31
|
+
}
|
|
32
|
+
};
|
|
33
|
+
this.handleCopySuccess = () => {
|
|
34
|
+
this.setState({ copied: true });
|
|
35
|
+
setTimeout(() => {
|
|
36
|
+
this.setState({ copied: false });
|
|
37
|
+
}, 1500);
|
|
38
|
+
};
|
|
39
|
+
this.updateHighlightedCode = () => {
|
|
40
|
+
const { code, language } = this.getProps();
|
|
41
|
+
let highlightedCode = '';
|
|
42
|
+
try {
|
|
43
|
+
const typeIndex = code.indexOf(`<span class="mc-typewriter`);
|
|
44
|
+
if (language && hljs.getLanguage(language)) {
|
|
45
|
+
if (typeIndex !== -1) {
|
|
46
|
+
highlightedCode =
|
|
47
|
+
hljs.highlight(code.slice(0, typeIndex), {
|
|
48
|
+
language,
|
|
49
|
+
}).value + code.slice(typeIndex);
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
highlightedCode = hljs.highlight(code, {
|
|
53
|
+
language,
|
|
54
|
+
}).value;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
else {
|
|
58
|
+
if (typeof hljs.highlightAuto !== 'undefined') {
|
|
59
|
+
if (typeIndex !== -1) {
|
|
60
|
+
highlightedCode =
|
|
61
|
+
hljs.highlightAuto(code.slice(0, typeIndex)).value +
|
|
62
|
+
code.slice(typeIndex);
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
highlightedCode = hljs.highlightAuto(code).value;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
highlightedCode = this.mdCardService.filterHtml(code);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
catch (_) {
|
|
74
|
+
highlightedCode = code;
|
|
75
|
+
}
|
|
76
|
+
this._adapter.highlightCodeChange(highlightedCode, language);
|
|
77
|
+
};
|
|
78
|
+
this.renderMermaid = async () => {
|
|
79
|
+
const { code, theme, mermaidConfig, } = this.getProps();
|
|
80
|
+
const { mermaidContentRef } = this.getStates();
|
|
81
|
+
const isMermaid = this.checkIsMermaid();
|
|
82
|
+
if (!isMermaid || !code || !mermaidContentRef) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
if (!this.mermaidService) {
|
|
86
|
+
try {
|
|
87
|
+
this.mermaidService = new MermaidService();
|
|
88
|
+
const config = {
|
|
89
|
+
theme: theme === 'dark' ? 'dark' : 'default',
|
|
90
|
+
...mermaidConfig,
|
|
91
|
+
};
|
|
92
|
+
this.mermaidService.setConfig(config);
|
|
93
|
+
}
|
|
94
|
+
catch (error) {
|
|
95
|
+
console.error('Failed to load MermaidService:', error);
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
this.nextTick(async () => {
|
|
100
|
+
const container = mermaidContentRef.nativeElement;
|
|
101
|
+
if (container) {
|
|
102
|
+
// 移除打字效果相关的span标签
|
|
103
|
+
const cleanCode = code.replace(/<span[^>]*\bclass\s*=\s*['"]mc-typewriter[^>]*>([\s\S]*?)<\/span>/g, `$1`);
|
|
104
|
+
await this.mermaidService?.renderToContainer(container, cleanCode, theme);
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
};
|
|
108
|
+
this.mdCardService = new MDCardService();
|
|
109
|
+
}
|
|
110
|
+
copyCodeInternal() {
|
|
111
|
+
const { code } = this.getProps();
|
|
112
|
+
if (navigator.clipboard) {
|
|
113
|
+
navigator.clipboard.writeText(code).then(() => {
|
|
114
|
+
this.handleCopySuccess();
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
const textarea = document.createElement('textarea');
|
|
119
|
+
textarea.style.position = 'fixed';
|
|
120
|
+
textarea.style.top = '-9999px';
|
|
121
|
+
textarea.style.left = '-9999px';
|
|
122
|
+
textarea.style.zIndex = '-1';
|
|
123
|
+
textarea.value = code;
|
|
124
|
+
document.body.appendChild(textarea);
|
|
125
|
+
textarea.select();
|
|
126
|
+
document.execCommand('copy');
|
|
127
|
+
document.body.removeChild(textarea);
|
|
128
|
+
this.handleCopySuccess();
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codeblock-foundation.js","sourceRoot":"","sources":["../../../../../projects/components-ng/src/components-common/MarkdownCard/codeblock-foundation.ts"],"names":[],"mappings":"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","sourcesContent":["import BaseFoundation, { DefaultAdapter } from '../Base/foundation';\nimport type { Token } from 'markdown-it';\nimport type { ASTNode, MermaidConfig } from './common/mdCard.types';\nimport { defaultTypingConfig } from './common/mdCard.types';\nimport { MermaidService } from './common/MermaidService';\nimport hljs from 'highlight.js';\nimport { MDCardService } from './common/MDCardService';\n\nexport interface CodeBlockAdapter extends DefaultAdapter {\n  getContainer(): HTMLElement | null;\n  highlightCodeChange(code: string, language: string): void;\n}\n\nexport class CodeBlockFoundation extends BaseFoundation<CodeBlockAdapter> {\n  mermaidService?: MermaidService;\n  mdCardService: MDCardService;\n  constructor(adapter: CodeBlockAdapter) {\n    super({ ...adapter });\n    this.mdCardService = new MDCardService();\n  }\n\n  toggleExpand = () => {\n    this.setState({ expanded: !this.getStates().expanded });\n  };\n\n  zoomOut = () => {\n    const container = this._adapter.getContainer();\n    if (container && this.mermaidService) {\n      this.mermaidService.zoomOut(container);\n    }\n  };\n\n  zoomIn = () => {\n    const container = this._adapter.getContainer();\n    if (container && this.mermaidService) {\n      this.mermaidService.zoomIn(container);\n    }\n  };\n\n  checkIsMermaid = () => {\n    const { enableMermaid, language } = this.getProps();\n    return enableMermaid && language?.toLowerCase() === 'mermaid';\n  };\n\n  download = () => {\n    const container = this._adapter.getContainer();\n    if (container && this.mermaidService) {\n      this.mermaidService.download(container);\n    }\n  };\n\n  handleCopySuccess = () => {\n    this.setState({ copied: true });\n    setTimeout(() => {\n      this.setState({ copied: false });\n    }, 1500);\n  };\n\n  copyCodeInternal() {\n    const { code } = this.getProps();\n    if (navigator.clipboard) {\n      navigator.clipboard.writeText(code).then(() => {\n        this.handleCopySuccess();\n      });\n    } else {\n      const textarea = document.createElement('textarea');\n      textarea.style.position = 'fixed';\n      textarea.style.top = '-9999px';\n      textarea.style.left = '-9999px';\n      textarea.style.zIndex = '-1';\n      textarea.value = code;\n      document.body.appendChild(textarea);\n      textarea.select();\n      document.execCommand('copy');\n      document.body.removeChild(textarea);\n      this.handleCopySuccess();\n    }\n  }\n\n  updateHighlightedCode = () => {\n    const { code, language } = this.getProps();\n    let highlightedCode = '';\n    try {\n      const typeIndex = code.indexOf(`<span class=\"mc-typewriter`);\n\n      if (language && hljs.getLanguage(language)) {\n        if (typeIndex !== -1) {\n          highlightedCode =\n            hljs.highlight(code.slice(0, typeIndex), {\n              language,\n            }).value + code.slice(typeIndex);\n        } else {\n          highlightedCode = hljs.highlight(code, {\n            language,\n          }).value;\n        }\n      } else {\n        if (typeof hljs.highlightAuto !== 'undefined') {\n          if (typeIndex !== -1) {\n            highlightedCode =\n              hljs.highlightAuto(code.slice(0, typeIndex)).value +\n              code.slice(typeIndex);\n          } else {\n            highlightedCode = hljs.highlightAuto(code).value;\n          }\n        } else {\n          highlightedCode = this.mdCardService.filterHtml(code);\n        }\n      }\n    } catch (_) {\n      highlightedCode = code;\n    }\n    this._adapter.highlightCodeChange(highlightedCode, language);\n  };\n\n  renderMermaid = async () => {\n    const {\n      code,\n      theme,\n      mermaidConfig,\n    } = this.getProps();\n    const { mermaidContentRef} = this.getStates();\n    const isMermaid = this.checkIsMermaid();\n    if (!isMermaid || !code || !mermaidContentRef) {\n      return;\n    }\n\n    if (!this.mermaidService) {\n      try {\n        this.mermaidService = new MermaidService();\n        const config: MermaidConfig = {\n          theme: theme === 'dark' ? 'dark' : 'default',\n          ...mermaidConfig,\n        };\n        this.mermaidService.setConfig(config);\n      } catch (error) {\n        console.error('Failed to load MermaidService:', error);\n        return;\n      }\n    }\n    this.nextTick(async () => {\n      const container = mermaidContentRef.nativeElement;\n      if (container) {\n        // 移除打字效果相关的span标签\n        const cleanCode = code.replace(\n          /<span[^>]*\\bclass\\s*=\\s*['\"]mc-typewriter[^>]*>([\\s\\S]*?)<\\/span>/g,\n          `$1`\n        );\n        await this.mermaidService?.renderToContainer(\n          container,\n          cleanCode,\n          theme\n        );\n      }\n    });\n  };\n}\n"]}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { filterXSS, getDefaultCSSWhiteList, getDefaultWhiteList } from 'xss';
|
|
2
|
+
export class MDCardService {
|
|
3
|
+
constructor() {
|
|
4
|
+
this.xssWhiteList = getDefaultWhiteList();
|
|
5
|
+
this.cssWhiteList = getDefaultCSSWhiteList();
|
|
6
|
+
this.setDefaultXss();
|
|
7
|
+
}
|
|
8
|
+
setDefaultXss() {
|
|
9
|
+
this.xssWhiteList['input'] = ['type', 'checked', 'disabled', 'class'];
|
|
10
|
+
this.xssWhiteList['label'] = ['for'];
|
|
11
|
+
this.xssWhiteList['ul'] = ['class'];
|
|
12
|
+
this.xssWhiteList['div'] = ['class'];
|
|
13
|
+
this.xssWhiteList['a'] = ['href', 'class', 'target', 'name'];
|
|
14
|
+
this.xssWhiteList['ol'] = ['start'];
|
|
15
|
+
this.xssWhiteList['p'] = ['class'];
|
|
16
|
+
this.xssWhiteList['span'] = ['style', 'class', 'title', 'id'];
|
|
17
|
+
this.xssWhiteList['svg'] = ['style', 'class', 'width', 'height', 'viewbox', 'preserveaspectratio', 'id', 'fill', 'stroke'];
|
|
18
|
+
this.xssWhiteList['path'] = ['style', 'class', 'd', 'id', 'fill', 'stroke'];
|
|
19
|
+
this.xssWhiteList['th'] = ['style'];
|
|
20
|
+
this.xssWhiteList['td'] = ['style'];
|
|
21
|
+
}
|
|
22
|
+
onIgnoreTagAttr(tag, name, value, isWhiteAttr) {
|
|
23
|
+
if (!isWhiteAttr && (name === 'id' || (tag === 'span' && name === 'style'))) {
|
|
24
|
+
return name + '=' + value;
|
|
25
|
+
}
|
|
26
|
+
return undefined;
|
|
27
|
+
}
|
|
28
|
+
getXssWhiteList() {
|
|
29
|
+
return this.xssWhiteList;
|
|
30
|
+
}
|
|
31
|
+
setXssWhiteList(list) {
|
|
32
|
+
this.xssWhiteList = list;
|
|
33
|
+
}
|
|
34
|
+
setCustomXssRules(rules) {
|
|
35
|
+
if (rules) {
|
|
36
|
+
rules.forEach((rule) => {
|
|
37
|
+
if (rule['value'] === null) {
|
|
38
|
+
delete this.xssWhiteList[rule['key']];
|
|
39
|
+
}
|
|
40
|
+
else {
|
|
41
|
+
this.xssWhiteList[rule['key']] = rule['value'];
|
|
42
|
+
}
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
setMdPlugins(plugins, mdt) {
|
|
47
|
+
if (plugins && plugins.length) {
|
|
48
|
+
plugins.forEach(item => {
|
|
49
|
+
const { plugin, opts } = item;
|
|
50
|
+
mdt.use(plugin, opts);
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
filterHtml(html) {
|
|
55
|
+
return filterXSS(html, {
|
|
56
|
+
whiteList: this.xssWhiteList,
|
|
57
|
+
onIgnoreTagAttr: this.onIgnoreTagAttr,
|
|
58
|
+
css: {
|
|
59
|
+
whiteList: Object.assign({}, this.cssWhiteList, {
|
|
60
|
+
top: true,
|
|
61
|
+
left: true,
|
|
62
|
+
bottom: true,
|
|
63
|
+
right: true,
|
|
64
|
+
}),
|
|
65
|
+
},
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTURDYXJkU2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMtbmcvc3JjL2NvbXBvbmVudHMtY29tbW9uL01hcmtkb3duQ2FyZC9jb21tb24vTURDYXJkU2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLHNCQUFzQixFQUFFLG1CQUFtQixFQUFtQixNQUFNLEtBQUssQ0FBQztBQUU5RixNQUFNLE9BQU8sYUFBYTtJQUl4QjtRQUhRLGlCQUFZLEdBQUcsbUJBQW1CLEVBQUUsQ0FBQztRQUNyQyxpQkFBWSxHQUFHLHNCQUFzQixFQUFFLENBQUM7UUFHOUMsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO0lBQ3ZCLENBQUM7SUFFTyxhQUFhO1FBQ25CLElBQUksQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsU0FBUyxFQUFFLFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUN0RSxJQUFJLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ3BDLElBQUksQ0FBQyxZQUFZLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNyQyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFDN0QsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBRXBDLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNuQyxJQUFJ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dCI6WyJpbXBvcnQgeyBmaWx0ZXJYU1MsIGdldERlZmF1bHRDU1NXaGl0ZUxpc3QsIGdldERlZmF1bHRXaGl0ZUxpc3QsIHR5cGUgSVdoaXRlTGlzdCB9IGZyb20gJ3hzcyc7XG5pbXBvcnQgdHlwZSB7IEN1c3RvbVhzc1J1bGUsIE1kUGx1Z2luIH0gZnJvbSAnLi9tZENhcmQudHlwZXMudHMnO1xuZXhwb3J0IGNsYXNzIE1EQ2FyZFNlcnZpY2Uge1xuICBwcml2YXRlIHhzc1doaXRlTGlzdCA9IGdldERlZmF1bHRXaGl0ZUxpc3QoKTtcbiAgcHJpdmF0ZSBjc3NXaGl0ZUxpc3QgPSBnZXREZWZhdWx0Q1NTV2hpdGVMaXN0KCk7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5zZXREZWZhdWx0WHNzKCk7XG4gIH1cblxuICBwcml2YXRlIHNldERlZmF1bHRYc3MoKSB7XG4gICAgdGhpcy54c3NXaGl0ZUxpc3RbJ2lucHV0J10gPSBbJ3R5cGUnLCAnY2hlY2tlZCcsICdkaXNhYmxlZCcsICdjbGFzcyddO1xuICAgIHRoaXMueHNzV2hpdGVMaXN0WydsYWJlbCddID0gWydmb3InXTtcbiAgICB0aGlzLnhzc1doaXRlTGlzdFsndWwnXSA9IFsnY2xhc3MnXTtcbiAgICB0aGlzLnhzc1doaXRlTGlzdFsnZGl2J10gPSBbJ2NsYXNzJ107XG4gICAgdGhpcy54c3NXaGl0ZUxpc3RbJ2EnXSA9IFsnaHJlZicsICdjbGFzcycsICd0YXJnZXQnLCAnbmFtZSddO1xuICAgIHRoaXMueHNzV2hpdGVMaXN0WydvbCddID0gWydzdGFydCddO1xuXG4gICAgdGhpcy54c3NXaGl0ZUxpc3RbJ3AnXSA9IFsnY2xhc3MnXTtcbiAgICB0aGlzLnhzc1doaXRlTGlzdFsnc3BhbiddID0gWydzdHlsZScsICdjbGFzcycsICd0aXRsZScsICdpZCddO1xuICAgIHRoaXMueHNzV2hpdGVMaXN0WydzdmcnXSA9IFsnc3R5bGUnLCAnY2xhc3MnLCAnd2lkdGgnLCAnaGVpZ2h0JywgJ3ZpZXdib3gnLCAncHJlc2VydmVhc3BlY3RyYXRpbycsICdpZCcsICdmaWxsJywgJ3N0cm9rZSddO1xuICAgIHRoaXMueHNzV2hpdGVMaXN0WydwYXRoJ10gPSBbJ3N0eWxlJywgJ2NsYXNzJywgJ2QnLCAnaWQnLCAnZmlsbCcsICdzdHJva2UnXTtcbiAgICB0aGlzLnhzc1doaXRlTGlzdFsndGgnXSA9IFsnc3R5bGUnXTtcbiAgICB0aGlzLnhzc1doaXRlTGlzdFsndGQnXSA9IFsnc3R5bGUnXTtcbiAgfVxuXG4gIHByaXZhdGUgb25JZ25vcmVUYWdBdHRyKHRhZzogc3RyaW5nLCBuYW1lOiBzdHJpbmcsIHZhbHVlOiBzdHJpbmcsIGlzV2hpdGVBdHRyOiBib29sZWFuKSB7XG4gICAgaWYgKCFpc1doaXRlQXR0ciAmJiAobmFtZSA9PT0gJ2lkJyB8fCAodGFnID09PSAnc3BhbicgJiYgbmFtZSA9PT0gJ3N0eWxlJykpKSB7XG4gICAgICByZXR1cm4gbmFtZSArICc9JyArIHZhbHVlO1xuICAgIH1cbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG5cbiAgZ2V0WHNzV2hpdGVMaXN0KCkge1xuICAgIHJldHVybiB0aGlzLnhzc1doaXRlTGlzdDtcbiAgfVxuXG4gIHNldFhzc1doaXRlTGlzdChsaXN0OiBJV2hpdGVMaXN0KSB7XG4gICAgdGhpcy54c3NXaGl0ZUxpc3QgPSBsaXN0O1xuICB9XG5cbiAgc2V0Q3VzdG9tWHNzUnVsZXMocnVsZXM6IEN1c3RvbVhzc1J1bGVbXSkge1xuICAgIGlmIChydWxlcykge1xuICAgICAgcnVsZXMuZm9yRWFjaCgocnVsZSkgPT4ge1xuICAgICAgICBpZiAocnVsZVsndmFsdWUnXSA9PT0gbnVsbCkge1xuICAgICAgICAgIGRlbGV0ZSB0aGlzLnhzc1doaXRlTGlzdFtydWxlWydrZXknXV07XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy54c3NXaGl0ZUxpc3RbcnVsZVsna2V5J11dID0gcnVsZVsndmFsdWUnXTtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cblxuICBzZXRNZFBsdWdpbnMocGx1Z2luczogTWRQbHVnaW5bXSwgbWR0OiBhbnkpIHtcbiAgICBpZiAocGx1Z2lucyAmJiBwbHVnaW5zLmxlbmd0aCkge1xuICAgICAgcGx1Z2lucy5mb3JFYWNoKGl0ZW0gPT4ge1xuICAgICAgICBjb25zdCB7IHBsdWdpbiwgb3B0cyB9ID0gaXRlbTtcbiAgICAgICAgbWR0LnVzZShwbHVnaW4sIG9wdHMpO1xuICAgICAgfSlcbiAgICB9XG4gIH1cblxuICBmaWx0ZXJIdG1sKGh0bWw6IHN0cmluZykge1xuICAgIHJldHVybiBmaWx0ZXJYU1MoaHRtbCwge1xuICAgICAgd2hpdGVMaXN0OiB0aGlzLnhzc1doaXRlTGlzdCxcbiAgICAgIG9uSWdub3JlVGFnQXR0cjogdGhpcy5vbklnbm9yZVRhZ0F0dHIsXG4gICAgICBjc3M6IHtcbiAgICAgICAgd2hpdGVMaXN0OiBPYmplY3QuYXNzaWduKHt9LCB0aGlzLmNzc1doaXRlTGlzdCwge1xuICAgICAgICAgIHRvcDogdHJ1ZSxcbiAgICAgICAgICBsZWZ0OiB0cnVlLFxuICAgICAgICAgIGJvdHRvbTogdHJ1ZSxcbiAgICAgICAgICByaWdodDogdHJ1ZSxcbiAgICAgICAgfSksXG4gICAgICB9LFxuICAgIH0pO1xuICB9XG59Il19
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
export class MermaidService {
|
|
2
|
+
constructor(config = {}) {
|
|
3
|
+
this.config = config;
|
|
4
|
+
this.mermaidInstance = null;
|
|
5
|
+
this.isLoading = false;
|
|
6
|
+
this.lastValidResult = '';
|
|
7
|
+
this.viewStateMap = new WeakMap();
|
|
8
|
+
this.containerHeight = 400;
|
|
9
|
+
}
|
|
10
|
+
async loadMermaid() {
|
|
11
|
+
if (this.mermaidInstance) {
|
|
12
|
+
return this.mermaidInstance;
|
|
13
|
+
}
|
|
14
|
+
if (this.isLoading) {
|
|
15
|
+
return new Promise((resolve) => {
|
|
16
|
+
const checkInstance = () => {
|
|
17
|
+
if (this.mermaidInstance) {
|
|
18
|
+
resolve(this.mermaidInstance);
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
setTimeout(checkInstance, 50);
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
checkInstance();
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
this.isLoading = true;
|
|
28
|
+
try {
|
|
29
|
+
const { default: mermaid } = await import('mermaid');
|
|
30
|
+
mermaid.initialize({
|
|
31
|
+
theme: this.config.theme || 'default',
|
|
32
|
+
startOnLoad: false,
|
|
33
|
+
suppressErrorRendering: true,
|
|
34
|
+
...this.config
|
|
35
|
+
});
|
|
36
|
+
this.mermaidInstance = mermaid;
|
|
37
|
+
return mermaid;
|
|
38
|
+
}
|
|
39
|
+
catch (error) {
|
|
40
|
+
console.error('Failed to load mermaid:', error);
|
|
41
|
+
throw new Error('Failed to load mermaid library');
|
|
42
|
+
}
|
|
43
|
+
finally {
|
|
44
|
+
this.isLoading = false;
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
async renderToContainer(container, code, theme = 'light') {
|
|
48
|
+
const svgStr = await this.renderMermaid(code, theme);
|
|
49
|
+
container.innerHTML = svgStr;
|
|
50
|
+
const svg = container.querySelector('svg');
|
|
51
|
+
if (svg) {
|
|
52
|
+
this.initViewState(container, svg);
|
|
53
|
+
this.applyTransform(container, svg);
|
|
54
|
+
svg.addEventListener('mousedown', (e) => this.onSvgMouseDown(e, container, svg));
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
initViewState(container, svg) {
|
|
58
|
+
// 获取svg的viewBox或宽高
|
|
59
|
+
let vb = svg.getAttribute('viewBox');
|
|
60
|
+
let svgW = 0, svgH = 0;
|
|
61
|
+
if (vb) {
|
|
62
|
+
const arr = vb.split(/\s+/);
|
|
63
|
+
svgW = parseFloat(arr[2]);
|
|
64
|
+
svgH = parseFloat(arr[3]);
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
svgW = svg.width.baseVal.value || svg.getBoundingClientRect().width;
|
|
68
|
+
svgH = svg.height.baseVal.value || svg.getBoundingClientRect().height;
|
|
69
|
+
}
|
|
70
|
+
const contW = container.clientWidth || 0;
|
|
71
|
+
const contH = this.containerHeight;
|
|
72
|
+
let scale = 1;
|
|
73
|
+
if (svgW && svgH && contW && contH) {
|
|
74
|
+
scale = Math.min(contW / svgW, contH / svgH, 1);
|
|
75
|
+
}
|
|
76
|
+
this.viewStateMap.set(container, {
|
|
77
|
+
scale,
|
|
78
|
+
offsetX: 0,
|
|
79
|
+
offsetY: 0,
|
|
80
|
+
dragging: false,
|
|
81
|
+
dragStart: { x: 0, y: 0 },
|
|
82
|
+
lastOffset: { x: 0, y: 0 },
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
applyTransform(container, svg) {
|
|
86
|
+
const state = this.viewStateMap.get(container);
|
|
87
|
+
if (!state)
|
|
88
|
+
return;
|
|
89
|
+
svg.style.position = 'absolute';
|
|
90
|
+
svg.style.left = '50%';
|
|
91
|
+
svg.style.top = '50%';
|
|
92
|
+
svg.style.transform = `translate(-50%, -50%) translate(${state.offsetX}px, ${state.offsetY}px) scale(${state.scale})`;
|
|
93
|
+
svg.style.transformOrigin = 'center center';
|
|
94
|
+
svg.style.cursor = state.dragging ? 'grabbing' : 'grab';
|
|
95
|
+
}
|
|
96
|
+
zoomIn(container) {
|
|
97
|
+
const svg = container.querySelector('svg');
|
|
98
|
+
const state = this.viewStateMap.get(container);
|
|
99
|
+
if (svg && state) {
|
|
100
|
+
state.scale = Math.min(state.scale + 0.2, 3);
|
|
101
|
+
this.applyTransform(container, svg);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
zoomOut(container) {
|
|
105
|
+
const svg = container.querySelector('svg');
|
|
106
|
+
const state = this.viewStateMap.get(container);
|
|
107
|
+
if (svg && state) {
|
|
108
|
+
state.scale = Math.max(state.scale - 0.2, 0.2);
|
|
109
|
+
this.applyTransform(container, svg);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
reset(container) {
|
|
113
|
+
const svg = container.querySelector('svg');
|
|
114
|
+
if (svg) {
|
|
115
|
+
this.initViewState(container, svg);
|
|
116
|
+
this.applyTransform(container, svg);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
async download(container, filename = 'diagram.png') {
|
|
120
|
+
const svg = container.querySelector('svg');
|
|
121
|
+
if (!svg)
|
|
122
|
+
return;
|
|
123
|
+
try {
|
|
124
|
+
const clonedSvg = svg.cloneNode(true);
|
|
125
|
+
const svgData = new XMLSerializer().serializeToString(clonedSvg);
|
|
126
|
+
const svgUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svgData)}`;
|
|
127
|
+
const img = new Image();
|
|
128
|
+
await new Promise((resolve, reject) => {
|
|
129
|
+
img.onload = () => resolve();
|
|
130
|
+
img.onerror = (e) => reject(new Error('Image loading failed'));
|
|
131
|
+
img.src = svgUrl;
|
|
132
|
+
});
|
|
133
|
+
const canvas = document.createElement('canvas');
|
|
134
|
+
const ctx = canvas.getContext('2d');
|
|
135
|
+
if (!ctx)
|
|
136
|
+
throw new Error('Canvas context not available');
|
|
137
|
+
canvas.width = img.width * 2;
|
|
138
|
+
canvas.height = img.height * 2;
|
|
139
|
+
ctx.fillStyle = 'white';
|
|
140
|
+
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
141
|
+
ctx.drawImage(img, 0, 0);
|
|
142
|
+
canvas.toBlob((blob) => {
|
|
143
|
+
if (!blob) {
|
|
144
|
+
console.error('Failed to create blob from canvas');
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
const url = URL.createObjectURL(blob);
|
|
148
|
+
const a = document.createElement('a');
|
|
149
|
+
a.href = url;
|
|
150
|
+
a.download = filename;
|
|
151
|
+
document.body.appendChild(a);
|
|
152
|
+
a.click();
|
|
153
|
+
setTimeout(() => {
|
|
154
|
+
document.body.removeChild(a);
|
|
155
|
+
URL.revokeObjectURL(url);
|
|
156
|
+
}, 100);
|
|
157
|
+
}, 'image/png');
|
|
158
|
+
}
|
|
159
|
+
catch (error) {
|
|
160
|
+
console.error('Failed to download diagram:', error);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
onSvgMouseDown(e, container, svg) {
|
|
164
|
+
const state = this.viewStateMap.get(container);
|
|
165
|
+
if (!state)
|
|
166
|
+
return;
|
|
167
|
+
state.dragging = true;
|
|
168
|
+
state.dragStart = { x: e.clientX, y: e.clientY };
|
|
169
|
+
state.lastOffset = { x: state.offsetX, y: state.offsetY };
|
|
170
|
+
const move = (ev) => this.onSvgMouseMove(ev, container, svg);
|
|
171
|
+
const up = () => this.onSvgMouseUp(container, svg, move, up);
|
|
172
|
+
document.addEventListener('mousemove', move);
|
|
173
|
+
document.addEventListener('mouseup', up);
|
|
174
|
+
this.applyTransform(container, svg);
|
|
175
|
+
}
|
|
176
|
+
onSvgMouseMove(e, container, svg) {
|
|
177
|
+
const state = this.viewStateMap.get(container);
|
|
178
|
+
if (!state || !state.dragging)
|
|
179
|
+
return;
|
|
180
|
+
state.offsetX = state.lastOffset.x + (e.clientX - state.dragStart.x);
|
|
181
|
+
state.offsetY = state.lastOffset.y + (e.clientY - state.dragStart.y);
|
|
182
|
+
this.applyTransform(container, svg);
|
|
183
|
+
}
|
|
184
|
+
onSvgMouseUp(container, svg, move, up) {
|
|
185
|
+
const state = this.viewStateMap.get(container);
|
|
186
|
+
if (!state)
|
|
187
|
+
return;
|
|
188
|
+
state.dragging = false;
|
|
189
|
+
document.removeEventListener('mousemove', move);
|
|
190
|
+
document.removeEventListener('mouseup', up);
|
|
191
|
+
this.applyTransform(container, svg);
|
|
192
|
+
}
|
|
193
|
+
async renderMermaid(code, theme = 'light') {
|
|
194
|
+
try {
|
|
195
|
+
const mermaid = await this.loadMermaid();
|
|
196
|
+
if (this.config.theme !== theme) {
|
|
197
|
+
this.config.theme = theme;
|
|
198
|
+
mermaid.initialize({
|
|
199
|
+
startOnLoad: false,
|
|
200
|
+
suppressErrorRendering: true,
|
|
201
|
+
theme: theme === 'dark' ? 'dark' : 'default',
|
|
202
|
+
...this.config
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
const id = `mc_mermaid_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
|
|
206
|
+
const { svg } = await mermaid.render(id, code);
|
|
207
|
+
this.lastValidResult = svg;
|
|
208
|
+
return svg;
|
|
209
|
+
}
|
|
210
|
+
catch (error) {
|
|
211
|
+
return this.lastValidResult;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
// 设置配置
|
|
215
|
+
setConfig(config = {}) {
|
|
216
|
+
this.config = {
|
|
217
|
+
theme: 'default',
|
|
218
|
+
...config
|
|
219
|
+
};
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MermaidService.js","sourceRoot":"","sources":["../../../../../../projects/components-ng/src/components-common/MarkdownCard/common/MermaidService.ts"],"names":[],"mappings":"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","sourcesContent":["import type { MermaidConfig } from './mdCard.types';\n\ninterface MermaidViewState {\n  scale: number;\n  offsetX: number;\n  offsetY: number;\n  dragging: boolean;\n  dragStart: { x: number; y: number };\n  lastOffset: { x: number; y: number };\n}\n\nexport class MermaidService {\n  private mermaidInstance: any = null;\n  private isLoading = false;\n  private lastValidResult: string = '';\n  private viewStateMap = new WeakMap<HTMLElement, MermaidViewState>();\n  private containerHeight = 400;\n\n  constructor(private config: MermaidConfig = {}) {}\n\n  private async loadMermaid() {\n    if (this.mermaidInstance) {\n      return this.mermaidInstance;\n    }\n\n    if (this.isLoading) {\n      return new Promise((resolve) => {\n        const checkInstance = () => {\n          if (this.mermaidInstance) {\n            resolve(this.mermaidInstance);\n          } else {\n            setTimeout(checkInstance, 50);\n          }\n        };\n        checkInstance();\n      });\n    }\n\n    this.isLoading = true;\n    try {\n      const { default: mermaid } = await import('mermaid') as any;\n      mermaid.initialize({\n        theme: this.config.theme || 'default',\n        startOnLoad: false,\n        suppressErrorRendering: true,\n        ...this.config\n      });\n      this.mermaidInstance = mermaid;\n      return mermaid;\n    } catch (error) {\n      console.error('Failed to load mermaid:', error);\n      throw new Error('Failed to load mermaid library');\n    } finally {\n      this.isLoading = false;\n    }\n  }\n\n  async renderToContainer(container: HTMLElement, code: string, theme: 'light' | 'dark' = 'light') {\n    const svgStr = await this.renderMermaid(code, theme);\n    container.innerHTML = svgStr;\n    const svg = container.querySelector('svg');\n    if (svg) {\n      this.initViewState(container, svg);\n      this.applyTransform(container, svg);\n      svg.addEventListener('mousedown', (e) => this.onSvgMouseDown(e, container, svg));\n    }\n  }\n\n  private initViewState(container: HTMLElement, svg: SVGSVGElement) {\n    // 获取svg的viewBox或宽高\n    let vb = svg.getAttribute('viewBox');\n    let svgW = 0, svgH = 0;\n    if (vb) {\n      const arr = vb.split(/\\s+/);\n      svgW = parseFloat(arr[2]);\n      svgH = parseFloat(arr[3]);\n    } else {\n      svgW = svg.width.baseVal.value || svg.getBoundingClientRect().width;\n      svgH = svg.height.baseVal.value || svg.getBoundingClientRect().height;\n    }\n    const contW = container.clientWidth || 0;\n    const contH = this.containerHeight;\n    let scale = 1;\n    if (svgW && svgH && contW && contH) {\n      scale = Math.min(contW / svgW, contH / svgH, 1);\n    }\n    this.viewStateMap.set(container, {\n      scale,\n      offsetX: 0,\n      offsetY: 0,\n      dragging: false,\n      dragStart: { x: 0, y: 0 },\n      lastOffset: { x: 0, y: 0 },\n    });\n  }\n\n  private applyTransform(container: HTMLElement, svg: SVGSVGElement) {\n    const state = this.viewStateMap.get(container);\n    if (!state) return;\n    svg.style.position = 'absolute';\n    svg.style.left = '50%';\n    svg.style.top = '50%';\n    svg.style.transform = `translate(-50%, -50%) translate(${state.offsetX}px, ${state.offsetY}px) scale(${state.scale})`;\n    svg.style.transformOrigin = 'center center';\n    svg.style.cursor = state.dragging ? 'grabbing' : 'grab';\n  }\n\n  zoomIn(container: HTMLElement) {\n    const svg = container.querySelector('svg');\n    const state = this.viewStateMap.get(container);\n    if (svg && state) {\n      state.scale = Math.min(state.scale + 0.2, 3);\n      this.applyTransform(container, svg);\n    }\n  }\n\n  zoomOut(container: HTMLElement) {\n    const svg = container.querySelector('svg');\n    const state = this.viewStateMap.get(container);\n    if (svg && state) {\n      state.scale = Math.max(state.scale - 0.2, 0.2);\n      this.applyTransform(container, svg);\n    }\n  }\n\n  reset(container: HTMLElement) {\n    const svg = container.querySelector('svg');\n    if (svg) {\n      this.initViewState(container, svg);\n      this.applyTransform(container, svg);\n    }\n  }\n\n  async download(container: HTMLElement, filename: string = 'diagram.png'): Promise<void> {\n    const svg = container.querySelector('svg');\n    if (!svg) return;\n\n    try {\n      const clonedSvg = svg.cloneNode(true) as SVGSVGElement;\n\n      const svgData = new XMLSerializer().serializeToString(clonedSvg);\n      \n      const svgUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svgData)}`;\n\n      const img = new Image();\n      await new Promise<void>((resolve, reject) => {\n        img.onload = () => resolve();\n        img.onerror = (e) => reject(new Error('Image loading failed'));\n        img.src = svgUrl;\n      });\n\n      const canvas = document.createElement('canvas');\n      const ctx = canvas.getContext('2d');\n      if (!ctx) throw new Error('Canvas context not available');\n\n      canvas.width = img.width * 2;\n      canvas.height = img.height * 2;\n      \n      ctx.fillStyle = 'white';\n      ctx.fillRect(0, 0, canvas.width, canvas.height);\n      \n      ctx.drawImage(img, 0, 0);\n\n      canvas.toBlob((blob) => {\n        if (!blob) {\n          console.error('Failed to create blob from canvas');\n          return;\n        }\n        \n        const url = URL.createObjectURL(blob);\n        const a = document.createElement('a');\n        a.href = url;\n        a.download = filename;\n        document.body.appendChild(a);\n        a.click();\n        \n        setTimeout(() => {\n          document.body.removeChild(a);\n          URL.revokeObjectURL(url);\n        }, 100);\n      }, 'image/png');\n    } catch (error) {\n      console.error('Failed to download diagram:', error);\n    }\n  }\n\n  private onSvgMouseDown(e: MouseEvent, container: HTMLElement, svg: SVGSVGElement) {\n    const state = this.viewStateMap.get(container);\n    if (!state) return;\n    state.dragging = true;\n    state.dragStart = { x: e.clientX, y: e.clientY };\n    state.lastOffset = { x: state.offsetX, y: state.offsetY };\n    const move = (ev: MouseEvent) => this.onSvgMouseMove(ev, container, svg);\n    const up = () => this.onSvgMouseUp(container, svg, move, up);\n    document.addEventListener('mousemove', move);\n    document.addEventListener('mouseup', up);\n    this.applyTransform(container, svg);\n  }\n  private onSvgMouseMove(e: MouseEvent, container: HTMLElement, svg: SVGSVGElement) {\n    const state = this.viewStateMap.get(container);\n    if (!state || !state.dragging) return;\n    state.offsetX = state.lastOffset.x + (e.clientX - state.dragStart.x);\n    state.offsetY = state.lastOffset.y + (e.clientY - state.dragStart.y);\n    this.applyTransform(container, svg);\n  }\n  private onSvgMouseUp(container: HTMLElement, svg: SVGSVGElement, move: any, up: any) {\n    const state = this.viewStateMap.get(container);\n    if (!state) return;\n    state.dragging = false;\n    document.removeEventListener('mousemove', move);\n    document.removeEventListener('mouseup', up);\n    this.applyTransform(container, svg);\n  }\n\n  async renderMermaid(code: string, theme: 'light' | 'dark' = 'light'): Promise<string> {\n    try {\n      const mermaid = await this.loadMermaid();\n      if (this.config.theme !== theme) {\n        this.config.theme = theme;\n          mermaid.initialize({\n          startOnLoad: false,\n          suppressErrorRendering: true,\n          theme: theme === 'dark' ? 'dark' : 'default',\n          ...this.config\n        });\n      }\n      const id = `mc_mermaid_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;\n      const { svg } = await mermaid.render(id, code);\n      this.lastValidResult = svg;\n      return svg;\n    } catch (error) {\n      return this.lastValidResult;\n    }\n  }\n\n    // 设置配置\n  setConfig(config: MermaidConfig = {}): void {\n    this.config = {\n      theme: 'default',\n      ...config\n    };\n  }\n}\n"]}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export const defaultTypingConfig = {
|
|
2
|
+
step: 2,
|
|
3
|
+
interval: 50,
|
|
4
|
+
style: 'normal',
|
|
5
|
+
};
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWRDYXJkLnR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50cy1uZy9zcmMvY29tcG9uZW50cy1jb21tb24vTWFya2Rvd25DYXJkL2NvbW1vbi9tZENhcmQudHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEJBLE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUFHO0lBQ2pDLElBQUksRUFBRSxDQUFDO0lBQ1AsUUFBUSxFQUFFLEVBQUU7SUFDWixLQUFLLEVBQUUsUUFBdUI7Q0FDL0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgT3B0aW9ucywgUGx1Z2luU2ltcGxlLCBQbHVnaW5XaXRoT3B0aW9ucywgUGx1Z2luV2l0aFBhcmFtcyB9IGZyb20gJ21hcmtkb3duLWl0JztcbmltcG9ydCB0eXBlIHsgVG9rZW4gfSBmcm9tICdtYXJrZG93bi1pdCc7XG5leHBvcnQgaW50ZXJmYWNlIE1lcm1haWRDb25maWcge1xuICB0aGVtZT86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDdXN0b21Yc3NSdWxlIHtcbiAga2V5OiBzdHJpbmc7XG4gIHZhbHVlOiBzdHJpbmdbXSB8IG51bGw7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29kQmxvY2tEYXRhIHtcbiAgY29kZTogc3RyaW5nO1xuICBsYW5ndWFnZTogc3RyaW5nO1xufVxuXG5leHBvcnQgdHlwZSBDb2RlQmxvY2tTbG90ID0ge1xuICBhY3Rpb25zPzogKCkgPT4gdm9pZDtcbiAgaGVhZGVyPzogKCkgPT4gdm9pZDtcbiAgY29udGVudD86ICgpID0+IHZvaWQ7XG59O1xuXG5leHBvcnQgdHlwZSBUaGVtZSA9ICdsaWdodCcgfCAnZGFyayc7XG5cbmV4cG9ydCB0eXBlIFR5cGluZ1N0eWxlID0gJ25vcm1hbCcgfCAnY3Vyc29yJyB8ICdjb2xvcicgfCAnZ3JhZGllbnQnO1xuXG5leHBvcnQgdHlwZSBJbnRlcnZhbFR5cGUgPSBudW1iZXIgfCBbbnVtYmVyLCBudW1iZXJdO1xuXG5leHBvcnQgY29uc3QgZGVmYXVsdFR5cGluZ0NvbmZpZyA9IHtcbiAgc3RlcDogMixcbiAgaW50ZXJ2YWw6IDUwLFxuICBzdHlsZTogJ25vcm1hbCcgYXMgVHlwaW5nU3R5bGUsXG59O1xuXG5leHBvcnQgaW50ZXJmYWNlIE1kUGx1Z2luIHtcbiAgcGx1Z2luOiBQbHVnaW5TaW1wbGUgfCBQbHVnaW5XaXRoT3B0aW9ucyB8IFBsdWdpbldpdGhQYXJhbXM7XG4gIG9wdHM/OiB1bmtub3duO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1hcmtkb3duQ2FyZFByb3BzIHtcbiAgY29udGVudD86IHN0cmluZztcbiAgdHlwaW5nPzogYm9vbGVhbjtcbiAgZW5hYmxlVGhpbms/OiBib29sZWFuO1xuICB0eXBpbmdPcHRpb25zPzoge1xuICAgIHN0ZXA/OiBudW1iZXI7XG4gICAgaW50ZXJ2YWw/OiBudW1iZXIgfCBbbnVtYmVyLCBudW1iZXJdO1xuICAgIHN0eWxlPzogVHlwaW5nU3R5bGU7XG4gIH07XG4gIHRoaW5rT3B0aW9ucz86IHtcbiAgICBjdXN0b21DbGFzcz86IHN0cmluZztcbiAgfTtcbiAgbWRPcHRpb25zPzogT3B0aW9ucztcbiAgbWRQbHVnaW5zPzogQXJyYXk8TWRQbHVnaW4+O1xuICBjdXN0b21Yc3NSdWxlcz86IEFycmF5PEN1c3RvbVhzc1J1bGU+O1xuICB0aGVtZT86IFRoZW1lO1xuICBlbmFibGVNZXJtYWlkPzogYm9vbGVhbjtcbiAgbWVybWFpZENvbmZpZz86IE1lcm1haWRDb25maWc7XG59XG5cbi8vIOWumuS5iSBBU1Qg6IqC54K55o6l5Y+jXG5leHBvcnQgaW50ZXJmYWNlIEFTVE5vZGUge1xuICBub2RlVHlwZTogc3RyaW5nO1xuICBvcGVuTm9kZTogVG9rZW4gfCBudWxsO1xuICBjbG9zZU5vZGU6IFRva2VuIHwgbnVsbDtcbiAgY2hpbGRyZW46IChBU1ROb2RlIHwgVG9rZW4pW107XG4gIHZOb2RlS2V5OiBzdHJpbmc7XG59Il19
|