@matechat/ng 20.0.1-alpha.0 → 20.1.0-alpha.1

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.
Files changed (80) hide show
  1. package/README.md +206 -41
  2. package/fesm2022/matechat-ng.mjs +71 -70
  3. package/fesm2022/matechat-ng.mjs.map +1 -1
  4. package/index.d.ts +612 -3
  5. package/package.json +14 -3
  6. package/Base/base.component.d.ts +0 -11
  7. package/Bubble/avatar/avatar.component.d.ts +0 -23
  8. package/Bubble/avatar-body-icon/avatar-body-icon.component.d.ts +0 -7
  9. package/Bubble/avatar-no-body-icon/avatar-no-body-icon.component.d.ts +0 -7
  10. package/Bubble/bubble-loading/bubble-loading.component.d.ts +0 -5
  11. package/Bubble/bubble.component.d.ts +0 -26
  12. package/Bubble/bubble.module.d.ts +0 -10
  13. package/Bubble/index.d.ts +0 -6
  14. package/Input/button/button.component.d.ts +0 -32
  15. package/Input/index.d.ts +0 -2
  16. package/Input/input.component.d.ts +0 -65
  17. package/Input/input.module.d.ts +0 -11
  18. package/Input/send-icon/send-icon.component.d.ts +0 -5
  19. package/Locale/index.d.ts +0 -5
  20. package/Locale/locale.module.d.ts +0 -17
  21. package/Locale/locale.service.d.ts +0 -50
  22. package/Locale/translate.pipe.d.ts +0 -23
  23. package/MarkdownCard/code-block.component.d.ts +0 -49
  24. package/MarkdownCard/index.d.ts +0 -3
  25. package/MarkdownCard/markdown-card.component.d.ts +0 -200
  26. package/MarkdownCard/markdown-card.module.d.ts +0 -13
  27. package/components-common/Base/foundation.d.ts +0 -45
  28. package/components-common/Bubble/common/bubble-constants.d.ts +0 -4
  29. package/components-common/Bubble/common/bubble-types.d.ts +0 -12
  30. package/components-common/Bubble/foundation.d.ts +0 -9
  31. package/components-common/Input/button-foundation.d.ts +0 -15
  32. package/components-common/Input/common/types.d.ts +0 -16
  33. package/components-common/Input/foundation.d.ts +0 -20
  34. package/components-common/Locale/lang/en-us.d.ts +0 -26
  35. package/components-common/Locale/lang/zh-cn.d.ts +0 -26
  36. package/components-common/MarkdownCard/codeblock-foundation.d.ts +0 -21
  37. package/components-common/MarkdownCard/common/MDCardService.d.ts +0 -14
  38. package/components-common/MarkdownCard/common/MermaidService.d.ts +0 -23
  39. package/components-common/MarkdownCard/common/mdCard.types.d.ts +0 -56
  40. package/components-common/MarkdownCard/common/parser.d.ts +0 -150
  41. package/components-common/MarkdownCard/foundation.d.ts +0 -38
  42. package/esm2022/Base/base.component.mjs +0 -46
  43. package/esm2022/Bubble/avatar/avatar.component.mjs +0 -121
  44. package/esm2022/Bubble/avatar-body-icon/avatar-body-icon.component.mjs +0 -19
  45. package/esm2022/Bubble/avatar-no-body-icon/avatar-no-body-icon.component.mjs +0 -19
  46. package/esm2022/Bubble/bubble-loading/bubble-loading.component.mjs +0 -11
  47. package/esm2022/Bubble/bubble.component.mjs +0 -89
  48. package/esm2022/Bubble/bubble.module.mjs +0 -36
  49. package/esm2022/Bubble/index.mjs +0 -7
  50. package/esm2022/Input/button/button.component.mjs +0 -83
  51. package/esm2022/Input/index.mjs +0 -3
  52. package/esm2022/Input/input.component.mjs +0 -285
  53. package/esm2022/Input/input.module.mjs +0 -34
  54. package/esm2022/Input/send-icon/send-icon.component.mjs +0 -11
  55. package/esm2022/Locale/index.mjs +0 -6
  56. package/esm2022/Locale/locale.module.mjs +0 -39
  57. package/esm2022/Locale/locale.service.mjs +0 -140
  58. package/esm2022/Locale/translate.pipe.mjs +0 -38
  59. package/esm2022/MarkdownCard/code-block.component.mjs +0 -175
  60. package/esm2022/MarkdownCard/index.mjs +0 -4
  61. package/esm2022/MarkdownCard/markdown-card.component.mjs +0 -436
  62. package/esm2022/MarkdownCard/markdown-card.module.mjs +0 -44
  63. package/esm2022/components-common/Base/foundation.mjs +0 -66
  64. package/esm2022/components-common/Bubble/common/bubble-constants.mjs +0 -5
  65. package/esm2022/components-common/Bubble/common/bubble-types.mjs +0 -2
  66. package/esm2022/components-common/Bubble/foundation.mjs +0 -30
  67. package/esm2022/components-common/Input/button-foundation.mjs +0 -33
  68. package/esm2022/components-common/Input/common/types.mjs +0 -21
  69. package/esm2022/components-common/Input/foundation.mjs +0 -71
  70. package/esm2022/components-common/Locale/lang/en-us.mjs +0 -26
  71. package/esm2022/components-common/Locale/lang/zh-cn.mjs +0 -26
  72. package/esm2022/components-common/MarkdownCard/codeblock-foundation.mjs +0 -132
  73. package/esm2022/components-common/MarkdownCard/common/MDCardService.mjs +0 -69
  74. package/esm2022/components-common/MarkdownCard/common/MermaidService.mjs +0 -222
  75. package/esm2022/components-common/MarkdownCard/common/mdCard.types.mjs +0 -6
  76. package/esm2022/components-common/MarkdownCard/common/parser.mjs +0 -194
  77. package/esm2022/components-common/MarkdownCard/foundation.mjs +0 -84
  78. package/esm2022/matechat-ng.mjs +0 -5
  79. package/esm2022/public-api.mjs +0 -8
  80. package/public-api.d.ts +0 -4
