obsidian-dev-utils 37.2.0 → 38.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.
- package/CHANGELOG.md +11 -0
- package/dist/lib/cjs/Error.cjs +12 -41
- package/dist/lib/cjs/Error.d.cts +1 -2
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/CodeBlockMarkdownInformation.cjs +1 -1
- package/dist/lib/cjs/obsidian/CodeBlockMarkdownInformation.d.cts +4 -0
- package/dist/lib/cjs/obsidian/Link.cjs +33 -10
- package/dist/lib/cjs/obsidian/Link.d.cts +39 -18
- package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.cjs +86 -79
- package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.d.cts +0 -4
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +32 -43
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.d.cts +2 -6
- package/dist/lib/cjs/obsidian/Vault.cjs +10 -1
- package/dist/lib/cjs/obsidian/Vault.d.cts +8 -0
- package/dist/lib/esm/Error.d.mts +1 -2
- package/dist/lib/esm/Error.mjs +12 -41
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/CodeBlockMarkdownInformation.d.mts +4 -0
- package/dist/lib/esm/obsidian/Link.d.mts +39 -18
- package/dist/lib/esm/obsidian/Link.mjs +32 -10
- package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.d.mts +0 -4
- package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.mjs +87 -79
- package/dist/lib/esm/obsidian/RenameDeleteHandler.d.mts +2 -6
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +34 -47
- package/dist/lib/esm/obsidian/Vault.d.mts +8 -0
- package/dist/lib/esm/obsidian/Vault.mjs +9 -1
- package/package.json +1 -1
|
@@ -42,82 +42,79 @@ async function getCodeBlockMarkdownInfo(options) {
|
|
|
42
42
|
if (!sourceFile) {
|
|
43
43
|
throw new Error(`Source file ${ctx.sourcePath} not found.`);
|
|
44
44
|
}
|
|
45
|
-
|
|
46
|
-
if (options.noteContent) {
|
|
47
|
-
content = options.noteContent;
|
|
48
|
-
} else {
|
|
49
|
-
await (0, import_Vault.saveNote)(app, sourceFile);
|
|
50
|
-
content = await app.vault.read(sourceFile);
|
|
51
|
-
}
|
|
52
|
-
const contentLf = (0, import_String.ensureLfEndings)(content);
|
|
53
|
-
if (contentLf !== content) {
|
|
54
|
-
const infoLf = await getCodeBlockMarkdownInfo({
|
|
55
|
-
...options,
|
|
56
|
-
noteContent: contentLf
|
|
57
|
-
});
|
|
58
|
-
if (!infoLf) {
|
|
59
|
-
return null;
|
|
60
|
-
}
|
|
61
|
-
const lfOffsetMapper = (0, import_String.getLfNormalizedOffsetToOriginalOffsetMapper)(content);
|
|
62
|
-
infoLf.positionInNote.start.offset = lfOffsetMapper(infoLf.positionInNote.start.offset);
|
|
63
|
-
infoLf.positionInNote.end.offset = lfOffsetMapper(infoLf.positionInNote.end.offset);
|
|
64
|
-
return infoLf;
|
|
65
|
-
}
|
|
66
|
-
const approximateSectionInfo = ctx.getSectionInfo(el) ?? {
|
|
67
|
-
lineEnd: content.split("\n").length - 1,
|
|
68
|
-
lineStart: 0,
|
|
69
|
-
text: content
|
|
70
|
-
};
|
|
71
|
-
approximateSectionInfo.text = (0, import_String.ensureLfEndings)(approximateSectionInfo.text);
|
|
72
|
-
if (!(0, import_String.hasSingleOccurrence)(content, approximateSectionInfo.text)) {
|
|
73
|
-
return null;
|
|
74
|
-
}
|
|
75
|
-
const sectionOffset = content.indexOf(approximateSectionInfo.text);
|
|
76
|
-
const linesBeforeSectionCount = content.slice(0, sectionOffset).split("\n").length - 1;
|
|
77
|
-
const isInCallout = !!el.parentElement?.classList.contains("callout-content");
|
|
78
|
-
const language = getLanguageFromElement(el);
|
|
79
|
-
const sourceLines = source.split("\n");
|
|
80
|
-
const textLines = approximateSectionInfo.text.split("\n");
|
|
81
|
-
const textLineOffsets = /* @__PURE__ */ new Map();
|
|
82
|
-
textLineOffsets.set(linesBeforeSectionCount, sectionOffset);
|
|
83
|
-
let lastTextLineOffset = sectionOffset;
|
|
84
|
-
for (let i = 0; i < textLines.length; i++) {
|
|
85
|
-
const textLine = textLines[i];
|
|
86
|
-
const line = textLine ?? "";
|
|
87
|
-
const lineOffset = lastTextLineOffset + line.length + 1;
|
|
88
|
-
textLineOffsets.set(linesBeforeSectionCount + i + 1, lineOffset);
|
|
89
|
-
lastTextLineOffset = lineOffset;
|
|
90
|
-
}
|
|
91
|
-
const potentialCodeBlockTextLines = textLines.map(
|
|
92
|
-
(line, index) => approximateSectionInfo.lineStart <= index && index <= approximateSectionInfo.lineEnd ? line : ""
|
|
93
|
-
);
|
|
94
|
-
const potentialCodeBlockText = potentialCodeBlockTextLines.join("\n");
|
|
95
|
-
const REG_EXP = /(?<=^|\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;
|
|
45
|
+
await (0, import_Vault.saveNote)(app, sourceFile);
|
|
96
46
|
let markdownInfo = null;
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
47
|
+
await (0, import_Vault.invokeWithFileSystemLock)(app, sourceFile, (noteContent) => {
|
|
48
|
+
const noteContentLf = (0, import_String.ensureLfEndings)(noteContent);
|
|
49
|
+
const approximateSectionInfo = ctx.getSectionInfo(el) ?? {
|
|
50
|
+
lineEnd: noteContentLf.split("\n").length - 1,
|
|
51
|
+
lineStart: 0,
|
|
52
|
+
text: noteContentLf
|
|
53
|
+
};
|
|
54
|
+
approximateSectionInfo.text = (0, import_String.ensureLfEndings)(approximateSectionInfo.text);
|
|
55
|
+
const sourceLf = (0, import_String.ensureLfEndings)(source);
|
|
56
|
+
if (!(0, import_String.hasSingleOccurrence)(noteContentLf, approximateSectionInfo.text)) {
|
|
57
|
+
return;
|
|
100
58
|
}
|
|
101
|
-
|
|
102
|
-
|
|
59
|
+
const sectionOffset = noteContentLf.indexOf(approximateSectionInfo.text);
|
|
60
|
+
const linesBeforeSectionCount = noteContentLf.slice(0, sectionOffset).split("\n").length - 1;
|
|
61
|
+
const isInCallout = !!el.parentElement?.classList.contains("callout-content");
|
|
62
|
+
const language = getLanguageFromElement(el);
|
|
63
|
+
const sourceLines = sourceLf.split("\n");
|
|
64
|
+
const textLines = approximateSectionInfo.text.split("\n");
|
|
65
|
+
const textLineOffsets = /* @__PURE__ */ new Map();
|
|
66
|
+
textLineOffsets.set(linesBeforeSectionCount, sectionOffset);
|
|
67
|
+
let lastTextLineOffset = sectionOffset;
|
|
68
|
+
for (let i = 0; i < textLines.length; i++) {
|
|
69
|
+
const textLine = textLines[i] ?? "";
|
|
70
|
+
const lineOffset = lastTextLineOffset + textLine.length + 1;
|
|
71
|
+
textLineOffsets.set(linesBeforeSectionCount + i + 1, lineOffset);
|
|
72
|
+
lastTextLineOffset = lineOffset;
|
|
103
73
|
}
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
74
|
+
const potentialCodeBlockTextLines = textLines.map(
|
|
75
|
+
(line, index) => approximateSectionInfo.lineStart <= index && index <= approximateSectionInfo.lineEnd ? line : ""
|
|
76
|
+
);
|
|
77
|
+
const potentialCodeBlockText = potentialCodeBlockTextLines.join("\n");
|
|
78
|
+
const REG_EXP = /(?<=^|\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;
|
|
79
|
+
for (const match of potentialCodeBlockText.matchAll(REG_EXP)) {
|
|
80
|
+
if (!isSuitableCodeBlock(match, language, sourceLf, isInCallout)) {
|
|
81
|
+
continue;
|
|
82
|
+
}
|
|
83
|
+
if (markdownInfo) {
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
markdownInfo = createMarkdownInfoFromMatch({
|
|
87
|
+
approximateSectionInfo,
|
|
88
|
+
linesBeforeSectionCount,
|
|
89
|
+
match,
|
|
90
|
+
noteContent,
|
|
91
|
+
potentialCodeBlockText,
|
|
92
|
+
sourceLinesCount: sourceLines.length,
|
|
93
|
+
textLineOffsets
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
if (!markdownInfo) {
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
if (noteContentLf === noteContent) {
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
const lfOffsetMapper = (0, import_String.getLfNormalizedOffsetToOriginalOffsetMapper)(noteContent);
|
|
103
|
+
markdownInfo.positionInNote.start.offset = lfOffsetMapper(markdownInfo.positionInNote.start.offset);
|
|
104
|
+
markdownInfo.positionInNote.end.offset = lfOffsetMapper(markdownInfo.positionInNote.end.offset);
|
|
105
|
+
});
|
|
109
106
|
return markdownInfo;
|
|
110
107
|
}
|
|
111
108
|
async function insertAfterCodeBlock(options) {
|
|
112
109
|
const { app, ctx, lineOffset = 0, text } = options;
|
|
113
110
|
await (0, import_Vault.process)(app, ctx.sourcePath, async (_abortSignal, content) => {
|
|
114
|
-
const markdownInfo = await getCodeBlockMarkdownInfo(
|
|
115
|
-
...options,
|
|
116
|
-
noteContent: content
|
|
117
|
-
});
|
|
111
|
+
const markdownInfo = await getCodeBlockMarkdownInfo(options);
|
|
118
112
|
if (!markdownInfo) {
|
|
119
113
|
throw new Error("Could not uniquely identify the code block.");
|
|
120
114
|
}
|
|
115
|
+
if (content !== markdownInfo.noteContent) {
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
121
118
|
const insertLineIndex = markdownInfo.positionInNote.end.line + lineOffset + 1;
|
|
122
119
|
return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
|
|
123
120
|
});
|
|
@@ -125,13 +122,13 @@ async function insertAfterCodeBlock(options) {
|
|
|
125
122
|
async function insertBeforeCodeBlock(options) {
|
|
126
123
|
const { app, ctx, lineOffset = 0, text } = options;
|
|
127
124
|
await (0, import_Vault.process)(app, ctx.sourcePath, async (_abortSignal, content) => {
|
|
128
|
-
const markdownInfo = await getCodeBlockMarkdownInfo(
|
|
129
|
-
...options,
|
|
130
|
-
noteContent: content
|
|
131
|
-
});
|
|
125
|
+
const markdownInfo = await getCodeBlockMarkdownInfo(options);
|
|
132
126
|
if (!markdownInfo) {
|
|
133
127
|
throw new Error("Could not uniquely identify the code block.");
|
|
134
128
|
}
|
|
129
|
+
if (content !== markdownInfo.noteContent) {
|
|
130
|
+
return null;
|
|
131
|
+
}
|
|
135
132
|
const insertLineIndex = markdownInfo.positionInNote.start.line - lineOffset;
|
|
136
133
|
return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
|
|
137
134
|
});
|
|
@@ -149,13 +146,13 @@ async function replaceCodeBlock(options) {
|
|
|
149
146
|
await (0, import_Vault.process)(app, ctx.sourcePath, async (abortSignal, content) => {
|
|
150
147
|
abortSignal = (0, import_AbortController.abortSignalAny)(abortSignal, options.abortSignal);
|
|
151
148
|
abortSignal.throwIfAborted();
|
|
152
|
-
const markdownInfo = await getCodeBlockMarkdownInfo(
|
|
153
|
-
...options,
|
|
154
|
-
noteContent: content
|
|
155
|
-
});
|
|
149
|
+
const markdownInfo = await getCodeBlockMarkdownInfo(options);
|
|
156
150
|
if (!markdownInfo) {
|
|
157
151
|
throw new Error("Could not uniquely identify the code block.");
|
|
158
152
|
}
|
|
153
|
+
if (content !== markdownInfo.noteContent) {
|
|
154
|
+
return null;
|
|
155
|
+
}
|
|
159
156
|
let oldCodeBlock = content.slice(markdownInfo.positionInNote.start.offset, markdownInfo.positionInNote.end.offset);
|
|
160
157
|
if (options.shouldPreserveLinePrefix) {
|
|
161
158
|
oldCodeBlock = (0, import_String.unindent)(oldCodeBlock, markdownInfo.linePrefix);
|
|
@@ -178,7 +175,16 @@ async function replaceCodeBlock(options) {
|
|
|
178
175
|
return `${textBeforeCodeBlock}${textAfterCodeBlock.slice(1)}`;
|
|
179
176
|
});
|
|
180
177
|
}
|
|
181
|
-
function createMarkdownInfoFromMatch(
|
|
178
|
+
function createMarkdownInfoFromMatch(options) {
|
|
179
|
+
const {
|
|
180
|
+
approximateSectionInfo,
|
|
181
|
+
linesBeforeSectionCount,
|
|
182
|
+
match,
|
|
183
|
+
noteContent,
|
|
184
|
+
potentialCodeBlockText,
|
|
185
|
+
sourceLinesCount,
|
|
186
|
+
textLineOffsets
|
|
187
|
+
} = options;
|
|
182
188
|
const linePrefix = match.groups?.["LinePrefix"] ?? "";
|
|
183
189
|
const codeBlockStartDelimiter = match.groups?.["CodeBlockStartDelimiter"] ?? "";
|
|
184
190
|
const codeBlockEndDelimiter = match.groups?.["CodeBlockEndDelimiter"] ?? "";
|
|
@@ -187,12 +193,13 @@ function createMarkdownInfoFromMatch(potentialCodeBlockText, match, approximateS
|
|
|
187
193
|
const previousText = potentialCodeBlockText.slice(0, match.index);
|
|
188
194
|
const previousTextLinesCount = previousText.split("\n").length - 1;
|
|
189
195
|
const startLine = linesBeforeSectionCount + previousTextLinesCount;
|
|
190
|
-
const endLine = startLine +
|
|
196
|
+
const endLine = startLine + sourceLinesCount + 1;
|
|
191
197
|
return {
|
|
192
198
|
args: codeBlockArgsStr.split(/\s+/).filter(Boolean),
|
|
193
199
|
endDelimiter: codeBlockEndDelimiter,
|
|
194
200
|
language,
|
|
195
201
|
linePrefix,
|
|
202
|
+
noteContent,
|
|
196
203
|
positionInNote: {
|
|
197
204
|
end: {
|
|
198
205
|
col: (textLineOffsets.get(endLine + 1) ?? 0) - (textLineOffsets.get(endLine) ?? 0) - 1,
|
|
@@ -207,7 +214,7 @@ function createMarkdownInfoFromMatch(potentialCodeBlockText, match, approximateS
|
|
|
207
214
|
},
|
|
208
215
|
rawArgsStr: codeBlockArgsStr,
|
|
209
216
|
sectionInfo: {
|
|
210
|
-
lineEnd: previousTextLinesCount +
|
|
217
|
+
lineEnd: previousTextLinesCount + sourceLinesCount + 1,
|
|
211
218
|
lineStart: previousTextLinesCount,
|
|
212
219
|
text: approximateSectionInfo.text
|
|
213
220
|
},
|
|
@@ -234,7 +241,7 @@ function insertText(content, insertLineIndex, text, shouldPreserveLinePrefix) {
|
|
|
234
241
|
newLines.splice(insertLineIndex, 0, ...shouldPreserveLinePrefix ? textLines.map((line) => (0, import_String.indent)(line, linePrefix)) : textLines);
|
|
235
242
|
return newLines.join("\n");
|
|
236
243
|
}
|
|
237
|
-
function isSuitableCodeBlock(match, language,
|
|
244
|
+
function isSuitableCodeBlock(match, language, sourceLf, isInCallout) {
|
|
238
245
|
const codeBlockLanguage = match.groups?.["CodeBlockLanguage"] ?? "";
|
|
239
246
|
if (codeBlockLanguage !== language) {
|
|
240
247
|
return false;
|
|
@@ -245,7 +252,7 @@ function isSuitableCodeBlock(match, language, source, isInCallout) {
|
|
|
245
252
|
}
|
|
246
253
|
const codeBlockContent = match.groups?.["CodeBlockContent"] ?? "";
|
|
247
254
|
const cleanCodeBlockContent = codeBlockContent.split("\n").map((line) => line.slice(linePrefix.length)).join("\n");
|
|
248
|
-
return cleanCodeBlockContent ===
|
|
255
|
+
return cleanCodeBlockContent === sourceLf;
|
|
249
256
|
}
|
|
250
257
|
// Annotate the CommonJS export names for ESM import in node:
|
|
251
258
|
0 && (module.exports = {
|
|
@@ -255,4 +262,4 @@ function isSuitableCodeBlock(match, language, source, isInCallout) {
|
|
|
255
262
|
removeCodeBlock,
|
|
256
263
|
replaceCodeBlock
|
|
257
264
|
});
|
|
258
|
-
//# 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\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 {\n  ensureLfEndings,\n  getLfNormalizedOffsetToOriginalOffsetMapper,\n  hasSingleOccurrence,\n  indent,\n  unindent\n} from '../String.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\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  app: App;\n\n  /**\n   * A {@link MarkdownPostProcessorContext} object.\n   */\n  ctx: MarkdownPostProcessorContext;\n\n  /**\n   * A {@link HTMLElement} representing the code block.\n   */\n  el: HTMLElement;\n\n  /**\n   * A content of the note.\n   */\n  noteContent?: string;\n\n  /**\n   * A source of the code block.\n   */\n  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  lineOffset?: number;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  shouldPreserveLinePrefix?: boolean;\n\n  /**\n   * A text to insert after the code block.\n   */\n  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  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  abortSignal?: AbortSignal;\n\n  /**\n   * Provides a new code block.\n   */\n  codeBlockProvider: ValueProvider<string, [string]>;\n\n  /**\n   * Whether to keep the gap when the new code block is empty. Default is `false`.\n   */\n  shouldKeepGapWhenEmpty?: boolean;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  shouldPreserveLinePrefix?: boolean;\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 = app.vault.getFileByPath(ctx.sourcePath);\n  if (!sourceFile) {\n    throw new Error(`Source file ${ctx.sourcePath} not found.`);\n  }\n\n  let content: string;\n  if (options.noteContent) {\n    content = options.noteContent;\n  } else {\n    await saveNote(app, sourceFile);\n    content = await app.vault.read(sourceFile);\n  }\n\n  const contentLf = ensureLfEndings(content);\n  if (contentLf !== content) {\n    const infoLf = await getCodeBlockMarkdownInfo({\n      ...options,\n      noteContent: contentLf\n    });\n    if (!infoLf) {\n      return null;\n    }\n\n    const lfOffsetMapper = getLfNormalizedOffsetToOriginalOffsetMapper(content);\n    infoLf.positionInNote.start.offset = lfOffsetMapper(infoLf.positionInNote.start.offset);\n    infoLf.positionInNote.end.offset = lfOffsetMapper(infoLf.positionInNote.end.offset);\n    return infoLf;\n  }\n\n  const approximateSectionInfo = ctx.getSectionInfo(el) ?? {\n    lineEnd: content.split('\\n').length - 1,\n    lineStart: 0,\n    text: content\n  };\n\n  approximateSectionInfo.text = ensureLfEndings(approximateSectionInfo.text);\n\n  if (!hasSingleOccurrence(content, approximateSectionInfo.text)) {\n    return null;\n  }\n\n  const sectionOffset = content.indexOf(approximateSectionInfo.text);\n  const linesBeforeSectionCount = content.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 = source.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 line = textLine ?? '';\n    const lineOffset = lastTextLineOffset + line.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  let markdownInfo: CodeBlockMarkdownInformation | null = null;\n\n  for (const match of potentialCodeBlockText.matchAll(REG_EXP)) {\n    if (!isSuitableCodeBlock(match, language, source, isInCallout)) {\n      continue;\n    }\n\n    if (markdownInfo) {\n      return null;\n    }\n\n    markdownInfo = createMarkdownInfoFromMatch(potentialCodeBlockText, match, approximateSectionInfo, sourceLines, textLineOffsets, linesBeforeSectionCount);\n  }\n\n  if (!markdownInfo) {\n    return null;\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({\n      ...options,\n      noteContent: content\n    });\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\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({\n      ...options,\n      noteContent: content\n    });\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\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({\n      ...options,\n      noteContent: content\n    });\n    if (!markdownInfo) {\n      throw new Error('Could not uniquely identify the code block.');\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    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(\n  potentialCodeBlockText: string,\n  match: RegExpMatchArray,\n  approximateSectionInfo: MarkdownSectionInformation,\n  sourceLines: string[],\n  textLineOffsets: Map<number, number>,\n  linesBeforeSectionCount: number\n): CodeBlockMarkdownInformation {\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 + sourceLines.length + 1;\n\n  return {\n    args: codeBlockArgsStr.split(/\\s+/).filter(Boolean),\n    endDelimiter: codeBlockEndDelimiter,\n    language,\n    linePrefix,\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 + sourceLines.length + 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  source: 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 === source;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
265
|
+
//# 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\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 {\n  ensureLfEndings,\n  getLfNormalizedOffsetToOriginalOffsetMapper,\n  hasSingleOccurrence,\n  indent,\n  unindent\n} from '../String.ts';\nimport { resolveValue } from '../ValueProvider.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  app: App;\n\n  /**\n   * A {@link MarkdownPostProcessorContext} object.\n   */\n  ctx: MarkdownPostProcessorContext;\n\n  /**\n   * A {@link HTMLElement} representing the code block.\n   */\n  el: HTMLElement;\n\n  /**\n   * A source of the code block.\n   */\n  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  lineOffset?: number;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  shouldPreserveLinePrefix?: boolean;\n\n  /**\n   * A text to insert after the code block.\n   */\n  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  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  abortSignal?: AbortSignal;\n\n  /**\n   * Provides a new code block.\n   */\n  codeBlockProvider: ValueProvider<string, [string]>;\n\n  /**\n   * Whether to keep the gap when the new code block is empty. Default is `false`.\n   */\n  shouldKeepGapWhenEmpty?: boolean;\n\n  /**\n   * Whether to preserve the line prefix of the code block. Default is `false`.\n   */\n  shouldPreserveLinePrefix?: boolean;\n}\n\ninterface CreateMarkdownInfoFromMatchOptions {\n  approximateSectionInfo: MarkdownSectionInformation;\n  linesBeforeSectionCount: number;\n  match: RegExpMatchArray;\n  noteContent: string;\n  potentialCodeBlockText: string;\n  sourceLinesCount: number;\n  textLineOffsets: Map<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 = app.vault.getFileByPath(ctx.sourcePath);\n  if (!sourceFile) {\n    throw new Error(`Source file ${ctx.sourcePath} not found.`);\n  }\n\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 = ctx.getSectionInfo(el) ?? {\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    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.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    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"],
  "mappings": "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",
  "names": []
}

|