UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

147 lines 17.4 kB
/** * 自动折叠 managedFields 字段 * @param monacoEditor Monaco 编辑器实例 */ export function foldManagedFields(monacoEditor) { if (!monacoEditor) { return; } const model = monacoEditor.getModel(); if (!model) { return; } try { // 查找所有 managedFields 字段(只在 metadata 下的) const managedFieldsMatches = model.findMatches('^(\\s+)managedFields:', false, true, false, null, true); if (managedFieldsMatches && managedFieldsMatches.length > 0) { const foldingRanges = []; for (const match of managedFieldsMatches) { const startLine = match.range.startLineNumber; const endLine = findManagedFieldsEndLine(model, startLine); if (endLine > startLine) { foldingRanges.push({ startLineNumber: startLine, startColumn: 1, endLineNumber: endLine, endColumn: model.getLineMaxColumn(endLine), }); } } if (foldingRanges.length > 0) { // 递归折叠每个 managedFields 区域,确保每次折叠完成后再进行下一次 foldRangesSequentially(monacoEditor, foldingRanges?.reverse(), 0); } } } catch (error) { console.warn('Failed to fold managedFields:', error); } } /** * 递归折叠多个区域,确保每次折叠完成后再进行下一次 * @param monacoEditor Monaco 编辑器实例 * @param foldingRanges 折叠区域数组 * @param index 当前折叠的索引 */ async function foldRangesSequentially(monacoEditor, foldingRanges, index) { if (index >= foldingRanges.length) { return; } const range = foldingRanges[index]; monacoEditor.setSelection(range); await Promise.resolve(monacoEditor.getAction('editor.fold')?.run()); return foldRangesSequentially(monacoEditor, foldingRanges, index + 1); } /** * 查找 managedFields 字段的结束行 * @param model Monaco 编辑器模型 * @param startLine 起始行号 * @returns 结束行号 */ function findManagedFieldsEndLine(model, startLine) { const totalLines = model.getLineCount(); const startIndent = getLineIndentation(model, startLine); // 从下一行开始查找 for (let line = startLine + 1; line <= totalLines; line++) { const lineContent = model.getLineContent(line).trim(); // 跳过空行 if (!lineContent) { continue; } const currentIndent = getLineIndentation(model, line); // 如果缩进小于等于起始行,说明 managedFields 字段结束 if (currentIndent <= startIndent) { return line - 1; } } return totalLines; } /** * 获取行的缩进级别 * @param model Monaco 编辑器模型 * @param lineNumber 行号 * @returns 缩进级别(空格数) */ function getLineIndentation(model, lineNumber) { const lineContent = model.getLineContent(lineNumber); const match = lineContent.match(/^(\s*)/); return match ? match[1].length : 0; } /** * ManagedFields 自动折叠器 * 用于在内容变化时自动折叠 managedFields 字段 */ export class ManagedFieldsAutoFolder { constructor() { this.lastFoldedContent = ''; } /** * 开始监听内容变化并自动折叠 * @param editorProvider 编辑器实例提供者 * @param contentObservable 内容变化的 Observable */ startAutoFold(editorProvider, contentObservable) { this.stopAutoFold(); this.subscription = contentObservable.subscribe(() => { setTimeout(() => { const editor = editorProvider.getEditorInstance(); if (editor) { const model = editor.getModel(); if (model) { const currentContent = model.getValue(); // 只有当内容真正发生变化时才执行折叠 if (currentContent.includes('managedFields:') && currentContent !== this.lastFoldedContent) { foldManagedFields(editor); this.lastFoldedContent = currentContent; } } } }, 100); }); } /** * 停止自动折叠 */ stopAutoFold() { if (this.subscription) { this.subscription.unsubscribe(); this.subscription = undefined; } } /** * 手动执行一次折叠 * @param editorProvider 编辑器实例提供者 */ foldOnce(editorProvider) { const editor = editorProvider.getEditorInstance(); if (editor) { foldManagedFields(editor); const model = editor.getModel(); if (model) { this.lastFoldedContent = model.getValue(); } } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"managed-fields-folder.util.js","sourceRoot":"","sources":["../../../../../libs/common/src/code/managed-fields-folder.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { Subscription } from 'rxjs';\n\n/**\n * 自动折叠 managedFields 字段\n * @param monacoEditor Monaco 编辑器实例\n */\nexport function foldManagedFields(\n  monacoEditor: import('monaco-editor/esm/vs/editor/editor.api').editor.IStandaloneCodeEditor,\n): void {\n  if (!monacoEditor) {\n    return;\n  }\n\n  const model = monacoEditor.getModel();\n  if (!model) {\n    return;\n  }\n\n  try {\n    // 查找所有 managedFields 字段（只在 metadata 下的）\n    const managedFieldsMatches = model.findMatches(\n      '^(\\\\s+)managedFields:',\n      false,\n      true,\n      false,\n      null,\n      true,\n    );\n\n    if (managedFieldsMatches && managedFieldsMatches.length > 0) {\n      const foldingRanges: any[] = [];\n\n      for (const match of managedFieldsMatches) {\n        const startLine = match.range.startLineNumber;\n\n        const endLine = findManagedFieldsEndLine(model, startLine);\n\n        if (endLine > startLine) {\n          foldingRanges.push({\n            startLineNumber: startLine,\n            startColumn: 1,\n            endLineNumber: endLine,\n            endColumn: model.getLineMaxColumn(endLine),\n          });\n        }\n      }\n      if (foldingRanges.length > 0) {\n        // 递归折叠每个 managedFields 区域，确保每次折叠完成后再进行下一次\n        foldRangesSequentially(monacoEditor, foldingRanges?.reverse(), 0);\n      }\n    }\n  } catch (error) {\n    console.warn('Failed to fold managedFields:', error);\n  }\n}\n\n/**\n * 递归折叠多个区域，确保每次折叠完成后再进行下一次\n * @param monacoEditor Monaco 编辑器实例\n * @param foldingRanges 折叠区域数组\n * @param index 当前折叠的索引\n */\nasync function foldRangesSequentially(\n  monacoEditor: import('monaco-editor/esm/vs/editor/editor.api').editor.IStandaloneCodeEditor,\n  foldingRanges: any[],\n  index: number,\n): Promise<void> {\n  if (index >= foldingRanges.length) {\n    return;\n  }\n\n  const range = foldingRanges[index];\n  monacoEditor.setSelection(range);\n  await Promise.resolve(monacoEditor.getAction('editor.fold')?.run());\n\n  return foldRangesSequentially(monacoEditor, foldingRanges, index + 1);\n}\n\n/**\n * 查找 managedFields 字段的结束行\n * @param model Monaco 编辑器模型\n * @param startLine 起始行号\n * @returns 结束行号\n */\nfunction findManagedFieldsEndLine(\n  model: import('monaco-editor/esm/vs/editor/editor.api').editor.ITextModel,\n  startLine: number,\n): number {\n  const totalLines = model.getLineCount();\n  const startIndent = getLineIndentation(model, startLine);\n\n  // 从下一行开始查找\n  for (let line = startLine + 1; line <= totalLines; line++) {\n    const lineContent = model.getLineContent(line).trim();\n\n    // 跳过空行\n    if (!lineContent) {\n      continue;\n    }\n\n    const currentIndent = getLineIndentation(model, line);\n\n    // 如果缩进小于等于起始行，说明 managedFields 字段结束\n    if (currentIndent <= startIndent) {\n      return line - 1;\n    }\n  }\n\n  return totalLines;\n}\n\n/**\n * 获取行的缩进级别\n * @param model Monaco 编辑器模型\n * @param lineNumber 行号\n * @returns 缩进级别（空格数）\n */\nfunction getLineIndentation(\n  model: import('monaco-editor/esm/vs/editor/editor.api').editor.ITextModel,\n  lineNumber: number,\n): number {\n  const lineContent = model.getLineContent(lineNumber);\n  const match = lineContent.match(/^(\\s*)/);\n  return match ? match[1].length : 0;\n}\n\n/**\n * 编辑器实例获取器接口\n */\nexport interface EditorInstanceProvider {\n  getEditorInstance():\n    | import('monaco-editor/esm/vs/editor/editor.api').editor.IStandaloneCodeEditor\n    | null;\n}\n\n/**\n * ManagedFields 自动折叠器\n * 用于在内容变化时自动折叠 managedFields 字段\n */\nexport class ManagedFieldsAutoFolder {\n  private subscription?: Subscription;\n  private lastFoldedContent = '';\n\n  /**\n   * 开始监听内容变化并自动折叠\n   * @param editorProvider 编辑器实例提供者\n   * @param contentObservable 内容变化的 Observable\n   */\n  startAutoFold(\n    editorProvider: EditorInstanceProvider,\n    contentObservable: any,\n  ): void {\n    this.stopAutoFold();\n\n    this.subscription = contentObservable.subscribe(() => {\n      setTimeout(() => {\n        const editor = editorProvider.getEditorInstance();\n        if (editor) {\n          const model = editor.getModel();\n          if (model) {\n            const currentContent = model.getValue();\n            // 只有当内容真正发生变化时才执行折叠\n            if (\n              currentContent.includes('managedFields:') &&\n              currentContent !== this.lastFoldedContent\n            ) {\n              foldManagedFields(editor);\n              this.lastFoldedContent = currentContent;\n            }\n          }\n        }\n      }, 100);\n    });\n  }\n\n  /**\n   * 停止自动折叠\n   */\n  stopAutoFold(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n      this.subscription = undefined;\n    }\n  }\n\n  /**\n   * 手动执行一次折叠\n   * @param editorProvider 编辑器实例提供者\n   */\n  foldOnce(editorProvider: EditorInstanceProvider): void {\n    const editor = editorProvider.getEditorInstance();\n    if (editor) {\n      foldManagedFields(editor);\n      const model = editor.getModel();\n      if (model) {\n        this.lastFoldedContent = model.getValue();\n      }\n    }\n  }\n}\n"]}