obsidian-dev-utils 48.1.4 → 49.0.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.
Files changed (106) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/dist/lib/cjs/Async.cjs +24 -24
  3. package/dist/lib/cjs/Async.d.cts +6 -6
  4. package/dist/lib/cjs/Library.cjs +1 -1
  5. package/dist/lib/cjs/ObjectUtils.cjs +8 -9
  6. package/dist/lib/cjs/ScriptUtils/ESLint/eslint.config.cjs +13 -1
  7. package/dist/lib/cjs/ScriptUtils/esbuild/ObsidianPluginBuilder.cjs +6 -6
  8. package/dist/lib/cjs/ScriptUtils/esbuild/ObsidianPluginBuilder.d.cts +3 -3
  9. package/dist/lib/cjs/ScriptUtils/version.cjs +1 -1
  10. package/dist/lib/cjs/obsidian/AsyncWithNotice.cjs +7 -7
  11. package/dist/lib/cjs/obsidian/AsyncWithNotice.d.cts +6 -6
  12. package/dist/lib/cjs/obsidian/AttachmentPath.cjs +9 -9
  13. package/dist/lib/cjs/obsidian/AttachmentPath.d.cts +10 -10
  14. package/dist/lib/cjs/obsidian/Backlink.cjs +5 -5
  15. package/dist/lib/cjs/obsidian/Backlink.d.cts +6 -6
  16. package/dist/lib/cjs/obsidian/Callout.cjs +4 -4
  17. package/dist/lib/cjs/obsidian/Callout.d.cts +3 -3
  18. package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.cjs +11 -11
  19. package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.d.cts +4 -4
  20. package/dist/lib/cjs/obsidian/Commands/CommandBase.cjs +7 -7
  21. package/dist/lib/cjs/obsidian/Commands/CommandBase.d.cts +3 -3
  22. package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.cjs +8 -8
  23. package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.d.cts +4 -4
  24. package/dist/lib/cjs/obsidian/Commands/FileCommandBase.cjs +4 -4
  25. package/dist/lib/cjs/obsidian/Commands/FileCommandBase.d.cts +3 -3
  26. package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.cjs +4 -4
  27. package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.d.cts +3 -3
  28. package/dist/lib/cjs/obsidian/Commands/NonEditorCommandBase.cjs +4 -4
  29. package/dist/lib/cjs/obsidian/Commands/NonEditorCommandBase.d.cts +3 -3
  30. package/dist/lib/cjs/obsidian/Dataview.cjs +10 -10
  31. package/dist/lib/cjs/obsidian/Dataview.d.cts +10 -10
  32. package/dist/lib/cjs/obsidian/Link.cjs +58 -58
  33. package/dist/lib/cjs/obsidian/Link.d.cts +22 -22
  34. package/dist/lib/cjs/obsidian/Loop.cjs +4 -4
  35. package/dist/lib/cjs/obsidian/Loop.d.cts +3 -3
  36. package/dist/lib/cjs/obsidian/Markdown.cjs +9 -9
  37. package/dist/lib/cjs/obsidian/Markdown.d.cts +4 -4
  38. package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.cjs +24 -24
  39. package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.d.cts +14 -14
  40. package/dist/lib/cjs/obsidian/Modals/Alert.cjs +12 -12
  41. package/dist/lib/cjs/obsidian/Modals/Alert.d.cts +3 -3
  42. package/dist/lib/cjs/obsidian/Modals/Confirm.cjs +13 -13
  43. package/dist/lib/cjs/obsidian/Modals/Confirm.d.cts +3 -3
  44. package/dist/lib/cjs/obsidian/Modals/ModalBase.cjs +1 -1
  45. package/dist/lib/cjs/obsidian/Modals/ModalBase.d.cts +2 -2
  46. package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +14 -14
  47. package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +3 -3
  48. package/dist/lib/cjs/obsidian/Modals/SelectItem.cjs +11 -11
  49. package/dist/lib/cjs/obsidian/Modals/SelectItem.d.cts +3 -3
  50. package/dist/lib/cjs/obsidian/Queue.cjs +13 -13
  51. package/dist/lib/cjs/obsidian/Queue.d.cts +6 -6
  52. package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +19 -19
  53. package/dist/lib/cjs/obsidian/i18n/i18n.cjs +1 -1
  54. package/dist/lib/esm/Async.d.mts +6 -6
  55. package/dist/lib/esm/Async.mjs +24 -24
  56. package/dist/lib/esm/Library.mjs +1 -1
  57. package/dist/lib/esm/ObjectUtils.mjs +8 -9
  58. package/dist/lib/esm/ScriptUtils/ESLint/eslint.config.mjs +13 -1
  59. package/dist/lib/esm/ScriptUtils/esbuild/ObsidianPluginBuilder.d.mts +3 -3
  60. package/dist/lib/esm/ScriptUtils/esbuild/ObsidianPluginBuilder.mjs +6 -6
  61. package/dist/lib/esm/ScriptUtils/version.mjs +1 -1
  62. package/dist/lib/esm/obsidian/AsyncWithNotice.d.mts +6 -6
  63. package/dist/lib/esm/obsidian/AsyncWithNotice.mjs +7 -7
  64. package/dist/lib/esm/obsidian/AttachmentPath.d.mts +10 -10
  65. package/dist/lib/esm/obsidian/AttachmentPath.mjs +9 -9
  66. package/dist/lib/esm/obsidian/Backlink.d.mts +6 -6
  67. package/dist/lib/esm/obsidian/Backlink.mjs +5 -5
  68. package/dist/lib/esm/obsidian/Callout.d.mts +3 -3
  69. package/dist/lib/esm/obsidian/Callout.mjs +4 -4
  70. package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.d.mts +4 -4
  71. package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.mjs +11 -11
  72. package/dist/lib/esm/obsidian/Commands/CommandBase.d.mts +3 -3
  73. package/dist/lib/esm/obsidian/Commands/CommandBase.mjs +7 -7
  74. package/dist/lib/esm/obsidian/Commands/EditorCommandBase.d.mts +4 -4
  75. package/dist/lib/esm/obsidian/Commands/EditorCommandBase.mjs +8 -8
  76. package/dist/lib/esm/obsidian/Commands/FileCommandBase.d.mts +3 -3
  77. package/dist/lib/esm/obsidian/Commands/FileCommandBase.mjs +4 -4
  78. package/dist/lib/esm/obsidian/Commands/FolderCommandBase.d.mts +3 -3
  79. package/dist/lib/esm/obsidian/Commands/FolderCommandBase.mjs +4 -4
  80. package/dist/lib/esm/obsidian/Commands/NonEditorCommandBase.d.mts +3 -3
  81. package/dist/lib/esm/obsidian/Commands/NonEditorCommandBase.mjs +4 -4
  82. package/dist/lib/esm/obsidian/Dataview.d.mts +10 -10
  83. package/dist/lib/esm/obsidian/Dataview.mjs +10 -10
  84. package/dist/lib/esm/obsidian/Link.d.mts +22 -22
  85. package/dist/lib/esm/obsidian/Link.mjs +58 -58
  86. package/dist/lib/esm/obsidian/Loop.d.mts +3 -3
  87. package/dist/lib/esm/obsidian/Loop.mjs +4 -4
  88. package/dist/lib/esm/obsidian/Markdown.d.mts +4 -4
  89. package/dist/lib/esm/obsidian/Markdown.mjs +9 -9
  90. package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.d.mts +14 -14
  91. package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.mjs +24 -24
  92. package/dist/lib/esm/obsidian/Modals/Alert.d.mts +3 -3
  93. package/dist/lib/esm/obsidian/Modals/Alert.mjs +12 -12
  94. package/dist/lib/esm/obsidian/Modals/Confirm.d.mts +3 -3
  95. package/dist/lib/esm/obsidian/Modals/Confirm.mjs +13 -13
  96. package/dist/lib/esm/obsidian/Modals/ModalBase.d.mts +2 -2
  97. package/dist/lib/esm/obsidian/Modals/ModalBase.mjs +1 -1
  98. package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +3 -3
  99. package/dist/lib/esm/obsidian/Modals/Prompt.mjs +14 -14
  100. package/dist/lib/esm/obsidian/Modals/SelectItem.d.mts +3 -3
  101. package/dist/lib/esm/obsidian/Modals/SelectItem.mjs +11 -11
  102. package/dist/lib/esm/obsidian/Queue.d.mts +6 -6
  103. package/dist/lib/esm/obsidian/Queue.mjs +13 -13
  104. package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +19 -19
  105. package/dist/lib/esm/obsidian/i18n/i18n.mjs +1 -1
  106. package/package.json +1 -1
