obsidian-dev-utils 33.1.1 → 34.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 -11
- package/dist/lib/cjs/Error.d.cts +2 -1
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/String.cjs +28 -3
- package/dist/lib/cjs/String.d.cts +25 -0
- package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.cjs +161 -28
- package/dist/lib/cjs/obsidian/MarkdownCodeBlockProcessor.d.cts +108 -12
- package/dist/lib/esm/Error.d.mts +2 -1
- package/dist/lib/esm/Error.mjs +12 -11
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/String.d.mts +25 -0
- package/dist/lib/esm/String.mjs +24 -2
- package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.d.mts +108 -12
- package/dist/lib/esm/obsidian/MarkdownCodeBlockProcessor.mjs +163 -28
- package/package.json +1 -1
package/dist/lib/esm/String.mjs
CHANGED
|
@@ -32,6 +32,14 @@ function ensureStartsWith(str, prefix) {
|
|
|
32
32
|
function escape(str) {
|
|
33
33
|
return replace(str, ESCAPE_MAP);
|
|
34
34
|
}
|
|
35
|
+
function hasSingleOccurrence(str, searchValue) {
|
|
36
|
+
const firstIndex = str.indexOf(searchValue);
|
|
37
|
+
const lastIndex = str.lastIndexOf(searchValue);
|
|
38
|
+
return firstIndex !== -1 && firstIndex === lastIndex;
|
|
39
|
+
}
|
|
40
|
+
function indent(text, prefix) {
|
|
41
|
+
return text.split("\n").map((line) => `${prefix}${line}`).join("\n");
|
|
42
|
+
}
|
|
35
43
|
function insertAt(str, substring, startIndex, endIndex) {
|
|
36
44
|
endIndex ??= startIndex;
|
|
37
45
|
return str.slice(0, startIndex) + substring + str.slice(endIndex);
|
|
@@ -120,10 +128,23 @@ function trimStart(str, prefix, validate) {
|
|
|
120
128
|
function unescape(str) {
|
|
121
129
|
return replace(str, UNESCAPE_MAP);
|
|
122
130
|
}
|
|
131
|
+
function unindent(text, prefix, shouldThrowIfNotIndented = false) {
|
|
132
|
+
return text.split("\n").map((line) => {
|
|
133
|
+
if (line.startsWith(prefix)) {
|
|
134
|
+
return line.slice(prefix.length);
|
|
135
|
+
}
|
|
136
|
+
if (shouldThrowIfNotIndented) {
|
|
137
|
+
throw new Error(`Line "${line}" is not indented with "${prefix}"`);
|
|
138
|
+
}
|
|
139
|
+
return line;
|
|
140
|
+
}).join("\n");
|
|
141
|
+
}
|
|
123
142
|
export {
|
|
124
143
|
ensureEndsWith,
|
|
125
144
|
ensureStartsWith,
|
|
126
145
|
escape,
|
|
146
|
+
hasSingleOccurrence,
|
|
147
|
+
indent,
|
|
127
148
|
insertAt,
|
|
128
149
|
makeValidVariableName,
|
|
129
150
|
normalize,
|
|
@@ -132,6 +153,7 @@ export {
|
|
|
132
153
|
replaceAllAsync,
|
|
133
154
|
trimEnd,
|
|
134
155
|
trimStart,
|
|
135
|
-
unescape
|
|
156
|
+
unescape,
|
|
157
|
+
unindent
|
|
136
158
|
};
|
|
137
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/String.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for string operations.\n */\n\nimport type { MaybeReturn } from './Type.ts';\nimport type { ValueProvider } from './ValueProvider.ts';\n\nimport { abortSignalNever } from './AbortController.ts';\nimport { throwExpression } from './Error.ts';\nimport { escapeRegExp } from './RegExp.ts';\nimport { resolveValue } from './ValueProvider.ts';\n\n/**\n * A synchronous/asynchronous function that generates replacement strings, or a string to replace with.\n */\nexport type AsyncReplacer<ReplaceGroupArgs extends string[]> = ValueProvider<StringReplacement, [ReplaceCommonArgs, ...ReplaceGroupArgs]>;\n\n/**\n * Common arguments for the `replaceAll`/`replaceAllAsync` functions.\n */\nexport interface ReplaceCommonArgs {\n  /**\n   * The groups of the match.\n   */\n  groups: Record<string, string | undefined> | undefined;\n\n  /**\n   * The indices of the groups that were not found in the match.\n   */\n  missingGroupIndices: number[];\n\n  /**\n   * The offset of the match.\n   */\n  offset: number;\n\n  /**\n   * The source of the match.\n   */\n  source: string;\n\n  /**\n   * The substring of the match.\n   */\n  substring: string;\n}\n\n/**\n * A synchronous function that generates replacement strings, or a string to replace with.\n */\nexport type Replacer<ReplaceGroupArgs extends string[]> = ((...args: [ReplaceCommonArgs, ...ReplaceGroupArgs]) => StringReplacement) | StringReplacement;\n\ntype StringReplacement = MaybeReturn<string>;\n\n/**\n * Mapping of special characters to their escaped counterparts.\n */\nconst ESCAPE_MAP: Record<string, string> = {\n  '\\n': '\\\\n',\n  '\\r': '\\\\r',\n  '\\t': '\\\\t',\n  '\\b': '\\\\b',\n  '\\f': '\\\\f',\n  '\\'': '\\\\\\'',\n  '\"': '\\\\\"',\n  '\\\\': '\\\\\\\\'\n} as const;\n\n/**\n * Mapping of escaped special characters to their unescaped counterparts.\n */\nconst UNESCAPE_MAP: Record<string, string> = {};\nfor (const [key, value] of Object.entries(ESCAPE_MAP)) {\n  UNESCAPE_MAP[value] = key;\n}\n\n/**\n * Ensures that a string ends with the specified suffix, adding it if necessary.\n *\n * @param str - The string to check.\n * @param suffix - The suffix to ensure.\n * @returns The string that ends with the suffix.\n */\nexport function ensureEndsWith(str: string, suffix: string): string {\n  return str.endsWith(suffix) ? str : str + suffix;\n}\n\n/**\n * Ensures that a string starts with the specified prefix, adding it if necessary.\n *\n * @param str - The string to check.\n * @param prefix - The prefix to ensure.\n * @returns The string that starts with the prefix.\n */\nexport function ensureStartsWith(str: string, prefix: string): string {\n  return str.startsWith(prefix) ? str : prefix + str;\n}\n\n/**\n * Escapes special characters in a string.\n *\n * @param str - The string to escape.\n * @returns The escaped string.\n */\nexport function escape(str: string): string {\n  return replace(str, ESCAPE_MAP);\n}\n\n/**\n * Inserts a substring at a specified position in a string.\n *\n * @param str - The string to insert the substring into.\n * @param substring - The substring to insert.\n * @param startIndex - The index to insert the substring at.\n * @param endIndex - The index to end the substring at.\n * @returns The modified string with the substring inserted.\n */\nexport function insertAt(str: string, substring: string, startIndex: number, endIndex?: number): string {\n  endIndex ??= startIndex;\n  return str.slice(0, startIndex) + substring + str.slice(endIndex);\n}\n\n/**\n * Converts a string into a valid JavaScript variable name by replacing invalid characters with underscores.\n *\n * @param str - The string to convert.\n * @returns The valid variable name.\n */\nexport function makeValidVariableName(str: string): string {\n  return replaceAll(str, /[^a-zA-Z0-9_]/g, '_');\n}\n\n/**\n * Normalizes a string by converting it to the NFC form and replacing non-breaking spaces with regular spaces.\n *\n * @param str - The string to normalize.\n * @returns The normalized string.\n */\nexport function normalize(str: string): string {\n  return replaceAll(str, /\\u00A0|\\u202F/g, ' ').normalize('NFC');\n}\n\n/**\n * Replaces occurrences of strings in a given string based on a replacements map.\n *\n * @param str - The string to perform replacements on.\n * @param replacementsMap - An object mapping strings to their replacement values.\n * @returns The modified string with replacements applied.\n */\nexport function replace(str: string, replacementsMap: Record<string, string>): string {\n  const regExp = new RegExp(Object.keys(replacementsMap).map((source) => escapeRegExp(source)).join('|'), 'g');\n  return replaceAll(str, regExp, ({ substring: source }) => replacementsMap[source] ?? throwExpression(new Error(`Unexpected replacement source: ${source}`)));\n}\n\n/**\n * Replaces all occurrences of a search string or pattern with the results of an replacer function.\n *\n * @typeParam ReplaceGroupArgs - The type of additional arguments passed to the replacer function.\n * @param str - The string in which to perform replacements.\n * @param searchValue - The string or regular expression to search for.\n * @param replacer - A replacer function that generates replacement strings, or a string to replace with.\n * @returns The string with all replacements made.\n */\nexport function replaceAll<ReplaceGroupArgs extends string[]>(\n  str: string,\n  searchValue: RegExp | string,\n  replacer: Replacer<ReplaceGroupArgs>\n): string {\n  if (typeof replacer === 'undefined') {\n    return str;\n  }\n\n  if (searchValue instanceof RegExp && !searchValue.global) {\n    searchValue = new RegExp(searchValue.source, `${searchValue.flags}g`);\n  }\n\n  if (typeof replacer === 'string') {\n    return str.replaceAll(searchValue, replacer);\n  }\n\n  return str.replaceAll(searchValue, (substring: string, ...args: unknown[]) => {\n    const SOURCE_INDEX_OFFSET_FOR_GROUP_ARG = 2;\n    const hasGroupsArg = typeof args.at(-1) === 'object';\n    const sourceIndex = hasGroupsArg ? args.length - SOURCE_INDEX_OFFSET_FOR_GROUP_ARG : args.length - 1;\n\n    const commonArgs: ReplaceCommonArgs = {\n      groups: hasGroupsArg ? args.at(-1) as Record<string, string | undefined> : undefined,\n      missingGroupIndices: [],\n      offset: args.at(sourceIndex - 1) as number,\n      source: args.at(sourceIndex) as string,\n      substring\n    };\n\n    const groupArgs = args.slice(0, sourceIndex - 1).map((arg, index) => {\n      if (typeof arg === 'string') {\n        return arg;\n      }\n\n      if (typeof arg === 'undefined') {\n        commonArgs.missingGroupIndices.push(index);\n        return '';\n      }\n\n      throw new Error(`Unexpected argument type: ${typeof arg}`);\n    }) as ReplaceGroupArgs;\n\n    return (replacer(commonArgs, ...groupArgs) as string | undefined) ?? commonArgs.substring;\n  });\n}\n\n/**\n * Asynchronously replaces all occurrences of a search string or pattern with the results of an asynchronous replacer function.\n *\n * @typeParam ReplaceGroupArgs - The type of additional arguments passed to the replacer function.\n * @param str - The string in which to perform replacements.\n * @param searchValue - The string or regular expression to search for.\n * @param replacer - A synchronous/asynchronous function that generates replacement strings, or a string to replace with.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns A {@link Promise} that resolves to the string with all replacements made.\n */\nexport async function replaceAllAsync<ReplaceGroupArgs extends string[]>(\n  str: string,\n  searchValue: RegExp | string,\n  replacer: AsyncReplacer<ReplaceGroupArgs>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  if (typeof replacer === 'string') {\n    return replaceAll(str, searchValue, replacer);\n  }\n\n  const replacementAsyncFns: (() => Promise<StringReplacement>)[] = [];\n\n  replaceAll<ReplaceGroupArgs>(str, searchValue, (commonArgs, ...groupArgs) => {\n    replacementAsyncFns.push(() => resolveValue(replacer, abortSignal, commonArgs, ...groupArgs));\n    return '';\n  });\n\n  const replacements: StringReplacement[] = [];\n\n  for (const asyncFn of replacementAsyncFns) {\n    abortSignal.throwIfAborted();\n    replacements.push(await asyncFn());\n  }\n\n  abortSignal.throwIfAborted();\n  return replaceAll(str, searchValue, (args): string => replacements.shift() ?? args.substring);\n}\n\n/**\n * Trims the specified suffix from the end of a string.\n *\n * @param str - The string to trim.\n * @param suffix - The suffix to remove from the end of the string.\n * @param shouldValidate - If true, throws an error if the string does not end with the suffix.\n * @returns The trimmed string.\n * @throws If `validate` is true and the string does not end with the suffix.\n */\nexport function trimEnd(str: string, suffix: string, shouldValidate?: boolean): string {\n  if (str.endsWith(suffix)) {\n    return str.slice(0, -suffix.length);\n  }\n\n  if (shouldValidate) {\n    throw new Error(`String ${str} does not end with suffix ${suffix}`);\n  }\n\n  return str;\n}\n\n/**\n * Trims the specified prefix from the start of a string.\n *\n * @param str - The string to trim.\n * @param prefix - The prefix to remove from the start of the string.\n * @param validate - If true, throws an error if the string does not start with the prefix.\n * @returns The trimmed string.\n * @throws If `validate` is true and the string does not start with the prefix.\n */\nexport function trimStart(str: string, prefix: string, validate?: boolean): string {\n  if (str.startsWith(prefix)) {\n    return str.slice(prefix.length);\n  }\n\n  if (validate) {\n    throw new Error(`String ${str} does not start with prefix ${prefix}`);\n  }\n\n  return str;\n}\n\n/**\n * Unescapes a string by replacing escape sequences with their corresponding characters.\n *\n * @param str - The string to unescape.\n * @returns The unescaped string.\n */\nexport function unescape(str: string): string {\n  return replace(str, UNESCAPE_MAP);\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
159
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/String.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for string operations.\n */\n\nimport type { MaybeReturn } from './Type.ts';\nimport type { ValueProvider } from './ValueProvider.ts';\n\nimport { abortSignalNever } from './AbortController.ts';\nimport { throwExpression } from './Error.ts';\nimport { escapeRegExp } from './RegExp.ts';\nimport { resolveValue } from './ValueProvider.ts';\n\n/**\n * A synchronous/asynchronous function that generates replacement strings, or a string to replace with.\n */\nexport type AsyncReplacer<ReplaceGroupArgs extends string[]> = ValueProvider<StringReplacement, [ReplaceCommonArgs, ...ReplaceGroupArgs]>;\n\n/**\n * Common arguments for the `replaceAll`/`replaceAllAsync` functions.\n */\nexport interface ReplaceCommonArgs {\n  /**\n   * The groups of the match.\n   */\n  groups: Record<string, string | undefined> | undefined;\n\n  /**\n   * The indices of the groups that were not found in the match.\n   */\n  missingGroupIndices: number[];\n\n  /**\n   * The offset of the match.\n   */\n  offset: number;\n\n  /**\n   * The source of the match.\n   */\n  source: string;\n\n  /**\n   * The substring of the match.\n   */\n  substring: string;\n}\n\n/**\n * A synchronous function that generates replacement strings, or a string to replace with.\n */\nexport type Replacer<ReplaceGroupArgs extends string[]> = ((...args: [ReplaceCommonArgs, ...ReplaceGroupArgs]) => StringReplacement) | StringReplacement;\n\ntype StringReplacement = MaybeReturn<string>;\n\n/**\n * Mapping of special characters to their escaped counterparts.\n */\nconst ESCAPE_MAP: Record<string, string> = {\n  '\\n': '\\\\n',\n  '\\r': '\\\\r',\n  '\\t': '\\\\t',\n  '\\b': '\\\\b',\n  '\\f': '\\\\f',\n  '\\'': '\\\\\\'',\n  '\"': '\\\\\"',\n  '\\\\': '\\\\\\\\'\n} as const;\n\n/**\n * Mapping of escaped special characters to their unescaped counterparts.\n */\nconst UNESCAPE_MAP: Record<string, string> = {};\nfor (const [key, value] of Object.entries(ESCAPE_MAP)) {\n  UNESCAPE_MAP[value] = key;\n}\n\n/**\n * Ensures that a string ends with the specified suffix, adding it if necessary.\n *\n * @param str - The string to check.\n * @param suffix - The suffix to ensure.\n * @returns The string that ends with the suffix.\n */\nexport function ensureEndsWith(str: string, suffix: string): string {\n  return str.endsWith(suffix) ? str : str + suffix;\n}\n\n/**\n * Ensures that a string starts with the specified prefix, adding it if necessary.\n *\n * @param str - The string to check.\n * @param prefix - The prefix to ensure.\n * @returns The string that starts with the prefix.\n */\nexport function ensureStartsWith(str: string, prefix: string): string {\n  return str.startsWith(prefix) ? str : prefix + str;\n}\n\n/**\n * Escapes special characters in a string.\n *\n * @param str - The string to escape.\n * @returns The escaped string.\n */\nexport function escape(str: string): string {\n  return replace(str, ESCAPE_MAP);\n}\n\n/**\n * Checks if a string has a single occurrence of a search value.\n *\n * @param str - The string to check.\n * @param searchValue - The search value to check for.\n * @returns `true` if the string has a single occurrence of the search value, `false` otherwise.\n */\nexport function hasSingleOccurrence(str: string, searchValue: string): boolean {\n  const firstIndex = str.indexOf(searchValue);\n  const lastIndex = str.lastIndexOf(searchValue);\n  return firstIndex !== -1 && firstIndex === lastIndex;\n}\n\n/**\n * Indents a string by adding a prefix to each line.\n *\n * @param text - The string to indent.\n * @param prefix - The prefix to add to each line.\n * @returns The indented string.\n */\nexport function indent(text: string, prefix: string): string {\n  return text.split('\\n').map((line) => `${prefix}${line}`).join('\\n');\n}\n\n/**\n * Inserts a substring at a specified position in a string.\n *\n * @param str - The string to insert the substring into.\n * @param substring - The substring to insert.\n * @param startIndex - The index to insert the substring at.\n * @param endIndex - The index to end the substring at.\n * @returns The modified string with the substring inserted.\n */\nexport function insertAt(str: string, substring: string, startIndex: number, endIndex?: number): string {\n  endIndex ??= startIndex;\n  return str.slice(0, startIndex) + substring + str.slice(endIndex);\n}\n\n/**\n * Converts a string into a valid JavaScript variable name by replacing invalid characters with underscores.\n *\n * @param str - The string to convert.\n * @returns The valid variable name.\n */\nexport function makeValidVariableName(str: string): string {\n  return replaceAll(str, /[^a-zA-Z0-9_]/g, '_');\n}\n\n/**\n * Normalizes a string by converting it to the NFC form and replacing non-breaking spaces with regular spaces.\n *\n * @param str - The string to normalize.\n * @returns The normalized string.\n */\nexport function normalize(str: string): string {\n  return replaceAll(str, /\\u00A0|\\u202F/g, ' ').normalize('NFC');\n}\n\n/**\n * Replaces occurrences of strings in a given string based on a replacements map.\n *\n * @param str - The string to perform replacements on.\n * @param replacementsMap - An object mapping strings to their replacement values.\n * @returns The modified string with replacements applied.\n */\nexport function replace(str: string, replacementsMap: Record<string, string>): string {\n  const regExp = new RegExp(Object.keys(replacementsMap).map((source) => escapeRegExp(source)).join('|'), 'g');\n  return replaceAll(str, regExp, ({ substring: source }) => replacementsMap[source] ?? throwExpression(new Error(`Unexpected replacement source: ${source}`)));\n}\n\n/**\n * Replaces all occurrences of a search string or pattern with the results of an replacer function.\n *\n * @typeParam ReplaceGroupArgs - The type of additional arguments passed to the replacer function.\n * @param str - The string in which to perform replacements.\n * @param searchValue - The string or regular expression to search for.\n * @param replacer - A replacer function that generates replacement strings, or a string to replace with.\n * @returns The string with all replacements made.\n */\nexport function replaceAll<ReplaceGroupArgs extends string[]>(\n  str: string,\n  searchValue: RegExp | string,\n  replacer: Replacer<ReplaceGroupArgs>\n): string {\n  if (typeof replacer === 'undefined') {\n    return str;\n  }\n\n  if (searchValue instanceof RegExp && !searchValue.global) {\n    searchValue = new RegExp(searchValue.source, `${searchValue.flags}g`);\n  }\n\n  if (typeof replacer === 'string') {\n    return str.replaceAll(searchValue, replacer);\n  }\n\n  return str.replaceAll(searchValue, (substring: string, ...args: unknown[]) => {\n    const SOURCE_INDEX_OFFSET_FOR_GROUP_ARG = 2;\n    const hasGroupsArg = typeof args.at(-1) === 'object';\n    const sourceIndex = hasGroupsArg ? args.length - SOURCE_INDEX_OFFSET_FOR_GROUP_ARG : args.length - 1;\n\n    const commonArgs: ReplaceCommonArgs = {\n      groups: hasGroupsArg ? args.at(-1) as Record<string, string | undefined> : undefined,\n      missingGroupIndices: [],\n      offset: args.at(sourceIndex - 1) as number,\n      source: args.at(sourceIndex) as string,\n      substring\n    };\n\n    const groupArgs = args.slice(0, sourceIndex - 1).map((arg, index) => {\n      if (typeof arg === 'string') {\n        return arg;\n      }\n\n      if (typeof arg === 'undefined') {\n        commonArgs.missingGroupIndices.push(index);\n        return '';\n      }\n\n      throw new Error(`Unexpected argument type: ${typeof arg}`);\n    }) as ReplaceGroupArgs;\n\n    return (replacer(commonArgs, ...groupArgs) as string | undefined) ?? commonArgs.substring;\n  });\n}\n\n/**\n * Asynchronously replaces all occurrences of a search string or pattern with the results of an asynchronous replacer function.\n *\n * @typeParam ReplaceGroupArgs - The type of additional arguments passed to the replacer function.\n * @param str - The string in which to perform replacements.\n * @param searchValue - The string or regular expression to search for.\n * @param replacer - A synchronous/asynchronous function that generates replacement strings, or a string to replace with.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns A {@link Promise} that resolves to the string with all replacements made.\n */\nexport async function replaceAllAsync<ReplaceGroupArgs extends string[]>(\n  str: string,\n  searchValue: RegExp | string,\n  replacer: AsyncReplacer<ReplaceGroupArgs>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  if (typeof replacer === 'string') {\n    return replaceAll(str, searchValue, replacer);\n  }\n\n  const replacementAsyncFns: (() => Promise<StringReplacement>)[] = [];\n\n  replaceAll<ReplaceGroupArgs>(str, searchValue, (commonArgs, ...groupArgs) => {\n    replacementAsyncFns.push(() => resolveValue(replacer, abortSignal, commonArgs, ...groupArgs));\n    return '';\n  });\n\n  const replacements: StringReplacement[] = [];\n\n  for (const asyncFn of replacementAsyncFns) {\n    abortSignal.throwIfAborted();\n    replacements.push(await asyncFn());\n  }\n\n  abortSignal.throwIfAborted();\n  return replaceAll(str, searchValue, (args): string => replacements.shift() ?? args.substring);\n}\n\n/**\n * Trims the specified suffix from the end of a string.\n *\n * @param str - The string to trim.\n * @param suffix - The suffix to remove from the end of the string.\n * @param shouldValidate - If true, throws an error if the string does not end with the suffix.\n * @returns The trimmed string.\n * @throws If `validate` is true and the string does not end with the suffix.\n */\nexport function trimEnd(str: string, suffix: string, shouldValidate?: boolean): string {\n  if (str.endsWith(suffix)) {\n    return str.slice(0, -suffix.length);\n  }\n\n  if (shouldValidate) {\n    throw new Error(`String ${str} does not end with suffix ${suffix}`);\n  }\n\n  return str;\n}\n\n/**\n * Trims the specified prefix from the start of a string.\n *\n * @param str - The string to trim.\n * @param prefix - The prefix to remove from the start of the string.\n * @param validate - If true, throws an error if the string does not start with the prefix.\n * @returns The trimmed string.\n * @throws If `validate` is true and the string does not start with the prefix.\n */\nexport function trimStart(str: string, prefix: string, validate?: boolean): string {\n  if (str.startsWith(prefix)) {\n    return str.slice(prefix.length);\n  }\n\n  if (validate) {\n    throw new Error(`String ${str} does not start with prefix ${prefix}`);\n  }\n\n  return str;\n}\n\n/**\n * Unescapes a string by replacing escape sequences with their corresponding characters.\n *\n * @param str - The string to unescape.\n * @returns The unescaped string.\n */\nexport function unescape(str: string): string {\n  return replace(str, UNESCAPE_MAP);\n}\n\n/**\n * Unindents a string by removing a prefix from each line.\n *\n * @param text - The string to unindent.\n * @param prefix - The prefix to remove from each line.\n * @param shouldThrowIfNotIndented - If `true`, throws an error if a line is not indented with the prefix.\n * @returns The unindented string.\n */\nexport function unindent(text: string, prefix: string, shouldThrowIfNotIndented = false): string {\n  return text.split('\\n').map((line) => {\n    if (line.startsWith(prefix)) {\n      return line.slice(prefix.length);\n    }\n    if (shouldThrowIfNotIndented) {\n      throw new Error(`Line \"${line}\" is not indented with \"${prefix}\"`);\n    }\n    return line;\n  }).join('\\n');\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -3,23 +3,119 @@
|
|
|
3
3
|
*
|
|
4
4
|
* This module provides utility functions for processing code blocks in Obsidian.
|
|
5
5
|
*/
|
|
6
|
-
import type { App, MarkdownPostProcessorContext } from 'obsidian';
|
|
6
|
+
import type { App, MarkdownPostProcessorContext, MarkdownSectionInformation } from 'obsidian';
|
|
7
7
|
import type { ValueProvider } from '../ValueProvider.mjs';
|
|
8
8
|
/**
|
|
9
|
-
*
|
|
9
|
+
* Represents the information about a code block in a Markdown section.
|
|
10
|
+
*/
|
|
11
|
+
export interface CodeBlockMarkdownSectionInformation extends MarkdownSectionInformation {
|
|
12
|
+
/**
|
|
13
|
+
* The arguments of the code block.
|
|
14
|
+
*/
|
|
15
|
+
args: string;
|
|
16
|
+
/**
|
|
17
|
+
* The end delimiter of the code block.
|
|
18
|
+
*/
|
|
19
|
+
endDelimiter: string;
|
|
20
|
+
/**
|
|
21
|
+
* The language of the code block.
|
|
22
|
+
*/
|
|
23
|
+
language: string;
|
|
24
|
+
/**
|
|
25
|
+
* The prefix of the code block.
|
|
26
|
+
*/
|
|
27
|
+
prefix: string;
|
|
28
|
+
/**
|
|
29
|
+
* The start delimiter of the code block.
|
|
30
|
+
*/
|
|
31
|
+
startDelimiter: string;
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Represents the options for getting the information about a code block in a Markdown section.
|
|
35
|
+
*/
|
|
36
|
+
export interface GetCodeBlockSectionInfoOptions {
|
|
37
|
+
/**
|
|
38
|
+
* The Obsidian App object.
|
|
39
|
+
*/
|
|
40
|
+
app: App;
|
|
41
|
+
/**
|
|
42
|
+
* The MarkdownPostProcessorContext object.
|
|
43
|
+
*/
|
|
44
|
+
ctx: MarkdownPostProcessorContext;
|
|
45
|
+
/**
|
|
46
|
+
* The HTMLElement representing the code block.
|
|
47
|
+
*/
|
|
48
|
+
el: HTMLElement;
|
|
49
|
+
/**
|
|
50
|
+
* The source of the code block.
|
|
51
|
+
*/
|
|
52
|
+
source: string;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Represents the options for inserting text after a code block.
|
|
56
|
+
*/
|
|
57
|
+
export interface InsertCodeBlockOptions extends GetCodeBlockSectionInfoOptions {
|
|
58
|
+
/**
|
|
59
|
+
* The number of lines to offset the insertion by. Default is `0`.
|
|
60
|
+
*/
|
|
61
|
+
lineOffset?: number;
|
|
62
|
+
/**
|
|
63
|
+
* Whether to preserve the line prefix of the code block. Default is `false`.
|
|
64
|
+
*/
|
|
65
|
+
shouldPreserveLinePrefix?: boolean;
|
|
66
|
+
/**
|
|
67
|
+
* The text to insert after the code block.
|
|
68
|
+
*/
|
|
69
|
+
text: string;
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Represents the options for replacing a code block.
|
|
73
|
+
*/
|
|
74
|
+
export interface ReplaceCodeBlockOptions extends GetCodeBlockSectionInfoOptions {
|
|
75
|
+
/**
|
|
76
|
+
* The abort signal to control the execution of the function.
|
|
77
|
+
*/
|
|
78
|
+
abortSignal?: AbortSignal;
|
|
79
|
+
/**
|
|
80
|
+
* The provider that provides the new code block.
|
|
81
|
+
*/
|
|
82
|
+
codeBlockProvider: ValueProvider<string, [string]>;
|
|
83
|
+
/**
|
|
84
|
+
* Whether to preserve the line prefix of the code block. Default is `false`.
|
|
85
|
+
*/
|
|
86
|
+
shouldPreserveLinePrefix?: boolean;
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Gets the information about a code block in a Markdown section.
|
|
90
|
+
*
|
|
91
|
+
* @param options - The options for the function.
|
|
92
|
+
* @returns The information about the code block in the Markdown section.
|
|
93
|
+
*
|
|
94
|
+
* @throws If no suitable code block is found.
|
|
95
|
+
* @throws If multiple suitable code blocks are found. Happens when the code block is in a callout. Caused by the bug in Obsidian: {@link https://forum.obsidian.md/t/bug-getsectioninfo-is-inaccurate-inside-callouts/104289}
|
|
96
|
+
*/
|
|
97
|
+
export declare function getCodeBlockSectionInfo(options: GetCodeBlockSectionInfoOptions): Promise<CodeBlockMarkdownSectionInformation>;
|
|
98
|
+
/**
|
|
99
|
+
* Inserts text after the code block.
|
|
100
|
+
*
|
|
101
|
+
* @param options - The options for the function.
|
|
102
|
+
*/
|
|
103
|
+
export declare function insertAfterCodeBlock(options: InsertCodeBlockOptions): Promise<void>;
|
|
104
|
+
/**
|
|
105
|
+
* Inserts text before the code block.
|
|
106
|
+
*
|
|
107
|
+
* @param options - The options for the function.
|
|
108
|
+
*/
|
|
109
|
+
export declare function insertBeforeCodeBlock(options: InsertCodeBlockOptions): Promise<void>;
|
|
110
|
+
/**
|
|
111
|
+
* Removes the code block.
|
|
10
112
|
*
|
|
11
|
-
* @param
|
|
12
|
-
* @param el - The HTMLElement representing the code block.
|
|
13
|
-
* @returns The argument of the code block as a string, or null if no argument is found.
|
|
113
|
+
* @param options - The options for the function.
|
|
14
114
|
*/
|
|
15
|
-
export declare function
|
|
115
|
+
export declare function removeCodeBlock(options: GetCodeBlockSectionInfoOptions): Promise<void>;
|
|
16
116
|
/**
|
|
17
117
|
* Replaces the code block.
|
|
18
118
|
*
|
|
19
|
-
* @param
|
|
20
|
-
* @param ctx - The MarkdownPostProcessorContext object.
|
|
21
|
-
* @param el - The HTMLElement representing the code block.
|
|
22
|
-
* @param codeBlockProvider - The ValueProvider that provides the new code block.
|
|
23
|
-
* @param abortSignal - The abort signal to control the execution of the function.
|
|
119
|
+
* @param options - The options for the function.
|
|
24
120
|
*/
|
|
25
|
-
export declare function replaceCodeBlock(
|
|
121
|
+
export declare function replaceCodeBlock(options: ReplaceCodeBlockOptions): Promise<void>;
|
|
@@ -5,44 +5,179 @@ if you want to view the source, please visit the github repository of this plugi
|
|
|
5
5
|
|
|
6
6
|
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
|
|
7
7
|
|
|
8
|
-
import {
|
|
8
|
+
import { getFrontMatterInfo } from "obsidian";
|
|
9
9
|
import { abortSignalNever } from "../AbortController.mjs";
|
|
10
|
-
import {
|
|
11
|
-
|
|
10
|
+
import {
|
|
11
|
+
hasSingleOccurrence,
|
|
12
|
+
indent,
|
|
13
|
+
replaceAll,
|
|
14
|
+
unindent
|
|
15
|
+
} from "../String.mjs";
|
|
12
16
|
import { resolveValue } from "../ValueProvider.mjs";
|
|
13
17
|
import { process } from "./Vault.mjs";
|
|
14
|
-
function
|
|
15
|
-
const
|
|
16
|
-
|
|
17
|
-
|
|
18
|
+
async function getCodeBlockSectionInfo(options) {
|
|
19
|
+
const { app, ctx, el, source } = options;
|
|
20
|
+
const approximateSectionInfo = ctx.getSectionInfo(el) ?? await createApproximateSectionInfo(app, ctx);
|
|
21
|
+
const isInCallout = !!el.parentElement?.classList.contains("callout-content");
|
|
22
|
+
const language = getLanguageFromElement(el);
|
|
23
|
+
const sourceLines = source.split("\n");
|
|
24
|
+
const textLines = approximateSectionInfo.text.split("\n");
|
|
25
|
+
const potentialCodeBlockTextLines = textLines.map(
|
|
26
|
+
(line, index) => approximateSectionInfo.lineStart <= index && index <= approximateSectionInfo.lineEnd ? line : ""
|
|
27
|
+
);
|
|
28
|
+
const potentialCodeBlockText = potentialCodeBlockTextLines.join("\n");
|
|
29
|
+
const REG_EXP = /(?<=^|\n)(?<LinePrefix> {0,3}(?:> {1,3})*)(?<CodeBlockStartDelimiter>(?<CodeBlockStartDelimiterChar>[`~])(?:\k<CodeBlockStartDelimiterChar>{2,}))(?<CodeBlockLanguage>\S*)(?:[ \t]+(?<CodeBlockArgs>.*?)[ \t]+)?(?:\n(?<CodeBlockContent>(?:\n?\k<LinePrefix>.*)+?))?\n\k<LinePrefix>(?<CodeBlockEndDelimiter>\k<CodeBlockStartDelimiter>\k<CodeBlockStartDelimiterChar>*)[ \t]*(?:\n|$)/g;
|
|
30
|
+
let sectionInfo = null;
|
|
31
|
+
for (const match of potentialCodeBlockText.matchAll(REG_EXP)) {
|
|
32
|
+
if (!isSuitableCodeBlock(match, language, source, isInCallout)) {
|
|
33
|
+
continue;
|
|
34
|
+
}
|
|
35
|
+
if (sectionInfo) {
|
|
36
|
+
throw new Error("Multiple suitable code blocks found.");
|
|
37
|
+
}
|
|
38
|
+
sectionInfo = createSectionInfoFromMatch(potentialCodeBlockText, match, approximateSectionInfo, sourceLines);
|
|
18
39
|
}
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
const match = /^(?:`{3,}|~{3,})\S+\s+(?<Arguments>.*)$/.exec(codeBlockHeader);
|
|
22
|
-
if (!match) {
|
|
23
|
-
return [];
|
|
40
|
+
if (!sectionInfo) {
|
|
41
|
+
throw new Error("No suitable code block found.");
|
|
24
42
|
}
|
|
25
|
-
return
|
|
43
|
+
return sectionInfo;
|
|
44
|
+
}
|
|
45
|
+
async function insertAfterCodeBlock(options) {
|
|
46
|
+
const { app, ctx, lineOffset = 0, text } = options;
|
|
47
|
+
const sectionInfo = await getCodeBlockSectionInfo(options);
|
|
48
|
+
await process(app, ctx.sourcePath, (_abortSignal, content) => {
|
|
49
|
+
if (!hasSingleOccurrence(content, sectionInfo.text)) {
|
|
50
|
+
throw new Error("Multiple suitable code blocks found.");
|
|
51
|
+
}
|
|
52
|
+
const index = content.indexOf(sectionInfo.text);
|
|
53
|
+
const textBeforeSection = content.slice(0, index);
|
|
54
|
+
const linesBeforeSection = textBeforeSection.split("\n");
|
|
55
|
+
const insertLineIndex = linesBeforeSection.length + sectionInfo.lineEnd + lineOffset;
|
|
56
|
+
return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
async function insertBeforeCodeBlock(options) {
|
|
60
|
+
const { app, ctx, lineOffset = 0, text } = options;
|
|
61
|
+
const sectionInfo = await getCodeBlockSectionInfo(options);
|
|
62
|
+
await process(app, ctx.sourcePath, (_abortSignal, content) => {
|
|
63
|
+
if (!hasSingleOccurrence(content, sectionInfo.text)) {
|
|
64
|
+
throw new Error("Multiple suitable code blocks found.");
|
|
65
|
+
}
|
|
66
|
+
const index = content.indexOf(sectionInfo.text);
|
|
67
|
+
const textBeforeSection = content.slice(0, index);
|
|
68
|
+
const linesBeforeSection = textBeforeSection.split("\n");
|
|
69
|
+
const insertLineIndex = linesBeforeSection.length + sectionInfo.lineStart - lineOffset - 1;
|
|
70
|
+
return insertText(content, insertLineIndex, text, options.shouldPreserveLinePrefix);
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
async function removeCodeBlock(options) {
|
|
74
|
+
await replaceCodeBlock({
|
|
75
|
+
...options,
|
|
76
|
+
codeBlockProvider: ""
|
|
77
|
+
});
|
|
26
78
|
}
|
|
27
|
-
async function replaceCodeBlock(
|
|
28
|
-
|
|
79
|
+
async function replaceCodeBlock(options) {
|
|
80
|
+
const { app, codeBlockProvider, ctx } = options;
|
|
81
|
+
const abortSignal = options.abortSignal ?? abortSignalNever();
|
|
29
82
|
abortSignal.throwIfAborted();
|
|
30
|
-
const sectionInfo =
|
|
31
|
-
if (!sectionInfo) {
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
83
|
+
const sectionInfo = await getCodeBlockSectionInfo(options);
|
|
34
84
|
const lines = sectionInfo.text.split("\n");
|
|
35
|
-
const
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
85
|
+
const textBeforeCodeBlock = lines.slice(0, sectionInfo.lineStart).join("\n");
|
|
86
|
+
let oldCodeBlock = lines.slice(sectionInfo.lineStart, sectionInfo.lineEnd + 1).join("\n");
|
|
87
|
+
if (options.shouldPreserveLinePrefix) {
|
|
88
|
+
oldCodeBlock = unindent(oldCodeBlock, sectionInfo.prefix);
|
|
89
|
+
}
|
|
90
|
+
const textAfterCodeBlock = lines.slice(sectionInfo.lineEnd + 1).join("\n");
|
|
91
|
+
let newCodeBlock = await resolveValue(codeBlockProvider, abortSignal, oldCodeBlock);
|
|
39
92
|
abortSignal.throwIfAborted();
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
93
|
+
if (newCodeBlock && options.shouldPreserveLinePrefix) {
|
|
94
|
+
newCodeBlock = indent(newCodeBlock, sectionInfo.prefix);
|
|
95
|
+
}
|
|
96
|
+
const newSectionText = `${appendNewLine(textBeforeCodeBlock)}${appendNewLine(newCodeBlock)}${textAfterCodeBlock}`;
|
|
97
|
+
await process(app, ctx.sourcePath, (_abortSignal, content) => {
|
|
98
|
+
if (!hasSingleOccurrence(content, sectionInfo.text)) {
|
|
99
|
+
throw new Error("Multiple suitable code blocks found.");
|
|
100
|
+
}
|
|
101
|
+
return replaceAll(content, sectionInfo.text, newSectionText);
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
function appendNewLine(text) {
|
|
105
|
+
return text === "" ? "" : `${text}
|
|
106
|
+
`;
|
|
107
|
+
}
|
|
108
|
+
async function createApproximateSectionInfo(app, ctx) {
|
|
109
|
+
const sourceFile = app.vault.getFileByPath(ctx.sourcePath);
|
|
110
|
+
if (!sourceFile) {
|
|
111
|
+
throw new Error(`Source file ${ctx.sourcePath} not found.`);
|
|
112
|
+
}
|
|
113
|
+
const cache = app.metadataCache.getFileCache(sourceFile);
|
|
114
|
+
const frontmatterEndOffset = cache?.frontmatterPosition?.end.offset;
|
|
115
|
+
const content = await app.vault.cachedRead(sourceFile);
|
|
116
|
+
const contentStartOffset = frontmatterEndOffset === void 0 ? getFrontMatterInfo(content).contentStart : frontmatterEndOffset + 1;
|
|
117
|
+
const text = content.slice(contentStartOffset);
|
|
118
|
+
return {
|
|
119
|
+
lineEnd: text.split("\n").length - 1,
|
|
120
|
+
lineStart: 0,
|
|
121
|
+
text
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
function createSectionInfoFromMatch(potentialCodeBlockText, match, approximateSectionInfo, sourceLines) {
|
|
125
|
+
const linePrefix = match.groups?.["LinePrefix"] ?? "";
|
|
126
|
+
const codeBlockStartDelimiter = match.groups?.["CodeBlockStartDelimiter"] ?? "";
|
|
127
|
+
const codeBlockEndDelimiter = match.groups?.["CodeBlockEndDelimiter"] ?? "";
|
|
128
|
+
const codeBlockArgs = match.groups?.["CodeBlockArgs"] ?? "";
|
|
129
|
+
const language = match.groups?.["CodeBlockLanguage"] ?? "";
|
|
130
|
+
const previousText = potentialCodeBlockText.slice(0, match.index);
|
|
131
|
+
const previousTextLines = previousText.split("\n");
|
|
132
|
+
return {
|
|
133
|
+
args: codeBlockArgs,
|
|
134
|
+
endDelimiter: codeBlockEndDelimiter,
|
|
135
|
+
language,
|
|
136
|
+
lineEnd: previousTextLines.length + sourceLines.length,
|
|
137
|
+
lineStart: previousTextLines.length - 1,
|
|
138
|
+
prefix: linePrefix,
|
|
139
|
+
startDelimiter: codeBlockStartDelimiter,
|
|
140
|
+
text: approximateSectionInfo.text
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
function getLanguageFromElement(el) {
|
|
144
|
+
const BLOCK_LANGUAGE_PREFIX = "block-language-";
|
|
145
|
+
return Array.from(el.classList).find((cls) => cls.startsWith(BLOCK_LANGUAGE_PREFIX))?.slice(BLOCK_LANGUAGE_PREFIX.length) ?? "";
|
|
146
|
+
}
|
|
147
|
+
function insertText(content, insertLineIndex, text, shouldPreserveLinePrefix) {
|
|
148
|
+
const lines = content.split("\n");
|
|
149
|
+
const newLines = lines.slice();
|
|
150
|
+
const textLines = text.split("\n");
|
|
151
|
+
if (insertLineIndex < 0) {
|
|
152
|
+
insertLineIndex = 0;
|
|
153
|
+
}
|
|
154
|
+
if (insertLineIndex > lines.length) {
|
|
155
|
+
insertLineIndex = lines.length;
|
|
156
|
+
}
|
|
157
|
+
const PREFIX_LINE_REG_EXP = /^ {0,3}(?:> {1,3})*/g;
|
|
158
|
+
const match = (lines[insertLineIndex] ?? "").match(PREFIX_LINE_REG_EXP);
|
|
159
|
+
const linePrefix = match?.[0] ?? "";
|
|
160
|
+
newLines.splice(insertLineIndex, 0, ...shouldPreserveLinePrefix ? textLines.map((line) => indent(line, linePrefix)) : textLines);
|
|
161
|
+
return newLines.join("\n");
|
|
162
|
+
}
|
|
163
|
+
function isSuitableCodeBlock(match, language, source, isInCallout) {
|
|
164
|
+
const codeBlockLanguage = match.groups?.["CodeBlockLanguage"] ?? "";
|
|
165
|
+
if (codeBlockLanguage !== language) {
|
|
166
|
+
return false;
|
|
167
|
+
}
|
|
168
|
+
const linePrefix = match.groups?.["LinePrefix"] ?? "";
|
|
169
|
+
if (isInCallout && !linePrefix.includes("> ")) {
|
|
170
|
+
return false;
|
|
171
|
+
}
|
|
172
|
+
const codeBlockContent = match.groups?.["CodeBlockContent"] ?? "";
|
|
173
|
+
const cleanCodeBlockContent = codeBlockContent.split("\n").map((line) => line.slice(linePrefix.length)).join("\n");
|
|
174
|
+
return cleanCodeBlockContent === source;
|
|
43
175
|
}
|
|
44
176
|
export {
|
|
45
|
-
|
|
177
|
+
getCodeBlockSectionInfo,
|
|
178
|
+
insertAfterCodeBlock,
|
|
179
|
+
insertBeforeCodeBlock,
|
|
180
|
+
removeCodeBlock,
|
|
46
181
|
replaceCodeBlock
|
|
47
182
|
};
|
|
48
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL01hcmtkb3duQ29kZUJsb2NrUHJvY2Vzc29yLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEBwYWNrYWdlRG9jdW1lbnRhdGlvblxuICpcbiAqIFRoaXMgbW9kdWxlIHByb3ZpZGVzIHV0aWxpdHkgZnVuY3Rpb25zIGZvciBwcm9jZXNzaW5nIGNvZGUgYmxvY2tzIGluIE9ic2lkaWFuLlxuICovXG5cbmltcG9ydCB0eXBlIHtcbiAgQXBwLFxuICBNYXJrZG93blBvc3RQcm9jZXNzb3JDb250ZXh0XG59IGZyb20gJ29ic2lkaWFuJztcblxuaW1wb3J0IHsgcGFyc2UgfSBmcm9tICdzaGVsbC1xdW90ZSc7XG5cbmltcG9ydCB0eXBlIHsgVmFsdWVQcm92aWRlciB9IGZyb20gJy4uL1ZhbHVlUHJvdmlkZXIudHMnO1xuXG5pbXBvcnQgeyBhYm9ydFNpZ25hbE5ldmVyIH0gZnJvbSAnLi4vQWJvcnRDb250cm9sbGVyLnRzJztcbmltcG9ydCB7IHRocm93RXhwcmVzc2lvbiB9IGZyb20gJy4uL0Vycm9yLnRzJztcbmltcG9ydCB7IHJlcGxhY2VBbGwgfSBmcm9tICcuLi9TdHJpbmcudHMnO1xuaW1wb3J0IHsgcmVzb2x2ZVZhbHVlIH0gZnJvbSAnLi4vVmFsdWVQcm92aWRlci50cyc7XG5pbXBvcnQgeyBwcm9jZXNzIH0gZnJvbSAnLi9WYXVsdC50cyc7XG5cbi8qKlxuICogUmV0cmlldmVzIHRoZSBhcmd1bWVudCBvZiBhIGNvZGUgYmxvY2sgZnJvbSB0aGUgZ2l2ZW4gTWFya2Rvd25Qb3N0UHJvY2Vzc29yQ29udGV4dCBhbmQgSFRNTEVsZW1lbnQuXG4gKlxuICogQHBhcmFtIGN0eCAtIFRoZSBNYXJrZG93blBvc3RQcm9jZXNzb3JDb250ZXh0IG9iamVjdC5cbiAqIEBwYXJhbSBlbCAtIFRoZSBIVE1MRWxlbWVudCByZXByZXNlbnRpbmcgdGhlIGNvZGUgYmxvY2suXG4gKiBAcmV0dXJucyBUaGUgYXJndW1lbnQgb2YgdGhlIGNvZGUgYmxvY2sgYXMgYSBzdHJpbmcsIG9yIG51bGwgaWYgbm8gYXJndW1lbnQgaXMgZm91bmQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb2RlQmxvY2tBcmd1bWVudHMoY3R4OiBNYXJrZG93blBvc3RQcm9jZXNzb3JDb250ZXh0LCBlbDogSFRNTEVsZW1lbnQpOiBzdHJpbmdbXSB7XG4gIGNvbnN0IHNlY3Rpb25JbmZvID0gY3R4LmdldFNlY3Rpb25JbmZvKGVsKTtcbiAgaWYgKCFzZWN0aW9uSW5mbykge1xuICAgIHJldHVybiBbXTtcbiAgfVxuICBjb25zdCBsaW5lcyA9IHNlY3Rpb25JbmZvLnRleHQuc3BsaXQoJ1xcbicpO1xuICBjb25zdCBjb2RlQmxvY2tIZWFkZXIgPSBsaW5lc1tzZWN0aW9uSW5mby5saW5lU3RhcnRdID8/IHRocm93RXhwcmVzc2lvbihuZXcgRXJyb3IoJ0NvZGUgYmxvY2sgaGVhZGVyIG5vdCBmb3VuZCcpKTtcbiAgY29uc3QgbWF0Y2ggPSAvXig/OmB7Myx9fH57Myx9KVxcUytcXHMrKD88QXJndW1lbnRzPi4qKSQvLmV4ZWMoY29kZUJsb2NrSGVhZGVyKTtcbiAgaWYgKCFtYXRjaCkge1xuICAgIHJldHVybiBbXTtcbiAgfVxuICByZXR1cm4gcGFyc2UobWF0Y2guZ3JvdXBzPy5bJ0FyZ3VtZW50cyddID8/ICcnKS5tYXAoU3RyaW5nKTtcbn1cblxuLyoqXG4gKiBSZXBsYWNlcyB0aGUgY29kZSBibG9jay5cbiAqXG4gKiBAcGFyYW0gYXBwIC0gVGhlIE9ic2lkaWFuIEFwcCBvYmplY3QuXG4gKiBAcGFyYW0gY3R4IC0gVGhlIE1hcmtkb3duUG9zdFByb2Nlc3NvckNvbnRleHQgb2JqZWN0LlxuICogQHBhcmFtIGVsIC0gVGhlIEhUTUxFbGVtZW50IHJlcHJlc2VudGluZyB0aGUgY29kZSBibG9jay5cbiAqIEBwYXJhbSBjb2RlQmxvY2tQcm92aWRlciAtIFRoZSBWYWx1ZVByb3ZpZGVyIHRoYXQgcHJvdmlkZXMgdGhlIG5ldyBjb2RlIGJsb2NrLlxuICogQHBhcmFtIGFib3J0U2lnbmFsIC0gVGhlIGFib3J0IHNpZ25hbCB0byBjb250cm9sIHRoZSBleGVjdXRpb24gb2YgdGhlIGZ1bmN0aW9uLlxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcmVwbGFjZUNvZGVCbG9jayhcbiAgYXBwOiBBcHAsXG4gIGN0eDogTWFya2Rvd25Qb3N0UHJvY2Vzc29yQ29udGV4dCxcbiAgZWw6IEhUTUxFbGVtZW50LFxuICBjb2RlQmxvY2tQcm92aWRlcjogVmFsdWVQcm92aWRlcjxzdHJpbmcsIFtzdHJpbmddPixcbiAgYWJvcnRTaWduYWw/OiBBYm9ydFNpZ25hbFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGFib3J0U2lnbmFsID8/PSBhYm9ydFNpZ25hbE5ldmVyKCk7XG4gIGFib3J0U2lnbmFsLnRocm93SWZBYm9ydGVkKCk7XG5cbiAgY29uc3Qgc2VjdGlvbkluZm8gPSBjdHguZ2V0U2VjdGlvbkluZm8oZWwpO1xuICBpZiAoIXNlY3Rpb25JbmZvKSB7XG4gICAgcmV0dXJuO1xuICB9XG4gIGNvbnN0IGxpbmVzID0gc2VjdGlvbkluZm8udGV4dC5zcGxpdCgnXFxuJyk7XG4gIGNvbnN0IHByZWZpeCA9IGxpbmVzLnNsaWNlKDAsIHNlY3Rpb25JbmZvLmxpbmVTdGFydCkuam9pbignXFxuJyk7XG4gIGNvbnN0IG9sZENvZGVCbG9jayA9IGxpbmVzLnNsaWNlKHNlY3Rpb25JbmZvLmxpbmVTdGFydCwgc2VjdGlvbkluZm8ubGluZUVuZCArIDEpLmpvaW4oJ1xcbicpO1xuICBjb25zdCBzdWZmaXggPSBsaW5lcy5zbGljZShzZWN0aW9uSW5mby5saW5lRW5kICsgMSkuam9pbignXFxuJyk7XG4gIGNvbnN0IG5ld0NvZGVCbG9jayA9IGF3YWl0IHJlc29sdmVWYWx1ZShjb2RlQmxvY2tQcm92aWRlciwgYWJvcnRTaWduYWwsIG9sZENvZGVCbG9jayk7XG4gIGFib3J0U2lnbmFsLnRocm93SWZBYm9ydGVkKCk7XG4gIGNvbnN0IG5ld1NlY3Rpb25UZXh0ID0gYCR7cHJlZml4fVxcbiR7bmV3Q29kZUJsb2NrfSR7bmV3Q29kZUJsb2NrID8gJ1xcbicgOiAnJ30ke3N1ZmZpeH1gO1xuICBhd2FpdCBwcm9jZXNzKGFwcCwgY3R4LnNvdXJjZVBhdGgsIChfYWJvcnRTaWduYWwsIGNvbnRlbnQpID0+IHJlcGxhY2VBbGwoY29udGVudCwgc2VjdGlvbkluZm8udGV4dCwgbmV3U2VjdGlvblRleHQpKTtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7QUFXQSxTQUFTLGFBQWE7QUFJdEIsU0FBUyx3QkFBd0I7QUFDakMsU0FBUyx1QkFBdUI7QUFDaEMsU0FBUyxrQkFBa0I7QUFDM0IsU0FBUyxvQkFBb0I7QUFDN0IsU0FBUyxlQUFlO0FBU2pCLFNBQVMsc0JBQXNCLEtBQW1DLElBQTJCO0FBQ2xHLFFBQU0sY0FBYyxJQUFJLGVBQWUsRUFBRTtBQUN6QyxNQUFJLENBQUMsYUFBYTtBQUNoQixXQUFPLENBQUM7QUFBQSxFQUNWO0FBQ0EsUUFBTSxRQUFRLFlBQVksS0FBSyxNQUFNLElBQUk7QUFDekMsUUFBTSxrQkFBa0IsTUFBTSxZQUFZLFNBQVMsS0FBSyxnQkFBZ0IsSUFBSSxNQUFNLDZCQUE2QixDQUFDO0FBQ2hILFFBQU0sUUFBUSwwQ0FBMEMsS0FBSyxlQUFlO0FBQzVFLE1BQUksQ0FBQyxPQUFPO0FBQ1YsV0FBTyxDQUFDO0FBQUEsRUFDVjtBQUNBLFNBQU8sTUFBTSxNQUFNLFNBQVMsV0FBVyxLQUFLLEVBQUUsRUFBRSxJQUFJLE1BQU07QUFDNUQ7QUFXQSxlQUFzQixpQkFDcEIsS0FDQSxLQUNBLElBQ0EsbUJBQ0EsYUFDZTtBQUNmLGtCQUFnQixpQkFBaUI7QUFDakMsY0FBWSxlQUFlO0FBRTNCLFFBQU0sY0FBYyxJQUFJLGVBQWUsRUFBRTtBQUN6QyxNQUFJLENBQUMsYUFBYTtBQUNoQjtBQUFBLEVBQ0Y7QUFDQSxRQUFNLFFBQVEsWUFBWSxLQUFLLE1BQU0sSUFBSTtBQUN6QyxRQUFNLFNBQVMsTUFBTSxNQUFNLEdBQUcsWUFBWSxTQUFTLEVBQUUsS0FBSyxJQUFJO0FBQzlELFFBQU0sZUFBZSxNQUFNLE1BQU0sWUFBWSxXQUFXLFlBQVksVUFBVSxDQUFDLEVBQUUsS0FBSyxJQUFJO0FBQzFGLFFBQU0sU0FBUyxNQUFNLE1BQU0sWUFBWSxVQUFVLENBQUMsRUFBRSxLQUFLLElBQUk7QUFDN0QsUUFBTSxlQUFlLE1BQU0sYUFBYSxtQkFBbUIsYUFBYSxZQUFZO0FBQ3BGLGNBQVksZUFBZTtBQUMzQixRQUFNLGlCQUFpQixHQUFHLE1BQU07QUFBQSxFQUFLLFlBQVksR0FBRyxlQUFlLE9BQU8sRUFBRSxHQUFHLE1BQU07QUFDckYsUUFBTSxRQUFRLEtBQUssSUFBSSxZQUFZLENBQUMsY0FBYyxZQUFZLFdBQVcsU0FBUyxZQUFZLE1BQU0sY0FBYyxDQUFDO0FBQ3JIOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
183
|
+
//# 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 { getFrontMatterInfo } from 'obsidian';\n\nimport type { ValueProvider } from '../ValueProvider.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  hasSingleOccurrence,\n  indent,\n  replaceAll,\n  unindent\n} from '../String.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport { process } from './Vault.ts';\n\n/**\n * Represents the information about a code block in a Markdown section.\n */\nexport interface CodeBlockMarkdownSectionInformation extends MarkdownSectionInformation {\n  /**\n   * The arguments of the code block.\n   */\n  args: string;\n\n  /**\n   * The end delimiter of the code block.\n   */\n  endDelimiter: string;\n\n  /**\n   * The language of the code block.\n   */\n  language: string;\n\n  /**\n   * The prefix of the code block.\n   */\n  prefix: string;\n\n  /**\n   * The start delimiter of the code block.\n   */\n  startDelimiter: string;\n}\n\n/**\n * Represents the options for getting the information about a code block in a Markdown section.\n */\nexport interface GetCodeBlockSectionInfoOptions {\n  /**\n   * The Obsidian App object.\n   */\n  app: App;\n\n  /**\n   * The MarkdownPostProcessorContext object.\n   */\n  ctx: MarkdownPostProcessorContext;\n\n  /**\n   * The HTMLElement representing the code block.\n   */\n  el: HTMLElement;\n\n  /**\n   * The source of the code block.\n   */\n  source: string;\n}\n\n/**\n * Represents the options for inserting text after a code block.\n */\nexport interface InsertCodeBlockOptions extends GetCodeBlockSectionInfoOptions {\n  /**\n   * The 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   * The text to insert after the code block.\n   */\n  text: string;\n}\n\n/**\n * Represents the options for replacing a code block.\n */\nexport interface ReplaceCodeBlockOptions extends GetCodeBlockSectionInfoOptions {\n  /**\n   * The abort signal to control the execution of the function.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * The provider that provides the new code block.\n   */\n  codeBlockProvider: ValueProvider<string, [string]>;\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 *\n * @throws If no suitable code block is found.\n * @throws If multiple suitable code blocks are found. Happens when the code block is in a callout. Caused by the bug in Obsidian: {@link https://forum.obsidian.md/t/bug-getsectioninfo-is-inaccurate-inside-callouts/104289}\n */\nexport async function getCodeBlockSectionInfo(options: GetCodeBlockSectionInfoOptions): Promise<CodeBlockMarkdownSectionInformation> {\n  const { app, ctx, el, source } = options;\n\n  const approximateSectionInfo = ctx.getSectionInfo(el) ?? await createApproximateSectionInfo(app, ctx);\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 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>.*?)[ \\t]+)?(?:\\n(?<CodeBlockContent>(?:\\n?\\k<LinePrefix>.*)+?))?\\n\\k<LinePrefix>(?<CodeBlockEndDelimiter>\\k<CodeBlockStartDelimiter>\\k<CodeBlockStartDelimiterChar>*)[ \\t]*(?:\\n|$)/g;\n\n  let sectionInfo: CodeBlockMarkdownSectionInformation | 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 (sectionInfo) {\n      throw new Error('Multiple suitable code blocks found.');\n    }\n\n    sectionInfo = createSectionInfoFromMatch(potentialCodeBlockText, match, approximateSectionInfo, sourceLines);\n  }\n\n  if (!sectionInfo) {\n    throw new Error('No suitable code block found.');\n  }\n\n  return sectionInfo;\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  const sectionInfo = await getCodeBlockSectionInfo(options);\n  await process(app, ctx.sourcePath, (_abortSignal, content) => {\n    if (!hasSingleOccurrence(content, sectionInfo.text)) {\n      throw new Error('Multiple suitable code blocks found.');\n    }\n\n    const index = content.indexOf(sectionInfo.text);\n    const textBeforeSection = content.slice(0, index);\n    const linesBeforeSection = textBeforeSection.split('\\n');\n    const insertLineIndex = linesBeforeSection.length + sectionInfo.lineEnd + lineOffset;\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  const sectionInfo = await getCodeBlockSectionInfo(options);\n  await process(app, ctx.sourcePath, (_abortSignal, content) => {\n    if (!hasSingleOccurrence(content, sectionInfo.text)) {\n      throw new Error('Multiple suitable code blocks found.');\n    }\n\n    const index = content.indexOf(sectionInfo.text);\n    const textBeforeSection = content.slice(0, index);\n    const linesBeforeSection = textBeforeSection.split('\\n');\n    const insertLineIndex = linesBeforeSection.length + sectionInfo.lineStart - lineOffset - 1;\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: GetCodeBlockSectionInfoOptions): Promise<void> {\n  await replaceCodeBlock({\n    ...options,\n    codeBlockProvider: ''\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\n  const abortSignal = options.abortSignal ?? abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const sectionInfo = await getCodeBlockSectionInfo(options);\n\n  const lines = sectionInfo.text.split('\\n');\n  const textBeforeCodeBlock = lines.slice(0, sectionInfo.lineStart).join('\\n');\n  let oldCodeBlock = lines.slice(sectionInfo.lineStart, sectionInfo.lineEnd + 1).join('\\n');\n  if (options.shouldPreserveLinePrefix) {\n    oldCodeBlock = unindent(oldCodeBlock, sectionInfo.prefix);\n  }\n  const textAfterCodeBlock = lines.slice(sectionInfo.lineEnd + 1).join('\\n');\n  let newCodeBlock = await resolveValue(codeBlockProvider, abortSignal, oldCodeBlock);\n  abortSignal.throwIfAborted();\n\n  if (newCodeBlock && options.shouldPreserveLinePrefix) {\n    newCodeBlock = indent(newCodeBlock, sectionInfo.prefix);\n  }\n\n  const newSectionText = `${appendNewLine(textBeforeCodeBlock)}${appendNewLine(newCodeBlock)}${textAfterCodeBlock}`;\n  await process(app, ctx.sourcePath, (_abortSignal, content) => {\n    if (!hasSingleOccurrence(content, sectionInfo.text)) {\n      throw new Error('Multiple suitable code blocks found.');\n    }\n    return replaceAll(content, sectionInfo.text, newSectionText);\n  });\n}\n\nfunction appendNewLine(text: string): string {\n  return text === '' ? '' : `${text}\\n`;\n}\n\nasync function createApproximateSectionInfo(app: App, ctx: MarkdownPostProcessorContext): Promise<MarkdownSectionInformation> {\n  const sourceFile = app.vault.getFileByPath(ctx.sourcePath);\n  if (!sourceFile) {\n    throw new Error(`Source file ${ctx.sourcePath} not found.`);\n  }\n  const cache = app.metadataCache.getFileCache(sourceFile);\n  const frontmatterEndOffset = cache?.frontmatterPosition?.end.offset;\n  const content = await app.vault.cachedRead(sourceFile);\n  const contentStartOffset = frontmatterEndOffset === undefined ? getFrontMatterInfo(content).contentStart : frontmatterEndOffset + 1;\n  const text = content.slice(contentStartOffset);\n  return {\n    lineEnd: text.split('\\n').length - 1,\n    lineStart: 0,\n    text\n  };\n}\n\nfunction createSectionInfoFromMatch(\n  potentialCodeBlockText: string,\n  match: RegExpMatchArray,\n  approximateSectionInfo: MarkdownSectionInformation,\n  sourceLines: string[]\n): CodeBlockMarkdownSectionInformation {\n  const linePrefix = match.groups?.['LinePrefix'] ?? '';\n  const codeBlockStartDelimiter = match.groups?.['CodeBlockStartDelimiter'] ?? '';\n  const codeBlockEndDelimiter = match.groups?.['CodeBlockEndDelimiter'] ?? '';\n  const codeBlockArgs = match.groups?.['CodeBlockArgs'] ?? '';\n  const language = match.groups?.['CodeBlockLanguage'] ?? '';\n\n  const previousText = potentialCodeBlockText.slice(0, match.index);\n  const previousTextLines = previousText.split('\\n');\n\n  return {\n    args: codeBlockArgs,\n    endDelimiter: codeBlockEndDelimiter,\n    language,\n    lineEnd: previousTextLines.length + sourceLines.length,\n    lineStart: previousTextLines.length - 1,\n    prefix: linePrefix,\n    startDelimiter: codeBlockStartDelimiter,\n    text: approximateSectionInfo.text\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": []
}

|