@@ -1,222 +0,0 @@
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"]}
@@ -1,6 +0,0 @@
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
@@ -1,194 +0,0 @@
1
- // 判断是否自闭合标签
2
- export const isSelfClosingTag = (token) => {
3
- // 判断token.content里面是否包含完整的HTML结构(所有开始标签都有对应的结束标签)
4
- const content = token.content || '';
5
- // 检查是否是自闭合标签(如 <img />, <br /> 等)
6
- if (content.match(/<(\w+)[^>]*\/>/)) {
7
- return true;
8
- }
9
- // 检查是否包含完整的HTML结构
10
- const tagStack = [];
11
- const openTagRegex = /<(\w+)[^>]*>/g;
12
- const closeTagRegex = /<\/(\w+)>/g;
13
- let openMatch;
14
- let closeMatch;
15
- // 重置正则表达式的lastIndex
16
- openTagRegex.lastIndex = 0;
17
- closeTagRegex.lastIndex = 0;
18
- // 按顺序处理所有标签
19
- const allMatches = [];
20
- // 收集所有开始标签
21
- while ((openMatch = openTagRegex.exec(content)) !== null) {
22
- allMatches.push({
23
- type: 'open',
24
- tagName: openMatch[1],
25
- index: openMatch.index
26
- });
27
- }
28
- // 收集所有结束标签
29
- while ((closeMatch = closeTagRegex.exec(content)) !== null) {
30
- allMatches.push({
31
- type: 'close',
32
- tagName: closeMatch[1],
33
- index: closeMatch.index
34
- });
35
- }
36
- // 按位置排序
37
- allMatches.sort((a, b) => a.index - b.index);
38
- // 检查标签是否完全匹配
39
- for (const match of allMatches) {
40
- if (match.type === 'open') {
41
- tagStack.push(match.tagName);
42
- }
43
- else {
44
- if (tagStack.length === 0) {
45
- return false; // 没有对应的开始标签
46
- }
47
- const lastOpenTag = tagStack[tagStack.length - 1];
48
- if (lastOpenTag !== match.tagName) {
49
- return false; // 标签不匹配
50
- }
51
- tagStack.pop();
52
- }
53
- }
54
- // 只有当所有标签都正确匹配时,才认为是自闭合的
55
- return tagStack.length === 0;
56
- };
57
- // 判断是否是结束标签
58
- export const isClosingTag = (openToken, closeToken) => {
59
- const openContent = openToken?.content || '';
60
- const closeContent = closeToken?.content || '';
61
- const openTagMatch = openContent.match(/<(\w+)/);
62
- const closeTagMatch = closeContent.match(/<\/(\w+)/);
63
- if (openTagMatch && closeTagMatch) {
64
- return openTagMatch[1] === closeTagMatch[1];
65
- }
66
- return false;
67
- };
68
- // 创建ast树节点
69
- export const genTreeNode = (node) => {
70
- return {
71
- nodeType: node ? node.type.replace('_open', '') : 'root',
72
- openNode: node,
73
- closeNode: null,
74
- children: [],
75
- vNodeKey: node?.vNodeKey || ''
76
- };
77
- };
78
- // 匹配成对html token
79
- export const matchHtmlToken = (token, stack) => {
80
- // 简单排除单独的闭合标签
81
- const isCloseTag = token.content.startsWith('</');
82
- if (!stack.length) {
83
- token.nesting = isCloseTag ? 0 : 1;
84
- stack.push(token);
85
- return;
86
- }
87
- // 判断当前token是否是上一个html token的闭合标签
88
- const prevToken = stack[stack.length - 1];
89
- const closing = isClosingTag(prevToken, token);
90
- if (closing) {
91
- token.nesting = -1;
92
- stack.pop();
93
- }
94
- else {
95
- if (isCloseTag) {
96
- token.nesting = 0;
97
- }
98
- else {
99
- token.nesting = 1;
100
- stack.push(token);
101
- }
102
- }
103
- };
104
- export const isValidTagName = (tagName) => {
105
- if (!tagName)
106
- return false;
107
- try {
108
- document.createElement(tagName);
109
- return true;
110
- }
111
- catch (error) {
112
- return false;
113
- }
114
- };
115
- export const tokensToAst = (tokens) => {
116
- // 递归处理 inline 类型的 token
117
- const processInlineToken = (token) => {
118
- const node = genTreeNode(token);
119
- // 如果 token 有 children,递归处理它们
120
- if (token.children && token.children.length > 0) {
121
- node.children = tokensToAst(token.children);
122
- }
123
- return node;
124
- };
125
- // 创建根节点
126
- const rootNode = genTreeNode(null);
127
- let curr = rootNode;
128
- const stack = [];
129
- const htmlInlineTokenStack = [];
130
- const htmlBlockTokenStack = [];
131
- // 处理html token nesting值
132
- tokens.forEach((tok, idx) => {
133
- tok.vNodeKey = `mc-markdown-content-key-${idx}`;
134
- tok.tokenIndex = idx;
135
- if (tok.type.includes('html_')) {
136
- if (isSelfClosingTag(tok)) {
137
- tok.nesting = 0;
138
- return;
139
- }
140
- const stack = tok.type === 'html_block'
141
- ? htmlBlockTokenStack
142
- : htmlInlineTokenStack;
143
- matchHtmlToken(tok, stack);
144
- }
145
- });
146
- tokens.forEach((tok, idx) => {
147
- let tmp;
148
- if (tok.nesting === 1) {
149
- // 开始标签
150
- tmp = genTreeNode(tok);
151
- curr.children.push(tmp);
152
- stack.push(curr);
153
- curr = tmp;
154
- }
155
- else if (tok.nesting === -1) {
156
- // 结束标签
157
- curr.closeNode = tok;
158
- if (!stack.length) {
159
- throw new Error('AST stack underflow.');
160
- }
161
- tmp = stack.pop();
162
- curr = tmp;
163
- }
164
- else if (tok.nesting === 0) {
165
- // 自闭合标签或 inline 内容
166
- if (tok.type === 'inline' && tok.children && tok.children.length > 0) {
167
- // 对于 inline 类型,递归处理其 children
168
- const inlineNode = processInlineToken(tok);
169
- curr.children.push(inlineNode);
170
- }
171
- else {
172
- // 普通 token,直接添加
173
- curr.children.push(tok);
174
- }
175
- }
176
- else {
177
- throw new Error(`Invalid nesting level found in token index ${idx}.`);
178
- }
179
- });
180
- if (stack.length !== 0) {
181
- // throw new Error('Unbalanced block open/close tokens.');
182
- }
183
- return rootNode.children;
184
- };
185
- // 声明一个utils静态默认导出
186
- export default {
187
- isSelfClosingTag,
188
- isClosingTag,
189
- tokensToAst,
190
- genTreeNode,
191
- matchHtmlToken,
192
- isValidTagName
193
- };
194
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parser.js","sourceRoot":"","sources":["../../../../../../projects/components-ng/src/components-common/MarkdownCard/common/parser.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Token } from \"markdown-it\";\nimport type { ASTNode } from './mdCard.types';\n// 判断是否自闭合标签\nexport const isSelfClosingTag = (token: Token): boolean => {\n    // 判断token.content里面是否包含完整的HTML结构（所有开始标签都有对应的结束标签）\n    const content = token.content || '';\n\n    // 检查是否是自闭合标签（如 <img />, <br /> 等）\n    if (content.match(/<(\\w+)[^>]*\\/>/)) {\n        return true;\n    }\n\n    // 检查是否包含完整的HTML结构\n    const tagStack: string[] = [];\n    const openTagRegex = /<(\\w+)[^>]*>/g;\n    const closeTagRegex = /<\\/(\\w+)>/g;\n\n    let openMatch;\n    let closeMatch;\n\n    // 重置正则表达式的lastIndex\n    openTagRegex.lastIndex = 0;\n    closeTagRegex.lastIndex = 0;\n\n    // 按顺序处理所有标签\n    const allMatches: Array<{ type: 'open' | 'close', tagName: string, index: number }> = [];\n\n    // 收集所有开始标签\n    while ((openMatch = openTagRegex.exec(content)) !== null) {\n        allMatches.push({\n            type: 'open',\n            tagName: openMatch[1],\n            index: openMatch.index\n        });\n    }\n\n    // 收集所有结束标签\n    while ((closeMatch = closeTagRegex.exec(content)) !== null) {\n        allMatches.push({\n            type: 'close',\n            tagName: closeMatch[1],\n            index: closeMatch.index\n        });\n    }\n\n    // 按位置排序\n    allMatches.sort((a, b) => a.index - b.index);\n\n    // 检查标签是否完全匹配\n    for (const match of allMatches) {\n        if (match.type === 'open') {\n            tagStack.push(match.tagName);\n        } else {\n            if (tagStack.length === 0) {\n                return false; // 没有对应的开始标签\n            }\n            const lastOpenTag = tagStack[tagStack.length - 1];\n            if (lastOpenTag !== match.tagName) {\n                return false; // 标签不匹配\n            }\n            tagStack.pop();\n        }\n    }\n\n    // 只有当所有标签都正确匹配时，才认为是自闭合的\n    return tagStack.length === 0;\n}\n\n// 判断是否是结束标签\nexport const isClosingTag = (openToken: Token, closeToken: Token): boolean => {\n    const openContent = openToken?.content || '';\n    const closeContent = closeToken?.content || '';\n\n    const openTagMatch = openContent.match(/<(\\w+)/);\n    const closeTagMatch = closeContent.match(/<\\/(\\w+)/);\n\n    if (openTagMatch && closeTagMatch) {\n        return openTagMatch[1] === closeTagMatch[1];\n    }\n\n    return false;\n}\n\n// 创建ast树节点\nexport const genTreeNode = (node: Token | null): ASTNode => {\n    return {\n        nodeType: node ? node.type.replace('_open', '') : 'root',\n        openNode: node,\n        closeNode: null,\n        children: [],\n        vNodeKey: (node as any)?.vNodeKey || ''\n    };\n};\n\n// 匹配成对html token\nexport const matchHtmlToken = (token: Token, stack: Token[]) => {\n    // 简单排除单独的闭合标签\n    const isCloseTag = token.content.startsWith('</');\n    if (!stack.length) {\n        token.nesting = isCloseTag ? 0 : 1;\n        stack.push(token);\n        return;\n    }\n    // 判断当前token是否是上一个html token的闭合标签\n    const prevToken = stack[stack.length - 1];\n    const closing = isClosingTag(prevToken, token);\n    if (closing) {\n        token.nesting = -1;\n        stack.pop();\n    } else {\n        if (isCloseTag) {\n            token.nesting = 0;\n        } else {\n            token.nesting = 1;\n            stack.push(token);\n        }\n    }\n}\n\nexport const isValidTagName = (tagName: string | undefined): boolean => {\n    if (!tagName) return false\n    try {\n        document.createElement(tagName)\n        return true\n    } catch (error) {\n        return false\n    }\n}\n\nexport const tokensToAst = (tokens: Token[]) => {\n    // 递归处理 inline 类型的 token\n    const processInlineToken = (token: Token): ASTNode => {\n      const node = genTreeNode(token);\n\n      // 如果 token 有 children，递归处理它们\n      if (token.children && token.children.length > 0) {\n        node.children = tokensToAst(token.children);\n      }\n\n      return node;\n    };\n\n    // 创建根节点\n    const rootNode = genTreeNode(null);\n    let curr: ASTNode = rootNode;\n    const stack: ASTNode[] = [];\n    const htmlInlineTokenStack: Token[] = [];\n    const htmlBlockTokenStack: Token[] = [];\n    // 处理html token nesting值\n    tokens.forEach((tok: Token, idx: number) => {\n      (tok as any).vNodeKey = `mc-markdown-content-key-${idx}`;\n      (tok as any).tokenIndex = idx;\n\n      if (tok.type.includes('html_')) {\n        if (isSelfClosingTag(tok)) {\n          tok.nesting = 0;\n          return;\n        }\n\n        const stack =\n          tok.type === 'html_block'\n            ? htmlBlockTokenStack\n            : htmlInlineTokenStack;\n\n        matchHtmlToken(tok, stack);\n      }\n    });\n\n    tokens.forEach((tok: Token, idx: number) => {\n      let tmp: ASTNode;\n      if (tok.nesting === 1) {\n        // 开始标签\n        tmp = genTreeNode(tok);\n        curr.children.push(tmp);\n        stack.push(curr);\n        curr = tmp;\n      } else if (tok.nesting === -1) {\n        // 结束标签\n        curr.closeNode = tok;\n        if (!stack.length) {\n          throw new Error('AST stack underflow.');\n        }\n        tmp = stack.pop()!;\n        curr = tmp;\n      } else if (tok.nesting === 0) {\n        // 自闭合标签或 inline 内容\n        if (tok.type === 'inline' && tok.children && tok.children.length > 0) {\n          // 对于 inline 类型，递归处理其 children\n          const inlineNode = processInlineToken(tok);\n          curr.children.push(inlineNode);\n        } else {\n          // 普通 token，直接添加\n          curr.children.push(tok);\n        }\n      } else {\n        throw new Error(`Invalid nesting level found in token index ${idx}.`);\n      }\n    });\n\n    if (stack.length !== 0) {\n      // throw new Error('Unbalanced block open/close tokens.');\n    }\n    return rootNode.children as ASTNode[];\n  };\n\n// 声明一个utils静态默认导出\nexport default {\n    isSelfClosingTag,\n    isClosingTag,\n    tokensToAst,\n    genTreeNode,\n    matchHtmlToken,\n    isValidTagName\n}"]}
@@ -1,84 +0,0 @@
1
- import BaseFoundation from '../Base/foundation';
2
- import { defaultTypingConfig } from './common/mdCard.types';
3
- export class MarkdownCardFoundation extends BaseFoundation {
4
- constructor(adapter) {
5
- super({ ...adapter });
6
- this.isToken = (node) => {
7
- return 'type' in node && 'content' in node;
8
- };
9
- this.typewriterEnd = () => {
10
- this.setState({ isTyping: false });
11
- this._adapter.typingEnd?.();
12
- };
13
- this.getThinkContent = (content, thinkOptions) => {
14
- const thinkClass = thinkOptions?.customClass || 'mc-think-block';
15
- return (content
16
- ?.replace('<think>', `<div class="${thinkClass}">`)
17
- ?.replace('</think>', '</div>') || '');
18
- };
19
- this.parseTypingContent = (content) => {
20
- const { typingOptions } = this.getProps();
21
- const { typingIndex } = this.getStates();
22
- content = content.slice(0, typingIndex) || '';
23
- const options = { ...defaultTypingConfig, ...typingOptions };
24
- if (options.style === 'cursor') {
25
- content += `<span class="mc-typewriter mc-typewriter-cursor">|</span>`;
26
- }
27
- else if (options.style === 'color' || options.style === 'gradient') {
28
- content =
29
- content.slice(0, -5) +
30
- `<span class="mc-typewriter mc-typewriter-${options.style}">${content.slice(-5)}</span>`;
31
- }
32
- return content || '';
33
- };
34
- this.parseContent = () => {
35
- const { content, thinkOptions, enableThink } = this.getProps();
36
- const { typing, isTyping } = this.getStates();
37
- let parseContent = content || '';
38
- if (typing && isTyping) {
39
- parseContent = this.parseTypingContent(content);
40
- }
41
- if (enableThink) {
42
- parseContent = this.getThinkContent(content, thinkOptions);
43
- }
44
- parseContent = this._adapter.parseContent(parseContent);
45
- };
46
- this.typewriterStart = () => {
47
- const { typingOptions } = this.getProps();
48
- const { timer } = this.getStates();
49
- if (timer) {
50
- clearTimeout(timer);
51
- }
52
- this.setState({ isTyping: true });
53
- this._adapter.typingStart?.();
54
- const options = { ...defaultTypingConfig, ...typingOptions };
55
- const typingStep = () => {
56
- const { typingIndex, content } = this.getStates();
57
- let step = options.step || 2;
58
- if (Array.isArray(options.step)) {
59
- step =
60
- options.step[0] +
61
- Math.floor(Math.random() * (options.step[1] - options.step[0]));
62
- }
63
- let index = typingIndex + step;
64
- this.setState({ typingIndex: index });
65
- this.parseContent();
66
- this._adapter.typingEvent();
67
- if (index >= content.length) {
68
- this.typewriterEnd();
69
- this.parseContent();
70
- return;
71
- }
72
- let typingTimeout = setTimeout(typingStep, typeof options.interval === 'number' ? options.interval : 50);
73
- this.setState({
74
- timer: typingTimeout,
75
- });
76
- };
77
- let typingStepTimeout = setTimeout(typingStep);
78
- this.setState({
79
- timer: typingStepTimeout,
80
- });
81
- };
82
- }
83
- }
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"foundation.js","sourceRoot":"","sources":["../../../../../projects/components-ng/src/components-common/MarkdownCard/foundation.ts"],"names":[],"mappings":"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","sourcesContent":["import BaseFoundation, { DefaultAdapter } from '../Base/foundation';\nimport type { Token } from 'markdown-it';\nimport type { ASTNode } from './common/mdCard.types';\nimport { defaultTypingConfig } from './common/mdCard.types';\n\nexport interface MarkdownCardAdapter extends DefaultAdapter {\n  locale(key: string, params?: Record<string, string>): string;\n  typingStart: () => void;\n  typingEnd: () => void;\n  typingEvent: () => void;\n  parseContent: (content: string) => void;\n}\n\nexport class MarkdownCardFoundation extends BaseFoundation<MarkdownCardAdapter> {\n  constructor(adapter: MarkdownCardAdapter) {\n    super({ ...adapter });\n  }\n\n  isToken = (node: ASTNode | Token): node is Token => {\n    return 'type' in node && 'content' in node;\n  };\n\n  typewriterEnd = () => {\n    this.setState({ isTyping: false });\n    this._adapter.typingEnd?.();\n  };\n\n  getThinkContent = (content: string, thinkOptions: any) => {\n    const thinkClass = thinkOptions?.customClass || 'mc-think-block';\n    return (\n      content\n        ?.replace('<think>', `<div class=\"${thinkClass}\">`)\n        ?.replace('</think>', '</div>') || ''\n    );\n  };\n\n  parseTypingContent = (content: string) => {\n    const { typingOptions } = this.getProps();\n    const { typingIndex } = this.getStates();\n    content = content.slice(0, typingIndex) || '';\n    const options = { ...defaultTypingConfig, ...typingOptions };\n\n    if (options.style === 'cursor') {\n      content += `<span class=\"mc-typewriter mc-typewriter-cursor\">|</span>`;\n    } else if (options.style === 'color' || options.style === 'gradient') {\n      content =\n        content.slice(0, -5) +\n        `<span class=\"mc-typewriter mc-typewriter-${\n          options.style\n        }\">${content.slice(-5)}</span>`;\n    }\n    return content || '';\n  };\n\n  parseContent = () => {\n    const { content, thinkOptions, enableThink } = this.getProps();\n    const { typing, isTyping } = this.getStates();\n    let parseContent = content || '';\n    if (typing && isTyping) {\n      parseContent = this.parseTypingContent(content);\n    }\n\n    if (enableThink) {\n      parseContent = this.getThinkContent(content, thinkOptions);\n    }\n\n    parseContent = this._adapter.parseContent(parseContent);\n  };\n\n  typewriterStart = () => {\n    const { typingOptions } = this.getProps();\n    const { timer } = this.getStates();\n    if (timer) {\n      clearTimeout(timer);\n    }\n\n    this.setState({ isTyping: true });\n    this._adapter.typingStart?.();\n    const options = { ...defaultTypingConfig, ...typingOptions };\n\n    const typingStep = () => {\n      const { typingIndex, content } = this.getStates();\n\n      let step = options.step || 2;\n      if (Array.isArray(options.step)) {\n        step =\n          options.step[0] +\n          Math.floor(Math.random() * (options.step[1] - options.step[0]));\n      }\n      let index = typingIndex + step;\n      this.setState({ typingIndex: index });\n      this.parseContent();\n      this._adapter.typingEvent();\n\n      if (index >= content!.length) {\n        this.typewriterEnd();\n        this.parseContent();\n        return;\n      }\n\n      let typingTimeout = setTimeout(\n        typingStep,\n        typeof options.interval === 'number' ? options.interval : 50\n      );\n      this.setState({\n        timer: typingTimeout,\n      });\n    };\n\n    let typingStepTimeout = setTimeout(typingStep);\n    this.setState({\n      timer: typingStepTimeout,\n    });\n  };\n}\n"]}
@@ -1,5 +0,0 @@
1
- /**
2
- * Generated bundle index. Do not edit.
3
- */
4
- export * from './public-api';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWF0ZWNoYXQtbmcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9jb21wb25lbnRzLW5nL3NyYy9tYXRlY2hhdC1uZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -1,8 +0,0 @@
1
- /*
2
- * Public API Surface of components-ng
3
- */
4
- export * from './Bubble/index';
5
- export * from './Input/index';
6
- export * from './Locale/index';
7
- export * from './MarkdownCard/index';
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMtbmcvc3JjL3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxzQkFBc0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgY29tcG9uZW50cy1uZ1xuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vQnViYmxlL2luZGV4JztcbmV4cG9ydCAqIGZyb20gJy4vSW5wdXQvaW5kZXgnO1xuZXhwb3J0ICogZnJvbSAnLi9Mb2NhbGUvaW5kZXgnO1xuZXhwb3J0ICogZnJvbSAnLi9NYXJrZG93bkNhcmQvaW5kZXgnO1xuIl19
package/public-api.d.ts DELETED
@@ -1,4 +0,0 @@
1
- export * from './Bubble/index';
2
- export * from './Input/index';
3
- export * from './Locale/index';
4
- export * from './MarkdownCard/index';