@xw-components/formula-editor 18.1.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.
@@ -0,0 +1,318 @@
1
+ import { inject, Injectable } from '@angular/core';
2
+ import { DomSanitizer } from '@angular/platform-browser';
3
+ import { Node, mergeAttributes } from '@tiptap/core';
4
+ import * as i0 from "@angular/core";
5
+ export class FormulaEditorService {
6
+ constructor() {
7
+ this.sanitizer = inject(DomSanitizer);
8
+ }
9
+ initNodes() {
10
+ this.createFuncNode();
11
+ this.createFieldNode();
12
+ }
13
+ createFieldNode() {
14
+ this.fieldNode = Node.create({
15
+ name: 'field',
16
+ inline: true,
17
+ group: 'inline',
18
+ selectable: false,
19
+ draggable: true,
20
+ atom: true,
21
+ addAttributes() {
22
+ return {
23
+ id: {
24
+ default: null
25
+ },
26
+ name: {
27
+ default: ''
28
+ },
29
+ className: {
30
+ default: ''
31
+ },
32
+ dataType: {
33
+ default: 'field'
34
+ }
35
+ };
36
+ },
37
+ renderText({ node }) {
38
+ return `${node.attrs['name']}`;
39
+ },
40
+ parseHTML() {
41
+ return [{ tag: 'div[data-type="field"]' }];
42
+ },
43
+ renderHTML({ HTMLAttributes, node }) {
44
+ return [
45
+ 'div',
46
+ mergeAttributes({
47
+ 'data-type': 'field',
48
+ class: `field-node ${node.attrs['className']}` || '',
49
+ contenteditable: 'false',
50
+ style: `display: inline-block;margin: 0 4px;font-size: 13px;height: 18px;line-height: 18px;`
51
+ }, HTMLAttributes),
52
+ node.attrs['name']
53
+ ];
54
+ }
55
+ });
56
+ }
57
+ createFuncNode() {
58
+ this.funcNode = Node.create({
59
+ name: 'func',
60
+ inline: true,
61
+ group: 'inline',
62
+ selectable: false,
63
+ draggable: false,
64
+ content: 'inline*',
65
+ atom: true,
66
+ addAttributes() {
67
+ return {
68
+ id: {
69
+ default: null
70
+ },
71
+ name: {
72
+ default: ''
73
+ },
74
+ className: {
75
+ default: ''
76
+ },
77
+ dataType: {
78
+ default: 'func'
79
+ }
80
+ };
81
+ },
82
+ renderText({ node }) {
83
+ const funcName = node.attrs['name'] || '';
84
+ const childrenText = node.content.content
85
+ .map(child => {
86
+ if (child.text) {
87
+ return child.text;
88
+ }
89
+ if (child.type && child.type.spec && child.type.spec['toText']) {
90
+ return child.type.spec['toText']({ node: child });
91
+ }
92
+ return '';
93
+ })
94
+ .join('');
95
+ return `${funcName}(${childrenText})`;
96
+ },
97
+ parseHTML() {
98
+ return [{ tag: 'div[data-type="func"]' }];
99
+ },
100
+ addNodeView() {
101
+ return ({ node }) => {
102
+ const createElement = (tag, props = {}) => {
103
+ const element = document.createElement(tag);
104
+ Object.entries(props).forEach(([key, value]) => {
105
+ if (key === 'style') {
106
+ Object.entries(value).forEach(([styleKey, styleValue]) => {
107
+ element.style[styleKey] = styleValue;
108
+ });
109
+ }
110
+ else if (key === 'innerText') {
111
+ element.innerText = value;
112
+ }
113
+ else if (key === 'className') {
114
+ element.className = value;
115
+ }
116
+ else {
117
+ element.setAttribute(key, value);
118
+ }
119
+ });
120
+ return element;
121
+ };
122
+ const className = node.attrs['className'] || '';
123
+ const funcName = node.attrs['name'];
124
+ const container = createElement('span', {
125
+ style: {
126
+ display: 'inline-block',
127
+ margin: '0 4px',
128
+ fontSize: '13px'
129
+ }
130
+ });
131
+ const content = createElement('span', {
132
+ style: {
133
+ whiteSpace: 'pre-wrap',
134
+ display: 'inline-block'
135
+ },
136
+ contenteditable: 'true'
137
+ });
138
+ const before = createElement('span', {
139
+ innerText: `${funcName} `,
140
+ contenteditable: 'false',
141
+ className
142
+ });
143
+ const bracket = (value, style = {}) => {
144
+ return createElement('span', {
145
+ innerText: value,
146
+ contenteditable: 'false',
147
+ style: {
148
+ ...style
149
+ }
150
+ });
151
+ };
152
+ container.append(before, bracket('(', { marginRight: '2px' }), content, bracket(')', { marginLeft: '2px' }));
153
+ return {
154
+ dom: container,
155
+ contentDOM: content
156
+ };
157
+ };
158
+ }
159
+ });
160
+ }
161
+ /**
162
+ * 校验公式的JSON表示是否有效
163
+ * @param formula 公式的JSON表示
164
+ * @returns 校验结果,包含是否有效和错误信息数组
165
+ */
166
+ validateFormula(formula) {
167
+ const errors = [];
168
+ if (!formula) {
169
+ errors.push('公式不能为空');
170
+ return { isValid: false, errors };
171
+ }
172
+ if (formula.type !== 'doc') {
173
+ errors.push('公式根节点类型必须是 doc');
174
+ }
175
+ if (!formula.content || !Array.isArray(formula.content)) {
176
+ errors.push('公式必须包含 content 数组');
177
+ return { isValid: false, errors };
178
+ }
179
+ for (const paragraph of formula.content) {
180
+ if (paragraph.type !== 'paragraph') {
181
+ errors.push('段落类型必须是 paragraph');
182
+ continue;
183
+ }
184
+ if (!paragraph.content || !Array.isArray(paragraph.content)) {
185
+ continue;
186
+ }
187
+ for (const node of paragraph.content) {
188
+ this.validateNode(node, errors);
189
+ }
190
+ }
191
+ return {
192
+ isValid: errors.length === 0,
193
+ errors
194
+ };
195
+ }
196
+ /**
197
+ * 从JSON数据中提取内容文本或HTML
198
+ * @param jsonData 公式的JSON表示
199
+ * @param outputFormat 输出格式,'text'或'html',默认'text'
200
+ * @returns 提取的内容文本或HTML字符串
201
+ */
202
+ getContentFromJson(jsonData, outputFormat = 'text') {
203
+ if (!jsonData || !jsonData.content)
204
+ return '';
205
+ return outputFormat === 'html'
206
+ ? this.sanitizer.bypassSecurityTrustHtml(this.processContentToHtml(jsonData.content))
207
+ : this.processContent(jsonData.content);
208
+ }
209
+ processContent(content) {
210
+ if (!content || !Array.isArray(content))
211
+ return '';
212
+ return content.map(item => this.processNode(item)).join('');
213
+ }
214
+ processNode(node) {
215
+ if (!node)
216
+ return '';
217
+ switch (node.type) {
218
+ case 'func':
219
+ return this.processFuncNode(node);
220
+ case 'field':
221
+ return `{${node.attrs?.name || ''}}`;
222
+ case 'metric':
223
+ return `[${node.attrs?.name || ''}]`;
224
+ case 'text':
225
+ return node.text || '';
226
+ case 'paragraph':
227
+ return this.processContent(node.content);
228
+ case 'doc':
229
+ return this.processContent(node.content);
230
+ default:
231
+ return '';
232
+ }
233
+ }
234
+ processFuncNode(node) {
235
+ const funcName = node.attrs?.name || '';
236
+ const content = this.processContent(node.content);
237
+ return `${funcName}(${content})`;
238
+ }
239
+ processContentToHtml(content) {
240
+ if (!content || !Array.isArray(content))
241
+ return '';
242
+ return content.map(item => this.processNodeToHtml(item)).join('');
243
+ }
244
+ processNodeToHtml(node) {
245
+ if (!node)
246
+ return '';
247
+ switch (node.type) {
248
+ case 'func':
249
+ return this.processFuncNodeToHtml(node);
250
+ case 'field':
251
+ return `<span class="${node.attrs?.className || ''}" style="display: inline-block; margin: 0 4px;height: 18px;line-height: 18px;">${node.attrs?.name || ''}</span>`;
252
+ case 'text':
253
+ return this.escapeHtml(node.text || '');
254
+ case 'paragraph':
255
+ return this.processContentToHtml(node.content);
256
+ case 'doc':
257
+ return this.processContentToHtml(node.content);
258
+ default:
259
+ return '';
260
+ }
261
+ }
262
+ processFuncNodeToHtml(node) {
263
+ const funcName = node.attrs?.name || '';
264
+ const className = node.attrs?.className || '';
265
+ const content = this.processContentToHtml(node.content);
266
+ return `<span style="display: inline-block; margin: 0 4px;"><span contenteditable="false" class="${className}">${funcName}(</span><span style="white-space: pre-wrap; display: inline-block; margin: 0 4px;" contenteditable="true">${content}</span><span contenteditable="false" class="${className}">)</span></span>`;
267
+ }
268
+ escapeHtml(text) {
269
+ const div = document.createElement('div');
270
+ div.textContent = text;
271
+ return div.innerHTML;
272
+ }
273
+ validateNode(node, errors, path = '') {
274
+ if (!node || !node.type) {
275
+ return;
276
+ }
277
+ const currentPath = path ? `${path} > ${node.type}` : node.type;
278
+ if (node.type === 'func') {
279
+ this.validateFuncNode(node, errors, currentPath);
280
+ }
281
+ }
282
+ validateFuncNode(node, errors, path) {
283
+ if (!node.content || !Array.isArray(node.content)) {
284
+ return;
285
+ }
286
+ const content = node.content;
287
+ const length = content.length;
288
+ for (let i = 0; i < length; i++) {
289
+ const currentItem = content[i];
290
+ const nextItem = content[i + 1];
291
+ if (currentItem.type === 'field' || currentItem.type === 'func') {
292
+ if (i < length - 1) {
293
+ if (nextItem && nextItem.type === 'text') {
294
+ const text = nextItem.text || '';
295
+ if (!text.startsWith(',')) {
296
+ errors.push(`${path}: 参数之间必须用逗号分隔,位置 ${i + 1}`);
297
+ }
298
+ }
299
+ else {
300
+ errors.push(`${path}: 参数之间必须用逗号分隔,位置 ${i + 1}`);
301
+ }
302
+ }
303
+ }
304
+ if (currentItem.type === 'func') {
305
+ this.validateFuncNode(currentItem, errors, `${path} > ${currentItem.attrs?.name || 'func'}`);
306
+ }
307
+ }
308
+ }
309
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormulaEditorService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
310
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormulaEditorService, providedIn: 'root' }); }
311
+ }
312
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormulaEditorService, decorators: [{
313
+ type: Injectable,
314
+ args: [{
315
+ providedIn: 'root'
316
+ }]
317
+ }] });
318
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formula-editor.service.js","sourceRoot":"","sources":["../../../../components/formula-editor/formula-editor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { style } from '@angular/animations';\nimport { inject, Injectable } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { Editor, Node, mergeAttributes } from '@tiptap/core';\nimport { NzSafeAny } from 'ng-zorro-antd/core/types';\nexport interface ValidationResult {\n  isValid: boolean;\n  errors: string[];\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class FormulaEditorService {\n  private readonly sanitizer: DomSanitizer = inject(DomSanitizer);\n  funcNode?: Node;\n  fieldNode?: Node;\n  paramNode?: Node;\n\n  initNodes() {\n    this.createFuncNode();\n    this.createFieldNode();\n  }\n\n  createFieldNode() {\n    this.fieldNode = Node.create({\n      name: 'field',\n      inline: true,\n      group: 'inline',\n      selectable: false,\n      draggable: true,\n      atom: true,\n      addAttributes() {\n        return {\n          id: {\n            default: null\n          },\n          name: {\n            default: ''\n          },\n          className: {\n            default: ''\n          },\n          dataType: {\n            default: 'field'\n          }\n        };\n      },\n      renderText({ node }) {\n        return `${node.attrs['name']}`;\n      },\n      parseHTML() {\n        return [{ tag: 'div[data-type=\"field\"]' }];\n      },\n      renderHTML({ HTMLAttributes, node }) {\n        return [\n          'div',\n          mergeAttributes(\n            {\n              'data-type': 'field',\n              class: `field-node ${node.attrs['className']}` || '',\n              contenteditable: 'false',\n              style: `display: inline-block;margin: 0 4px;font-size: 13px;height: 18px;line-height: 18px;`\n            },\n            HTMLAttributes\n          ),\n          node.attrs['name']\n        ];\n      }\n    });\n  }\n\n  createFuncNode() {\n    this.funcNode = Node.create({\n      name: 'func',\n      inline: true,\n      group: 'inline',\n      selectable: false,\n      draggable: false,\n      content: 'inline*',\n      atom: true,\n      addAttributes() {\n        return {\n          id: {\n            default: null\n          },\n          name: {\n            default: ''\n          },\n          className: {\n            default: ''\n          },\n          dataType: {\n            default: 'func'\n          }\n        };\n      },\n      renderText({ node }) {\n        const funcName = node.attrs['name'] || '';\n        const childrenText = node.content.content\n          .map(child => {\n            if (child.text) {\n              return child.text;\n            }\n            if (child.type && child.type.spec && child.type.spec['toText']) {\n              return child.type.spec['toText']({ node: child });\n            }\n            return '';\n          })\n          .join('');\n        return `${funcName}(${childrenText})`;\n      },\n      parseHTML() {\n        return [{ tag: 'div[data-type=\"func\"]' }];\n      },\n      addNodeView() {\n        return ({ node }) => {\n          const createElement = (tag: string, props: Record<string, any> = {}) => {\n            const element = document.createElement(tag);\n            Object.entries(props).forEach(([key, value]) => {\n              if (key === 'style') {\n                Object.entries(value).forEach(([styleKey, styleValue]) => {\n                  (element.style as any)[styleKey] = styleValue;\n                });\n              } else if (key === 'innerText') {\n                element.innerText = value;\n              } else if (key === 'className') {\n                element.className = value;\n              } else {\n                element.setAttribute(key, value);\n              }\n            });\n            return element;\n          };\n\n          const className = node.attrs['className'] || '';\n          const funcName = node.attrs['name'];\n\n          const container = createElement('span', {\n            style: {\n              display: 'inline-block',\n              margin: '0 4px',\n              fontSize: '13px'\n            }\n          });\n\n          const content = createElement('span', {\n            style: {\n              whiteSpace: 'pre-wrap',\n              display: 'inline-block'\n            },\n            contenteditable: 'true'\n          });\n\n          const before = createElement('span', {\n            innerText: `${funcName} `,\n            contenteditable: 'false',\n            className\n          });\n\n          const bracket = (value: string, style: Record<string, any> = {}) => {\n            return createElement('span', {\n              innerText: value,\n              contenteditable: 'false',\n              style: {\n                ...style\n              }\n            });\n          };\n\n          container.append(before, bracket('(', { marginRight: '2px' }), content, bracket(')', { marginLeft: '2px' }));\n\n          return {\n            dom: container,\n            contentDOM: content\n          };\n        };\n      }\n    });\n  }\n\n  /**\n   * 校验公式的JSON表示是否有效\n   * @param formula 公式的JSON表示\n   * @returns 校验结果，包含是否有效和错误信息数组\n   */\n  public validateFormula(formula: any): ValidationResult {\n    const errors: string[] = [];\n\n    if (!formula) {\n      errors.push('公式不能为空');\n      return { isValid: false, errors };\n    }\n\n    if (formula.type !== 'doc') {\n      errors.push('公式根节点类型必须是 doc');\n    }\n\n    if (!formula.content || !Array.isArray(formula.content)) {\n      errors.push('公式必须包含 content 数组');\n      return { isValid: false, errors };\n    }\n\n    for (const paragraph of formula.content) {\n      if (paragraph.type !== 'paragraph') {\n        errors.push('段落类型必须是 paragraph');\n        continue;\n      }\n\n      if (!paragraph.content || !Array.isArray(paragraph.content)) {\n        continue;\n      }\n\n      for (const node of paragraph.content) {\n        this.validateNode(node, errors);\n      }\n    }\n\n    return {\n      isValid: errors.length === 0,\n      errors\n    };\n  }\n\n  /**\n   * 从JSON数据中提取内容文本或HTML\n   * @param jsonData 公式的JSON表示\n   * @param outputFormat 输出格式，'text'或'html'，默认'text'\n   * @returns 提取的内容文本或HTML字符串\n   */\n  public getContentFromJson(jsonData: NzSafeAny, outputFormat: 'text' | 'html' = 'text'): string | SafeHtml {\n    if (!jsonData || !jsonData.content) return '';\n    return outputFormat === 'html'\n      ? this.sanitizer.bypassSecurityTrustHtml(this.processContentToHtml(jsonData.content))\n      : this.processContent(jsonData.content);\n  }\n\n  private processContent(content: any[]): string {\n    if (!content || !Array.isArray(content)) return '';\n    return content.map(item => this.processNode(item)).join('');\n  }\n\n  private processNode(node: any): string {\n    if (!node) return '';\n\n    switch (node.type) {\n      case 'func':\n        return this.processFuncNode(node);\n      case 'field':\n        return `{${node.attrs?.name || ''}}`;\n      case 'metric':\n        return `[${node.attrs?.name || ''}]`;\n      case 'text':\n        return node.text || '';\n      case 'paragraph':\n        return this.processContent(node.content);\n      case 'doc':\n        return this.processContent(node.content);\n      default:\n        return '';\n    }\n  }\n\n  private processFuncNode(node: any): string {\n    const funcName = node.attrs?.name || '';\n    const content = this.processContent(node.content);\n    return `${funcName}(${content})`;\n  }\n\n  private processContentToHtml(content: any[]): string {\n    if (!content || !Array.isArray(content)) return '';\n\n    return content.map(item => this.processNodeToHtml(item)).join('');\n  }\n\n  private processNodeToHtml(node: any): string {\n    if (!node) return '';\n\n    switch (node.type) {\n      case 'func':\n        return this.processFuncNodeToHtml(node);\n      case 'field':\n        return `<span class=\"${node.attrs?.className || ''}\" style=\"display: inline-block; margin: 0 4px;height: 18px;line-height: 18px;\">${node.attrs?.name || ''}</span>`;\n      case 'text':\n        return this.escapeHtml(node.text || '');\n      case 'paragraph':\n        return this.processContentToHtml(node.content);\n      case 'doc':\n        return this.processContentToHtml(node.content);\n      default:\n        return '';\n    }\n  }\n\n  private processFuncNodeToHtml(node: any): string {\n    const funcName = node.attrs?.name || '';\n    const className = node.attrs?.className || '';\n    const content = this.processContentToHtml(node.content);\n\n    return `<span style=\"display: inline-block; margin: 0 4px;\"><span contenteditable=\"false\" class=\"${className}\">${funcName}(</span><span style=\"white-space: pre-wrap; display: inline-block; margin: 0 4px;\" contenteditable=\"true\">${content}</span><span contenteditable=\"false\" class=\"${className}\">)</span></span>`;\n  }\n\n  private escapeHtml(text: string): string {\n    const div = document.createElement('div');\n    div.textContent = text;\n    return div.innerHTML;\n  }\n\n  private validateNode(node: any, errors: string[], path: string = ''): void {\n    if (!node || !node.type) {\n      return;\n    }\n\n    const currentPath = path ? `${path} > ${node.type}` : node.type;\n\n    if (node.type === 'func') {\n      this.validateFuncNode(node, errors, currentPath);\n    }\n  }\n\n  private validateFuncNode(node: any, errors: string[], path: string): void {\n    if (!node.content || !Array.isArray(node.content)) {\n      return;\n    }\n\n    const content = node.content;\n    const length = content.length;\n\n    for (let i = 0; i < length; i++) {\n      const currentItem = content[i];\n      const nextItem = content[i + 1];\n\n      if (currentItem.type === 'field' || currentItem.type === 'func') {\n        if (i < length - 1) {\n          if (nextItem && nextItem.type === 'text') {\n            const text = nextItem.text || '';\n            if (!text.startsWith(',')) {\n              errors.push(`${path}: 参数之间必须用逗号分隔，位置 ${i + 1}`);\n            }\n          } else {\n            errors.push(`${path}: 参数之间必须用逗号分隔，位置 ${i + 1}`);\n          }\n        }\n      }\n\n      if (currentItem.type === 'func') {\n        this.validateFuncNode(currentItem, errors, `${path} > ${currentItem.attrs?.name || 'func'}`);\n      }\n    }\n  }\n}\n"]}
@@ -0,0 +1,6 @@
1
+ export * from './formula-editor.component';
2
+ export * from './formula-editor.model';
3
+ export * from './formula-editor-request.service';
4
+ export * from './formula-editor.service';
5
+ export * from './formula-editor-icon';
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9jb21wb25lbnRzL2Zvcm11bGEtZWRpdG9yL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNEJBQTRCLENBQUM7QUFDM0MsY0FBYyx3QkFBd0IsQ0FBQztBQUN2QyxjQUFjLGtDQUFrQyxDQUFDO0FBQ2pELGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyx1QkFBdUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZm9ybXVsYS1lZGl0b3IuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZm9ybXVsYS1lZGl0b3IubW9kZWwnO1xuZXhwb3J0ICogZnJvbSAnLi9mb3JtdWxhLWVkaXRvci1yZXF1ZXN0LnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9mb3JtdWxhLWVkaXRvci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vZm9ybXVsYS1lZGl0b3ItaWNvbic7XG4iXX0=