obsidian-dev-utils 46.3.2 → 46.4.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 +9 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/AsyncWithNotice.cjs +13 -3
- package/dist/lib/cjs/obsidian/AsyncWithNotice.d.cts +8 -0
- package/dist/lib/cjs/obsidian/Components/SettingComponents/CodeHighlighterComponent.cjs +2 -2
- package/dist/lib/cjs/obsidian/Loop.cjs +5 -1
- package/dist/lib/cjs/obsidian/Loop.d.cts +10 -6
- package/dist/lib/cjs/obsidian/MetadataCache.cjs +4 -3
- package/dist/lib/cjs/obsidian/MetadataCache.d.cts +11 -2
- package/dist/lib/cjs/obsidian/Queue.cjs +10 -2
- package/dist/lib/cjs/obsidian/Queue.d.cts +8 -0
- package/dist/lib/cjs/obsidian/Vault.cjs +4 -2
- package/dist/lib/cjs/obsidian/Vault.d.cts +4 -0
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/AsyncWithNotice.d.mts +8 -0
- package/dist/lib/esm/obsidian/AsyncWithNotice.mjs +13 -3
- package/dist/lib/esm/obsidian/Components/SettingComponents/CodeHighlighterComponent.mjs +2 -2
- package/dist/lib/esm/obsidian/Loop.d.mts +10 -6
- package/dist/lib/esm/obsidian/Loop.mjs +5 -1
- package/dist/lib/esm/obsidian/MetadataCache.d.mts +11 -2
- package/dist/lib/esm/obsidian/MetadataCache.mjs +4 -3
- package/dist/lib/esm/obsidian/Queue.d.mts +8 -0
- package/dist/lib/esm/obsidian/Queue.mjs +10 -2
- package/dist/lib/esm/obsidian/Vault.d.mts +4 -0
- package/dist/lib/esm/obsidian/Vault.mjs +4 -2
- package/package.json +1 -1
|
@@ -25,6 +25,10 @@ export interface AddToQueueAndWaitOptions {
|
|
|
25
25
|
* Optional name of the operation.
|
|
26
26
|
*/
|
|
27
27
|
operationName?: string;
|
|
28
|
+
/**
|
|
29
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
30
|
+
*/
|
|
31
|
+
shouldShowTimeoutNotice?: boolean;
|
|
28
32
|
/**
|
|
29
33
|
* Optional stack trace.
|
|
30
34
|
*/
|
|
@@ -54,6 +58,10 @@ export interface AddToQueueOptions {
|
|
|
54
58
|
* Optional name of the operation.
|
|
55
59
|
*/
|
|
56
60
|
operationName?: string;
|
|
61
|
+
/**
|
|
62
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
63
|
+
*/
|
|
64
|
+
shouldShowTimeoutNotice?: boolean;
|
|
57
65
|
/**
|
|
58
66
|
* Optional stack trace.
|
|
59
67
|
*/
|
|
@@ -279,6 +279,7 @@ async function process(app, pathOrFile, newContentProvider, options = {}) {
|
|
|
279
279
|
const DEFAULT_RETRY_OPTIONS = {
|
|
280
280
|
shouldFailOnMissingFile: true,
|
|
281
281
|
shouldLockEditorWhileProcessing: true,
|
|
282
|
+
shouldShowTimeoutNotice: true,
|
|
282
283
|
// eslint-disable-next-line no-magic-numbers -- Default value.
|
|
283
284
|
timeoutInMilliseconds: 500
|
|
284
285
|
};
|
|
@@ -343,7 +344,8 @@ async function process(app, pathOrFile, newContentProvider, options = {}) {
|
|
|
343
344
|
}
|
|
344
345
|
},
|
|
345
346
|
operationName: (0, import_i18n.t)(($) => $.obsidianDevUtils.vault.processFile, { filePath: path }),
|
|
346
|
-
retryOptions: fullOptions
|
|
347
|
+
retryOptions: fullOptions,
|
|
348
|
+
shouldShowTimeoutNotice: fullOptions.shouldShowTimeoutNotice
|
|
347
349
|
});
|
|
348
350
|
} finally {
|
|
349
351
|
activeLeafChangeEventRef?.e.offref(activeLeafChangeEventRef);
|
|
@@ -428,4 +430,4 @@ async function invokeFileActionSafe(app, pathOrFile, fileAction) {
|
|
|
428
430
|
renameSafe,
|
|
429
431
|
saveNote
|
|
430
432
|
});
|
|
431
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  EventRef,\n  ListedFiles,\n  TFile,\n  TFolder\n} from 'obsidian';\n\nimport { MarkdownView } from 'obsidian';\nimport {\n  parentFolderPath,\n  ViewType\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport { noopAsync } from '../Function.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join\n} from '../Path.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  lockEditor,\n  unlockEditor\n} from './Editor.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport { t } from './i18n/i18n.ts';\n\n/**\n * Options for {@link process}.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * Whether to fail if the file is missing or deleted. Default is `true`.\n   */\n  shouldFailOnMissingFile?: boolean;\n\n  /**\n   * Whether to lock the editor while processing the file. Applicable only for markdown files. Default is `true`.\n   */\n  shouldLockEditorWhileProcessing?: boolean;\n}\n\n/**\n * Copies a file safely in the vault.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to copy.\n * @param newPath - The new path to copy the file to.\n * @returns A {@link Promise} that resolves to the new path of the copied file.\n */\nexport async function copySafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const file = getFile(app, oldPathOrFile);\n\n  if (file.path === newPath) {\n    return newPath;\n  }\n\n  const newFolderPath = parentFolderPath(newPath);\n  await createFolderSafe(app, newFolderPath);\n\n  const newAvailablePath = getAvailablePath(app, newPath);\n\n  try {\n    await app.vault.copy(file, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.createFolder(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n    return true;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n *\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A {@link Promise} that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = getFileOrNull(app, path);\n  if (file) {\n    return noopAsync;\n  }\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  try {\n    await app.vault.create(path, '');\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  return async () => {\n    file = getFile(app, path);\n    if (!file.deleted) {\n      await app.fileManager.trashFile(file);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link Promise} that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = getFolderOrNull(app, path);\n  if (folder) {\n    return noopAsync;\n  }\n\n  const folderPath = parentFolderPath(path);\n  await createTempFolder(app, folderPath);\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  await createFolderSafe(app, path);\n\n  return async () => {\n    folder = getFolder(app, path);\n    if (!folder.deleted) {\n      await app.fileManager.trashFile(folder);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Gets an available path for a file in the vault.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get an available path for.\n * @returns The available path for the file.\n */\nexport function getAvailablePath(app: App, path: string): string {\n  const ext = extname(path);\n  return app.vault.getAvailablePath(join(dirname(path), basename(path, ext)), ext.slice(1));\n}\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Retrieves an array of all note files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of all note files in the vault sorted by file path.\n */\nexport function getNoteFilesSorted(app: App): TFile[] {\n  return app.vault.getAllLoadedFiles().filter((file) => isFile(file) && isNote(app, file)).sort((a, b) => a.path.localeCompare(b.path)) as TFile[];\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns The safe rename path for the file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrFile: PathOrFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrFile);\n\n  if (app.vault.adapter.insensitive) {\n    let folderPath = dirname(newPath);\n    let nonExistingPath = basename(newPath);\n    let folder: null | TFolder;\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- There is no elegant way to perform infinite loops.\n    while (true) {\n      folder = getFolderOrNull(app, folderPath, true);\n      if (folder) {\n        break;\n      }\n      nonExistingPath = join(basename(folderPath), nonExistingPath);\n      folderPath = dirname(folderPath);\n    }\n    newPath = join(folder.getParentPrefix(), nonExistingPath);\n  }\n\n  if (oldPath.toLowerCase() === newPath.toLowerCase()) {\n    return newPath;\n  }\n\n  return getAvailablePath(app, newPath);\n}\n\n/**\n * Invokes a function with the file system lock.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to execute the function with the file system lock of.\n * @param fn - The function to execute.\n */\nexport async function invokeWithFileSystemLock(app: App, pathOrFile: PathOrFile, fn: (content: string) => void): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  await app.vault.process(file, (content) => {\n    fn(content);\n    return content;\n  });\n}\n\n/**\n * Checks if a folder is empty.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the folder is empty.\n */\nexport async function isEmptyFolder(app: App, pathOrFolder: PathOrFolder): Promise<boolean> {\n  const listedFiles = await listSafe(app, getPath(app, pathOrFolder));\n  return listedFiles.files.length === 0 && listedFiles.folders.length === 0;\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to list.\n * @returns A {@link Promise} that resolves to a {@link ListedFiles} object containing the listed files and folders.\n */\nexport async function listSafe(app: App, pathOrFolder: PathOrFolder): Promise<ListedFiles> {\n  const path = getPath(app, pathOrFolder);\n  const EMPTY = { files: [], folders: [] };\n\n  if ((await app.vault.adapter.stat(path))?.type !== 'folder') {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param options - Optional options for processing/retrying the operation.\n *\n * @returns A {@link Promise} that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function process(\n  app: App,\n  pathOrFile: PathOrFile,\n  newContentProvider: ValueProvider<null | string, [string]>,\n  options: ProcessOptions = {}\n): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS = {\n    shouldFailOnMissingFile: true,\n    shouldLockEditorWhileProcessing: true,\n    // eslint-disable-next-line no-magic-numbers -- Default value.\n    timeoutInMilliseconds: 500\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  const abortController = new AbortController();\n  fullOptions.abortSignal = abortSignalAny(fullOptions.abortSignal, abortController.signal);\n  const path = getPath(app, pathOrFile);\n\n  let activeLeafChangeEventRef: EventRef | null = null;\n\n  if (fullOptions.shouldLockEditorWhileProcessing) {\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    }\n\n    activeLeafChangeEventRef = app.workspace.on('active-leaf-change', (leaf) => {\n      if (leaf?.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    });\n  }\n\n  try {\n    await retryWithTimeoutNotice({\n      async operationFn(abortSignal) {\n        abortSignal.throwIfAborted();\n\n        const oldContent = await readSafe(app, pathOrFile);\n        abortSignal.throwIfAborted();\n\n        if (oldContent === null) {\n          return handleMissingFile();\n        }\n\n        const newContent = await resolveValue(newContentProvider, abortSignal, oldContent);\n        abortSignal.throwIfAborted();\n\n        if (newContent === null) {\n          return false;\n        }\n\n        let isSuccess = true;\n        const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n          abortSignal.throwIfAborted();\n          await app.vault.process(file, (content) => {\n            abortSignal.throwIfAborted();\n            if (content !== oldContent) {\n              getLibDebugger('Vault:process')('Content has changed since it was read. Retrying...', {\n                actualContent: content,\n                expectedContent: oldContent,\n                path: file.path\n              });\n              isSuccess = false;\n              return content;\n            }\n\n            return newContent;\n          });\n\n          abortSignal.throwIfAborted();\n        });\n\n        if (!doesFileExist) {\n          return handleMissingFile();\n        }\n\n        return isSuccess;\n\n        function handleMissingFile(): boolean {\n          if (fullOptions.shouldFailOnMissingFile) {\n            throw new Error(`File '${path}' not found`);\n          }\n          return true;\n        }\n      },\n      operationName: t(($) => $.obsidianDevUtils.vault.processFile, { filePath: path }),\n      retryOptions: fullOptions\n    });\n  } finally {\n    activeLeafChangeEventRef?.e.offref(activeLeafChangeEventRef);\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        unlockEditor(leaf.view.editor);\n      }\n    }\n  }\n}\n\n/**\n * Reads the content of a file safely from the vault.\n *\n * It covers the case when the file was removed during the reading.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to read.\n * @returns A {@link Promise} that resolves to the content of the file or `null` if the file is missing or deleted.\n */\nexport async function readSafe(app: App, pathOrFile: PathOrFile): Promise<null | string> {\n  let content: null | string = null;\n  await invokeFileActionSafe(app, pathOrFile, async (file) => {\n    await saveNote(app, file);\n    content = await app.vault.read(file);\n  });\n  return content;\n}\n\n/**\n * Renames a file safely in the vault.\n * If the new path already exists, the file will be renamed to an available path.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A {@link Promise} that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const oldFile = getFile(app, oldPathOrFile, false, true);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrFile, newPath);\n\n  if (oldFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldFile.path !== newPath) {\n      await app.vault.rename(oldFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.vault.rename(oldFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Saves the specified note in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The note to be saved.\n * @returns A {@link Promise} that resolves when the note is saved.\n */\nexport async function saveNote(app: App, pathOrFile: PathOrFile): Promise<void> {\n  if (!isMarkdownFile(app, pathOrFile)) {\n    return;\n  }\n\n  const path = getPath(app, pathOrFile);\n\n  for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n    if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path && leaf.view.dirty) {\n      await leaf.view.save();\n    }\n  }\n}\n\nasync function invokeFileActionSafe(app: App, pathOrFile: PathOrFile, fileAction: (file: TFile) => Promise<void>): Promise<boolean> {\n  const path = getPath(app, pathOrFile);\n  let file = getFileOrNull(app, path);\n  if (!file || file.deleted) {\n    return false;\n  }\n  try {\n    await fileAction(file);\n    return true;\n  } catch (e) {\n    file = getFileOrNull(app, path);\n    if (!file || file.deleted) {\n      return false;\n    }\n    throw e;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
433
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  EventRef,\n  ListedFiles,\n  TFile,\n  TFolder\n} from 'obsidian';\n\nimport { MarkdownView } from 'obsidian';\nimport {\n  parentFolderPath,\n  ViewType\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport { noopAsync } from '../Function.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join\n} from '../Path.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  lockEditor,\n  unlockEditor\n} from './Editor.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport { t } from './i18n/i18n.ts';\n\n/**\n * Options for {@link process}.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * Whether to fail if the file is missing or deleted. Default is `true`.\n   */\n  shouldFailOnMissingFile?: boolean;\n\n  /**\n   * Whether to lock the editor while processing the file. Applicable only for markdown files. Default is `true`.\n   */\n  shouldLockEditorWhileProcessing?: boolean;\n\n  /**\n   * Whether to show a timeout notice. Default is `true`.\n   */\n  shouldShowTimeoutNotice?: boolean;\n}\n\n/**\n * Copies a file safely in the vault.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to copy.\n * @param newPath - The new path to copy the file to.\n * @returns A {@link Promise} that resolves to the new path of the copied file.\n */\nexport async function copySafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const file = getFile(app, oldPathOrFile);\n\n  if (file.path === newPath) {\n    return newPath;\n  }\n\n  const newFolderPath = parentFolderPath(newPath);\n  await createFolderSafe(app, newFolderPath);\n\n  const newAvailablePath = getAvailablePath(app, newPath);\n\n  try {\n    await app.vault.copy(file, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.createFolder(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n    return true;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n *\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A {@link Promise} that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = getFileOrNull(app, path);\n  if (file) {\n    return noopAsync;\n  }\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  try {\n    await app.vault.create(path, '');\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  return async () => {\n    file = getFile(app, path);\n    if (!file.deleted) {\n      await app.fileManager.trashFile(file);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link Promise} that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = getFolderOrNull(app, path);\n  if (folder) {\n    return noopAsync;\n  }\n\n  const folderPath = parentFolderPath(path);\n  await createTempFolder(app, folderPath);\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  await createFolderSafe(app, path);\n\n  return async () => {\n    folder = getFolder(app, path);\n    if (!folder.deleted) {\n      await app.fileManager.trashFile(folder);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Gets an available path for a file in the vault.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get an available path for.\n * @returns The available path for the file.\n */\nexport function getAvailablePath(app: App, path: string): string {\n  const ext = extname(path);\n  return app.vault.getAvailablePath(join(dirname(path), basename(path, ext)), ext.slice(1));\n}\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Retrieves an array of all note files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of all note files in the vault sorted by file path.\n */\nexport function getNoteFilesSorted(app: App): TFile[] {\n  return app.vault.getAllLoadedFiles().filter((file) => isFile(file) && isNote(app, file)).sort((a, b) => a.path.localeCompare(b.path)) as TFile[];\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns The safe rename path for the file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrFile: PathOrFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrFile);\n\n  if (app.vault.adapter.insensitive) {\n    let folderPath = dirname(newPath);\n    let nonExistingPath = basename(newPath);\n    let folder: null | TFolder;\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- There is no elegant way to perform infinite loops.\n    while (true) {\n      folder = getFolderOrNull(app, folderPath, true);\n      if (folder) {\n        break;\n      }\n      nonExistingPath = join(basename(folderPath), nonExistingPath);\n      folderPath = dirname(folderPath);\n    }\n    newPath = join(folder.getParentPrefix(), nonExistingPath);\n  }\n\n  if (oldPath.toLowerCase() === newPath.toLowerCase()) {\n    return newPath;\n  }\n\n  return getAvailablePath(app, newPath);\n}\n\n/**\n * Invokes a function with the file system lock.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to execute the function with the file system lock of.\n * @param fn - The function to execute.\n */\nexport async function invokeWithFileSystemLock(app: App, pathOrFile: PathOrFile, fn: (content: string) => void): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  await app.vault.process(file, (content) => {\n    fn(content);\n    return content;\n  });\n}\n\n/**\n * Checks if a folder is empty.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A {@link Promise} that resolves to a boolean indicating whether the folder is empty.\n */\nexport async function isEmptyFolder(app: App, pathOrFolder: PathOrFolder): Promise<boolean> {\n  const listedFiles = await listSafe(app, getPath(app, pathOrFolder));\n  return listedFiles.files.length === 0 && listedFiles.folders.length === 0;\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to list.\n * @returns A {@link Promise} that resolves to a {@link ListedFiles} object containing the listed files and folders.\n */\nexport async function listSafe(app: App, pathOrFolder: PathOrFolder): Promise<ListedFiles> {\n  const path = getPath(app, pathOrFolder);\n  const EMPTY = { files: [], folders: [] };\n\n  if ((await app.vault.adapter.stat(path))?.type !== 'folder') {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param options - Optional options for processing/retrying the operation.\n *\n * @returns A {@link Promise} that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function process(\n  app: App,\n  pathOrFile: PathOrFile,\n  newContentProvider: ValueProvider<null | string, [string]>,\n  options: ProcessOptions = {}\n): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS = {\n    shouldFailOnMissingFile: true,\n    shouldLockEditorWhileProcessing: true,\n    shouldShowTimeoutNotice: true,\n    // eslint-disable-next-line no-magic-numbers -- Default value.\n    timeoutInMilliseconds: 500\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  const abortController = new AbortController();\n  fullOptions.abortSignal = abortSignalAny(fullOptions.abortSignal, abortController.signal);\n  const path = getPath(app, pathOrFile);\n\n  let activeLeafChangeEventRef: EventRef | null = null;\n\n  if (fullOptions.shouldLockEditorWhileProcessing) {\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    }\n\n    activeLeafChangeEventRef = app.workspace.on('active-leaf-change', (leaf) => {\n      if (leaf?.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    });\n  }\n\n  try {\n    await retryWithTimeoutNotice({\n      async operationFn(abortSignal) {\n        abortSignal.throwIfAborted();\n\n        const oldContent = await readSafe(app, pathOrFile);\n        abortSignal.throwIfAborted();\n\n        if (oldContent === null) {\n          return handleMissingFile();\n        }\n\n        const newContent = await resolveValue(newContentProvider, abortSignal, oldContent);\n        abortSignal.throwIfAborted();\n\n        if (newContent === null) {\n          return false;\n        }\n\n        let isSuccess = true;\n        const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n          abortSignal.throwIfAborted();\n          await app.vault.process(file, (content) => {\n            abortSignal.throwIfAborted();\n            if (content !== oldContent) {\n              getLibDebugger('Vault:process')('Content has changed since it was read. Retrying...', {\n                actualContent: content,\n                expectedContent: oldContent,\n                path: file.path\n              });\n              isSuccess = false;\n              return content;\n            }\n\n            return newContent;\n          });\n\n          abortSignal.throwIfAborted();\n        });\n\n        if (!doesFileExist) {\n          return handleMissingFile();\n        }\n\n        return isSuccess;\n\n        function handleMissingFile(): boolean {\n          if (fullOptions.shouldFailOnMissingFile) {\n            throw new Error(`File '${path}' not found`);\n          }\n          return true;\n        }\n      },\n      operationName: t(($) => $.obsidianDevUtils.vault.processFile, { filePath: path }),\n      retryOptions: fullOptions,\n      shouldShowTimeoutNotice: fullOptions.shouldShowTimeoutNotice\n    });\n  } finally {\n    activeLeafChangeEventRef?.e.offref(activeLeafChangeEventRef);\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        unlockEditor(leaf.view.editor);\n      }\n    }\n  }\n}\n\n/**\n * Reads the content of a file safely from the vault.\n *\n * It covers the case when the file was removed during the reading.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to read.\n * @returns A {@link Promise} that resolves to the content of the file or `null` if the file is missing or deleted.\n */\nexport async function readSafe(app: App, pathOrFile: PathOrFile): Promise<null | string> {\n  let content: null | string = null;\n  await invokeFileActionSafe(app, pathOrFile, async (file) => {\n    await saveNote(app, file);\n    content = await app.vault.read(file);\n  });\n  return content;\n}\n\n/**\n * Renames a file safely in the vault.\n * If the new path already exists, the file will be renamed to an available path.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A {@link Promise} that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const oldFile = getFile(app, oldPathOrFile, false, true);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrFile, newPath);\n\n  if (oldFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldFile.path !== newPath) {\n      await app.vault.rename(oldFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.vault.rename(oldFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Saves the specified note in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The note to be saved.\n * @returns A {@link Promise} that resolves when the note is saved.\n */\nexport async function saveNote(app: App, pathOrFile: PathOrFile): Promise<void> {\n  if (!isMarkdownFile(app, pathOrFile)) {\n    return;\n  }\n\n  const path = getPath(app, pathOrFile);\n\n  for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n    if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path && leaf.view.dirty) {\n      await leaf.view.save();\n    }\n  }\n}\n\nasync function invokeFileActionSafe(app: App, pathOrFile: PathOrFile, fileAction: (file: TFile) => Promise<void>): Promise<boolean> {\n  const path = getPath(app, pathOrFile);\n  let file = getFileOrNull(app, path);\n  if (!file || file.deleted) {\n    return false;\n  }\n  try {\n    await fileAction(file);\n    return true;\n  } catch (e) {\n    file = getFileOrNull(app, path);\n    if (!file || file.deleted) {\n      return false;\n    }\n    throw e;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -19,6 +19,10 @@ export interface ProcessOptions extends RetryOptions {
|
|
|
19
19
|
* Whether to lock the editor while processing the file. Applicable only for markdown files. Default is `true`.
|
|
20
20
|
*/
|
|
21
21
|
shouldLockEditorWhileProcessing?: boolean;
|
|
22
|
+
/**
|
|
23
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
24
|
+
*/
|
|
25
|
+
shouldShowTimeoutNotice?: boolean;
|
|
22
26
|
}
|
|
23
27
|
/**
|
|
24
28
|
* Copies a file safely in the vault.
|
package/dist/lib/esm/Library.mjs
CHANGED
|
@@ -19,7 +19,7 @@ if you want to view the source, please visit the github repository of this plugi
|
|
|
19
19
|
globalThis.process = browserProcess;
|
|
20
20
|
})();
|
|
21
21
|
|
|
22
|
-
const LIBRARY_VERSION = "46.
|
|
22
|
+
const LIBRARY_VERSION = "46.4.0";
|
|
23
23
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
|
24
24
|
const LIBRARY_STYLES = ".obsidian-dev-utils.code-highlighter-component textarea, .obsidian-dev-utils.code-highlighter-component pre, .obsidian-dev-utils.code-highlighter-component code {\n font-family: var(--font-monospace);\n line-height: var(--line-height-normal);\n margin: 0;\n}\n.obsidian-dev-utils.code-highlighter-component textarea, .obsidian-dev-utils.code-highlighter-component code {\n font-size: var(--code-size);\n}\n.obsidian-dev-utils.code-highlighter-component textarea {\n background: transparent;\n color: transparent;\n z-index: 2;\n width: 20em;\n height: 10em;\n}\n.obsidian-dev-utils.code-highlighter-component pre {\n position: absolute;\n pointer-events: none;\n border: var(--input-border-width) solid transparent;\n overflow: auto;\n inset: 0;\n padding: var(--size-4-1) var(--size-4-2);\n z-index: 1;\n}\n.obsidian-dev-utils.code-highlighter-component pre::after {\n content: \"\";\n display: block;\n height: var(--bottom-gap, 0);\n}\n.obsidian-dev-utils.code-highlighter-component pre.is-placeholder {\n opacity: 0.6;\n}\n.obsidian-dev-utils.code-highlighter-component code {\n display: block;\n padding: 0;\n}\n\n.obsidian-dev-utils input[type=url] {\n height: var(--input-height);\n}\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=tel],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=url],\n.obsidian-dev-utils input[type=week] {\n -webkit-app-region: no-drag;\n background: var(--background-modifier-form-field);\n border: var(--input-border-width) solid var(--background-modifier-border);\n color: var(--text-normal);\n font-family: inherit;\n padding: var(--size-4-1) var(--size-4-2);\n font-size: var(--font-ui-small);\n border-radius: var(--input-radius);\n outline: none;\n}\n@media (hover: hover) {\n .obsidian-dev-utils input[type=month]:hover,\n .obsidian-dev-utils input[type=tel]:hover,\n .obsidian-dev-utils input[type=time]:hover,\n .obsidian-dev-utils input[type=url]:hover,\n .obsidian-dev-utils input[type=week]:hover {\n border-color: var(--background-modifier-border-hover);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n }\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus,\n.obsidian-dev-utils input[type=tel]:active,\n.obsidian-dev-utils input[type=tel]:focus,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus {\n border-color: var(--background-modifier-border-focus);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus, .obsidian-dev-utils input[type=month]:focus-visible,\n.obsidian-dev-utils input[type=tel]:active,\n.obsidian-dev-utils input[type=tel]:focus,\n.obsidian-dev-utils input[type=tel]:focus-visible,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=time]:focus-visible,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=url]:focus-visible,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus,\n.obsidian-dev-utils input[type=week]:focus-visible {\n box-shadow: 0 0 0 2px var(--background-modifier-border-focus);\n}\n.obsidian-dev-utils input[type=month]::placeholder,\n.obsidian-dev-utils input[type=tel]::placeholder,\n.obsidian-dev-utils input[type=time]::placeholder,\n.obsidian-dev-utils input[type=url]::placeholder,\n.obsidian-dev-utils input[type=week]::placeholder {\n color: var(--text-faint);\n}\n.mod-rtl input[type=month],\n.mod-rtl input[type=time],\n.mod-rtl input[type=week],\n.is-rtl input[type=month],\n.is-rtl input[type=time],\n.is-rtl input[type=week],\n.rtl input[type=month],\n.rtl input[type=time],\n.rtl input[type=week] {\n direction: rtl;\n}\n.mod-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.rtl input[type=month]::-webkit-calendar-picker-indicator,\n.rtl input[type=time]::-webkit-calendar-picker-indicator,\n.rtl input[type=week]::-webkit-calendar-picker-indicator {\n right: var(--size-4-1);\n left: auto;\n}\n\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=week] {\n font-variant-numeric: tabular-nums;\n position: relative;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-text {\n color: var(--text-faint);\n padding-inline-end: 0;\n}\n.obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n position: absolute;\n left: var(--size-4-1);\n right: auto;\n opacity: 0.5;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:focus {\n background-color: var(--text-selection);\n color: var(--text-normal);\n cursor: text;\n}\n.mod-rtl .obsidian-dev-utils input[type=month], .is-rtl .obsidian-dev-utils input[type=month], .rtl .obsidian-dev-utils input[type=month],\n.mod-rtl .obsidian-dev-utils input[type=time],\n.is-rtl .obsidian-dev-utils input[type=time],\n.rtl .obsidian-dev-utils input[type=time],\n.mod-rtl .obsidian-dev-utils input[type=week],\n.is-rtl .obsidian-dev-utils input[type=week],\n.rtl .obsidian-dev-utils input[type=week] {\n direction: rtl;\n}\n.mod-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .is-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n left: auto;\n right: var(--size-4-1);\n}\n\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=month],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=time],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {\n padding-inline-start: var(--size-4-6);\n}\n\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator {\n margin-inline-start: 0;\n}\n\n.obsidian-dev-utilsprogress.loop {\n min-width: 200px;\n}\n\n.obsidian-dev-utils.modal-container .ok-button {\n margin-right: 10px;\n margin-top: 20px;\n}\n\n.obsidian-dev-utils .multiple-dropdown-component select,\n.obsidian-dev-utils .multiple-dropdown-component select:focus,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown {\n height: auto;\n padding-top: 3px;\n}\n.obsidian-dev-utils .multiple-dropdown-component select option:checked,\n.obsidian-dev-utils .multiple-dropdown-component select:focus option:checked,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown option:checked {\n background-color: #1967d2;\n color: #fff;\n}\n\n.obsidian-dev-utils.plugin-settings-tab a:focus {\n outline: 2px solid var(--link-color);\n}\n\n.obsidian-dev-utils.prompt-modal .text-box {\n width: 100%;\n}\n\n.obsidian-dev-utils.tri-state-checkbox-component input[type=checkbox]:indeterminate {\n appearance: checkbox;\n}\n\n.obsidian-dev-utils :invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n.obsidian-dev-utils input.metadata-input-text:active:invalid, .obsidian-dev-utils input.metadata-input-text:focus-visible:invalid, .obsidian-dev-utils input.metadata-input-text:focus:invalid,\n.obsidian-dev-utils input[type=date]:active:invalid,\n.obsidian-dev-utils input[type=date]:focus-visible:invalid,\n.obsidian-dev-utils input[type=date]:focus:invalid,\n.obsidian-dev-utils input[type=datetime-local]:active:invalid,\n.obsidian-dev-utils input[type=datetime-local]:focus-visible:invalid,\n.obsidian-dev-utils input[type=datetime-local]:focus:invalid,\n.obsidian-dev-utils input[type=email]:active:invalid,\n.obsidian-dev-utils input[type=email]:focus-visible:invalid,\n.obsidian-dev-utils input[type=email]:focus:invalid,\n.obsidian-dev-utils input[type=number]:active:invalid,\n.obsidian-dev-utils input[type=number]:focus-visible:invalid,\n.obsidian-dev-utils input[type=number]:focus:invalid,\n.obsidian-dev-utils input[type=password]:active:invalid,\n.obsidian-dev-utils input[type=password]:focus-visible:invalid,\n.obsidian-dev-utils input[type=password]:focus:invalid,\n.obsidian-dev-utils input[type=search]:active:invalid,\n.obsidian-dev-utils input[type=search]:focus-visible:invalid,\n.obsidian-dev-utils input[type=search]:focus:invalid,\n.obsidian-dev-utils input[type=text]:active:invalid,\n.obsidian-dev-utils input[type=text]:focus-visible:invalid,\n.obsidian-dev-utils input[type=text]:focus:invalid,\n.obsidian-dev-utils textarea:active:invalid,\n.obsidian-dev-utils textarea:focus-visible:invalid,\n.obsidian-dev-utils textarea:focus:invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n.obsidian-dev-utils.setting-component-wrapper {\n position: relative;\n display: inline-flex;\n}\n.obsidian-dev-utils.overlay-validator {\n caret-color: transparent;\n cursor: default;\n position: absolute;\n background-color: transparent;\n border: none;\n outline: none;\n pointer-events: none;\n z-index: 9999;\n left: 0;\n top: 0;\n width: 100%;\n height: 100%;\n}\n.obsidian-dev-utils.tooltip.tooltip-validator {\n position: absolute;\n top: calc(100% + 8px);\n width: max-content;\n}\n\n/*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/code-highlighter-component.scss%22,%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/loop.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/plugin-settings-tab.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/tri-state-checkbox-component.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AAEI;EACE;EACA;EACA;;AAGF;EACE;;AAGF;EACE;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA;;AAGF;EACE;;AAGF;EACE;EACA;;;ACzCJ;EACE;;AAGF;AAAA;AAAA;AAAA;AAAA;EAKE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC7DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;ACjDJ;EACE;;;ACAA;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;EACE;;;ACDF;EACE;;;ACDF;EACE;;;ACEJ;EAJA;;AAoBI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EApBJ;;AA0BA;EACE;EACA;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20&.code-highlighter-component%20%7B%5Cn%20%20%20%20textarea,%20pre,%20code%20%7B%5Cn%20%20%20%20%20%20font-family:%20var(--font-monospace);%5Cn%20%20%20%20%20%20line-height:%20var(--line-height-normal);%5Cn%20%20%20%20%20%20margin:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20textarea,%20code%20%7B%5Cn%20%20%20%20%20%20font-size:%20var(--code-size);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20textarea%20%7B%5Cn%20%20%20%20%20%20background:%20transparent;%5Cn%20%20%20%20%20%20color:%20transparent;%5Cn%20%20%20%20%20%20z-index:%202;%5Cn%20%20%20%20%20%20width:%2020em;%5Cn%20%20%20%20%20%20height:%2010em;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20%20%20border:%20var(--input-border-width)%20solid%20transparent;%5Cn%20%20%20%20%20%20overflow:%20auto;%5Cn%20%20%20%20%20%20inset:%200;%5Cn%20%20%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20%20%20z-index:%201;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre::after%20%7B%5Cn%20%20%20%20%20%20content:%20%5C%22%5C%22;%5Cn%20%20%20%20%20%20display:%20block;%5Cn%20%20%20%20%20%20height:%20var(--bottom-gap,%200);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre.is-placeholder%20%7B%5Cn%20%20%20%20%20%20opacity:%200.6;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20code%20%7B%5Cn%20%20%20%20%20%20display:%20block;%5Cn%20%20%20%20%20%20padding:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='tel'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&progress.loop%20%7B%5Cn%20%20%20%20min-width:%20200px;%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%20%20%20%20%20%20padding-top:%203px;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.plugin-settings-tab%20%7B%5Cn%20%20%20%20a:focus%20%7B%5Cn%20%20%20%20%20%20outline:%202px%20solid%20var(--link-color);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cr%5Cn%20%20&.tri-state-checkbox-component%20%7B%5Cr%5Cn%20%20%20%20input%5Btype='checkbox'%5D:indeterminate%20%7B%5Cr%5Cn%20%20%20%20%20%20appearance:%20checkbox;%5Cr%5Cn%20%20%20%20%7D%5Cr%5Cn%20%20%7D%5Cr%5Cn%7D%5Cr%5Cn%22,%22@mixin%20invalid%20%7B%5Cn%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%7D%5Cn%5Cn.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20@include%20invalid;%5Cn%20%20%7D%5Cn%5Cn%20%20input.metadata-input-text,%5Cn%20%20input%5Btype='date'%5D,%5Cn%20%20input%5Btype='datetime-local'%5D,%5Cn%20%20input%5Btype='email'%5D,%5Cn%20%20input%5Btype='number'%5D,%5Cn%20%20input%5Btype='password'%5D,%5Cn%20%20input%5Btype='search'%5D,%5Cn%20%20input%5Btype='text'%5D,%5Cn%20%20textarea%20%7B%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus-visible,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20&:invalid%20%7B%5Cn%20%20%20%20%20%20%20%20@include%20invalid;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20&.setting-component-wrapper%20%7B%5Cn%20%20%20%20position:%20relative;%5Cn%20%20%20%20display:%20inline-flex;%5Cn%20%20%7D%5Cn%5Cn%20%20&.overlay-validator%20%7B%5Cn%20%20%20%20caret-color:%20transparent;%5Cn%20%20%20%20cursor:%20default;%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20background-color:%20transparent;%5Cn%20%20%20%20border:%20none;%5Cn%20%20%20%20outline:%20none;%5Cn%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20z-index:%209999;%5Cn%20%20%20%20left:%200;%5Cn%20%20%20%20top:%200;%5Cn%20%20%20%20width:%20100%25;%5Cn%20%20%20%20height:%20100%25;%5Cn%20%20%7D%5Cn%5Cn%20%20&.tooltip.tooltip-validator%20%7B%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20top:%20calc(100%25%20+%208px);%5Cn%20%20%20%20width:%20max-content;%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */\n";
|
|
25
25
|
export {
|
|
@@ -24,6 +24,10 @@ export interface RetryWithTimeoutNoticeOptions {
|
|
|
24
24
|
* The retry options.
|
|
25
25
|
*/
|
|
26
26
|
retryOptions?: RetryOptions;
|
|
27
|
+
/**
|
|
28
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
29
|
+
*/
|
|
30
|
+
shouldShowTimeoutNotice?: boolean;
|
|
27
31
|
/**
|
|
28
32
|
* The stack trace of the source function.
|
|
29
33
|
*/
|
|
@@ -48,6 +52,10 @@ export interface RunWithTimeoutNoticeOptions<Result> {
|
|
|
48
52
|
* The name of the operation.
|
|
49
53
|
*/
|
|
50
54
|
operationName?: string;
|
|
55
|
+
/**
|
|
56
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
57
|
+
*/
|
|
58
|
+
shouldShowTimeoutNotice?: boolean;
|
|
51
59
|
/**
|
|
52
60
|
* The stack trace of the source function.
|
|
53
61
|
*/
|
|
@@ -24,17 +24,18 @@ import {
|
|
|
24
24
|
retryWithTimeout,
|
|
25
25
|
runWithTimeout
|
|
26
26
|
} from "../Async.mjs";
|
|
27
|
+
import { getDebugger } from "../Debug.mjs";
|
|
27
28
|
import { t } from "./i18n/i18n.mjs";
|
|
28
29
|
async function retryWithTimeoutNotice(options) {
|
|
29
30
|
return retryWithTimeout({
|
|
30
31
|
...options,
|
|
31
|
-
onTimeout: onTimeoutNotice
|
|
32
|
+
onTimeout: options.shouldShowTimeoutNotice ? onTimeoutNotice : onTimeoutWithoutNotice
|
|
32
33
|
});
|
|
33
34
|
}
|
|
34
35
|
async function runWithTimeoutNotice(options) {
|
|
35
36
|
return runWithTimeout({
|
|
36
37
|
...options,
|
|
37
|
-
onTimeout: onTimeoutNotice
|
|
38
|
+
onTimeout: options.shouldShowTimeoutNotice ? onTimeoutNotice : onTimeoutWithoutNotice
|
|
38
39
|
});
|
|
39
40
|
}
|
|
40
41
|
function onTimeoutNotice(ctx) {
|
|
@@ -79,8 +80,17 @@ function onTimeoutNotice(ctx) {
|
|
|
79
80
|
runningTimeEl.textContent = String(runningTimeInMilliseconds);
|
|
80
81
|
}
|
|
81
82
|
}
|
|
83
|
+
function onTimeoutWithoutNotice(ctx) {
|
|
84
|
+
const startTime = Math.trunc(performance.now() - ctx.duration);
|
|
85
|
+
ctx.onOperationCompleted(() => {
|
|
86
|
+
getDebugger("AsyncWithNotice:onTimeoutWithoutNotice")("Operation completed after timeout", {
|
|
87
|
+
operationName: ctx.operationName,
|
|
88
|
+
totalDuration: Math.trunc(performance.now() - startTime)
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
}
|
|
82
92
|
export {
|
|
83
93
|
retryWithTimeoutNotice,
|
|
84
94
|
runWithTimeoutNotice
|
|
85
95
|
};
|
|
86
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
96
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/AsyncWithNotice.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Provides a utility to execute an asynchronous function with a notice.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport type {\n  RetryOptions,\n  TimeoutContext\n} from '../Async.ts';\n\nimport {\n  retryWithTimeout,\n  runWithTimeout\n} from '../Async.ts';\nimport { getDebugger } from '../Debug.ts';\nimport { t } from './i18n/i18n.ts';\n\n/**\n * Options for {@link retryWithTimeoutNotice}.\n */\nexport interface RetryWithTimeoutNoticeOptions {\n  /**\n   * The operation function to execute.\n   *\n   * @param abortSignal - The abort signal to listen to.\n   * @returns The result of the function.\n   */\n  operationFn(this: void, abortSignal: AbortSignal): Promisable<boolean>;\n\n  /**\n   * The name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * The retry options.\n   */\n  retryOptions?: RetryOptions;\n\n  /**\n   * Whether to show a timeout notice. Default is `true`.\n   */\n  shouldShowTimeoutNotice?: boolean;\n\n  /**\n   * The stack trace of the source function.\n   */\n  stackTrace?: string;\n}\n\n/**\n * Options for {@link runWithTimeout}.\n */\nexport interface RunWithTimeoutNoticeOptions<Result> {\n  /**\n   * The context of the function.\n   */\n  context?: unknown;\n\n  /**\n   * The operation function to execute.\n   *\n   * @param abortSignal - The abort signal to listen to.\n   * @returns The result of the function.\n   */\n  operationFn(abortSignal: AbortSignal): Promisable<Result>;\n\n  /**\n   * The name of the operation.\n   */\n  operationName?: string;\n\n  /**\n   * Whether to show a timeout notice. Default is `true`.\n   */\n  shouldShowTimeoutNotice?: boolean;\n\n  /**\n   * The stack trace of the source function.\n   */\n  stackTrace?: string;\n\n  /**\n   * The maximum time to wait in milliseconds.\n   */\n  timeoutInMilliseconds: number;\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached and displays a notice if the function times out.\n *\n * @param options - The options for the function.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeoutNotice(options: RetryWithTimeoutNoticeOptions): Promise<void> {\n  return retryWithTimeout({\n    ...options,\n    onTimeout: options.shouldShowTimeoutNotice ? onTimeoutNotice : onTimeoutWithoutNotice\n  });\n}\n\n/**\n * Executes a function with a timeout and displays a notice if the function times out.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param options - The options for the function.\n * @returns The result of the function.\n */\nexport async function runWithTimeoutNotice<Result>(options: RunWithTimeoutNoticeOptions<Result>): Promise<Result> {\n  return runWithTimeout({\n    ...options,\n    onTimeout: options.shouldShowTimeoutNotice ? onTimeoutNotice : onTimeoutWithoutNotice\n  });\n}\n\nfunction onTimeoutNotice(ctx: TimeoutContext): void {\n  const startTime = Math.trunc(performance.now() - ctx.duration);\n  let runningTimeEl: HTMLSpanElement;\n  let intervalId: number;\n  const SECOND_IN_MILLISECONDS = 1000;\n\n  const notice = new Notice(createFragment((f) => {\n    if (ctx.operationName) {\n      f.appendText(t(($) => $.obsidianDevUtils.asyncWithNotice.operation));\n      f.appendText(': ');\n      f.appendText(ctx.operationName);\n      f.createEl('br');\n    }\n    f.appendText(t(($) => $.obsidianDevUtils.asyncWithNotice.timedOut, { duration: ctx.duration }));\n    f.createEl('br');\n    f.appendText(t(($) => $.obsidianDevUtils.asyncWithNotice.runningFor));\n    f.appendText(' ');\n    runningTimeEl = f.createSpan();\n    f.appendText(' ');\n    f.appendText(t(($) => $.obsidianDevUtils.asyncWithNotice.milliseconds));\n    f.createEl('br');\n    f.appendText(t(($) => $.obsidianDevUtils.asyncWithNotice.terminateOperation));\n    f.createEl('br');\n    const button = f.createEl('button', {\n      text: t(($) => $.obsidianDevUtils.buttons.cancel)\n    });\n    button.addEventListener('click', () => {\n      ctx.terminateOperation();\n      clearInterval(intervalId);\n      notice.hide();\n    });\n  }));\n\n  updateRunningTime();\n  intervalId = window.setInterval(updateRunningTime, SECOND_IN_MILLISECONDS);\n\n  ctx.onOperationCompleted(() => {\n    clearInterval(intervalId);\n    notice.hide();\n  });\n\n  function updateRunningTime(): void {\n    const runningTimeInMilliseconds = Math.max(ctx.duration, Math.round((performance.now() - startTime) / SECOND_IN_MILLISECONDS) * SECOND_IN_MILLISECONDS);\n    runningTimeEl.textContent = String(runningTimeInMilliseconds);\n  }\n}\n\nfunction onTimeoutWithoutNotice(ctx: TimeoutContext): void {\n  const startTime = Math.trunc(performance.now() - ctx.duration);\n\n  ctx.onOperationCompleted(() => {\n    getDebugger('AsyncWithNotice:onTimeoutWithoutNotice')('Operation completed after timeout', {\n      operationName: ctx.operationName,\n      totalDuration: Math.trunc(performance.now() - startTime)\n    });\n  });\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAQA,SAAS,cAAc;AAOvB;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,mBAAmB;AAC5B,SAAS,SAAS;AA+ElB,eAAsB,uBAAuB,SAAuD;AAClG,SAAO,iBAAiB;AAAA,IACtB,GAAG;AAAA,IACH,WAAW,QAAQ,0BAA0B,kBAAkB;AAAA,EACjE,CAAC;AACH;AASA,eAAsB,qBAA6B,SAA+D;AAChH,SAAO,eAAe;AAAA,IACpB,GAAG;AAAA,IACH,WAAW,QAAQ,0BAA0B,kBAAkB;AAAA,EACjE,CAAC;AACH;AAEA,SAAS,gBAAgB,KAA2B;AAClD,QAAM,YAAY,KAAK,MAAM,YAAY,IAAI,IAAI,IAAI,QAAQ;AAC7D,MAAI;AACJ,MAAI;AACJ,QAAM,yBAAyB;AAE/B,QAAM,SAAS,IAAI,OAAO,eAAe,CAAC,MAAM;AAC9C,QAAI,IAAI,eAAe;AACrB,QAAE,WAAW,EAAE,CAAC,MAAM,EAAE,iBAAiB,gBAAgB,SAAS,CAAC;AACnE,QAAE,WAAW,IAAI;AACjB,QAAE,WAAW,IAAI,aAAa;AAC9B,QAAE,SAAS,IAAI;AAAA,IACjB;AACA,MAAE,WAAW,EAAE,CAAC,MAAM,EAAE,iBAAiB,gBAAgB,UAAU,EAAE,UAAU,IAAI,SAAS,CAAC,CAAC;AAC9F,MAAE,SAAS,IAAI;AACf,MAAE,WAAW,EAAE,CAAC,MAAM,EAAE,iBAAiB,gBAAgB,UAAU,CAAC;AACpE,MAAE,WAAW,GAAG;AAChB,oBAAgB,EAAE,WAAW;AAC7B,MAAE,WAAW,GAAG;AAChB,MAAE,WAAW,EAAE,CAAC,MAAM,EAAE,iBAAiB,gBAAgB,YAAY,CAAC;AACtE,MAAE,SAAS,IAAI;AACf,MAAE,WAAW,EAAE,CAAC,MAAM,EAAE,iBAAiB,gBAAgB,kBAAkB,CAAC;AAC5E,MAAE,SAAS,IAAI;AACf,UAAM,SAAS,EAAE,SAAS,UAAU;AAAA,MAClC,MAAM,EAAE,CAAC,MAAM,EAAE,iBAAiB,QAAQ,MAAM;AAAA,IAClD,CAAC;AACD,WAAO,iBAAiB,SAAS,MAAM;AACrC,UAAI,mBAAmB;AACvB,oBAAc,UAAU;AACxB,aAAO,KAAK;AAAA,IACd,CAAC;AAAA,EACH,CAAC,CAAC;AAEF,oBAAkB;AAClB,eAAa,OAAO,YAAY,mBAAmB,sBAAsB;AAEzE,MAAI,qBAAqB,MAAM;AAC7B,kBAAc,UAAU;AACxB,WAAO,KAAK;AAAA,EACd,CAAC;AAED,WAAS,oBAA0B;AACjC,UAAM,4BAA4B,KAAK,IAAI,IAAI,UAAU,KAAK,OAAO,YAAY,IAAI,IAAI,aAAa,sBAAsB,IAAI,sBAAsB;AACtJ,kBAAc,cAAc,OAAO,yBAAyB;AAAA,EAC9D;AACF;AAEA,SAAS,uBAAuB,KAA2B;AACzD,QAAM,YAAY,KAAK,MAAM,YAAY,IAAI,IAAI,IAAI,QAAQ;AAE7D,MAAI,qBAAqB,MAAM;AAC7B,gBAAY,wCAAwC,EAAE,qCAAqC;AAAA,MACzF,eAAe,IAAI;AAAA,MACnB,eAAe,KAAK,MAAM,YAAY,IAAI,IAAI,SAAS;AAAA,IACzD,CAAC;AAAA,EACH,CAAC;AACH;",
  "names": []
}

|
|
@@ -189,7 +189,7 @@ class CodeHighlighterComponent extends ValueComponent {
|
|
|
189
189
|
}
|
|
190
190
|
evt.preventDefault();
|
|
191
191
|
if (evt.ctrlKey || evt.metaKey) {
|
|
192
|
-
const focusables = Array.from(
|
|
192
|
+
const focusables = Array.from(activeDocument.querySelectorAll(
|
|
193
193
|
':is(a, button, input, select, textarea, [tabindex]):not([tabindex="-1"]):not(:disabled):not([type="hidden"])'
|
|
194
194
|
));
|
|
195
195
|
const index = focusables.indexOf(this.inputEl);
|
|
@@ -238,4 +238,4 @@ class CodeHighlighterComponent extends ValueComponent {
|
|
|
238
238
|
export {
|
|
239
239
|
CodeHighlighterComponent
|
|
240
240
|
};
|
|
241
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/Components/SettingComponents/CodeHighlighterComponent.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains a component that displays and edits multiple text values.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  TextAreaComponent,\n  ValueComponent\n} from 'obsidian';\nimport { loadPrism } from 'obsidian-typings/implementations';\n\nimport type { ValidatorElement } from '../../../HTMLElement.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `initPluginContext` to use it in the tsdocs.\nimport type { initPluginContext } from '../../Plugin/PluginContext.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `SettingEx` to use it in the tsdocs.\nimport type { SettingEx } from '../../SettingEx.ts';\nimport type { TextBasedComponent } from './TextBasedComponent.ts';\nimport type { ValidatorComponent } from './ValidatorComponent.ts';\nimport type { ValueComponentWithChangeTracking } from './ValueComponentWithChangeTracking.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../../Async.ts';\nimport { CssClass } from '../../../CssClass.ts';\nimport { toPx } from '../../../HTMLElement.ts';\nimport { addPluginCssClasses } from '../../Plugin/PluginContext.ts';\n\n/**\n * A component that displays and edits code.\n *\n * You can add this component using {@link SettingEx.addCodeHighlighter}.\n *\n * In order to add the styles for the component, use {@link initPluginContext} in your plugin's `onload()` function.\n *\n * Alternatively, you can copy styles from {@link https://github.com/mnaoumov/obsidian-dev-utils/releases/latest/download/styles.css}.\n */\nexport class CodeHighlighterComponent extends ValueComponent<string>\n  implements TextBasedComponent<string>, ValidatorComponent, ValueComponentWithChangeTracking<string> {\n  /**\n   * An input element of the component.\n   *\n   * @returns The input element of the component.\n   */\n  public get inputEl(): HTMLTextAreaElement {\n    return this.textAreaComponent.inputEl;\n  }\n\n  /**\n   * Gets the validator element of the component.\n   *\n   * @returns The validator element of the component.\n   */\n  public get validatorEl(): ValidatorElement {\n    return this.inputEl;\n  }\n\n  private readonly codeEl: HTMLElement;\n  private placeholder = '';\n  private readonly preEl: HTMLElement;\n  private tabSize: number;\n  private readonly textAreaComponent: TextAreaComponent;\n\n  /**\n   * Creates a new multiple text component.\n   *\n   * @param containerEl - The container element of the component.\n   */\n  public constructor(containerEl: HTMLElement) {\n    super();\n    addPluginCssClasses(containerEl, CssClass.CodeHighlighterComponent);\n\n    const wrapper = containerEl.createDiv();\n    addPluginCssClasses(wrapper, CssClass.SettingComponentWrapper);\n\n    this.textAreaComponent = new TextAreaComponent(wrapper);\n    this.preEl = wrapper.createEl('pre', {\n      attr: {\n        tabIndex: '-1'\n      }\n    });\n    this.codeEl = this.preEl.createEl('code', {\n      attr: {\n        tabIndex: '-1'\n      }\n    });\n\n    this.inputEl.addEventListener('input', convertAsyncToSync(this.updateHighlightedCode.bind(this)));\n    this.inputEl.addEventListener('scroll', this.handleScroll.bind(this));\n    this.inputEl.addEventListener('keydown', this.handleKeyDown.bind(this));\n    const DEFAULT_TAB_SIZE = 2;\n    this.tabSize = DEFAULT_TAB_SIZE;\n  }\n\n  /**\n   * Empties the component.\n   */\n  public empty(): void {\n    this.setValue('');\n  }\n\n  /**\n   * Gets the value of the component.\n   *\n   * @returns The value of the component.\n   */\n  public override getValue(): string {\n    return this.textAreaComponent.getValue();\n  }\n\n  /**\n   * Checks if the component is empty.\n   *\n   * @returns `true` if the component is empty, `false` otherwise.\n   */\n  public isEmpty(): boolean {\n    return this.textAreaComponent.getValue() === '';\n  }\n\n  /**\n   * Adds a change listener to the component.\n   *\n   * @param callback - The callback to call when the value changes.\n   * @returns The component.\n   */\n  public onChange(callback: (newValue: string) => Promisable<void>): this {\n    this.textAreaComponent.onChange(() => callback(this.getValue()));\n    return this;\n  }\n\n  /**\n   * Sets the disabled state of the component.\n   *\n   * @param disabled - The disabled state to set.\n   * @returns The component.\n   */\n  public override setDisabled(disabled: boolean): this {\n    super.setDisabled(disabled);\n    this.textAreaComponent.setDisabled(disabled);\n    return this;\n  }\n\n  /**\n   * Sets the language for code highlighting.\n   *\n   * @param language - The language to set.\n   * @returns The component.\n   */\n  public setLanguage(language: string): this {\n    const LANGUAGE_CLASS_PREFIX = 'language-';\n    for (const el of [this.preEl, this.codeEl]) {\n      for (const cls of Array.from(el.classList)) {\n        if (cls.startsWith(LANGUAGE_CLASS_PREFIX)) {\n          el.classList.remove(cls);\n        }\n      }\n      el.classList.add(`${LANGUAGE_CLASS_PREFIX}${language}`);\n    }\n    return this;\n  }\n\n  /**\n   * Sets the placeholder of the component.\n   *\n   * @param placeholder - The placeholder to set.\n   * @returns The component.\n   */\n  public setPlaceholder(placeholder: string): this {\n    this.placeholder = placeholder;\n    invokeAsyncSafely(this.updateHighlightedCode.bind(this));\n    return this;\n  }\n\n  /**\n   * Sets the placeholder value of the component.\n   *\n   * @param placeholderValue - The placeholder value to set.\n   * @returns The component.\n   */\n  public setPlaceholderValue(placeholderValue: string): this {\n    this.setPlaceholder(placeholderValue);\n    return this;\n  }\n\n  /**\n   * Sets the tab size of the component.\n   *\n   * @param tabSize - The tab size to set.\n   * @returns The component.\n   */\n  public setTabSize(tabSize: number): this {\n    this.tabSize = tabSize;\n    return this;\n  }\n\n  /**\n   * Sets the value of the component.\n   *\n   * @param value - The value to set.\n   * @returns The component.\n   */\n  public override setValue(value: string): this {\n    this.textAreaComponent.setValue(value);\n    invokeAsyncSafely(this.updateHighlightedCode.bind(this));\n    return this;\n  }\n\n  private handleKeyDown(evt: KeyboardEvent): void {\n    if (evt.key !== 'Tab') {\n      return;\n    }\n\n    evt.preventDefault();\n\n    if (evt.ctrlKey || evt.metaKey) {\n      const focusables = Array.from(document.querySelectorAll<HTMLElement>(\n        ':is(a, button, input, select, textarea, [tabindex]):not([tabindex=\"-1\"]):not(:disabled):not([type=\"hidden\"])'\n      ));\n      const index = focusables.indexOf(this.inputEl);\n      const deltaIndex = evt.shiftKey ? -1 : 1;\n      const nextControl = focusables[(index + deltaIndex + focusables.length) % focusables.length];\n      nextControl?.focus();\n      return;\n    }\n\n    const oldValue = this.getValue();\n    const selectionStart = this.inputEl.selectionStart;\n    const selectionEnd = this.inputEl.selectionEnd;\n    const beforeSelection = oldValue.slice(0, selectionStart);\n    const afterSelection = oldValue.slice(selectionEnd);\n    const tabs = ' '.repeat(this.tabSize);\n    let newBeforeSelection = beforeSelection;\n\n    if (evt.shiftKey) {\n      if (beforeSelection.endsWith(tabs)) {\n        newBeforeSelection = beforeSelection.slice(0, -this.tabSize);\n      }\n    } else {\n      newBeforeSelection = beforeSelection + tabs;\n    }\n\n    const newValue = `${newBeforeSelection}${afterSelection}`;\n    this.setValue(newValue);\n    this.inputEl.selectionStart = newBeforeSelection.length;\n    this.inputEl.selectionEnd = newBeforeSelection.length;\n  }\n\n  private handleScroll(): void {\n    this.preEl.scrollTop = this.inputEl.scrollTop;\n    this.preEl.scrollLeft = this.inputEl.scrollLeft;\n  }\n\n  private async updateHighlightedCode(): Promise<void> {\n    this.codeEl.textContent = this.inputEl.value || this.placeholder;\n    const prism = await loadPrism();\n    prism.highlightElement(this.codeEl);\n    this.preEl.toggleClass(CssClass.IsPlaceholder, this.isEmpty());\n    requestAnimationFrame(() => {\n      const gap = Math.max(0, this.inputEl.scrollHeight - this.preEl.scrollHeight);\n      this.preEl.setCssProps({\n        '--bottom-gap': toPx(gap)\n      });\n      this.handleScroll();\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
241
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/Components/SettingComponents/CodeHighlighterComponent.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains a component that displays and edits multiple text values.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  TextAreaComponent,\n  ValueComponent\n} from 'obsidian';\nimport { loadPrism } from 'obsidian-typings/implementations';\n\nimport type { ValidatorElement } from '../../../HTMLElement.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `initPluginContext` to use it in the tsdocs.\nimport type { initPluginContext } from '../../Plugin/PluginContext.ts';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars -- We need to import `SettingEx` to use it in the tsdocs.\nimport type { SettingEx } from '../../SettingEx.ts';\nimport type { TextBasedComponent } from './TextBasedComponent.ts';\nimport type { ValidatorComponent } from './ValidatorComponent.ts';\nimport type { ValueComponentWithChangeTracking } from './ValueComponentWithChangeTracking.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../../Async.ts';\nimport { CssClass } from '../../../CssClass.ts';\nimport { toPx } from '../../../HTMLElement.ts';\nimport { addPluginCssClasses } from '../../Plugin/PluginContext.ts';\n\n/**\n * A component that displays and edits code.\n *\n * You can add this component using {@link SettingEx.addCodeHighlighter}.\n *\n * In order to add the styles for the component, use {@link initPluginContext} in your plugin's `onload()` function.\n *\n * Alternatively, you can copy styles from {@link https://github.com/mnaoumov/obsidian-dev-utils/releases/latest/download/styles.css}.\n */\nexport class CodeHighlighterComponent extends ValueComponent<string>\n  implements TextBasedComponent<string>, ValidatorComponent, ValueComponentWithChangeTracking<string> {\n  /**\n   * An input element of the component.\n   *\n   * @returns The input element of the component.\n   */\n  public get inputEl(): HTMLTextAreaElement {\n    return this.textAreaComponent.inputEl;\n  }\n\n  /**\n   * Gets the validator element of the component.\n   *\n   * @returns The validator element of the component.\n   */\n  public get validatorEl(): ValidatorElement {\n    return this.inputEl;\n  }\n\n  private readonly codeEl: HTMLElement;\n  private placeholder = '';\n  private readonly preEl: HTMLElement;\n  private tabSize: number;\n  private readonly textAreaComponent: TextAreaComponent;\n\n  /**\n   * Creates a new multiple text component.\n   *\n   * @param containerEl - The container element of the component.\n   */\n  public constructor(containerEl: HTMLElement) {\n    super();\n    addPluginCssClasses(containerEl, CssClass.CodeHighlighterComponent);\n\n    const wrapper = containerEl.createDiv();\n    addPluginCssClasses(wrapper, CssClass.SettingComponentWrapper);\n\n    this.textAreaComponent = new TextAreaComponent(wrapper);\n    this.preEl = wrapper.createEl('pre', {\n      attr: {\n        tabIndex: '-1'\n      }\n    });\n    this.codeEl = this.preEl.createEl('code', {\n      attr: {\n        tabIndex: '-1'\n      }\n    });\n\n    this.inputEl.addEventListener('input', convertAsyncToSync(this.updateHighlightedCode.bind(this)));\n    this.inputEl.addEventListener('scroll', this.handleScroll.bind(this));\n    this.inputEl.addEventListener('keydown', this.handleKeyDown.bind(this));\n    const DEFAULT_TAB_SIZE = 2;\n    this.tabSize = DEFAULT_TAB_SIZE;\n  }\n\n  /**\n   * Empties the component.\n   */\n  public empty(): void {\n    this.setValue('');\n  }\n\n  /**\n   * Gets the value of the component.\n   *\n   * @returns The value of the component.\n   */\n  public override getValue(): string {\n    return this.textAreaComponent.getValue();\n  }\n\n  /**\n   * Checks if the component is empty.\n   *\n   * @returns `true` if the component is empty, `false` otherwise.\n   */\n  public isEmpty(): boolean {\n    return this.textAreaComponent.getValue() === '';\n  }\n\n  /**\n   * Adds a change listener to the component.\n   *\n   * @param callback - The callback to call when the value changes.\n   * @returns The component.\n   */\n  public onChange(callback: (newValue: string) => Promisable<void>): this {\n    this.textAreaComponent.onChange(() => callback(this.getValue()));\n    return this;\n  }\n\n  /**\n   * Sets the disabled state of the component.\n   *\n   * @param disabled - The disabled state to set.\n   * @returns The component.\n   */\n  public override setDisabled(disabled: boolean): this {\n    super.setDisabled(disabled);\n    this.textAreaComponent.setDisabled(disabled);\n    return this;\n  }\n\n  /**\n   * Sets the language for code highlighting.\n   *\n   * @param language - The language to set.\n   * @returns The component.\n   */\n  public setLanguage(language: string): this {\n    const LANGUAGE_CLASS_PREFIX = 'language-';\n    for (const el of [this.preEl, this.codeEl]) {\n      for (const cls of Array.from(el.classList)) {\n        if (cls.startsWith(LANGUAGE_CLASS_PREFIX)) {\n          el.classList.remove(cls);\n        }\n      }\n      el.classList.add(`${LANGUAGE_CLASS_PREFIX}${language}`);\n    }\n    return this;\n  }\n\n  /**\n   * Sets the placeholder of the component.\n   *\n   * @param placeholder - The placeholder to set.\n   * @returns The component.\n   */\n  public setPlaceholder(placeholder: string): this {\n    this.placeholder = placeholder;\n    invokeAsyncSafely(this.updateHighlightedCode.bind(this));\n    return this;\n  }\n\n  /**\n   * Sets the placeholder value of the component.\n   *\n   * @param placeholderValue - The placeholder value to set.\n   * @returns The component.\n   */\n  public setPlaceholderValue(placeholderValue: string): this {\n    this.setPlaceholder(placeholderValue);\n    return this;\n  }\n\n  /**\n   * Sets the tab size of the component.\n   *\n   * @param tabSize - The tab size to set.\n   * @returns The component.\n   */\n  public setTabSize(tabSize: number): this {\n    this.tabSize = tabSize;\n    return this;\n  }\n\n  /**\n   * Sets the value of the component.\n   *\n   * @param value - The value to set.\n   * @returns The component.\n   */\n  public override setValue(value: string): this {\n    this.textAreaComponent.setValue(value);\n    invokeAsyncSafely(this.updateHighlightedCode.bind(this));\n    return this;\n  }\n\n  private handleKeyDown(evt: KeyboardEvent): void {\n    if (evt.key !== 'Tab') {\n      return;\n    }\n\n    evt.preventDefault();\n\n    if (evt.ctrlKey || evt.metaKey) {\n      const focusables = Array.from(activeDocument.querySelectorAll<HTMLElement>(\n        ':is(a, button, input, select, textarea, [tabindex]):not([tabindex=\"-1\"]):not(:disabled):not([type=\"hidden\"])'\n      ));\n      const index = focusables.indexOf(this.inputEl);\n      const deltaIndex = evt.shiftKey ? -1 : 1;\n      const nextControl = focusables[(index + deltaIndex + focusables.length) % focusables.length];\n      nextControl?.focus();\n      return;\n    }\n\n    const oldValue = this.getValue();\n    const selectionStart = this.inputEl.selectionStart;\n    const selectionEnd = this.inputEl.selectionEnd;\n    const beforeSelection = oldValue.slice(0, selectionStart);\n    const afterSelection = oldValue.slice(selectionEnd);\n    const tabs = ' '.repeat(this.tabSize);\n    let newBeforeSelection = beforeSelection;\n\n    if (evt.shiftKey) {\n      if (beforeSelection.endsWith(tabs)) {\n        newBeforeSelection = beforeSelection.slice(0, -this.tabSize);\n      }\n    } else {\n      newBeforeSelection = beforeSelection + tabs;\n    }\n\n    const newValue = `${newBeforeSelection}${afterSelection}`;\n    this.setValue(newValue);\n    this.inputEl.selectionStart = newBeforeSelection.length;\n    this.inputEl.selectionEnd = newBeforeSelection.length;\n  }\n\n  private handleScroll(): void {\n    this.preEl.scrollTop = this.inputEl.scrollTop;\n    this.preEl.scrollLeft = this.inputEl.scrollLeft;\n  }\n\n  private async updateHighlightedCode(): Promise<void> {\n    this.codeEl.textContent = this.inputEl.value || this.placeholder;\n    const prism = await loadPrism();\n    prism.highlightElement(this.codeEl);\n    this.preEl.toggleClass(CssClass.IsPlaceholder, this.isEmpty());\n    requestAnimationFrame(() => {\n      const gap = Math.max(0, this.inputEl.scrollHeight - this.preEl.scrollHeight);\n      this.preEl.setCssProps({\n        '--bottom-gap': toPx(gap)\n      });\n      this.handleScroll();\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -25,11 +25,11 @@ export interface LoopOptions<T> {
|
|
|
25
25
|
*/
|
|
26
26
|
items: T[];
|
|
27
27
|
/**
|
|
28
|
-
* A timeout for the notice before it is shown.
|
|
28
|
+
* A timeout for the notice before it is shown. Default is `500`.
|
|
29
29
|
*/
|
|
30
30
|
noticeBeforeShownTimeoutInMilliseconds?: number;
|
|
31
31
|
/**
|
|
32
|
-
* A minimum timeout for the notice.
|
|
32
|
+
* A minimum timeout for the notice. Default is `2000`.
|
|
33
33
|
*/
|
|
34
34
|
noticeMinTimeoutInMilliseconds?: number;
|
|
35
35
|
/**
|
|
@@ -39,19 +39,23 @@ export interface LoopOptions<T> {
|
|
|
39
39
|
*/
|
|
40
40
|
processItem(item: T): Promisable<void>;
|
|
41
41
|
/**
|
|
42
|
-
* A title of the progress bar.
|
|
42
|
+
* A title of the progress bar. Default is `''`.
|
|
43
43
|
*/
|
|
44
44
|
progressBarTitle?: string;
|
|
45
45
|
/**
|
|
46
|
-
* Whether to continue the loop on error.
|
|
46
|
+
* Whether to continue the loop on error. Default is `true`.
|
|
47
47
|
*/
|
|
48
48
|
shouldContinueOnError?: boolean;
|
|
49
49
|
/**
|
|
50
|
-
* Whether to show a
|
|
50
|
+
* Whether to show a notice. Default is `true`.
|
|
51
|
+
*/
|
|
52
|
+
shouldShowNotice?: boolean;
|
|
53
|
+
/**
|
|
54
|
+
* Whether to show a progress bar. Default is `true`.
|
|
51
55
|
*/
|
|
52
56
|
shouldShowProgressBar?: boolean;
|
|
53
57
|
/**
|
|
54
|
-
* A threshold for the UI update.
|
|
58
|
+
* A threshold for the UI update. Default is `100`.
|
|
55
59
|
*/
|
|
56
60
|
uiUpdateThresholdInMilliseconds?: number;
|
|
57
61
|
}
|
|
@@ -48,6 +48,7 @@ async function loop(options) {
|
|
|
48
48
|
processItem: noop,
|
|
49
49
|
progressBarTitle: "",
|
|
50
50
|
shouldContinueOnError: true,
|
|
51
|
+
shouldShowNotice: true,
|
|
51
52
|
shouldShowProgressBar: true,
|
|
52
53
|
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
|
|
53
54
|
uiUpdateThresholdInMilliseconds: 100
|
|
@@ -101,6 +102,9 @@ async function loop(options) {
|
|
|
101
102
|
notice?.hide();
|
|
102
103
|
isDone = true;
|
|
103
104
|
async function showNotice() {
|
|
105
|
+
if (!fullOptions.shouldShowNotice) {
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
104
108
|
await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);
|
|
105
109
|
if (isDone) {
|
|
106
110
|
return;
|
|
@@ -118,4 +122,4 @@ async function loop(options) {
|
|
|
118
122
|
export {
|
|
119
123
|
loop
|
|
120
124
|
};
|
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Loop.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for looping in Obsidian.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  invokeAsyncSafely,\n  requestAnimationFrameAsync\n} from '../Async.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace\n} from '../Error.ts';\nimport { noop } from '../Function.ts';\nimport { addPluginCssClasses } from './Plugin/PluginContext.ts';\n\n/**\n * Options for {@link loop}.\n */\nexport interface LoopOptions<T> {\n  /**\n   * An optional abort signal to cancel the loop.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * Build a notice message for each item.\n   *\n   * @param item - The current item.\n   * @param iterationStr - A string representing the current iteration.\n   * @returns A string to display in the notice.\n   */\n  buildNoticeMessage(item: T, iterationStr: string): string;\n\n  /**\n   * Items to loop over.\n   */\n  items: T[];\n\n  /**\n   * A timeout for the notice before it is shown.\n   */\n  noticeBeforeShownTimeoutInMilliseconds?: number;\n\n  /**\n   * A minimum timeout for the notice.\n   */\n  noticeMinTimeoutInMilliseconds?: number;\n\n  /**\n   * Process each item.\n   *\n   * @param item - The current item.\n   */\n  processItem(item: T): Promisable<void>;\n\n  /**\n   * A title of the progress bar.\n   */\n  progressBarTitle?: string;\n\n  /**\n   * Whether to continue the loop on error.\n   */\n  shouldContinueOnError?: boolean;\n\n  /**\n   * Whether to show a progress bar.\n   */\n  shouldShowProgressBar?: boolean;\n\n  /**\n   * A threshold for the UI update.\n   */\n  uiUpdateThresholdInMilliseconds?: number;\n}\n\n/**\n * Loops over a list of items and processes each item.\n *\n * @param options - The options for the loop.\n */\nexport async function loop<T>(options: LoopOptions<T>): Promise<void> {\n  const DEFAULT_OPTIONS: Required<LoopOptions<T>> = {\n    abortSignal: abortSignalNever(),\n    buildNoticeMessage() {\n      throw new Error('buildNoticeMessage is required');\n    },\n    items: [],\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeBeforeShownTimeoutInMilliseconds: 500,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeMinTimeoutInMilliseconds: 2000,\n    processItem: noop,\n    progressBarTitle: '',\n    shouldContinueOnError: true,\n    shouldShowProgressBar: true,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    uiUpdateThresholdInMilliseconds: 100\n  };\n\n  const fullOptions: Required<LoopOptions<T>> = {\n    ...DEFAULT_OPTIONS,\n    ...options\n  };\n\n  const stackTrace = getStackTrace(1);\n\n  const items = fullOptions.items;\n  let iterationCount = 0;\n  let notice = null as Notice | null;\n  let isDone = false;\n  invokeAsyncSafely(() => showNotice());\n\n  const noticeMinTimeoutPromise = sleep(fullOptions.noticeMinTimeoutInMilliseconds);\n  const progressBarEl = createEl('progress');\n  addPluginCssClasses(progressBarEl, 'loop');\n  progressBarEl.max = items.length;\n\n  let lastUIUpdateTimestamp = performance.now();\n\n  for (const item of items) {\n    if (fullOptions.abortSignal.aborted) {\n      notice?.hide();\n      return;\n    }\n    iterationCount++;\n    const iterationStr = `# ${String(iterationCount)} / ${String(items.length)}`;\n    const message = fullOptions.buildNoticeMessage(item, iterationStr);\n    if (!fullOptions.shouldShowProgressBar) {\n      notice?.setMessage(message);\n    }\n    getLibDebugger('Loop')(message);\n\n    try {\n      if (performance.now() - lastUIUpdateTimestamp > fullOptions.uiUpdateThresholdInMilliseconds) {\n        await requestAnimationFrameAsync();\n        lastUIUpdateTimestamp = performance.now();\n      }\n      await fullOptions.processItem(item);\n    } catch (error) {\n      console.error('Error processing item', item);\n      if (!fullOptions.shouldContinueOnError) {\n        notice?.hide();\n        throw new CustomStackTraceError('loop failed', stackTrace, error);\n      }\n\n      emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, error));\n    }\n    progressBarEl.value++;\n  }\n  if (notice) {\n    await noticeMinTimeoutPromise;\n  }\n  notice?.hide();\n  isDone = true;\n\n  async function showNotice(): Promise<void> {\n    await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);\n    if (isDone) {\n      return;\n    }\n    notice = new Notice('', 0);\n    if (!fullOptions.shouldShowProgressBar) {\n      return;\n    }\n    const fragment = createFragment();\n    fragment.createDiv({ text: fullOptions.progressBarTitle });\n    fragment.appendChild(progressBarEl);\n    notice.setMessage(fragment);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AAQA,SAAS,cAAc;AAEvB,SAAS,wBAAwB;AACjC;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,sBAAsB;AAC/B;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,YAAY;AACrB,SAAS,2BAA2B;AAoEpC,eAAsB,KAAQ,SAAwC;AACpE,QAAM,kBAA4C;AAAA,IAChD,aAAa,iBAAiB;AAAA,IAC9B,qBAAqB;AACnB,YAAM,IAAI,MAAM,gCAAgC;AAAA,IAClD;AAAA,IACA,OAAO,CAAC;AAAA;AAAA,IAER,wCAAwC;AAAA;AAAA,IAExC,gCAAgC;AAAA,IAChC,aAAa;AAAA,IACb,kBAAkB;AAAA,IAClB,uBAAuB;AAAA,IACvB,uBAAuB;AAAA;AAAA,IAEvB,iCAAiC;AAAA,EACnC;AAEA,QAAM,cAAwC;AAAA,IAC5C,GAAG;AAAA,IACH,GAAG;AAAA,EACL;AAEA,QAAM,aAAa,cAAc,CAAC;AAElC,QAAM,QAAQ,YAAY;AAC1B,MAAI,iBAAiB;AACrB,MAAI,SAAS;AACb,MAAI,SAAS;AACb,oBAAkB,MAAM,WAAW,CAAC;AAEpC,QAAM,0BAA0B,MAAM,YAAY,8BAA8B;AAChF,QAAM,gBAAgB,SAAS,UAAU;AACzC,sBAAoB,eAAe,MAAM;AACzC,gBAAc,MAAM,MAAM;AAE1B,MAAI,wBAAwB,YAAY,IAAI;AAE5C,aAAW,QAAQ,OAAO;AACxB,QAAI,YAAY,YAAY,SAAS;AACnC,cAAQ,KAAK;AACb;AAAA,IACF;AACA;AACA,UAAM,eAAe,KAAK,OAAO,cAAc,CAAC,MAAM,OAAO,MAAM,MAAM,CAAC;AAC1E,UAAM,UAAU,YAAY,mBAAmB,MAAM,YAAY;AACjE,QAAI,CAAC,YAAY,uBAAuB;AACtC,cAAQ,WAAW,OAAO;AAAA,IAC5B;AACA,mBAAe,MAAM,EAAE,OAAO;AAE9B,QAAI;AACF,UAAI,YAAY,IAAI,IAAI,wBAAwB,YAAY,iCAAiC;AAC3F,cAAM,2BAA2B;AACjC,gCAAwB,YAAY,IAAI;AAAA,MAC1C;AACA,YAAM,YAAY,YAAY,IAAI;AAAA,IACpC,SAAS,OAAO;AACd,cAAQ,MAAM,yBAAyB,IAAI;AAC3C,UAAI,CAAC,YAAY,uBAAuB;AACtC,gBAAQ,KAAK;AACb,cAAM,IAAI,sBAAsB,eAAe,YAAY,KAAK;AAAA,MAClE;AAEA,0BAAoB,IAAI,sBAAsB,6BAA6B,YAAY,KAAK,CAAC;AAAA,IAC/F;AACA,kBAAc;AAAA,EAChB;AACA,MAAI,QAAQ;AACV,UAAM;AAAA,EACR;AACA,UAAQ,KAAK;AACb,WAAS;AAET,iBAAe,aAA4B;AACzC,UAAM,MAAM,YAAY,sCAAsC;AAC9D,QAAI,QAAQ;AACV;AAAA,IACF;AACA,aAAS,IAAI,OAAO,IAAI,CAAC;AACzB,QAAI,CAAC,YAAY,uBAAuB;AACtC;AAAA,IACF;AACA,UAAM,WAAW,eAAe;AAChC,aAAS,UAAU,EAAE,MAAM,YAAY,iBAAiB,CAAC;AACzD,aAAS,YAAY,aAAa;AAClC,WAAO,WAAW,QAAQ;AAAA,EAC5B;AACF;",
  "names": []
}

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

|
|
@@ -8,6 +8,15 @@ import type { CustomArrayDict } from 'obsidian-typings';
|
|
|
8
8
|
import type { RetryOptions } from '../Async.mjs';
|
|
9
9
|
import type { PathOrFile } from './FileSystem.mjs';
|
|
10
10
|
import type { CombinedFrontmatter } from './Frontmatter.mjs';
|
|
11
|
+
/**
|
|
12
|
+
* Options for {@link getBacklinksForFileSafe}.
|
|
13
|
+
*/
|
|
14
|
+
export interface GetBacklinksForFileSafeOptions extends RetryOptions {
|
|
15
|
+
/**
|
|
16
|
+
* Whether to show a timeout notice. Default is `true`.
|
|
17
|
+
*/
|
|
18
|
+
shouldShowTimeoutNotice?: boolean;
|
|
19
|
+
}
|
|
11
20
|
/**
|
|
12
21
|
* Wrapper for the getBacklinksForFile method that provides a safe overload.
|
|
13
22
|
*/
|
|
@@ -48,10 +57,10 @@ export declare function getBacklinksForFileOrPath(app: App, pathOrFile: PathOrFi
|
|
|
48
57
|
*
|
|
49
58
|
* @param app - The Obsidian application instance.
|
|
50
59
|
* @param pathOrFile - The path or file object.
|
|
51
|
-
* @param
|
|
60
|
+
* @param options - Other options.
|
|
52
61
|
* @returns A {@link Promise} that resolves to an array dictionary of backlinks.
|
|
53
62
|
*/
|
|
54
|
-
export declare function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile,
|
|
63
|
+
export declare function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile, options?: GetBacklinksForFileSafeOptions): Promise<CustomArrayDict<Reference>>;
|
|
55
64
|
/**
|
|
56
65
|
* Retrieves the cached metadata for a given file or path.
|
|
57
66
|
*
|