@@ -65,7 +65,7 @@ export declare enum LinkStyle {
65
65
  /**
66
66
  * Options for {@link convertLink}.
67
67
  */
68
- export interface ConvertLinkOptions {
68
+ export interface ConvertLinkParams {
69
69
  /**
70
70
  * An Obsidian app instance.
71
71
  */
@@ -94,7 +94,7 @@ export interface ConvertLinkOptions {
94
94
  /**
95
95
  * Options for {@link generateMarkdownLink}.
96
96
  */
97
- export interface GenerateMarkdownLinkOptions {
97
+ export interface GenerateMarkdownLinkParams {
98
98
  /**
99
99
  * An alias for the link.
100
100
  *
@@ -234,7 +234,7 @@ export interface GenerateMarkdownLinkOptions {
234
234
  /**
235
235
  * Options for {@link generateRawMarkdownLink}.
236
236
  */
237
- export interface GenerateRawMarkdownLinkOptions {
237
+ export interface GenerateRawMarkdownLinkParams {
238
238
  /**
239
239
  * An alias of the link. Defaults to `undefined`.
240
240
  */
@@ -374,7 +374,7 @@ export interface ParseLinkResult {
374
374
  /**
375
375
  * Options for {@link shouldResetAlias}.
376
376
  */
377
- export interface ShouldResetAliasOptions {
377
+ export interface ShouldResetAliasParams {
378
378
  /**
379
379
  * An Obsidian app instance.
380
380
  */
@@ -420,7 +420,7 @@ export interface SplitSubpathResult {
420
420
  /**
421
421
  * Options for {@link updateLink}.
422
422
  */
423
- export interface UpdateLinkOptions {
423
+ export interface UpdateLinkParams {
424
424
  /**
425
425
  * An Obsidian app instance.
426
426
  */
@@ -457,7 +457,7 @@ export interface UpdateLinkOptions {
457
457
  /**
458
458
  * Options for {@link updateLinksInFile}.
459
459
  */
460
- export interface UpdateLinksInFileOptions extends ProcessOptions {
460
+ export interface UpdateLinksInFileParams extends ProcessOptions {
461
461
  /**
462
462
  * An Obsidian app instance.
463
463
  */
@@ -486,7 +486,7 @@ export interface UpdateLinksInFileOptions extends ProcessOptions {
486
486
  /**
487
487
  * Options for {@link updateLinksInContent}.
488
488
  */
489
- interface UpdateLinksInContentOptions {
489
+ interface UpdateLinksInContentParams {
490
490
  /**
491
491
  * An Obsidian app instance.
492
492
  */
@@ -519,10 +519,10 @@ interface UpdateLinksInContentOptions {
519
519
  /**
520
520
  * Converts a link to a new path.
521
521
  *
522
- * @param options - The options for converting the link.
522
+ * @param params - The parameters for converting the link.
523
523
  * @returns The converted link.
524
524
  */
525
- export declare function convertLink(options: ConvertLinkOptions): string;
525
+ export declare function convertLink(params: ConvertLinkParams): string;
526
526
  /**
527
527
  * Edits the backlinks for a file or path.
528
528
  *
@@ -592,17 +592,17 @@ export declare function fixFrontmatterMarkdownLinks(cache: CachedMetadata): bool
592
592
  /**
593
593
  * Generates a markdown link based on the provided parameters.
594
594
  *
595
- * @param options - The options for generating the markdown link.
595
+ * @param params - The parameters for generating the markdown link.
596
596
  * @returns The generated markdown link.
597
597
  */
598
- export declare function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string;
598
+ export declare function generateMarkdownLink(params: GenerateMarkdownLinkParams): string;
599
599
  /**
600
600
  * Generates a raw markdown link based on the provided options.
601
601
  *
602
- * @param options - The options for generating a raw markdown link.
602
+ * @param params - The parameters for generating a raw markdown link.
603
603
  * @returns A raw markdown link.
604
604
  */
605
- export declare function generateRawMarkdownLink(options: GenerateRawMarkdownLinkOptions): string;
605
+ export declare function generateRawMarkdownLink(params: GenerateRawMarkdownLinkParams): string;
606
606
  /**
607
607
  * Parses a link into its components.
608
608
  *
@@ -623,14 +623,14 @@ export declare function parseLinks(str: string): ParseLinkResult[];
623
623
  * @param plugin - The plugin instance.
624
624
  * @param fn - The function that returns the default options.
625
625
  */
626
- export declare function registerGenerateMarkdownLinkDefaultOptionsFn(plugin: Plugin, fn: () => Partial<GenerateMarkdownLinkOptions>): void;
626
+ export declare function registerGenerateMarkdownLinkDefaultOptionsFn(plugin: Plugin, fn: () => Partial<GenerateMarkdownLinkParams>): void;
627
627
  /**
628
628
  * Determines if the alias of a link should be reset.
629
629
  *
630
- * @param options - The options for determining if the alias should be reset.
630
+ * @param params - The parameters for determining if the alias should be reset.
631
631
  * @returns Whether the alias should be reset.
632
632
  */
633
- export declare function shouldResetAlias(options: ShouldResetAliasOptions): boolean;
633
+ export declare function shouldResetAlias(params: ShouldResetAliasParams): boolean;
634
634
  /**
635
635
  * Splits a link into its link path and subpath.
636
636
  *
@@ -695,22 +695,22 @@ export declare function unescapeAlias(escapedAlias: string): string;
695
695
  /**
696
696
  * Updates a link based on the provided parameters.
697
697
  *
698
- * @param options - The options for updating the link.
698
+ * @param params - The parameters for updating the link.
699
699
  * @returns The updated link.
700
700
  */
701
- export declare function updateLink(options: UpdateLinkOptions): string;
701
+ export declare function updateLink(params: UpdateLinkParams): string;
702
702
  /**
703
703
  * Updates the links in a content string based on the provided parameters.
704
704
  *
705
- * @param options - The options for updating the links.
705
+ * @param params - The parameters for updating the links.
706
706
  * @returns A {@link Promise} that resolves to the content with updated links.
707
707
  */
708
- export declare function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<string>;
708
+ export declare function updateLinksInContent(params: UpdateLinksInContentParams): Promise<string>;
709
709
  /**
710
710
  * Updates the links in a file based on the provided parameters.
711
711
  *
712
- * @param options - The options for updating the links.
712
+ * @param params - The parameters for updating the links.
713
713
  * @returns A {@link Promise} that resolves when the links are updated.
714
714
  */
715
- export declare function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void>;
715
+ export declare function updateLinksInFile(params: UpdateLinksInFileParams): Promise<void>;
716
716
  export {};
@@ -131,10 +131,10 @@ var import_Debug = require('../Debug.cjs');
131
131
  var import_Error = require('../Error.cjs');
132
132
  var import_Function = require('../Function.cjs');
133
133
  var import_PluginContext = require('./Plugin/PluginContext.cjs');
134
- async function loop(options) {
134
+ async function loop(params) {
135
135
  const DEFAULT_OPTIONS = {
136
136
  abortSignal: (0, import_AbortController.abortSignalNever)(),
137
- /* v8 ignore start -- buildNoticeMessage is required in LoopOptions and always overridden by the spread. */
137
+ /* v8 ignore start -- buildNoticeMessage is required in LoopParams and always overridden by the spread. */
138
138
  buildNoticeMessage() {
139
139
  throw new Error("buildNoticeMessage is required");
140
140
  },
@@ -154,7 +154,7 @@ async function loop(options) {
154
154
  };
155
155
  const fullOptions = {
156
156
  ...DEFAULT_OPTIONS,
157
- ...options
157
+ ...params
158
158
  };
159
159
  const stackTrace = (0, import_Error.getStackTrace)(1);
160
160
  const items = fullOptions.items;
@@ -222,4 +222,4 @@ async function loop(options) {
222
222
  0 && (module.exports = {
223
223
  loop
224
224
  });
225
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Loop.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for looping in Obsidian.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  invokeAsyncSafely,\n  requestAnimationFrameAsync\n} from '../Async.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace\n} from '../Error.ts';\nimport { noop } from '../Function.ts';\nimport { addPluginCssClasses } from './Plugin/PluginContext.ts';\n\n/**\n * Options for {@link loop}.\n */\nexport interface LoopOptions<T> {\n  /**\n   * An optional abort signal to cancel the loop.\n   */\n  readonly abortSignal?: AbortSignal;\n\n  /**\n   * Build a notice message for each item.\n   *\n   * @param item - The current item.\n   * @param iterationStr - A string representing the current iteration.\n   * @returns A string to display in the notice.\n   */\n  readonly buildNoticeMessage: (item: T, iterationStr: string) => string;\n\n  /**\n   * Items to loop over.\n   */\n  readonly items: T[];\n\n  /**\n   * A timeout for the notice before it is shown. Default is `500`.\n   */\n  readonly noticeBeforeShownTimeoutInMilliseconds?: number;\n\n  /**\n   * A minimum timeout for the notice. Default is `2000`.\n   */\n  readonly noticeMinTimeoutInMilliseconds?: number;\n\n  /**\n   * Process each item.\n   *\n   * @param item - The current item.\n   */\n  readonly processItem: (item: T) => Promisable<void>;\n\n  /**\n   * A title of the progress bar. Default is `''`.\n   */\n  readonly progressBarTitle?: string;\n\n  /**\n   * Whether to continue the loop on error. Default is `true`.\n   */\n  readonly shouldContinueOnError?: boolean;\n\n  /**\n   * Whether to show a notice. Default is `true`.\n   */\n  readonly shouldShowNotice?: boolean;\n\n  /**\n   * Whether to show a progress bar. Default is `true`.\n   */\n  readonly shouldShowProgressBar?: boolean;\n\n  /**\n   * A threshold for the UI update. Default is `100`.\n   */\n  readonly uiUpdateThresholdInMilliseconds?: number;\n}\n\n/**\n * Loops over a list of items and processes each item.\n *\n * @param options - The options for the loop.\n */\nexport async function loop<T>(options: LoopOptions<T>): Promise<void> {\n  const DEFAULT_OPTIONS: Required<LoopOptions<T>> = {\n    abortSignal: abortSignalNever(),\n    /* v8 ignore start -- buildNoticeMessage is required in LoopOptions and always overridden by the spread. */\n    buildNoticeMessage() {\n      throw new Error('buildNoticeMessage is required');\n    },\n    /* v8 ignore stop */\n    items: [],\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeBeforeShownTimeoutInMilliseconds: 500,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeMinTimeoutInMilliseconds: 2000,\n    processItem: noop,\n    progressBarTitle: '',\n    shouldContinueOnError: true,\n    shouldShowNotice: true,\n    shouldShowProgressBar: true,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    uiUpdateThresholdInMilliseconds: 100\n  };\n\n  const fullOptions: Required<LoopOptions<T>> = {\n    ...DEFAULT_OPTIONS,\n    ...options\n  };\n\n  const stackTrace = getStackTrace(1);\n\n  const items = fullOptions.items;\n  let iterationCount = 0;\n  let notice = null as Notice | null;\n  let isDone = false;\n  invokeAsyncSafely(() => showNotice());\n\n  const noticeMinTimeoutPromise = sleep(fullOptions.noticeMinTimeoutInMilliseconds);\n  const progressBarEl = createEl('progress');\n  addPluginCssClasses(progressBarEl, 'loop');\n  progressBarEl.max = items.length;\n\n  let lastUIUpdateTimestamp = performance.now();\n\n  for (const item of items) {\n    if (fullOptions.abortSignal.aborted) {\n      notice?.hide();\n      return;\n    }\n    iterationCount++;\n    const iterationStr = `# ${String(iterationCount)} / ${String(items.length)}`;\n    const message = fullOptions.buildNoticeMessage(item, iterationStr);\n    if (!fullOptions.shouldShowProgressBar) {\n      notice?.setMessage(message);\n    }\n    getLibDebugger('Loop')(message);\n\n    try {\n      if (performance.now() - lastUIUpdateTimestamp > fullOptions.uiUpdateThresholdInMilliseconds) {\n        await requestAnimationFrameAsync();\n        lastUIUpdateTimestamp = performance.now();\n      }\n      await fullOptions.processItem(item);\n    } catch (error) {\n      console.error('Error processing item', item);\n      if (!fullOptions.shouldContinueOnError) {\n        notice?.hide();\n        throw new CustomStackTraceError('loop failed', stackTrace, error);\n      }\n\n      emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, error));\n    }\n    progressBarEl.value++;\n  }\n  if (notice) {\n    await noticeMinTimeoutPromise;\n  }\n  notice?.hide();\n  isDone = true;\n\n  async function showNotice(): Promise<void> {\n    if (!fullOptions.shouldShowNotice) {\n      return;\n    }\n    await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);\n    if (isDone) {\n      return;\n    }\n    notice = new Notice('', 0);\n    if (!fullOptions.shouldShowProgressBar) {\n      return;\n    }\n    const fragment = createFragment();\n    fragment.createDiv({ text: fullOptions.progressBarTitle });\n    fragment.appendChild(progressBarEl);\n    notice.setMessage(fragment);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,sBAAuB;AAEvB,6BAAiC;AACjC,mBAGO;AACP,mBAA+B;AAC/B,mBAKO;AACP,sBAAqB;AACrB,2BAAoC;AAyEpC,eAAsB,KAAQ,SAAwC;AACpE,QAAM,kBAA4C;AAAA,IAChD,iBAAa,yCAAiB;AAAA;AAAA,IAE9B,qBAAqB;AACnB,YAAM,IAAI,MAAM,gCAAgC;AAAA,IAClD;AAAA;AAAA,IAEA,OAAO,CAAC;AAAA;AAAA,IAER,wCAAwC;AAAA;AAAA,IAExC,gCAAgC;AAAA,IAChC,aAAa;AAAA,IACb,kBAAkB;AAAA,IAClB,uBAAuB;AAAA,IACvB,kBAAkB;AAAA,IAClB,uBAAuB;AAAA;AAAA,IAEvB,iCAAiC;AAAA,EACnC;AAEA,QAAM,cAAwC;AAAA,IAC5C,GAAG;AAAA,IACH,GAAG;AAAA,EACL;AAEA,QAAM,iBAAa,4BAAc,CAAC;AAElC,QAAM,QAAQ,YAAY;AAC1B,MAAI,iBAAiB;AACrB,MAAI,SAAS;AACb,MAAI,SAAS;AACb,sCAAkB,MAAM,WAAW,CAAC;AAEpC,QAAM,0BAA0B,MAAM,YAAY,8BAA8B;AAChF,QAAM,gBAAgB,SAAS,UAAU;AACzC,gDAAoB,eAAe,MAAM;AACzC,gBAAc,MAAM,MAAM;AAE1B,MAAI,wBAAwB,YAAY,IAAI;AAE5C,aAAW,QAAQ,OAAO;AACxB,QAAI,YAAY,YAAY,SAAS;AACnC,cAAQ,KAAK;AACb;AAAA,IACF;AACA;AACA,UAAM,eAAe,KAAK,OAAO,cAAc,CAAC,MAAM,OAAO,MAAM,MAAM,CAAC;AAC1E,UAAM,UAAU,YAAY,mBAAmB,MAAM,YAAY;AACjE,QAAI,CAAC,YAAY,uBAAuB;AACtC,cAAQ,WAAW,OAAO;AAAA,IAC5B;AACA,qCAAe,MAAM,EAAE,OAAO;AAE9B,QAAI;AACF,UAAI,YAAY,IAAI,IAAI,wBAAwB,YAAY,iCAAiC;AAC3F,kBAAM,yCAA2B;AACjC,gCAAwB,YAAY,IAAI;AAAA,MAC1C;AACA,YAAM,YAAY,YAAY,IAAI;AAAA,IACpC,SAAS,OAAO;AACd,cAAQ,MAAM,yBAAyB,IAAI;AAC3C,UAAI,CAAC,YAAY,uBAAuB;AACtC,gBAAQ,KAAK;AACb,cAAM,IAAI,mCAAsB,eAAe,YAAY,KAAK;AAAA,MAClE;AAEA,4CAAoB,IAAI,mCAAsB,0CAA6B,YAAY,KAAK,CAAC;AAAA,IAC/F;AACA,kBAAc;AAAA,EAChB;AACA,MAAI,QAAQ;AACV,UAAM;AAAA,EACR;AACA,UAAQ,KAAK;AACb,WAAS;AAET,iBAAe,aAA4B;AACzC,QAAI,CAAC,YAAY,kBAAkB;AACjC;AAAA,IACF;AACA,UAAM,MAAM,YAAY,sCAAsC;AAC9D,QAAI,QAAQ;AACV;AAAA,IACF;AACA,aAAS,IAAI,uBAAO,IAAI,CAAC;AACzB,QAAI,CAAC,YAAY,uBAAuB;AACtC;AAAA,IACF;AACA,UAAM,WAAW,eAAe;AAChC,aAAS,UAAU,EAAE,MAAM,YAAY,iBAAiB,CAAC;AACzD,aAAS,YAAY,aAAa;AAClC,WAAO,WAAW,QAAQ;AAAA,EAC5B;AACF;",
  "names": []
}

225
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Loop.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for looping in Obsidian.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  invokeAsyncSafely,\n  requestAnimationFrameAsync\n} from '../Async.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace\n} from '../Error.ts';\nimport { noop } from '../Function.ts';\nimport { addPluginCssClasses } from './Plugin/PluginContext.ts';\n\n/**\n * Options for {@link loop}.\n */\nexport interface LoopParams<T> {\n  /**\n   * An optional abort signal to cancel the loop.\n   */\n  readonly abortSignal?: AbortSignal;\n\n  /**\n   * Build a notice message for each item.\n   *\n   * @param item - The current item.\n   * @param iterationStr - A string representing the current iteration.\n   * @returns A string to display in the notice.\n   */\n  readonly buildNoticeMessage: (item: T, iterationStr: string) => string;\n\n  /**\n   * Items to loop over.\n   */\n  readonly items: T[];\n\n  /**\n   * A timeout for the notice before it is shown. Default is `500`.\n   */\n  readonly noticeBeforeShownTimeoutInMilliseconds?: number;\n\n  /**\n   * A minimum timeout for the notice. Default is `2000`.\n   */\n  readonly noticeMinTimeoutInMilliseconds?: number;\n\n  /**\n   * Process each item.\n   *\n   * @param item - The current item.\n   */\n  readonly processItem: (item: T) => Promisable<void>;\n\n  /**\n   * A title of the progress bar. Default is `''`.\n   */\n  readonly progressBarTitle?: string;\n\n  /**\n   * Whether to continue the loop on error. Default is `true`.\n   */\n  readonly shouldContinueOnError?: boolean;\n\n  /**\n   * Whether to show a notice. Default is `true`.\n   */\n  readonly shouldShowNotice?: boolean;\n\n  /**\n   * Whether to show a progress bar. Default is `true`.\n   */\n  readonly shouldShowProgressBar?: boolean;\n\n  /**\n   * A threshold for the UI update. Default is `100`.\n   */\n  readonly uiUpdateThresholdInMilliseconds?: number;\n}\n\n/**\n * Loops over a list of items and processes each item.\n *\n * @param params - The parameters for the loop.\n */\nexport async function loop<T>(params: LoopParams<T>): Promise<void> {\n  const DEFAULT_OPTIONS: Required<LoopParams<T>> = {\n    abortSignal: abortSignalNever(),\n    /* v8 ignore start -- buildNoticeMessage is required in LoopParams and always overridden by the spread. */\n    buildNoticeMessage() {\n      throw new Error('buildNoticeMessage is required');\n    },\n    /* v8 ignore stop */\n    items: [],\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeBeforeShownTimeoutInMilliseconds: 500,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeMinTimeoutInMilliseconds: 2000,\n    processItem: noop,\n    progressBarTitle: '',\n    shouldContinueOnError: true,\n    shouldShowNotice: true,\n    shouldShowProgressBar: true,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    uiUpdateThresholdInMilliseconds: 100\n  };\n\n  const fullOptions: Required<LoopParams<T>> = {\n    ...DEFAULT_OPTIONS,\n    ...params\n  };\n\n  const stackTrace = getStackTrace(1);\n\n  const items = fullOptions.items;\n  let iterationCount = 0;\n  let notice = null as Notice | null;\n  let isDone = false;\n  invokeAsyncSafely(() => showNotice());\n\n  const noticeMinTimeoutPromise = sleep(fullOptions.noticeMinTimeoutInMilliseconds);\n  const progressBarEl = createEl('progress');\n  addPluginCssClasses(progressBarEl, 'loop');\n  progressBarEl.max = items.length;\n\n  let lastUIUpdateTimestamp = performance.now();\n\n  for (const item of items) {\n    if (fullOptions.abortSignal.aborted) {\n      notice?.hide();\n      return;\n    }\n    iterationCount++;\n    const iterationStr = `# ${String(iterationCount)} / ${String(items.length)}`;\n    const message = fullOptions.buildNoticeMessage(item, iterationStr);\n    if (!fullOptions.shouldShowProgressBar) {\n      notice?.setMessage(message);\n    }\n    getLibDebugger('Loop')(message);\n\n    try {\n      if (performance.now() - lastUIUpdateTimestamp > fullOptions.uiUpdateThresholdInMilliseconds) {\n        await requestAnimationFrameAsync();\n        lastUIUpdateTimestamp = performance.now();\n      }\n      await fullOptions.processItem(item);\n    } catch (error) {\n      console.error('Error processing item', item);\n      if (!fullOptions.shouldContinueOnError) {\n        notice?.hide();\n        throw new CustomStackTraceError('loop failed', stackTrace, error);\n      }\n\n      emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, error));\n    }\n    progressBarEl.value++;\n  }\n  if (notice) {\n    await noticeMinTimeoutPromise;\n  }\n  notice?.hide();\n  isDone = true;\n\n  async function showNotice(): Promise<void> {\n    if (!fullOptions.shouldShowNotice) {\n      return;\n    }\n    await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);\n    if (isDone) {\n      return;\n    }\n    notice = new Notice('', 0);\n    if (!fullOptions.shouldShowProgressBar) {\n      return;\n    }\n    const fragment = createFragment();\n    fragment.createDiv({ text: fullOptions.progressBarTitle });\n    fragment.appendChild(progressBarEl);\n    notice.setMessage(fragment);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,sBAAuB;AAEvB,6BAAiC;AACjC,mBAGO;AACP,mBAA+B;AAC/B,mBAKO;AACP,sBAAqB;AACrB,2BAAoC;AAyEpC,eAAsB,KAAQ,QAAsC;AAClE,QAAM,kBAA2C;AAAA,IAC/C,iBAAa,yCAAiB;AAAA;AAAA,IAE9B,qBAAqB;AACnB,YAAM,IAAI,MAAM,gCAAgC;AAAA,IAClD;AAAA;AAAA,IAEA,OAAO,CAAC;AAAA;AAAA,IAER,wCAAwC;AAAA;AAAA,IAExC,gCAAgC;AAAA,IAChC,aAAa;AAAA,IACb,kBAAkB;AAAA,IAClB,uBAAuB;AAAA,IACvB,kBAAkB;AAAA,IAClB,uBAAuB;AAAA;AAAA,IAEvB,iCAAiC;AAAA,EACnC;AAEA,QAAM,cAAuC;AAAA,IAC3C,GAAG;AAAA,IACH,GAAG;AAAA,EACL;AAEA,QAAM,iBAAa,4BAAc,CAAC;AAElC,QAAM,QAAQ,YAAY;AAC1B,MAAI,iBAAiB;AACrB,MAAI,SAAS;AACb,MAAI,SAAS;AACb,sCAAkB,MAAM,WAAW,CAAC;AAEpC,QAAM,0BAA0B,MAAM,YAAY,8BAA8B;AAChF,QAAM,gBAAgB,SAAS,UAAU;AACzC,gDAAoB,eAAe,MAAM;AACzC,gBAAc,MAAM,MAAM;AAE1B,MAAI,wBAAwB,YAAY,IAAI;AAE5C,aAAW,QAAQ,OAAO;AACxB,QAAI,YAAY,YAAY,SAAS;AACnC,cAAQ,KAAK;AACb;AAAA,IACF;AACA;AACA,UAAM,eAAe,KAAK,OAAO,cAAc,CAAC,MAAM,OAAO,MAAM,MAAM,CAAC;AAC1E,UAAM,UAAU,YAAY,mBAAmB,MAAM,YAAY;AACjE,QAAI,CAAC,YAAY,uBAAuB;AACtC,cAAQ,WAAW,OAAO;AAAA,IAC5B;AACA,qCAAe,MAAM,EAAE,OAAO;AAE9B,QAAI;AACF,UAAI,YAAY,IAAI,IAAI,wBAAwB,YAAY,iCAAiC;AAC3F,kBAAM,yCAA2B;AACjC,gCAAwB,YAAY,IAAI;AAAA,MAC1C;AACA,YAAM,YAAY,YAAY,IAAI;AAAA,IACpC,SAAS,OAAO;AACd,cAAQ,MAAM,yBAAyB,IAAI;AAC3C,UAAI,CAAC,YAAY,uBAAuB;AACtC,gBAAQ,KAAK;AACb,cAAM,IAAI,mCAAsB,eAAe,YAAY,KAAK;AAAA,MAClE;AAEA,4CAAoB,IAAI,mCAAsB,0CAA6B,YAAY,KAAK,CAAC;AAAA,IAC/F;AACA,kBAAc;AAAA,EAChB;AACA,MAAI,QAAQ;AACV,UAAM;AAAA,EACR;AACA,UAAQ,KAAK;AACb,WAAS;AAET,iBAAe,aAA4B;AACzC,QAAI,CAAC,YAAY,kBAAkB;AACjC;AAAA,IACF;AACA,UAAM,MAAM,YAAY,sCAAsC;AAC9D,QAAI,QAAQ;AACV;AAAA,IACF;AACA,aAAS,IAAI,uBAAO,IAAI,CAAC;AACzB,QAAI,CAAC,YAAY,uBAAuB;AACtC;AAAA,IACF;AACA,UAAM,WAAW,eAAe;AAChC,aAAS,UAAU,EAAE,MAAM,YAAY,iBAAiB,CAAC;AACzD,aAAS,YAAY,aAAa;AAClC,WAAO,WAAW,QAAQ;AAAA,EAC5B;AACF;",
  "names": []
}

@@ -7,7 +7,7 @@ import type { Promisable } from 'type-fest';
7
7
  /**
8
8
  * Options for {@link loop}.
9
9
  */
10
- export interface LoopOptions<T> {
10
+ export interface LoopParams<T> {
11
11
  /**
12
12
  * An optional abort signal to cancel the loop.
13
13
  */
@@ -62,6 +62,6 @@ export interface LoopOptions<T> {
62
62
  /**
63
63
  * Loops over a list of items and processes each item.
64
64
  *
65
- * @param options - The options for the loop.
65
+ * @param params - The parameters for the loop.
66
66
  */
67
- export declare function loop<T>(options: LoopOptions<T>): Promise<void>;
67
+ export declare function loop<T>(params: LoopParams<T>): Promise<void>;
@@ -165,30 +165,30 @@ class FixedZIndexDomEventsHandlersInfo {
165
165
  this.zIndex = (0, import_HTMLElement.getZIndex)(el) + 1;
166
166
  }
167
167
  }
168
- async function fullRender(options) {
169
- const sourcePath = options.sourcePath ?? "/";
168
+ async function fullRender(params) {
169
+ const sourcePath = params.sourcePath ?? "/";
170
170
  let shouldUnloadComponent = false;
171
171
  let component;
172
- if (options.component) {
173
- component = options.component;
172
+ if (params.component) {
173
+ component = params.component;
174
174
  } else {
175
175
  component = new import_obsidian.Component();
176
176
  component.load();
177
177
  shouldUnloadComponent = true;
178
178
  }
179
- await (0, import_MonkeyAround.invokeWithPatchAsync)(options.app.embedRegistry.embedByExtension, {
179
+ await (0, import_MonkeyAround.invokeWithPatchAsync)(params.app.embedRegistry.embedByExtension, {
180
180
  md: (next) => (context, file, subpath) => {
181
181
  context.displayMode = false;
182
182
  return next(context, file, subpath);
183
183
  }
184
184
  }, async () => {
185
- await import_obsidian.MarkdownRenderer.render(options.app, options.markdown, options.el, sourcePath, component);
185
+ await import_obsidian.MarkdownRenderer.render(params.app, params.markdown, params.el, sourcePath, component);
186
186
  });
187
187
  if (shouldUnloadComponent) {
188
188
  component.unload();
189
189
  }
190
- if (options.shouldRegisterLinkHandlers) {
191
- await registerLinkHandlers(options.app, options.el, options.sourcePath);
190
+ if (params.shouldRegisterLinkHandlers) {
191
+ await registerLinkHandlers(params.app, params.el, params.sourcePath);
192
192
  }
193
193
  }
194
194
  async function markdownToHtml(app, markdown, sourcePath) {
@@ -282,4 +282,4 @@ async function getDomEventsHandlersConstructor(app) {
282
282
  renderExternalLink,
283
283
  renderInternalLink
284
284
  });
285
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Markdown.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing Markdown content in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { App } from 'obsidian';\nimport type {\n  DomEventsHandlersConstructor,\n  DomEventsHandlersInfo,\n  EmbedCreator\n} from 'obsidian-typings';\n\nimport {\n  Component,\n  HoverPopover,\n  MarkdownPreviewRenderer,\n  MarkdownRenderer\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type { PathOrAbstractFile } from './FileSystem.ts';\n\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport { getZIndex } from '../HTMLElement.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport {\n  getAbstractFileOrNull,\n  getPath,\n  isFolder\n} from './FileSystem.ts';\nimport { invokeWithPatchAsync } from './MonkeyAround.ts';\n\nlet domEventsHandlersConstructor: DomEventsHandlersConstructor | null = null;\n\n/**\n * The options for the full render.\n */\nexport interface FullRenderOptions {\n  /**\n   * The Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * The Component instance to use for the render.\n   */\n  readonly component?: Component;\n\n  /**\n   * The HTMLElement to render to.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * The Markdown string to render.\n   */\n  readonly markdown: string;\n\n  /**\n   * Whether to register link handlers for the rendered element.\n   */\n  readonly shouldRegisterLinkHandlers?: boolean;\n\n  /**\n   * The source path to resolve relative links.\n   */\n  readonly sourcePath?: string;\n}\n\ntype RegisterDomEventsFn = typeof MarkdownPreviewRenderer.registerDomEvents;\n\nclass FixedZIndexDomEventsHandlersInfo implements DomEventsHandlersInfo {\n  public get hoverPopover(): HoverPopover | null {\n    return this._hoverPopover;\n  }\n\n  public set hoverPopover(hoverPopover: HoverPopover | null) {\n    this._hoverPopover = hoverPopover;\n    if (hoverPopover && this.zIndex !== undefined) {\n      hoverPopover.hoverEl.setCssStyles({\n        zIndex: String(this.zIndex)\n      });\n    }\n  }\n\n  private _hoverPopover: HoverPopover | null = null;\n\n  private zIndex?: number;\n\n  public constructor(public readonly app: App, public readonly path: string, el: HTMLElement) {\n    if (el.isConnected) {\n      this.updateZIndex(el);\n    } else {\n      el.onNodeInserted(() => {\n        this.updateZIndex(el);\n      });\n    }\n  }\n\n  private updateZIndex(el: HTMLElement): void {\n    this.zIndex = getZIndex(el) + 1;\n  }\n}\n\n/**\n * Render the markdown and embeds.\n *\n * @param options - The options for the full render.\n * @returns The {@link Promise} that resolves when the full render is complete.\n */\nexport async function fullRender(options: FullRenderOptions): Promise<void> {\n  const sourcePath = options.sourcePath ?? '/';\n  let shouldUnloadComponent = false;\n  let component: Component;\n  if (options.component) {\n    component = options.component;\n  } else {\n    component = new Component();\n    component.load();\n    shouldUnloadComponent = true;\n  }\n  await invokeWithPatchAsync(options.app.embedRegistry.embedByExtension, {\n    md: (next: EmbedCreator): EmbedCreator => (context, file, subpath) => {\n      context.displayMode = false;\n      return next(context, file, subpath);\n    }\n  }, async () => {\n    await MarkdownRenderer.render(options.app, options.markdown, options.el, sourcePath, component);\n  });\n\n  if (shouldUnloadComponent) {\n    component.unload();\n  }\n\n  if (options.shouldRegisterLinkHandlers) {\n    await registerLinkHandlers(options.app, options.el, options.sourcePath);\n  }\n}\n\n/**\n * Converts Markdown to HTML.\n *\n * @param app - The Obsidian app instance.\n * @param markdown - The Markdown string to convert.\n * @param sourcePath - (optional) The source path to resolve relative links.\n * @returns The HTML string.\n */\nexport async function markdownToHtml(app: App, markdown: string, sourcePath?: string): Promise<string> {\n  const component = new Component();\n  component.load();\n  const renderDiv = createDiv();\n  await MarkdownRenderer.render(app, markdown, renderDiv, sourcePath ?? '', component);\n  const html = renderDiv.innerHTML;\n  component.unload();\n  return html;\n}\n\n/**\n * Registers link handlers for the given element.\n *\n * @param app - The Obsidian app instance.\n * @param el - The HTMLElement to register link handlers for.\n * @param sourcePath - The source path to resolve relative links from.\n */\nexport async function registerLinkHandlers(app: App, el: HTMLElement, sourcePath?: string): Promise<void> {\n  // eslint-disable-next-line require-atomic-updates -- No race condition.\n  domEventsHandlersConstructor ??= await getDomEventsHandlersConstructor(app);\n  MarkdownPreviewRenderer.registerDomEvents(\n    el,\n    new domEventsHandlersConstructor(new FixedZIndexDomEventsHandlersInfo(app, sourcePath ?? '', el))\n  );\n}\n\n/**\n * Renders an external link.\n *\n * @param app - The Obsidian app instance.\n * @param url - The URL to render the external link for.\n * @param displayText - The text to display for the external link.\n * @returns The HTMLAnchorElement containing the rendered external link.\n */\nexport async function renderExternalLink(app: App, url: string, displayText?: string): Promise<HTMLAnchorElement> {\n  displayText ??= url;\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[${displayText}](${url})`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\n/**\n * Renders an internal link.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrAbstractFile - The path or abstract file to render the internal link for.\n * @param displayText - The text to display for the internal link.\n * @returns The HTMLAnchorElement containing the rendered internal link.\n */\nexport async function renderInternalLink(app: App, pathOrAbstractFile: PathOrAbstractFile, displayText?: string): Promise<HTMLAnchorElement> {\n  const abstractFile = getAbstractFileOrNull(app, pathOrAbstractFile);\n  const path = getPath(app, pathOrAbstractFile);\n  displayText ??= path;\n  if (isFolder(abstractFile)) {\n    return createEl('a', { text: displayText }, (aEl) => {\n      aEl.addEventListener('click', (evt) => {\n        evt.preventDefault();\n        app.internalPlugins.getEnabledPluginById(InternalPluginName.FileExplorer)?.revealInFolder(abstractFile);\n      });\n    });\n  }\n\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[[${path}|${displayText}]]`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\nasync function getDomEventsHandlersConstructor(app: App): Promise<DomEventsHandlersConstructor> {\n  let mdFile = app.vault.getMarkdownFiles()[0];\n  let shouldDelete = false;\n  if (!mdFile) {\n    // eslint-disable-next-line require-atomic-updates -- No race condition.\n    mdFile = await app.vault.create('__temp.md', '');\n    shouldDelete = true;\n  }\n  let ctor: DomEventsHandlersConstructor | null = null;\n  try {\n    await invokeWithPatchAsync(MarkdownPreviewRenderer, {\n      registerDomEvents: (next: RegisterDomEventsFn): RegisterDomEventsFn => {\n        return (el, handlers, childElFn) => {\n          ctor = handlers.constructor as DomEventsHandlersConstructor;\n          next(el, handlers, childElFn);\n        };\n      }\n    }, async () => {\n      const leaf = app.workspace.getLeaf(true);\n      await leaf.openFile(mdFile, {\n        active: true,\n        state: { mode: 'preview' }\n      });\n      await requestAnimationFrameAsync();\n      leaf.detach();\n    });\n\n    assertNonNullable(ctor, 'Failed to get register dom events handlers constructor');\n    return ctor;\n  } finally {\n    if (shouldDelete) {\n      await app.fileManager.trashFile(mdFile);\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["aEl"]
}

285
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Markdown.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing Markdown content in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type { App } from 'obsidian';\nimport type {\n  DomEventsHandlersConstructor,\n  DomEventsHandlersInfo,\n  EmbedCreator\n} from 'obsidian-typings';\n\nimport {\n  Component,\n  HoverPopover,\n  MarkdownPreviewRenderer,\n  MarkdownRenderer\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\n\nimport type { PathOrAbstractFile } from './FileSystem.ts';\n\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport { getZIndex } from '../HTMLElement.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport {\n  getAbstractFileOrNull,\n  getPath,\n  isFolder\n} from './FileSystem.ts';\nimport { invokeWithPatchAsync } from './MonkeyAround.ts';\n\nlet domEventsHandlersConstructor: DomEventsHandlersConstructor | null = null;\n\n/**\n * The params for the full render.\n */\nexport interface FullRenderParams {\n  /**\n   * The Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * The Component instance to use for the render.\n   */\n  readonly component?: Component;\n\n  /**\n   * The HTMLElement to render to.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * The Markdown string to render.\n   */\n  readonly markdown: string;\n\n  /**\n   * Whether to register link handlers for the rendered element.\n   */\n  readonly shouldRegisterLinkHandlers?: boolean;\n\n  /**\n   * The source path to resolve relative links.\n   */\n  readonly sourcePath?: string;\n}\n\ntype RegisterDomEventsFn = typeof MarkdownPreviewRenderer.registerDomEvents;\n\nclass FixedZIndexDomEventsHandlersInfo implements DomEventsHandlersInfo {\n  public get hoverPopover(): HoverPopover | null {\n    return this._hoverPopover;\n  }\n\n  public set hoverPopover(hoverPopover: HoverPopover | null) {\n    this._hoverPopover = hoverPopover;\n    if (hoverPopover && this.zIndex !== undefined) {\n      hoverPopover.hoverEl.setCssStyles({\n        zIndex: String(this.zIndex)\n      });\n    }\n  }\n\n  private _hoverPopover: HoverPopover | null = null;\n\n  private zIndex?: number;\n\n  public constructor(public readonly app: App, public readonly path: string, el: HTMLElement) {\n    if (el.isConnected) {\n      this.updateZIndex(el);\n    } else {\n      el.onNodeInserted(() => {\n        this.updateZIndex(el);\n      });\n    }\n  }\n\n  private updateZIndex(el: HTMLElement): void {\n    this.zIndex = getZIndex(el) + 1;\n  }\n}\n\n/**\n * Render the markdown and embeds.\n *\n * @param params - The parameters for the full render.\n * @returns The {@link Promise} that resolves when the full render is complete.\n */\nexport async function fullRender(params: FullRenderParams): Promise<void> {\n  const sourcePath = params.sourcePath ?? '/';\n  let shouldUnloadComponent = false;\n  let component: Component;\n  if (params.component) {\n    component = params.component;\n  } else {\n    component = new Component();\n    component.load();\n    shouldUnloadComponent = true;\n  }\n  await invokeWithPatchAsync(params.app.embedRegistry.embedByExtension, {\n    md: (next: EmbedCreator): EmbedCreator => (context, file, subpath) => {\n      context.displayMode = false;\n      return next(context, file, subpath);\n    }\n  }, async () => {\n    await MarkdownRenderer.render(params.app, params.markdown, params.el, sourcePath, component);\n  });\n\n  if (shouldUnloadComponent) {\n    component.unload();\n  }\n\n  if (params.shouldRegisterLinkHandlers) {\n    await registerLinkHandlers(params.app, params.el, params.sourcePath);\n  }\n}\n\n/**\n * Converts Markdown to HTML.\n *\n * @param app - The Obsidian app instance.\n * @param markdown - The Markdown string to convert.\n * @param sourcePath - (optional) The source path to resolve relative links.\n * @returns The HTML string.\n */\nexport async function markdownToHtml(app: App, markdown: string, sourcePath?: string): Promise<string> {\n  const component = new Component();\n  component.load();\n  const renderDiv = createDiv();\n  await MarkdownRenderer.render(app, markdown, renderDiv, sourcePath ?? '', component);\n  const html = renderDiv.innerHTML;\n  component.unload();\n  return html;\n}\n\n/**\n * Registers link handlers for the given element.\n *\n * @param app - The Obsidian app instance.\n * @param el - The HTMLElement to register link handlers for.\n * @param sourcePath - The source path to resolve relative links from.\n */\nexport async function registerLinkHandlers(app: App, el: HTMLElement, sourcePath?: string): Promise<void> {\n  // eslint-disable-next-line require-atomic-updates -- No race condition.\n  domEventsHandlersConstructor ??= await getDomEventsHandlersConstructor(app);\n  MarkdownPreviewRenderer.registerDomEvents(\n    el,\n    new domEventsHandlersConstructor(new FixedZIndexDomEventsHandlersInfo(app, sourcePath ?? '', el))\n  );\n}\n\n/**\n * Renders an external link.\n *\n * @param app - The Obsidian app instance.\n * @param url - The URL to render the external link for.\n * @param displayText - The text to display for the external link.\n * @returns The HTMLAnchorElement containing the rendered external link.\n */\nexport async function renderExternalLink(app: App, url: string, displayText?: string): Promise<HTMLAnchorElement> {\n  displayText ??= url;\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[${displayText}](${url})`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\n/**\n * Renders an internal link.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrAbstractFile - The path or abstract file to render the internal link for.\n * @param displayText - The text to display for the internal link.\n * @returns The HTMLAnchorElement containing the rendered internal link.\n */\nexport async function renderInternalLink(app: App, pathOrAbstractFile: PathOrAbstractFile, displayText?: string): Promise<HTMLAnchorElement> {\n  const abstractFile = getAbstractFileOrNull(app, pathOrAbstractFile);\n  const path = getPath(app, pathOrAbstractFile);\n  displayText ??= path;\n  if (isFolder(abstractFile)) {\n    return createEl('a', { text: displayText }, (aEl) => {\n      aEl.addEventListener('click', (evt) => {\n        evt.preventDefault();\n        app.internalPlugins.getEnabledPluginById(InternalPluginName.FileExplorer)?.revealInFolder(abstractFile);\n      });\n    });\n  }\n\n  const wrapperEl = createSpan();\n  await fullRender({\n    app,\n    el: wrapperEl,\n    markdown: `[[${path}|${displayText}]]`\n  });\n  const aEl = wrapperEl.find('a') as HTMLAnchorElement;\n  await registerLinkHandlers(app, aEl);\n  return aEl;\n}\n\nasync function getDomEventsHandlersConstructor(app: App): Promise<DomEventsHandlersConstructor> {\n  let mdFile = app.vault.getMarkdownFiles()[0];\n  let shouldDelete = false;\n  if (!mdFile) {\n    // eslint-disable-next-line require-atomic-updates -- No race condition.\n    mdFile = await app.vault.create('__temp.md', '');\n    shouldDelete = true;\n  }\n  let ctor: DomEventsHandlersConstructor | null = null;\n  try {\n    await invokeWithPatchAsync(MarkdownPreviewRenderer, {\n      registerDomEvents: (next: RegisterDomEventsFn): RegisterDomEventsFn => {\n        return (el, handlers, childElFn) => {\n          ctor = handlers.constructor as DomEventsHandlersConstructor;\n          next(el, handlers, childElFn);\n        };\n      }\n    }, async () => {\n      const leaf = app.workspace.getLeaf(true);\n      await leaf.openFile(mdFile, {\n        active: true,\n        state: { mode: 'preview' }\n      });\n      await requestAnimationFrameAsync();\n      leaf.detach();\n    });\n\n    assertNonNullable(ctor, 'Failed to get register dom events handlers constructor');\n    return ctor;\n  } finally {\n    if (shouldDelete) {\n      await app.fileManager.trashFile(mdFile);\n    }\n  }\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": ["aEl"]
}

@@ -7,9 +7,9 @@ import type { App } from 'obsidian';
7
7
  import { Component } from 'obsidian';
8
8
  import type { PathOrAbstractFile } from './FileSystem.cjs';
9
9
  /**
10
- * The options for the full render.
10
+ * The params for the full render.
11
11
  */
12
- export interface FullRenderOptions {
12
+ export interface FullRenderParams {
13
13
  /**
14
14
  * The Obsidian app instance.
15
15
  */
@@ -38,10 +38,10 @@ export interface FullRenderOptions {
38
38
  /**
39
39
  * Render the markdown and embeds.
40
40
  *
41
- * @param options - The options for the full render.
41
+ * @param params - The parameters for the full render.
42
42
  * @returns The {@link Promise} that resolves when the full render is complete.
43
43
  */
44
- export declare function fullRender(options: FullRenderOptions): Promise<void>;
44
+ export declare function fullRender(params: FullRenderParams): Promise<void>;
45
45
  /**
46
46
  * Converts Markdown to HTML.
47
47
  *
@@ -135,8 +135,8 @@ var import_TypeGuards = require('../TypeGuards.cjs');
135
135
  var import_ValueProvider = require('../ValueProvider.cjs');
136
136
  var import_FileSystem = require('./FileSystem.cjs');
137
137
  var import_Vault = require('./Vault.cjs');
138
- async function getCodeBlockMarkdownInfo(options) {
139
- const { app, ctx, el, source } = options;
138
+ async function getCodeBlockMarkdownInfo(params) {
139
+ const { app, ctx, el, source } = params;
140
140
  const sourceFile = (0, import_FileSystem.getFileOrNull)(app, ctx.sourcePath);
141
141
  (0, import_TypeGuards.assertNonNullable)(sourceFile, `Source file ${ctx.sourcePath} not found.`);
142
142
  await (0, import_Async.requestAnimationFrameAsync)();
@@ -203,22 +203,22 @@ async function getCodeBlockMarkdownInfo(options) {
203
203
  });
204
204
  return markdownInfo;
205
205
  }
206
- async function insertAfterCodeBlock(options) {
207
- const { app, ctx, lineOffset = 0, text } = options;
206
+ async function insertAfterCodeBlock(params) {
207
+ const { app, ctx, lineOffset = 0, text } = params;
208
208
  await (0, import_Vault.process)(app, ctx.sourcePath, async (_abortSignal, content) => {
209
- const markdownInfo = await getCodeBlockMarkdownInfo(options);
209
+ const markdownInfo = await getCodeBlockMarkdownInfo(params);
210
210
  (0, import_TypeGuards.assertNonNullable)(markdownInfo, "Could not uniquely identify the code block.");
211
211
  if (content !== markdownInfo.noteContent) {
212
212
  return null;
213
213
  }
214
214
  const insertLineIndex = markdownInfo.positionInNote.end.line + lineOffset + 1;
215
- return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
215
+ return insertText(content, insertLineIndex, text, params.shouldPreserveLinePrefix);
216
216
  });
217
217
  }
218
- async function insertBeforeCodeBlock(options) {
219
- const { app, ctx, lineOffset = 0, text } = options;
218
+ async function insertBeforeCodeBlock(params) {
219
+ const { app, ctx, lineOffset = 0, text } = params;
220
220
  await (0, import_Vault.process)(app, ctx.sourcePath, async (_abortSignal, content) => {
221
- const markdownInfo = await getCodeBlockMarkdownInfo(options);
221
+ const markdownInfo = await getCodeBlockMarkdownInfo(params);
222
222
  if (!markdownInfo) {
223
223
  throw new Error("Could not uniquely identify the code block.");
224
224
  }
@@ -226,23 +226,23 @@ async function insertBeforeCodeBlock(options) {
226
226
  return null;
227
227
  }
228
228
  const insertLineIndex = markdownInfo.positionInNote.start.line - lineOffset;
229
- return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
229
+ return insertText(content, insertLineIndex, text, params.shouldPreserveLinePrefix);
230
230
  });
231
231
  }
232
- async function removeCodeBlock(options) {
232
+ async function removeCodeBlock(params) {
233
233
  await replaceCodeBlock({
234
- ...options,
234
+ ...params,
235
235
  codeBlockProvider: "",
236
- shouldKeepGapWhenEmpty: options.shouldKeepGap ?? false
236
+ shouldKeepGapWhenEmpty: params.shouldKeepGap ?? false
237
237
  });
238
238
  }
239
- async function replaceCodeBlock(options) {
240
- const { app, codeBlockProvider, ctx } = options;
241
- options.abortSignal?.throwIfAborted();
239
+ async function replaceCodeBlock(params) {
240
+ const { app, codeBlockProvider, ctx } = params;
241
+ params.abortSignal?.throwIfAborted();
242
242
  await (0, import_Vault.process)(app, ctx.sourcePath, async (abortSignal, content) => {
243
- abortSignal = (0, import_AbortController.abortSignalAny)(abortSignal, options.abortSignal);
243
+ abortSignal = (0, import_AbortController.abortSignalAny)(abortSignal, params.abortSignal);
244
244
  abortSignal.throwIfAborted();
245
- const markdownInfo = await getCodeBlockMarkdownInfo(options);
245
+ const markdownInfo = await getCodeBlockMarkdownInfo(params);
246
246
  if (!markdownInfo) {
247
247
  throw new Error("Could not uniquely identify the code block.");
248
248
  }
@@ -250,17 +250,17 @@ async function replaceCodeBlock(options) {
250
250
  return null;
251
251
  }
252
252
  let oldCodeBlock = content.slice(markdownInfo.positionInNote.start.offset, markdownInfo.positionInNote.end.offset);
253
- if (options.shouldPreserveLinePrefix) {
253
+ if (params.shouldPreserveLinePrefix) {
254
254
  oldCodeBlock = (0, import_String.unindent)(oldCodeBlock, markdownInfo.linePrefix);
255
255
  }
256
256
  let newCodeBlock = await (0, import_ValueProvider.resolveValue)(codeBlockProvider, abortSignal, oldCodeBlock);
257
257
  abortSignal.throwIfAborted();
258
- if ((newCodeBlock || options.shouldKeepGapWhenEmpty) && options.shouldPreserveLinePrefix) {
258
+ if ((newCodeBlock || params.shouldKeepGapWhenEmpty) && params.shouldPreserveLinePrefix) {
259
259
  newCodeBlock = (0, import_String.indent)(newCodeBlock, markdownInfo.linePrefix);
260
260
  }
261
261
  const textBeforeCodeBlock = content.slice(0, markdownInfo.positionInNote.start.offset);
262
262
  const textAfterCodeBlock = content.slice(markdownInfo.positionInNote.end.offset);
263
- if (newCodeBlock || options.shouldKeepGapWhenEmpty) {
263
+ if (newCodeBlock || params.shouldKeepGapWhenEmpty) {
264
264
  return `${textBeforeCodeBlock}${newCodeBlock}${textAfterCodeBlock}`;
265
265
  }
266
266
  if (!textBeforeCodeBlock && !textAfterCodeBlock) {
@@ -272,7 +272,7 @@ async function replaceCodeBlock(options) {
272
272
  return `${textBeforeCodeBlock}${textAfterCodeBlock.slice(1)}`;
273
273
  });
274
274
  }
275
- function createMarkdownInfoFromMatch(options) {
275
+ function createMarkdownInfoFromMatch(params) {
276
276
  const {
277
277
  approximateSectionInfo,
278
278
  linesBeforeSectionCount,
@@ -281,7 +281,7 @@ function createMarkdownInfoFromMatch(options) {
281
281
  potentialCodeBlockText,
282
282
  sourceLinesCount,
283
283
  textLineOffsets
284
- } = options;
284
+ } = params;
285
285
  const linePrefix = match.groups?.["LinePrefix"] ?? "";
286
286
  const codeBlockStartDelimiter = match.groups?.["CodeBlockStartDelimiter"] ?? "";
287
287
  const codeBlockEndDelimiter = match.groups?.["CodeBlockEndDelimiter"] ?? "";
@@ -359,4 +359,4 @@ function isSuitableCodeBlock(match, language, sourceLf, isInCallout) {
359
359
  removeCodeBlock,
360
360
  replaceCodeBlock
361
361
  });
362
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MarkdownCodeBlockProcessor.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing code blocks in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type {\n  App,\n  MarkdownPostProcessorContext,\n  MarkdownSectionInformation\n} from 'obsidian';\n\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { CodeBlockMarkdownInformation } from './CodeBlockMarkdownInformation.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport {\n  ensureLfEndings,\n  getLfNormalizedOffsetToOriginalOffsetMapper,\n  hasSingleOccurrence,\n  indent,\n  unindent\n} from '../String.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport { getFileOrNull } from './FileSystem.ts';\nimport {\n  invokeWithFileSystemLock,\n  process,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Options for {@link getCodeBlockMarkdownInfo}.\n */\nexport interface GetCodeBlockMarkdownInfoOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * A {@link MarkdownPostProcessorContext} object.\n   */\n  readonly ctx: MarkdownPostProcessorContext;\n\n  /**\n   * A {@link HTMLElement} representing the code block.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * A source of the code block.\n   */\n  readonly source: string;\n}\n\n/**\n * Options for {@link insertAfterCodeBlock} / {@link insertBeforeCodeBlock}.\n */\nexport interface InsertCodeBlockOptions extends GetCodeBlockMarkdownInfoOptions {\n  /**\n   * A number of lines to offset the insertion by. Default is `0`.\n   */\n  readonly lineOffset?: number;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  readonly shouldPreserveLinePrefix?: boolean;\n\n  /**\n   * A text to insert after the code block.\n   */\n  readonly text: string;\n}\n\n/**\n * Options for {@link removeCodeBlock}.\n */\nexport interface RemoveCodeBlockOptions extends GetCodeBlockMarkdownInfoOptions {\n  /**\n   * Whether to keep the gap after removing the code block. Default is `false`.\n   */\n  readonly shouldKeepGap?: boolean;\n}\n\n/**\n * Options for {@link replaceCodeBlock}.\n */\nexport interface ReplaceCodeBlockOptions extends GetCodeBlockMarkdownInfoOptions {\n  /**\n   * An abort signal to control the execution of the function.\n   */\n  readonly abortSignal?: AbortSignal;\n\n  /**\n   * Provides a new code block.\n   */\n  readonly codeBlockProvider: ValueProvider<string, [string]>;\n\n  /**\n   * Whether to keep the gap when the new code block is empty. Default is `false`.\n   */\n  readonly shouldKeepGapWhenEmpty?: boolean;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  readonly shouldPreserveLinePrefix?: boolean;\n}\n\ninterface CreateMarkdownInfoFromMatchOptions {\n  readonly approximateSectionInfo: MarkdownSectionInformation;\n  readonly linesBeforeSectionCount: number;\n  readonly match: RegExpMatchArray;\n  readonly noteContent: string;\n  readonly potentialCodeBlockText: string;\n  readonly sourceLinesCount: number;\n  readonly textLineOffsets: ReadonlyMap<number, number>;\n}\n\n/**\n * Gets the information about a code block in a Markdown section.\n *\n * @param options - The options for the function.\n * @returns The information about the code block in the Markdown section.\n */\nexport async function getCodeBlockMarkdownInfo(options: GetCodeBlockMarkdownInfoOptions): Promise<CodeBlockMarkdownInformation | null> {\n  const { app, ctx, el, source } = options;\n\n  const sourceFile = getFileOrNull(app, ctx.sourcePath);\n  assertNonNullable(sourceFile, `Source file ${ctx.sourcePath} not found.`);\n\n  await requestAnimationFrameAsync();\n  await saveNote(app, sourceFile);\n\n  let markdownInfo: CodeBlockMarkdownInformation | null = null;\n\n  await invokeWithFileSystemLock(app, sourceFile, (noteContent) => {\n    const noteContentLf = ensureLfEndings(noteContent);\n\n    const approximateSectionInfo: MarkdownSectionInformation = {\n      lineEnd: noteContentLf.split('\\n').length - 1,\n      lineStart: 0,\n      text: noteContentLf\n    };\n\n    approximateSectionInfo.text = ensureLfEndings(approximateSectionInfo.text);\n    const sourceLf = ensureLfEndings(source);\n\n    if (!hasSingleOccurrence(noteContentLf, approximateSectionInfo.text)) {\n      return;\n    }\n\n    const sectionOffset = noteContentLf.indexOf(approximateSectionInfo.text);\n    const linesBeforeSectionCount = noteContentLf.slice(0, sectionOffset).split('\\n').length - 1;\n\n    const isInCallout = !!el.parentElement?.classList.contains('callout-content');\n\n    const language = getLanguageFromElement(el);\n    const sourceLines = sourceLf.split('\\n');\n\n    const textLines = approximateSectionInfo.text.split('\\n');\n    const textLineOffsets = new Map<number, number>();\n    textLineOffsets.set(linesBeforeSectionCount, sectionOffset);\n\n    let lastTextLineOffset = sectionOffset;\n    for (let i = 0; i < textLines.length; i++) {\n      const textLine = textLines[i] ?? '';\n      const lineOffset = lastTextLineOffset + textLine.length + 1;\n      textLineOffsets.set(linesBeforeSectionCount + i + 1, lineOffset);\n      lastTextLineOffset = lineOffset;\n    }\n\n    const potentialCodeBlockTextLines = textLines.map((line, index) =>\n      approximateSectionInfo.lineStart <= index && index <= approximateSectionInfo.lineEnd ? line : ''\n    );\n    const potentialCodeBlockText = potentialCodeBlockTextLines.join('\\n');\n\n    const REG_EXP =\n      /(?<=^|\\n)(?<LinePrefix> {0,3}(?:> {1,3})*)(?<CodeBlockStartDelimiter>(?<CodeBlockStartDelimiterChar>[`~])(?:\\k<CodeBlockStartDelimiterChar>{2,}))(?<CodeBlockLanguage>\\S*)(?:[ \\t](?<CodeBlockArgs>.*?))?(?:\\n(?<CodeBlockContent>(?:\\n?\\k<LinePrefix>.*)+?))?\\n\\k<LinePrefix>(?<CodeBlockEndDelimiter>\\k<CodeBlockStartDelimiter>\\k<CodeBlockStartDelimiterChar>*)[ \\t]*(?=\\n|$)/g;\n\n    for (const match of potentialCodeBlockText.matchAll(REG_EXP)) {\n      if (!isSuitableCodeBlock(match, language, sourceLf, isInCallout)) {\n        continue;\n      }\n\n      if (markdownInfo) {\n        return;\n      }\n\n      markdownInfo = createMarkdownInfoFromMatch({\n        approximateSectionInfo,\n        linesBeforeSectionCount,\n        match,\n        noteContent,\n        potentialCodeBlockText,\n        sourceLinesCount: sourceLines.length,\n        textLineOffsets\n      });\n    }\n\n    if (!markdownInfo) {\n      return;\n    }\n\n    if (noteContentLf === noteContent) {\n      return;\n    }\n\n    const lfOffsetMapper = getLfNormalizedOffsetToOriginalOffsetMapper(noteContent);\n    markdownInfo.positionInNote.start.offset = lfOffsetMapper(markdownInfo.positionInNote.start.offset);\n    markdownInfo.positionInNote.end.offset = lfOffsetMapper(markdownInfo.positionInNote.end.offset);\n  });\n\n  return markdownInfo;\n}\n\n/**\n * Inserts text after the code block.\n *\n * @param options - The options for the function.\n */\nexport async function insertAfterCodeBlock(options: InsertCodeBlockOptions): Promise<void> {\n  const { app, ctx, lineOffset = 0, text } = options;\n\n  await process(app, ctx.sourcePath, async (_abortSignal, content) => {\n    const markdownInfo = await getCodeBlockMarkdownInfo(options);\n    assertNonNullable(markdownInfo, 'Could not uniquely identify the code block.');\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    const insertLineIndex = markdownInfo.positionInNote.end.line + lineOffset + 1;\n    return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);\n  });\n}\n\n/**\n * Inserts text before the code block.\n *\n * @param options - The options for the function.\n */\nexport async function insertBeforeCodeBlock(options: InsertCodeBlockOptions): Promise<void> {\n  const { app, ctx, lineOffset = 0, text } = options;\n\n  await process(app, ctx.sourcePath, async (_abortSignal, content) => {\n    const markdownInfo = await getCodeBlockMarkdownInfo(options);\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\n    }\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    const insertLineIndex = markdownInfo.positionInNote.start.line - lineOffset;\n    return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);\n  });\n}\n\n/**\n * Removes the code block.\n *\n * @param options - The options for the function.\n */\nexport async function removeCodeBlock(options: RemoveCodeBlockOptions): Promise<void> {\n  await replaceCodeBlock({\n    ...options,\n    codeBlockProvider: '',\n    shouldKeepGapWhenEmpty: options.shouldKeepGap ?? false\n  });\n}\n\n/**\n * Replaces the code block.\n *\n * @param options - The options for the function.\n */\nexport async function replaceCodeBlock(options: ReplaceCodeBlockOptions): Promise<void> {\n  const { app, codeBlockProvider, ctx } = options;\n  options.abortSignal?.throwIfAborted();\n\n  await process(app, ctx.sourcePath, async (abortSignal, content) => {\n    abortSignal = abortSignalAny(abortSignal, options.abortSignal);\n    abortSignal.throwIfAborted();\n    const markdownInfo = await getCodeBlockMarkdownInfo(options);\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\n    }\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    let oldCodeBlock = content.slice(markdownInfo.positionInNote.start.offset, markdownInfo.positionInNote.end.offset);\n    if (options.shouldPreserveLinePrefix) {\n      oldCodeBlock = unindent(oldCodeBlock, markdownInfo.linePrefix);\n    }\n\n    let newCodeBlock = await resolveValue(codeBlockProvider, abortSignal, oldCodeBlock);\n    abortSignal.throwIfAborted();\n    if ((newCodeBlock || options.shouldKeepGapWhenEmpty) && options.shouldPreserveLinePrefix) {\n      newCodeBlock = indent(newCodeBlock, markdownInfo.linePrefix);\n    }\n\n    const textBeforeCodeBlock = content.slice(0, markdownInfo.positionInNote.start.offset);\n    const textAfterCodeBlock = content.slice(markdownInfo.positionInNote.end.offset);\n\n    if (newCodeBlock || options.shouldKeepGapWhenEmpty) {\n      return `${textBeforeCodeBlock}${newCodeBlock}${textAfterCodeBlock}`;\n    }\n\n    if (!textBeforeCodeBlock && !textAfterCodeBlock) {\n      return '';\n    }\n\n    if (textBeforeCodeBlock) {\n      return `${textBeforeCodeBlock.slice(0, -1)}${textAfterCodeBlock}`;\n    }\n\n    return `${textBeforeCodeBlock}${textAfterCodeBlock.slice(1)}`;\n  });\n}\n\nfunction createMarkdownInfoFromMatch(options: CreateMarkdownInfoFromMatchOptions): CodeBlockMarkdownInformation {\n  const {\n    approximateSectionInfo,\n    linesBeforeSectionCount,\n    match,\n    noteContent,\n    potentialCodeBlockText,\n    sourceLinesCount,\n    textLineOffsets\n  } = options;\n\n  const linePrefix = match.groups?.['LinePrefix'] ?? '';\n  const codeBlockStartDelimiter = match.groups?.['CodeBlockStartDelimiter'] ?? '';\n  const codeBlockEndDelimiter = match.groups?.['CodeBlockEndDelimiter'] ?? '';\n  const codeBlockArgsStr = match.groups?.['CodeBlockArgs'] ?? '';\n  const language = match.groups?.['CodeBlockLanguage'] ?? '';\n\n  const previousText = potentialCodeBlockText.slice(0, match.index);\n  const previousTextLinesCount = previousText.split('\\n').length - 1;\n\n  const startLine = linesBeforeSectionCount + previousTextLinesCount;\n  const endLine = startLine + sourceLinesCount + 1;\n\n  return {\n    args: codeBlockArgsStr.split(/\\s+/).filter(Boolean),\n    endDelimiter: codeBlockEndDelimiter,\n    language,\n    linePrefix,\n    noteContent,\n    positionInNote: {\n      end: {\n        col: (textLineOffsets.get(endLine + 1) ?? 0) - (textLineOffsets.get(endLine) ?? 0) - 1,\n        line: endLine,\n        offset: (textLineOffsets.get(endLine + 1) ?? 0) - 1\n      },\n      start: {\n        col: 0,\n        line: startLine,\n        offset: textLineOffsets.get(startLine) ?? 0\n      }\n    },\n    rawArgsStr: codeBlockArgsStr,\n    sectionInfo: {\n      lineEnd: previousTextLinesCount + sourceLinesCount + 1,\n      lineStart: previousTextLinesCount,\n      text: approximateSectionInfo.text\n    },\n    startDelimiter: codeBlockStartDelimiter\n  };\n}\n\nfunction getLanguageFromElement(el: HTMLElement): string {\n  const BLOCK_LANGUAGE_PREFIX = 'block-language-';\n  return Array.from(el.classList).find((cls) => cls.startsWith(BLOCK_LANGUAGE_PREFIX))?.slice(BLOCK_LANGUAGE_PREFIX.length) ?? '';\n}\n\nfunction insertText(content: string, insertLineIndex: number, text: string, shouldPreserveLinePrefix?: boolean): string {\n  const lines = content.split('\\n');\n  const newLines = lines.slice();\n  const textLines = text.split('\\n');\n\n  if (insertLineIndex < 0) {\n    insertLineIndex = 0;\n  }\n  if (insertLineIndex > lines.length) {\n    insertLineIndex = lines.length;\n  }\n\n  const PREFIX_LINE_REG_EXP = /^ {0,3}(?:> {1,3})*/g;\n  const match = (lines[insertLineIndex] ?? '').match(PREFIX_LINE_REG_EXP);\n  const linePrefix = match?.[0] ?? '';\n  newLines.splice(insertLineIndex, 0, ...(shouldPreserveLinePrefix ? textLines.map((line) => indent(line, linePrefix)) : textLines));\n  return newLines.join('\\n');\n}\n\nfunction isSuitableCodeBlock(\n  match: RegExpMatchArray,\n  language: string,\n  sourceLf: string,\n  isInCallout: boolean\n): boolean {\n  const codeBlockLanguage = match.groups?.['CodeBlockLanguage'] ?? '';\n  if (codeBlockLanguage !== language) {\n    return false;\n  }\n\n  const linePrefix = match.groups?.['LinePrefix'] ?? '';\n\n  if (isInCallout && !linePrefix.includes('> ')) {\n    return false;\n  }\n\n  const codeBlockContent = match.groups?.['CodeBlockContent'] ?? '';\n  const cleanCodeBlockContent = codeBlockContent.split('\\n').map((line) => line.slice(linePrefix.length)).join('\\n');\n\n  return cleanCodeBlockContent === sourceLf;\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}

362
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MarkdownCodeBlockProcessor.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for processing code blocks in Obsidian.\n */\n\n/* v8 ignore start -- Deeply coupled to Obsidian runtime; requires running vault for meaningful testing. */\n\nimport type {\n  App,\n  MarkdownPostProcessorContext,\n  MarkdownSectionInformation\n} from 'obsidian';\n\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { CodeBlockMarkdownInformation } from './CodeBlockMarkdownInformation.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { requestAnimationFrameAsync } from '../Async.ts';\nimport {\n  ensureLfEndings,\n  getLfNormalizedOffsetToOriginalOffsetMapper,\n  hasSingleOccurrence,\n  indent,\n  unindent\n} from '../String.ts';\nimport { assertNonNullable } from '../TypeGuards.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport { getFileOrNull } from './FileSystem.ts';\nimport {\n  invokeWithFileSystemLock,\n  process,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Options for {@link getCodeBlockMarkdownInfo}.\n */\nexport interface GetCodeBlockMarkdownInfoParams {\n  /**\n   * An Obsidian app instance.\n   */\n  readonly app: App;\n\n  /**\n   * A {@link MarkdownPostProcessorContext} object.\n   */\n  readonly ctx: MarkdownPostProcessorContext;\n\n  /**\n   * A {@link HTMLElement} representing the code block.\n   */\n  readonly el: HTMLElement;\n\n  /**\n   * A source of the code block.\n   */\n  readonly source: string;\n}\n\n/**\n * Options for {@link insertAfterCodeBlock} / {@link insertBeforeCodeBlock}.\n */\nexport interface InsertCodeBlockParams extends GetCodeBlockMarkdownInfoParams {\n  /**\n   * A number of lines to offset the insertion by. Default is `0`.\n   */\n  readonly lineOffset?: number;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  readonly shouldPreserveLinePrefix?: boolean;\n\n  /**\n   * A text to insert after the code block.\n   */\n  readonly text: string;\n}\n\n/**\n * Options for {@link removeCodeBlock}.\n */\nexport interface RemoveCodeBlockParams extends GetCodeBlockMarkdownInfoParams {\n  /**\n   * Whether to keep the gap after removing the code block. Default is `false`.\n   */\n  readonly shouldKeepGap?: boolean;\n}\n\n/**\n * Options for {@link replaceCodeBlock}.\n */\nexport interface ReplaceCodeBlockParams extends GetCodeBlockMarkdownInfoParams {\n  /**\n   * An abort signal to control the execution of the function.\n   */\n  readonly abortSignal?: AbortSignal;\n\n  /**\n   * Provides a new code block.\n   */\n  readonly codeBlockProvider: ValueProvider<string, [string]>;\n\n  /**\n   * Whether to keep the gap when the new code block is empty. Default is `false`.\n   */\n  readonly shouldKeepGapWhenEmpty?: boolean;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  readonly shouldPreserveLinePrefix?: boolean;\n}\n\ninterface CreateMarkdownInfoFromMatchParams {\n  readonly approximateSectionInfo: MarkdownSectionInformation;\n  readonly linesBeforeSectionCount: number;\n  readonly match: RegExpMatchArray;\n  readonly noteContent: string;\n  readonly potentialCodeBlockText: string;\n  readonly sourceLinesCount: number;\n  readonly textLineOffsets: ReadonlyMap<number, number>;\n}\n\n/**\n * Gets the information about a code block in a Markdown section.\n *\n * @param params - The parameters for the function.\n * @returns The information about the code block in the Markdown section.\n */\nexport async function getCodeBlockMarkdownInfo(params: GetCodeBlockMarkdownInfoParams): Promise<CodeBlockMarkdownInformation | null> {\n  const { app, ctx, el, source } = params;\n\n  const sourceFile = getFileOrNull(app, ctx.sourcePath);\n  assertNonNullable(sourceFile, `Source file ${ctx.sourcePath} not found.`);\n\n  await requestAnimationFrameAsync();\n  await saveNote(app, sourceFile);\n\n  let markdownInfo: CodeBlockMarkdownInformation | null = null;\n\n  await invokeWithFileSystemLock(app, sourceFile, (noteContent) => {\n    const noteContentLf = ensureLfEndings(noteContent);\n\n    const approximateSectionInfo: MarkdownSectionInformation = {\n      lineEnd: noteContentLf.split('\\n').length - 1,\n      lineStart: 0,\n      text: noteContentLf\n    };\n\n    approximateSectionInfo.text = ensureLfEndings(approximateSectionInfo.text);\n    const sourceLf = ensureLfEndings(source);\n\n    if (!hasSingleOccurrence(noteContentLf, approximateSectionInfo.text)) {\n      return;\n    }\n\n    const sectionOffset = noteContentLf.indexOf(approximateSectionInfo.text);\n    const linesBeforeSectionCount = noteContentLf.slice(0, sectionOffset).split('\\n').length - 1;\n\n    const isInCallout = !!el.parentElement?.classList.contains('callout-content');\n\n    const language = getLanguageFromElement(el);\n    const sourceLines = sourceLf.split('\\n');\n\n    const textLines = approximateSectionInfo.text.split('\\n');\n    const textLineOffsets = new Map<number, number>();\n    textLineOffsets.set(linesBeforeSectionCount, sectionOffset);\n\n    let lastTextLineOffset = sectionOffset;\n    for (let i = 0; i < textLines.length; i++) {\n      const textLine = textLines[i] ?? '';\n      const lineOffset = lastTextLineOffset + textLine.length + 1;\n      textLineOffsets.set(linesBeforeSectionCount + i + 1, lineOffset);\n      lastTextLineOffset = lineOffset;\n    }\n\n    const potentialCodeBlockTextLines = textLines.map((line, index) =>\n      approximateSectionInfo.lineStart <= index && index <= approximateSectionInfo.lineEnd ? line : ''\n    );\n    const potentialCodeBlockText = potentialCodeBlockTextLines.join('\\n');\n\n    const REG_EXP =\n      /(?<=^|\\n)(?<LinePrefix> {0,3}(?:> {1,3})*)(?<CodeBlockStartDelimiter>(?<CodeBlockStartDelimiterChar>[`~])(?:\\k<CodeBlockStartDelimiterChar>{2,}))(?<CodeBlockLanguage>\\S*)(?:[ \\t](?<CodeBlockArgs>.*?))?(?:\\n(?<CodeBlockContent>(?:\\n?\\k<LinePrefix>.*)+?))?\\n\\k<LinePrefix>(?<CodeBlockEndDelimiter>\\k<CodeBlockStartDelimiter>\\k<CodeBlockStartDelimiterChar>*)[ \\t]*(?=\\n|$)/g;\n\n    for (const match of potentialCodeBlockText.matchAll(REG_EXP)) {\n      if (!isSuitableCodeBlock(match, language, sourceLf, isInCallout)) {\n        continue;\n      }\n\n      if (markdownInfo) {\n        return;\n      }\n\n      markdownInfo = createMarkdownInfoFromMatch({\n        approximateSectionInfo,\n        linesBeforeSectionCount,\n        match,\n        noteContent,\n        potentialCodeBlockText,\n        sourceLinesCount: sourceLines.length,\n        textLineOffsets\n      });\n    }\n\n    if (!markdownInfo) {\n      return;\n    }\n\n    if (noteContentLf === noteContent) {\n      return;\n    }\n\n    const lfOffsetMapper = getLfNormalizedOffsetToOriginalOffsetMapper(noteContent);\n    markdownInfo.positionInNote.start.offset = lfOffsetMapper(markdownInfo.positionInNote.start.offset);\n    markdownInfo.positionInNote.end.offset = lfOffsetMapper(markdownInfo.positionInNote.end.offset);\n  });\n\n  return markdownInfo;\n}\n\n/**\n * Inserts text after the code block.\n *\n * @param params - The parameters for the function.\n */\nexport async function insertAfterCodeBlock(params: InsertCodeBlockParams): Promise<void> {\n  const { app, ctx, lineOffset = 0, text } = params;\n\n  await process(app, ctx.sourcePath, async (_abortSignal, content) => {\n    const markdownInfo = await getCodeBlockMarkdownInfo(params);\n    assertNonNullable(markdownInfo, 'Could not uniquely identify the code block.');\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    const insertLineIndex = markdownInfo.positionInNote.end.line + lineOffset + 1;\n    return insertText(content, insertLineIndex, text, params.shouldPreserveLinePrefix);\n  });\n}\n\n/**\n * Inserts text before the code block.\n *\n * @param params - The parameters for the function.\n */\nexport async function insertBeforeCodeBlock(params: InsertCodeBlockParams): Promise<void> {\n  const { app, ctx, lineOffset = 0, text } = params;\n\n  await process(app, ctx.sourcePath, async (_abortSignal, content) => {\n    const markdownInfo = await getCodeBlockMarkdownInfo(params);\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\n    }\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    const insertLineIndex = markdownInfo.positionInNote.start.line - lineOffset;\n    return insertText(content, insertLineIndex, text, params.shouldPreserveLinePrefix);\n  });\n}\n\n/**\n * Removes the code block.\n *\n * @param params - The parameters for the function.\n */\nexport async function removeCodeBlock(params: RemoveCodeBlockParams): Promise<void> {\n  await replaceCodeBlock({\n    ...params,\n    codeBlockProvider: '',\n    shouldKeepGapWhenEmpty: params.shouldKeepGap ?? false\n  });\n}\n\n/**\n * Replaces the code block.\n *\n * @param params - The parameters for the function.\n */\nexport async function replaceCodeBlock(params: ReplaceCodeBlockParams): Promise<void> {\n  const { app, codeBlockProvider, ctx } = params;\n  params.abortSignal?.throwIfAborted();\n\n  await process(app, ctx.sourcePath, async (abortSignal, content) => {\n    abortSignal = abortSignalAny(abortSignal, params.abortSignal);\n    abortSignal.throwIfAborted();\n    const markdownInfo = await getCodeBlockMarkdownInfo(params);\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\n    }\n\n    if (content !== markdownInfo.noteContent) {\n      return null;\n    }\n\n    let oldCodeBlock = content.slice(markdownInfo.positionInNote.start.offset, markdownInfo.positionInNote.end.offset);\n    if (params.shouldPreserveLinePrefix) {\n      oldCodeBlock = unindent(oldCodeBlock, markdownInfo.linePrefix);\n    }\n\n    let newCodeBlock = await resolveValue(codeBlockProvider, abortSignal, oldCodeBlock);\n    abortSignal.throwIfAborted();\n    if ((newCodeBlock || params.shouldKeepGapWhenEmpty) && params.shouldPreserveLinePrefix) {\n      newCodeBlock = indent(newCodeBlock, markdownInfo.linePrefix);\n    }\n\n    const textBeforeCodeBlock = content.slice(0, markdownInfo.positionInNote.start.offset);\n    const textAfterCodeBlock = content.slice(markdownInfo.positionInNote.end.offset);\n\n    if (newCodeBlock || params.shouldKeepGapWhenEmpty) {\n      return `${textBeforeCodeBlock}${newCodeBlock}${textAfterCodeBlock}`;\n    }\n\n    if (!textBeforeCodeBlock && !textAfterCodeBlock) {\n      return '';\n    }\n\n    if (textBeforeCodeBlock) {\n      return `${textBeforeCodeBlock.slice(0, -1)}${textAfterCodeBlock}`;\n    }\n\n    return `${textBeforeCodeBlock}${textAfterCodeBlock.slice(1)}`;\n  });\n}\n\nfunction createMarkdownInfoFromMatch(params: CreateMarkdownInfoFromMatchParams): CodeBlockMarkdownInformation {\n  const {\n    approximateSectionInfo,\n    linesBeforeSectionCount,\n    match,\n    noteContent,\n    potentialCodeBlockText,\n    sourceLinesCount,\n    textLineOffsets\n  } = params;\n\n  const linePrefix = match.groups?.['LinePrefix'] ?? '';\n  const codeBlockStartDelimiter = match.groups?.['CodeBlockStartDelimiter'] ?? '';\n  const codeBlockEndDelimiter = match.groups?.['CodeBlockEndDelimiter'] ?? '';\n  const codeBlockArgsStr = match.groups?.['CodeBlockArgs'] ?? '';\n  const language = match.groups?.['CodeBlockLanguage'] ?? '';\n\n  const previousText = potentialCodeBlockText.slice(0, match.index);\n  const previousTextLinesCount = previousText.split('\\n').length - 1;\n\n  const startLine = linesBeforeSectionCount + previousTextLinesCount;\n  const endLine = startLine + sourceLinesCount + 1;\n\n  return {\n    args: codeBlockArgsStr.split(/\\s+/).filter(Boolean),\n    endDelimiter: codeBlockEndDelimiter,\n    language,\n    linePrefix,\n    noteContent,\n    positionInNote: {\n      end: {\n        col: (textLineOffsets.get(endLine + 1) ?? 0) - (textLineOffsets.get(endLine) ?? 0) - 1,\n        line: endLine,\n        offset: (textLineOffsets.get(endLine + 1) ?? 0) - 1\n      },\n      start: {\n        col: 0,\n        line: startLine,\n        offset: textLineOffsets.get(startLine) ?? 0\n      }\n    },\n    rawArgsStr: codeBlockArgsStr,\n    sectionInfo: {\n      lineEnd: previousTextLinesCount + sourceLinesCount + 1,\n      lineStart: previousTextLinesCount,\n      text: approximateSectionInfo.text\n    },\n    startDelimiter: codeBlockStartDelimiter\n  };\n}\n\nfunction getLanguageFromElement(el: HTMLElement): string {\n  const BLOCK_LANGUAGE_PREFIX = 'block-language-';\n  return Array.from(el.classList).find((cls) => cls.startsWith(BLOCK_LANGUAGE_PREFIX))?.slice(BLOCK_LANGUAGE_PREFIX.length) ?? '';\n}\n\nfunction insertText(content: string, insertLineIndex: number, text: string, shouldPreserveLinePrefix?: boolean): string {\n  const lines = content.split('\\n');\n  const newLines = lines.slice();\n  const textLines = text.split('\\n');\n\n  if (insertLineIndex < 0) {\n    insertLineIndex = 0;\n  }\n  if (insertLineIndex > lines.length) {\n    insertLineIndex = lines.length;\n  }\n\n  const PREFIX_LINE_REG_EXP = /^ {0,3}(?:> {1,3})*/g;\n  const match = (lines[insertLineIndex] ?? '').match(PREFIX_LINE_REG_EXP);\n  const linePrefix = match?.[0] ?? '';\n  newLines.splice(insertLineIndex, 0, ...(shouldPreserveLinePrefix ? textLines.map((line) => indent(line, linePrefix)) : textLines));\n  return newLines.join('\\n');\n}\n\nfunction isSuitableCodeBlock(\n  match: RegExpMatchArray,\n  language: string,\n  sourceLf: string,\n  isInCallout: boolean\n): boolean {\n  const codeBlockLanguage = match.groups?.['CodeBlockLanguage'] ?? '';\n  if (codeBlockLanguage !== language) {\n    return false;\n  }\n\n  const linePrefix = match.groups?.['LinePrefix'] ?? '';\n\n  if (isInCallout && !linePrefix.includes('> ')) {\n    return false;\n  }\n\n  const codeBlockContent = match.groups?.['CodeBlockContent'] ?? '';\n  const cleanCodeBlockContent = codeBlockContent.split('\\n').map((line) => line.slice(linePrefix.length)).join('\\n');\n\n  return cleanCodeBlockContent === sourceLf;\n}\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}
