obsidian-dev-utils 37.1.2 → 37.2.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 +4 -0
- package/dist/lib/cjs/Async.cjs +11 -1
- package/dist/lib/cjs/Error.cjs +15 -1
- package/dist/lib/cjs/Error.d.cts +11 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +2 -2
- package/dist/lib/esm/Async.mjs +13 -2
- package/dist/lib/esm/Error.d.mts +11 -0
- package/dist/lib/esm/Error.mjs +14 -1
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +6 -3
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
package/dist/lib/cjs/Async.cjs
CHANGED
|
@@ -60,6 +60,10 @@ async function addErrorHandler(asyncFn, stackTrace) {
|
|
|
60
60
|
try {
|
|
61
61
|
await asyncFn();
|
|
62
62
|
} catch (asyncError) {
|
|
63
|
+
if (asyncError instanceof import_Error.SilentError) {
|
|
64
|
+
printSilentError(asyncError, stackTrace);
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
63
67
|
(0, import_Error.emitAsyncErrorEvent)(new import_Error.CustomStackTraceError(import_Error.ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));
|
|
64
68
|
}
|
|
65
69
|
}
|
|
@@ -146,6 +150,12 @@ async function promiseAllAsyncFnsSequentially(asyncFns) {
|
|
|
146
150
|
async function promiseAllSequentially(promises) {
|
|
147
151
|
return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));
|
|
148
152
|
}
|
|
153
|
+
function printSilentError(error, stackTrace) {
|
|
154
|
+
(0, import_Debug.getLibDebugger)("Async:printSilentError")("Silent error", {
|
|
155
|
+
error,
|
|
156
|
+
stackTrace
|
|
157
|
+
});
|
|
158
|
+
}
|
|
149
159
|
const terminateRetryErrors = /* @__PURE__ */ new WeakSet();
|
|
150
160
|
class Result {
|
|
151
161
|
constructor(result) {
|
|
@@ -328,4 +338,4 @@ async function toArray(iter) {
|
|
|
328
338
|
timeout,
|
|
329
339
|
toArray
|
|
330
340
|
});
|
|
331
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  abortSignalAny,\n  abortSignalNever,\n  abortSignalTimeout,\n  waitForAbort\n} from './AbortController.ts';\nimport {\n  getLibDebugger,\n  printWithStackTrace\n} from './Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError\n} from './Error.ts';\nimport { noop } from './Function.ts';\n\n/**\n * A type representing a function that resolves a {@link Promise}.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryOptions {\n  /**\n   * A abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * A delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * A maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Adds an error handler to a {@link Promise} that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>, stackTrace?: string): Promise<void> {\n  stackTrace ??= getStackTrace(1);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const ans: T[] = [];\n\n  const length = arr.length;\n  for (let i = 0; i < length; i++) {\n    if (!Object.hasOwn(arr, i)) {\n      continue;\n    }\n\n    const item = arr[i] as T;\n    if (await predicate(item, i, arr)) {\n      ans.push(item);\n    }\n  }\n\n  return ans;\n}\n\n/**\n * Filters an array asynchronously in place, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves when the array is filtered.\n */\nexport async function asyncFilterInPlace<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<void> {\n  const length = arr.length;\n  let writeIndex = 0;\n  for (let readIndex = 0; readIndex < length; readIndex++) {\n    if (!Object.hasOwn(arr, readIndex)) {\n      continue;\n    }\n\n    const current = arr[readIndex] as T;\n    if (await predicate(current, readIndex, arr)) {\n      // eslint-disable-next-line require-atomic-updates\n      arr[writeIndex++] = current;\n    }\n  }\n  arr.length = writeIndex;\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await promiseAllSequentially(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @param stackTrace - The stack trace of the source function.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>, stackTrace?: string): (...args: Args) => void {\n  stackTrace ??= getStackTrace(1);\n  return (...args: Args): void => {\n    const innerStackTrace = getStackTrace(1);\n    stackTrace = `${stackTrace ?? ''}\\n    at --- convertAsyncToSync --- (0)\\n${innerStackTrace}`;\n    invokeAsyncSafely(() => asyncFunc(...args), stackTrace);\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a {@link Promise}.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return async (...args: Args): Promise<Result> => {\n    await Promise.resolve();\n    return syncFn(...args);\n  };\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A {@link Promise} that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  const ignoreErrorDebugger = getLibDebugger('Async:ignoreError');\n  const stackTrace = getStackTrace(1);\n  try {\n    return await promise;\n  } catch (e) {\n    ignoreErrorDebugger('Ignored error', new CustomStackTraceError('Ignored error', stackTrace, e));\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a {@link Promise} and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n * @param stackTrace - The stack trace of the source function.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>, stackTrace?: string): void {\n  stackTrace ??= getStackTrace(1);\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn, stackTrace);\n}\n\n/**\n * Invokes an asynchronous function after a delay.\n *\n * @param asyncFn - The asynchronous function to invoke.\n * @param delayInMilliseconds - The delay in milliseconds.\n * @param stackTrace - The stack trace of the source function.\n * @param abortSignal - The abort signal to listen to.\n */\nexport function invokeAsyncSafelyAfterDelay(\n  asyncFn: (abortSignal: AbortSignal) => Promisable<void>,\n  delayInMilliseconds = 0,\n  stackTrace?: string,\n  abortSignal?: AbortSignal\n): void {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  stackTrace ??= getStackTrace(1);\n  invokeAsyncSafely(async () => {\n    await sleep(delayInMilliseconds, abortSignal, true);\n    await asyncFn(abortSignal);\n  }, stackTrace);\n}\n\n/**\n * Executes async functions sequentially.\n *\n * @typeParam T - The type of the value.\n * @param asyncFns - The async functions to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the async functions.\n */\nexport async function promiseAllAsyncFnsSequentially<T>(asyncFns: (() => Promisable<T>)[]): Promise<T[]> {\n  const results: T[] = [];\n  for (const asyncFn of asyncFns) {\n    results.push(await asyncFn());\n  }\n  return results;\n}\n\n/**\n * Executes promises sequentially.\n *\n * @typeParam T - The type of the value.\n * @param promises - The promises to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the promises.\n */\nexport async function promiseAllSequentially<T>(promises: Promisable<T>[]): Promise<T[]> {\n  return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));\n}\n\nconst terminateRetryErrors = new WeakSet<Error>();\n\nclass Result<R> {\n  public constructor(public readonly result: R) {}\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n */\nexport function marksAsTerminateRetry(error: Error): void {\n  terminateRetryErrors.add(error);\n}\n\n/**\n * An async function that never ends.\n *\n * @returns A {@link Promise} that never resolves.\n */\nexport async function neverEnds(): Promise<never> {\n  await new Promise(() => {\n    noop();\n  });\n  throw new Error('Should never happen');\n}\n\n/**\n * Gets the next tick.\n *\n * @returns A promise that resolves when the next tick is available.\n */\nexport async function nextTickAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    process.nextTick(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next queue microtask.\n *\n * @returns A promise that resolves when the next queue microtask is available.\n */\nexport async function queueMicrotaskAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    queueMicrotask(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next request animation frame.\n *\n * @returns A promise that resolves when the next request animation frame is available.\n */\nexport async function requestAnimationFrameAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    requestAnimationFrame(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\n      const combinedAbortSignal = abortSignalAny(fullOptions.abortSignal, abortSignal);\n      combinedAbortSignal.throwIfAborted();\n      let attempt = 0;\n      while (!combinedAbortSignal.aborted) {\n        attempt++;\n        let isSuccess: boolean;\n        try {\n          isSuccess = await fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n          if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error as Error)) {\n            throw new CustomStackTraceError('retryWithTimeout failed', stackTrace, error);\n          }\n          printError(error);\n          isSuccess = false;\n        }\n        if (isSuccess) {\n          printWithStackTrace(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {\n            fn\n          });\n          return;\n        }\n\n        printWithStackTrace(\n          retryWithTimeoutDebugger,\n          stackTrace,\n          `Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,\n          {\n            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\n  );\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<R>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<R> {\n  stackTrace ??= getStackTrace(1);\n  const startTime = performance.now();\n\n  const abortController = new AbortController();\n\n  await Promise.race([run(), innerTimeout()]);\n  if (abortController.signal.reason instanceof Result) {\n    return abortController.signal.reason.result as R;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, abortController.signal.reason as unknown);\n\n  async function run(): Promise<void> {\n    try {\n      const result = await fn(abortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      abortController.abort(new Result(result));\n    } catch (e) {\n      abortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    while (!abortController.signal.aborted) {\n      await sleep(timeoutInMilliseconds, abortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (abortController.signal.aborted) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        abortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\n  }\n}\n\n/**\n * Gets the next set immediate.\n *\n * @returns A promise that resolves when the next set immediate is available.\n */\nexport async function setImmediateAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    setImmediate(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param delay - The time to wait in milliseconds.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function setTimeoutAsync(delay?: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, delay);\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<void> {\n  await waitForAbort(abortSignalAny(abortSignal, abortSignalTimeout(milliseconds)));\n  if (shouldThrowOnAbort) {\n    abortSignal?.throwIfAborted();\n  }\n}\n\n/**\n * Returns a {@link Promise} that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<never> {\n  await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);\n  throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A {@link Promise} that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,6BAKO;AACP,mBAGO;AACP,mBAMO;AACP,sBAAqB;AA0CrB,eAAsB,gBAAgB,SAAiC,YAAoC;AACzG,qBAAe,4BAAc,CAAC;AAC9B,MAAI;AACF,UAAM,QAAQ;AAAA,EAChB,SAAS,YAAY;AACnB,0CAAoB,IAAI,mCAAsB,0CAA6B,YAAY,UAAU,CAAC;AAAA,EACpG;AACF;AAUA,eAAsB,YAAe,KAAU,WAAuF;AACpI,QAAM,MAAW,CAAC;AAElB,QAAM,SAAS,IAAI;AACnB,WAAS,IAAI,GAAG,IAAI,QAAQ,KAAK;AAC/B,QAAI,CAAC,OAAO,OAAO,KAAK,CAAC,GAAG;AAC1B;AAAA,IACF;AAEA,UAAM,OAAO,IAAI,CAAC;AAClB,QAAI,MAAM,UAAU,MAAM,GAAG,GAAG,GAAG;AACjC,UAAI,KAAK,IAAI;AAAA,IACf;AAAA,EACF;AAEA,SAAO;AACT;AAUA,eAAsB,mBAAsB,KAAU,WAAwF;AAC5I,QAAM,SAAS,IAAI;AACnB,MAAI,aAAa;AACjB,WAAS,YAAY,GAAG,YAAY,QAAQ,aAAa;AACvD,QAAI,CAAC,OAAO,OAAO,KAAK,SAAS,GAAG;AAClC;AAAA,IACF;AAEA,UAAM,UAAU,IAAI,SAAS;AAC7B,QAAI,MAAM,UAAU,SAAS,WAAW,GAAG,GAAG;AAE5C,UAAI,YAAY,IAAI;AAAA,IACtB;AAAA,EACF;AACA,MAAI,SAAS;AACf;AAWA,eAAsB,aAAmB,KAAU,UAAkF;AACnI,UAAQ,MAAM,SAAS,KAAK,QAAQ,GAAG,KAAK;AAC9C;AAWA,eAAsB,SAAe,KAAU,UAAgF;AAC7H,SAAO,MAAM,uBAAuB,IAAI,IAAI,QAAQ,CAAC;AACvD;AAUO,SAAS,mBAA2C,WAAgD,YAA8C;AACvJ,qBAAe,4BAAc,CAAC;AAC9B,SAAO,IAAI,SAAqB;AAC9B,UAAM,sBAAkB,4BAAc,CAAC;AACvC,iBAAa,GAAG,cAAc,EAAE;AAAA;AAAA,EAA4C,eAAe;AAC3F,sBAAkB,MAAM,UAAU,GAAG,IAAI,GAAG,UAAU;AAAA,EACxD;AACF;AAUO,SAAS,mBAAmD,QAAuE;AACxI,SAAO,UAAU,SAAgC;AAC/C,UAAM,QAAQ,QAAQ;AACtB,WAAO,OAAO,GAAG,IAAI;AAAA,EACvB;AACF;AAmBA,eAAsB,YAAe,SAAqB,eAA8B;AACtF,QAAM,0BAAsB,6BAAe,mBAAmB;AAC9D,QAAM,iBAAa,4BAAc,CAAC;AAClC,MAAI;AACF,WAAO,MAAM;AAAA,EACf,SAAS,GAAG;AACV,wBAAoB,iBAAiB,IAAI,mCAAsB,iBAAiB,YAAY,CAAC,CAAC;AAC9F,WAAO;AAAA,EACT;AACF;AAQO,SAAS,kBAAkB,SAAiC,YAA2B;AAC5F,qBAAe,4BAAc,CAAC;AAE9B,OAAK,gBAAgB,SAAS,UAAU;AAC1C;AAUO,SAAS,4BACd,SACA,sBAAsB,GACtB,YACA,aACM;AACN,sBAAgB,yCAAiB;AACjC,cAAY,eAAe;AAC3B,qBAAe,4BAAc,CAAC;AAC9B,oBAAkB,YAAY;AAC5B,UAAM,MAAM,qBAAqB,aAAa,IAAI;AAClD,UAAM,QAAQ,WAAW;AAAA,EAC3B,GAAG,UAAU;AACf;AASA,eAAsB,+BAAkC,UAAiD;AACvG,QAAM,UAAe,CAAC;AACtB,aAAW,WAAW,UAAU;AAC9B,YAAQ,KAAK,MAAM,QAAQ,CAAC;AAAA,EAC9B;AACA,SAAO;AACT;AASA,eAAsB,uBAA0B,UAAyC;AACvF,SAAO,MAAM,+BAA+B,SAAS,IAAI,CAAC,YAAY,MAAM,OAAO,CAAC;AACtF;AAEA,MAAM,uBAAuB,oBAAI,QAAe;AAEhD,MAAM,OAAU;AAAA,EACP,YAA4B,QAAW;AAAX;AAAA,EAAY;AACjD;AAOO,SAAS,sBAAsB,OAAoB;AACxD,uBAAqB,IAAI,KAAK;AAChC;AAOA,eAAsB,YAA4B;AAChD,QAAM,IAAI,QAAQ,MAAM;AACtB,8BAAK;AAAA,EACP,CAAC;AACD,QAAM,IAAI,MAAM,qBAAqB;AACvC;AAOA,eAAsB,gBAA+B;AACnD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,YAAQ,SAAS,MAAM;AACrB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,sBAAqC;AACzD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,mBAAe,MAAM;AACnB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,6BAA4C;AAChE,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,0BAAsB,MAAM;AAC1B,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAUA,eAAsB,iBAAiB,IAAuD,cAA6B,YAAoC;AAC7J,QAAM,+BAA2B,6BAAe,wBAAwB;AACxE,qBAAe,4BAAc,CAAC;AAC9B,QAAM,wBAAwB;AAAA;AAAA,IAE5B,0BAA0B;AAAA,IAC1B,oBAAoB;AAAA;AAAA,IAEpB,uBAAuB;AAAA,EACzB;AACA,QAAM,cAAc,EAAE,GAAG,uBAAuB,GAAG,aAAa;AAChE,cAAY,aAAa,eAAe;AAExC,QAAM;AAAA,IACJ,YAAY;AAAA,IACZ,OAAO,gBAA6B;AAClC,YAAM,0BAAsB,uCAAe,YAAY,aAAa,WAAW;AAC/E,0BAAoB,eAAe;AACnC,UAAI,UAAU;AACd,aAAO,CAAC,oBAAoB,SAAS;AACnC;AACA,YAAI;AACJ,YAAI;AACF,sBAAY,MAAM,GAAG,mBAAmB;AAAA,QAC1C,SAAS,OAAO;AAEd,cAAI,oBAAoB,WAAW,CAAC,YAAY,sBAAsB,qBAAqB,IAAI,KAAc,GAAG;AAC9G,kBAAM,IAAI,mCAAsB,2BAA2B,YAAY,KAAK;AAAA,UAC9E;AACA,uCAAW,KAAK;AAChB,sBAAY;AAAA,QACd;AACA,YAAI,WAAW;AACb,gDAAoB,0BAA0B,YAAY,sCAAsC,OAAO,OAAO,CAAC,aAAa;AAAA,YAC1H;AAAA,UACF,CAAC;AACD;AAAA,QACF;AAEA;AAAA,UACE;AAAA,UACA;AAAA,UACA,iBAAiB,OAAO,OAAO,CAAC,8CAA8C,OAAO,YAAY,wBAAwB,CAAC;AAAA,UAC1H;AAAA,YACE;AAAA,UACF;AAAA,QACF;AAEA,cAAM,MAAM,YAAY,0BAA0B,WAAW;AAAA,MAC/D;AAAA,IACF;AAAA,IACA,EAAE,SAAS,GAAG;AAAA,IACd;AAAA,EACF;AACF;AAcA,eAAsB,eACpB,uBACA,IACA,SACA,YACY;AACZ,qBAAe,4BAAc,CAAC;AAC9B,QAAM,YAAY,YAAY,IAAI;AAElC,QAAM,kBAAkB,IAAI,gBAAgB;AAE5C,QAAM,QAAQ,KAAK,CAAC,IAAI,GAAG,aAAa,CAAC,CAAC;AAC1C,MAAI,gBAAgB,OAAO,kBAAkB,QAAQ;AACnD,WAAO,gBAAgB,OAAO,OAAO;AAAA,EACvC;AAEA,QAAM,IAAI,mCAAsB,2BAA2B,YAAY,gBAAgB,OAAO,MAAiB;AAE/G,iBAAe,MAAqB;AAClC,QAAI;AACF,YAAM,SAAS,MAAM,GAAG,gBAAgB,MAAM;AAC9C,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,YAAM,6BAAyB,6BAAe,sBAAsB;AACpE,4CAAoB,wBAAwB,cAAc,IAAI,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AACjI,sBAAgB,MAAM,IAAI,OAAO,MAAM,CAAC;AAAA,IAC1C,SAAS,GAAG;AACV,sBAAgB,MAAM,CAAC;AAAA,IACzB;AAAA,EACF;AAEA,iBAAe,eAA8B;AAC3C,WAAO,CAAC,gBAAgB,OAAO,SAAS;AACtC,YAAM,MAAM,uBAAuB,gBAAgB,MAAM;AAEzD,UAAI,gBAAgB,OAAO,SAAS;AAClC;AAAA,MACF;AACA,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,cAAQ,KAAK,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AAChF,YAAM,sBAAkB,6BAAe,8BAA8B;AACrE,UAAI,CAAC,gBAAgB,SAAS;AAC5B,wBAAgB,MAAM,IAAI,MAAM,mBAAmB,OAAO,QAAQ,CAAC,eAAe,CAAC;AACnF;AAAA,MACF;AAEA;AAAA,QACE,oDAAoD,gBAAgB,SAAS;AAAA,MAC/E;AAAA,IACF;AAAA,EACF;AACF;AAOA,eAAsB,oBAAmC;AACvD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,iBAAa,MAAM;AACjB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAQA,eAAsB,gBAAgB,OAA+B;AACnE,QAAM,IAAI,QAAQ,CAAC,YAAY;AAC7B,eAAW,SAAS,KAAK;AAAA,EAC3B,CAAC;AACH;AAUA,eAAsB,MAAM,cAAsB,aAA2B,oBAA6C;AACxH,YAAM,yCAAa,uCAAe,iBAAa,2CAAmB,YAAY,CAAC,CAAC;AAChF,MAAI,oBAAoB;AACtB,iBAAa,eAAe;AAAA,EAC9B;AACF;AAUA,eAAsB,QAAQ,uBAA+B,aAA2B,oBAA8C;AACpI,QAAM,MAAM,uBAAuB,aAAa,kBAAkB;AAClE,QAAM,IAAI,MAAM,gBAAgB,OAAO,qBAAqB,CAAC,eAAe;AAC9E;AASA,eAAsB,QAAW,MAA8C;AAC7E,QAAM,MAAW,CAAC;AAClB,mBAAiB,QAAQ,MAAM;AAC7B,QAAI,KAAK,IAAI;AAAA,EACf;AACA,SAAO;AACT;",
  "names": []
}

|
|
341
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  abortSignalAny,\n  abortSignalNever,\n  abortSignalTimeout,\n  waitForAbort\n} from './AbortController.ts';\nimport {\n  getLibDebugger,\n  printWithStackTrace\n} from './Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError,\n  SilentError\n} from './Error.ts';\nimport { noop } from './Function.ts';\n\n/**\n * A type representing a function that resolves a {@link Promise}.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryOptions {\n  /**\n   * A abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * A delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * A maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Adds an error handler to a {@link Promise} that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>, stackTrace?: string): Promise<void> {\n  stackTrace ??= getStackTrace(1);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    if (asyncError instanceof SilentError) {\n      printSilentError(asyncError, stackTrace);\n      return;\n    }\n    emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const ans: T[] = [];\n\n  const length = arr.length;\n  for (let i = 0; i < length; i++) {\n    if (!Object.hasOwn(arr, i)) {\n      continue;\n    }\n\n    const item = arr[i] as T;\n    if (await predicate(item, i, arr)) {\n      ans.push(item);\n    }\n  }\n\n  return ans;\n}\n\n/**\n * Filters an array asynchronously in place, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves when the array is filtered.\n */\nexport async function asyncFilterInPlace<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<void> {\n  const length = arr.length;\n  let writeIndex = 0;\n  for (let readIndex = 0; readIndex < length; readIndex++) {\n    if (!Object.hasOwn(arr, readIndex)) {\n      continue;\n    }\n\n    const current = arr[readIndex] as T;\n    if (await predicate(current, readIndex, arr)) {\n      // eslint-disable-next-line require-atomic-updates\n      arr[writeIndex++] = current;\n    }\n  }\n  arr.length = writeIndex;\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await promiseAllSequentially(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @param stackTrace - The stack trace of the source function.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>, stackTrace?: string): (...args: Args) => void {\n  stackTrace ??= getStackTrace(1);\n  return (...args: Args): void => {\n    const innerStackTrace = getStackTrace(1);\n    stackTrace = `${stackTrace ?? ''}\\n    at --- convertAsyncToSync --- (0)\\n${innerStackTrace}`;\n    invokeAsyncSafely(() => asyncFunc(...args), stackTrace);\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a {@link Promise}.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return async (...args: Args): Promise<Result> => {\n    await Promise.resolve();\n    return syncFn(...args);\n  };\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A {@link Promise} that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  const ignoreErrorDebugger = getLibDebugger('Async:ignoreError');\n  const stackTrace = getStackTrace(1);\n  try {\n    return await promise;\n  } catch (e) {\n    ignoreErrorDebugger('Ignored error', new CustomStackTraceError('Ignored error', stackTrace, e));\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a {@link Promise} and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n * @param stackTrace - The stack trace of the source function.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>, stackTrace?: string): void {\n  stackTrace ??= getStackTrace(1);\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn, stackTrace);\n}\n\n/**\n * Invokes an asynchronous function after a delay.\n *\n * @param asyncFn - The asynchronous function to invoke.\n * @param delayInMilliseconds - The delay in milliseconds.\n * @param stackTrace - The stack trace of the source function.\n * @param abortSignal - The abort signal to listen to.\n */\nexport function invokeAsyncSafelyAfterDelay(\n  asyncFn: (abortSignal: AbortSignal) => Promisable<void>,\n  delayInMilliseconds = 0,\n  stackTrace?: string,\n  abortSignal?: AbortSignal\n): void {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  stackTrace ??= getStackTrace(1);\n  invokeAsyncSafely(async () => {\n    await sleep(delayInMilliseconds, abortSignal, true);\n    await asyncFn(abortSignal);\n  }, stackTrace);\n}\n\n/**\n * Executes async functions sequentially.\n *\n * @typeParam T - The type of the value.\n * @param asyncFns - The async functions to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the async functions.\n */\nexport async function promiseAllAsyncFnsSequentially<T>(asyncFns: (() => Promisable<T>)[]): Promise<T[]> {\n  const results: T[] = [];\n  for (const asyncFn of asyncFns) {\n    results.push(await asyncFn());\n  }\n  return results;\n}\n\n/**\n * Executes promises sequentially.\n *\n * @typeParam T - The type of the value.\n * @param promises - The promises to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the promises.\n */\nexport async function promiseAllSequentially<T>(promises: Promisable<T>[]): Promise<T[]> {\n  return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));\n}\n\nfunction printSilentError(error: SilentError, stackTrace: string): void {\n  getLibDebugger('Async:printSilentError')('Silent error', {\n    error,\n    stackTrace\n  });\n}\n\nconst terminateRetryErrors = new WeakSet<Error>();\n\nclass Result<R> {\n  public constructor(public readonly result: R) {}\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n */\nexport function marksAsTerminateRetry(error: Error): void {\n  terminateRetryErrors.add(error);\n}\n\n/**\n * An async function that never ends.\n *\n * @returns A {@link Promise} that never resolves.\n */\nexport async function neverEnds(): Promise<never> {\n  await new Promise(() => {\n    noop();\n  });\n  throw new Error('Should never happen');\n}\n\n/**\n * Gets the next tick.\n *\n * @returns A promise that resolves when the next tick is available.\n */\nexport async function nextTickAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    process.nextTick(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next queue microtask.\n *\n * @returns A promise that resolves when the next queue microtask is available.\n */\nexport async function queueMicrotaskAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    queueMicrotask(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next request animation frame.\n *\n * @returns A promise that resolves when the next request animation frame is available.\n */\nexport async function requestAnimationFrameAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    requestAnimationFrame(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\n      const combinedAbortSignal = abortSignalAny(fullOptions.abortSignal, abortSignal);\n      combinedAbortSignal.throwIfAborted();\n      let attempt = 0;\n      while (!combinedAbortSignal.aborted) {\n        attempt++;\n        let isSuccess: boolean;\n        try {\n          isSuccess = await fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n          if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error as Error)) {\n            throw new CustomStackTraceError('retryWithTimeout failed', stackTrace, error);\n          }\n          printError(error);\n          isSuccess = false;\n        }\n        if (isSuccess) {\n          printWithStackTrace(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {\n            fn\n          });\n          return;\n        }\n\n        printWithStackTrace(\n          retryWithTimeoutDebugger,\n          stackTrace,\n          `Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,\n          {\n            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\n  );\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<R>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<R> {\n  stackTrace ??= getStackTrace(1);\n  const startTime = performance.now();\n\n  const abortController = new AbortController();\n\n  await Promise.race([run(), innerTimeout()]);\n  if (abortController.signal.reason instanceof Result) {\n    return abortController.signal.reason.result as R;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, abortController.signal.reason as unknown);\n\n  async function run(): Promise<void> {\n    try {\n      const result = await fn(abortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      abortController.abort(new Result(result));\n    } catch (e) {\n      abortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    while (!abortController.signal.aborted) {\n      await sleep(timeoutInMilliseconds, abortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (abortController.signal.aborted) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        abortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\n  }\n}\n\n/**\n * Gets the next set immediate.\n *\n * @returns A promise that resolves when the next set immediate is available.\n */\nexport async function setImmediateAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    setImmediate(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param delay - The time to wait in milliseconds.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function setTimeoutAsync(delay?: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, delay);\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<void> {\n  await waitForAbort(abortSignalAny(abortSignal, abortSignalTimeout(milliseconds)));\n  if (shouldThrowOnAbort) {\n    abortSignal?.throwIfAborted();\n  }\n}\n\n/**\n * Returns a {@link Promise} that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<never> {\n  await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);\n  throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A {@link Promise} that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/dist/lib/cjs/Error.cjs
CHANGED
|
@@ -27,6 +27,7 @@ var Error_exports = {};
|
|
|
27
27
|
__export(Error_exports, {
|
|
28
28
|
ASYNC_WRAPPER_ERROR_MESSAGE: () => ASYNC_WRAPPER_ERROR_MESSAGE,
|
|
29
29
|
CustomStackTraceError: () => CustomStackTraceError,
|
|
30
|
+
SilentError: () => SilentError,
|
|
30
31
|
emitAsyncErrorEvent: () => emitAsyncErrorEvent,
|
|
31
32
|
errorToString: () => errorToString,
|
|
32
33
|
getStackTrace: () => getStackTrace,
|
|
@@ -72,6 +73,18 @@ class CustomStackTraceError extends Error {
|
|
|
72
73
|
this.stack = originalStackLines.join("\n");
|
|
73
74
|
}
|
|
74
75
|
}
|
|
76
|
+
class SilentError extends Error {
|
|
77
|
+
/**
|
|
78
|
+
* Creates a new SilentError.
|
|
79
|
+
*
|
|
80
|
+
* @param message - The message of the error.
|
|
81
|
+
*/
|
|
82
|
+
constructor(message) {
|
|
83
|
+
super(message);
|
|
84
|
+
this.name = "SilentError";
|
|
85
|
+
Error.captureStackTrace(this, SilentError);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
75
88
|
function emitAsyncErrorEvent(asyncError) {
|
|
76
89
|
asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);
|
|
77
90
|
}
|
|
@@ -140,6 +153,7 @@ function parseErrorEntries(error, level = 0, entries = []) {
|
|
|
140
153
|
0 && (module.exports = {
|
|
141
154
|
ASYNC_WRAPPER_ERROR_MESSAGE,
|
|
142
155
|
CustomStackTraceError,
|
|
156
|
+
SilentError,
|
|
143
157
|
emitAsyncErrorEvent,
|
|
144
158
|
errorToString,
|
|
145
159
|
getStackTrace,
|
|
@@ -147,4 +161,4 @@ function parseErrorEntries(error, level = 0, entries = []) {
|
|
|
147
161
|
registerAsyncErrorEventHandler,
|
|
148
162
|
throwExpression
|
|
149
163
|
});
|
|
150
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Error.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for error handling.\n */\n\nimport { AsyncEvents } from './AsyncEvents.ts';\n\nconst ASYNC_ERROR_EVENT = 'asyncError';\n\nconst ERROR_STACK_PREFIX = 'Error stack:\\n';\n\nconst asyncErrorEventEmitter = new AsyncEvents();\nasyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handleAsyncError);\n\ninterface ErrorEntry {\n  level: number;\n  message: string;\n  shouldClearAnsiSequence?: boolean;\n}\n\n/**\n * A message of the AsyncWrapperError.\n */\nexport const ASYNC_WRAPPER_ERROR_MESSAGE = 'An unhandled error occurred executing async operation';\n\n/**\n * An error that wraps an error with a custom stack trace.\n */\nexport class CustomStackTraceError extends Error {\n  /**\n   * Creates a new CustomStackTraceError.\n   *\n   * @param message - The message of the error.\n   * @param stackTrace - The stack trace of the error.\n   * @param cause - The cause of the error.\n   */\n  public constructor(message: string, stackTrace: string, cause: unknown) {\n    super(message, { cause });\n    this.name = 'CustomStackTraceError';\n    Error.captureStackTrace(this, CustomStackTraceError);\n\n    let rootCause = cause;\n    const parentCauses = new Set<CustomStackTraceError>();\n    while (rootCause instanceof CustomStackTraceError) {\n      if (parentCauses.has(rootCause)) {\n        throw new Error('Circular cause detected');\n      }\n      parentCauses.add(rootCause);\n      rootCause = rootCause.cause;\n    }\n\n    const originalStackLines = (this.stack ?? '').split('\\n');\n    const stackLines = stackTrace.split('\\n');\n    const ERROR_HEADER_REG_EXP = /^\\w*Error(?:: |$)/;\n    if (ERROR_HEADER_REG_EXP.test(stackLines[0] ?? '')) {\n      stackLines.splice(0, 1);\n    }\n    originalStackLines.splice(1, originalStackLines.length - 1, ...stackLines);\n    this.stack = originalStackLines.join('\\n');\n  }\n}\n\n/**\n * Emits an asynchronous error event.\n *\n * @param asyncError - The error to emit as an asynchronous error event.\n */\nexport function emitAsyncErrorEvent(asyncError: unknown): void {\n  asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);\n}\n\n/**\n * Converts an error to a string representation, including nested causes with indentation.\n *\n * @param error - The error to convert to a string.\n * @param shouldEnsureAnsiSequencesCleared - Whether to ensure ANSI sequences in the error message are cleared.\n * @returns The string representation of the error.\n */\nexport function errorToString(error: unknown, shouldEnsureAnsiSequencesCleared = false): string {\n  const ANSI_CLEAR_SEQUENCE = '\\x1b[0m';\n  return parseErrorEntries(error)\n    .map((entry) => {\n      const prefix = '  '.repeat(entry.level);\n      // NOTE: Cannot use `String.trimStart()` function here because of the circular dependency.\n      let message = entry.message.startsWith(ERROR_STACK_PREFIX) ? entry.message.slice(ERROR_STACK_PREFIX.length) : entry.message;\n      if (entry.shouldClearAnsiSequence && shouldEnsureAnsiSequencesCleared) {\n        message = `${ANSI_CLEAR_SEQUENCE}${message}${ANSI_CLEAR_SEQUENCE}`;\n      }\n      return prefix + message;\n    })\n    .join('\\n');\n}\n\n/**\n * Gets the current stack trace as a string, excluding the current function call.\n *\n * @param framesToSkip - The number of frames to skip in the stack trace.\n * @returns A string representation of the current stack trace, excluding the current function call.\n */\nexport function getStackTrace(framesToSkip = 0): string {\n  // Skipping Error prefix and `getStackTrace` function call\n  const ADDITIONAL_FRAMES_TO_SKIP = 2;\n  const stack = new Error().stack ?? '';\n  const lines = stack.split('\\n');\n  return lines.slice(framesToSkip + ADDITIONAL_FRAMES_TO_SKIP).join('\\n');\n}\n\n/**\n * Prints an error to the console, including nested causes and optional ANSI sequence clearing.\n *\n * @param error - The error to print.\n * @param console - The console to print to (default: `globalThis.console`).\n */\nexport function printError(error: unknown, console?: Console): void {\n  console ??= globalThis.console;\n  console.error(errorToString(error, true));\n}\n\n/**\n * Registers an event handler for asynchronous errors.\n *\n * @param handler - The handler function to be called when an asynchronous error event occurs.\n * @returns A function to unregister the handler.\n */\nexport function registerAsyncErrorEventHandler(handler: (asyncError: unknown) => void): () => void {\n  const eventRef = asyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handler);\n  return () => {\n    asyncErrorEventEmitter.offref(eventRef);\n  };\n}\n\n/**\n * Throws an error with the specified message.\n *\n * @param error - The error to throw.\n * @throws\n */\nexport function throwExpression(error: unknown): never {\n  throw error;\n}\n\n/**\n * Handles asynchronous errors by printing them.\n *\n * @param asyncError - The asynchronous error to handle.\n */\nfunction handleAsyncError(asyncError: unknown): void {\n  printError(asyncError);\n}\n\n/**\n * Parses an error into an array of ErrorEntry objects, including nested causes.\n *\n * @param error - The error to parse.\n * @param level - The current indentation level for nested causes.\n * @param entries - The array of ErrorEntry objects to populate.\n * @returns An array of ErrorEntry objects representing the error and its causes.\n */\nfunction parseErrorEntries(error: unknown, level = 0, entries: ErrorEntry[] = []): ErrorEntry[] {\n  if (error === undefined) {\n    return entries;\n  }\n\n  if (!(error instanceof Error)) {\n    let str: string;\n\n    if (error === null) {\n      str = '(null)';\n    } else if (typeof error === 'string') {\n      str = error;\n    } else {\n      str = JSON.stringify(error) ?? 'undefined';\n    }\n\n    entries.push({ level, message: str });\n    return entries;\n  }\n\n  const title = `${error.name}: ${error.message}`;\n  entries.push({ level, message: title, shouldClearAnsiSequence: true });\n\n  if (error.stack) {\n    const restStack = error.stack.startsWith(title) ? error.stack.slice(title.length + 1) : error.stack;\n    entries.push({ level, message: `${ERROR_STACK_PREFIX}${restStack}` });\n  }\n\n  if (error.cause !== undefined) {\n    entries.push({ level, message: 'Caused by:' });\n    parseErrorEntries(error.cause, level + 1, entries);\n  }\n\n  return entries;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Error.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for error handling.\n */\n\nimport { AsyncEvents } from './AsyncEvents.ts';\n\nconst ASYNC_ERROR_EVENT = 'asyncError';\n\nconst ERROR_STACK_PREFIX = 'Error stack:\\n';\n\nconst asyncErrorEventEmitter = new AsyncEvents();\nasyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handleAsyncError);\n\ninterface ErrorEntry {\n  level: number;\n  message: string;\n  shouldClearAnsiSequence?: boolean;\n}\n\n/**\n * A message of the AsyncWrapperError.\n */\nexport const ASYNC_WRAPPER_ERROR_MESSAGE = 'An unhandled error occurred executing async operation';\n\n/**\n * An error that wraps an error with a custom stack trace.\n */\nexport class CustomStackTraceError extends Error {\n  /**\n   * Creates a new CustomStackTraceError.\n   *\n   * @param message - The message of the error.\n   * @param stackTrace - The stack trace of the error.\n   * @param cause - The cause of the error.\n   */\n  public constructor(message: string, stackTrace: string, cause: unknown) {\n    super(message, { cause });\n    this.name = 'CustomStackTraceError';\n    Error.captureStackTrace(this, CustomStackTraceError);\n\n    let rootCause = cause;\n    const parentCauses = new Set<CustomStackTraceError>();\n    while (rootCause instanceof CustomStackTraceError) {\n      if (parentCauses.has(rootCause)) {\n        throw new Error('Circular cause detected');\n      }\n      parentCauses.add(rootCause);\n      rootCause = rootCause.cause;\n    }\n\n    const originalStackLines = (this.stack ?? '').split('\\n');\n    const stackLines = stackTrace.split('\\n');\n    const ERROR_HEADER_REG_EXP = /^\\w*Error(?:: |$)/;\n    if (ERROR_HEADER_REG_EXP.test(stackLines[0] ?? '')) {\n      stackLines.splice(0, 1);\n    }\n    originalStackLines.splice(1, originalStackLines.length - 1, ...stackLines);\n    this.stack = originalStackLines.join('\\n');\n  }\n}\n\n/**\n * An error that is not printed to the console.\n */\nexport class SilentError extends Error {\n  /**\n   * Creates a new SilentError.\n   *\n   * @param message - The message of the error.\n   */\n  public constructor(message: string) {\n    super(message);\n    this.name = 'SilentError';\n    Error.captureStackTrace(this, SilentError);\n  }\n}\n\n/**\n * Emits an asynchronous error event.\n *\n * @param asyncError - The error to emit as an asynchronous error event.\n */\nexport function emitAsyncErrorEvent(asyncError: unknown): void {\n  asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);\n}\n\n/**\n * Converts an error to a string representation, including nested causes with indentation.\n *\n * @param error - The error to convert to a string.\n * @param shouldEnsureAnsiSequencesCleared - Whether to ensure ANSI sequences in the error message are cleared.\n * @returns The string representation of the error.\n */\nexport function errorToString(error: unknown, shouldEnsureAnsiSequencesCleared = false): string {\n  const ANSI_CLEAR_SEQUENCE = '\\x1b[0m';\n  return parseErrorEntries(error)\n    .map((entry) => {\n      const prefix = '  '.repeat(entry.level);\n      // NOTE: Cannot use `String.trimStart()` function here because of the circular dependency.\n      let message = entry.message.startsWith(ERROR_STACK_PREFIX) ? entry.message.slice(ERROR_STACK_PREFIX.length) : entry.message;\n      if (entry.shouldClearAnsiSequence && shouldEnsureAnsiSequencesCleared) {\n        message = `${ANSI_CLEAR_SEQUENCE}${message}${ANSI_CLEAR_SEQUENCE}`;\n      }\n      return prefix + message;\n    })\n    .join('\\n');\n}\n\n/**\n * Gets the current stack trace as a string, excluding the current function call.\n *\n * @param framesToSkip - The number of frames to skip in the stack trace.\n * @returns A string representation of the current stack trace, excluding the current function call.\n */\nexport function getStackTrace(framesToSkip = 0): string {\n  // Skipping Error prefix and `getStackTrace` function call\n  const ADDITIONAL_FRAMES_TO_SKIP = 2;\n  const stack = new Error().stack ?? '';\n  const lines = stack.split('\\n');\n  return lines.slice(framesToSkip + ADDITIONAL_FRAMES_TO_SKIP).join('\\n');\n}\n\n/**\n * Prints an error to the console, including nested causes and optional ANSI sequence clearing.\n *\n * @param error - The error to print.\n * @param console - The console to print to (default: `globalThis.console`).\n */\nexport function printError(error: unknown, console?: Console): void {\n  console ??= globalThis.console;\n  console.error(errorToString(error, true));\n}\n\n/**\n * Registers an event handler for asynchronous errors.\n *\n * @param handler - The handler function to be called when an asynchronous error event occurs.\n * @returns A function to unregister the handler.\n */\nexport function registerAsyncErrorEventHandler(handler: (asyncError: unknown) => void): () => void {\n  const eventRef = asyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handler);\n  return () => {\n    asyncErrorEventEmitter.offref(eventRef);\n  };\n}\n\n/**\n * Throws an error with the specified message.\n *\n * @param error - The error to throw.\n * @throws\n */\nexport function throwExpression(error: unknown): never {\n  throw error;\n}\n\n/**\n * Handles asynchronous errors by printing them.\n *\n * @param asyncError - The asynchronous error to handle.\n */\nfunction handleAsyncError(asyncError: unknown): void {\n  printError(asyncError);\n}\n\n/**\n * Parses an error into an array of ErrorEntry objects, including nested causes.\n *\n * @param error - The error to parse.\n * @param level - The current indentation level for nested causes.\n * @param entries - The array of ErrorEntry objects to populate.\n * @returns An array of ErrorEntry objects representing the error and its causes.\n */\nfunction parseErrorEntries(error: unknown, level = 0, entries: ErrorEntry[] = []): ErrorEntry[] {\n  if (error === undefined) {\n    return entries;\n  }\n\n  if (!(error instanceof Error)) {\n    let str: string;\n\n    if (error === null) {\n      str = '(null)';\n    } else if (typeof error === 'string') {\n      str = error;\n    } else {\n      str = JSON.stringify(error) ?? 'undefined';\n    }\n\n    entries.push({ level, message: str });\n    return entries;\n  }\n\n  const title = `${error.name}: ${error.message}`;\n  entries.push({ level, message: title, shouldClearAnsiSequence: true });\n\n  if (error.stack) {\n    const restStack = error.stack.startsWith(title) ? error.stack.slice(title.length + 1) : error.stack;\n    entries.push({ level, message: `${ERROR_STACK_PREFIX}${restStack}` });\n  }\n\n  if (error.cause !== undefined) {\n    entries.push({ level, message: 'Caused by:' });\n    parseErrorEntries(error.cause, level + 1, entries);\n  }\n\n  return entries;\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/dist/lib/cjs/Error.d.cts
CHANGED
|
@@ -20,6 +20,17 @@ export declare class CustomStackTraceError extends Error {
|
|
|
20
20
|
*/
|
|
21
21
|
constructor(message: string, stackTrace: string, cause: unknown);
|
|
22
22
|
}
|
|
23
|
+
/**
|
|
24
|
+
* An error that is not printed to the console.
|
|
25
|
+
*/
|
|
26
|
+
export declare class SilentError extends Error {
|
|
27
|
+
/**
|
|
28
|
+
* Creates a new SilentError.
|
|
29
|
+
*
|
|
30
|
+
* @param message - The message of the error.
|
|
31
|
+
*/
|
|
32
|
+
constructor(message: string);
|
|
33
|
+
}
|
|
23
34
|
/**
|
|
24
35
|
* Emits an asynchronous error event.
|
|
25
36
|
*
|
package/dist/lib/cjs/Library.cjs
CHANGED
|
@@ -30,7 +30,7 @@ __export(Library_exports, {
|
|
|
30
30
|
LIBRARY_VERSION: () => LIBRARY_VERSION
|
|
31
31
|
});
|
|
32
32
|
module.exports = __toCommonJS(Library_exports);
|
|
33
|
-
const LIBRARY_VERSION = "37.
|
|
33
|
+
const LIBRARY_VERSION = "37.2.0";
|
|
34
34
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
|
35
35
|
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-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.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/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.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;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;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&.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&.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";
|
|
36
36
|
// Annotate the CommonJS export names for ESM import in node:
|
|
@@ -217,7 +217,7 @@ class PluginBase extends import_obsidian.Plugin {
|
|
|
217
217
|
const abortController = new AbortController();
|
|
218
218
|
this._abortSignal = abortController.signal;
|
|
219
219
|
this.register(() => {
|
|
220
|
-
abortController.abort(new
|
|
220
|
+
abortController.abort(new import_Error.SilentError(`Plugin ${this.manifest.id} had been unloaded`));
|
|
221
221
|
});
|
|
222
222
|
}
|
|
223
223
|
/**
|
|
@@ -280,4 +280,4 @@ ${message}`);
|
|
|
280
280
|
0 && (module.exports = {
|
|
281
281
|
PluginBase
|
|
282
282
|
});
|
|
283
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { ReadonlyDeep } from 'type-fest';\n\nimport {\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type {\n  ExtractPluginSettings,\n  ExtractPluginSettingsManager,\n  ExtractPluginSettingsTab,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely,\n  invokeAsyncSafelyAfterDelay\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { getDebugger } from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noopAsync } from '../../Function.ts';\nimport { registerAsyncEvent } from '../Components/AsyncEventsComponent.ts';\nimport { initPluginContext } from './PluginContext.ts';\n\ntype LifecycleEventName = 'layoutReady' | 'load' | 'unload';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginBase<PluginTypes extends PluginTypesBase> extends ObsidianPlugin {\n  public readonly events = new AsyncEvents();\n\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<ExtractPluginSettings<PluginTypes>> {\n    return this.settingsManager.settingsWrapper.safeSettings as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n  }\n\n  /**\n   * Gets the plugin settings manager.\n   *\n   * @returns The plugin settings manager.\n   */\n  public get settingsManager(): ExtractPluginSettingsManager<PluginTypes> {\n    if (!this._settingsManager) {\n      throw new Error('Settings manager not defined');\n    }\n\n    return this._settingsManager;\n  }\n\n  /**\n   * Gets the plugin settings tab.\n   *\n   * @returns The plugin settings tab.\n   */\n  public get settingsTab(): ExtractPluginSettingsTab<PluginTypes> {\n    if (!this._settingsTab) {\n      throw new Error('Settings tab not defined');\n    }\n\n    return this._settingsTab;\n  }\n\n  private _abortSignal!: AbortSignal;\n  private _settingsManager: ExtractPluginSettingsManager<PluginTypes> | null = null;\n  private _settingsTab: ExtractPluginSettingsTab<PluginTypes> | null = null;\n  private lifecycleEventNames = new Set<LifecycleEventName>();\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const pluginDebugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    pluginDebugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onLoadSettings} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this._settingsManager?.loadFromFile(false);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   *\n   * Usually, you don't need to override this method. Consider using {@link onloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onload()` first.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onunloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `super.onunload()` first.\n   */\n  public override onunload(): void {\n    super.onunload();\n    invokeAsyncSafely(async () => {\n      try {\n        await this.onunloadImpl();\n      } finally {\n        await this.triggerLifecycleEvent('unload');\n      }\n    });\n  }\n\n  /**\n   * Registers a callback to be executed when a lifecycle event is triggered.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback to execute.\n   */\n  public registerForLifecycleEvent(name: LifecycleEventName, callback: () => Promise<void>): void {\n    invokeAsyncSafely(async () => {\n      await this.waitForLifecycleEvent(name);\n      await callback();\n    });\n  }\n\n  /**\n   * Waits for a lifecycle event to be triggered.\n   *\n   * If you `await` this method during lifecycle event, it might cause a deadlock.\n   *\n   * Consider wrapping this call with {@link invokeAsyncSafely}.\n   *\n   * @param name - The name of the event.\n   * @returns A {@link Promise} that resolves when the event is triggered.\n   */\n  public async waitForLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    if (this.lifecycleEventNames.has(name)) {\n      return;\n    }\n\n    await new Promise<void>((resolve) => {\n      this.events.once(name, () => {\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected createSettingsManager(): ExtractPluginSettingsManager<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Creates a plugin settings tab.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected createSettingsTab(): ExtractPluginSettingsTab<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Executed when the plugin is loaded.\n   *\n   * If this method fails, the plugin will be automatically unloaded.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n    if (this._settingsManager) {\n      registerAsyncEvent(this, this._settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n      registerAsyncEvent(this, this._settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n    }\n\n    await this._settingsManager?.loadFromFile(true);\n    this._settingsTab = this.createSettingsTab();\n    if (this._settingsTab) {\n      this.addSettingTab(this._settingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort(new Error(`Plugin ${this.manifest.id} had been unloaded`));\n    });\n  }\n\n  /**\n   * Called when the plugin settings are loaded or reloaded.\n   *\n   * @param _loadedSettings - The loaded settings wrapper.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saved.\n   *\n   * @param _newSettings - The new settings.\n   * @param _oldSettings - The old settings.\n   * @param _context - The context.\n   */\n  protected async onSaveSettings(\n    _newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   */\n  protected async onunloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this._abortSignal.aborted) {\n      return;\n    }\n    await this.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.triggerLifecycleEvent('layoutReady');\n    }\n  }\n\n  private async triggerLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    this.lifecycleEventNames.add(name);\n    await this.events.triggerAsync(name);\n  }\n}\n"],
  "mappings": "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",
  "names": ["ObsidianPlugin"]
}

|
|
283
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { ReadonlyDeep } from 'type-fest';\n\nimport {\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type {\n  ExtractPluginSettings,\n  ExtractPluginSettingsManager,\n  ExtractPluginSettingsTab,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely,\n  invokeAsyncSafelyAfterDelay\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { getDebugger } from '../../Debug.ts';\nimport {\n  registerAsyncErrorEventHandler,\n  SilentError\n} from '../../Error.ts';\nimport { noopAsync } from '../../Function.ts';\nimport { registerAsyncEvent } from '../Components/AsyncEventsComponent.ts';\nimport { initPluginContext } from './PluginContext.ts';\n\ntype LifecycleEventName = 'layoutReady' | 'load' | 'unload';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginBase<PluginTypes extends PluginTypesBase> extends ObsidianPlugin {\n  public readonly events = new AsyncEvents();\n\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<ExtractPluginSettings<PluginTypes>> {\n    return this.settingsManager.settingsWrapper.safeSettings as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n  }\n\n  /**\n   * Gets the plugin settings manager.\n   *\n   * @returns The plugin settings manager.\n   */\n  public get settingsManager(): ExtractPluginSettingsManager<PluginTypes> {\n    if (!this._settingsManager) {\n      throw new Error('Settings manager not defined');\n    }\n\n    return this._settingsManager;\n  }\n\n  /**\n   * Gets the plugin settings tab.\n   *\n   * @returns The plugin settings tab.\n   */\n  public get settingsTab(): ExtractPluginSettingsTab<PluginTypes> {\n    if (!this._settingsTab) {\n      throw new Error('Settings tab not defined');\n    }\n\n    return this._settingsTab;\n  }\n\n  private _abortSignal!: AbortSignal;\n  private _settingsManager: ExtractPluginSettingsManager<PluginTypes> | null = null;\n  private _settingsTab: ExtractPluginSettingsTab<PluginTypes> | null = null;\n  private lifecycleEventNames = new Set<LifecycleEventName>();\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const pluginDebugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    pluginDebugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onLoadSettings} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this._settingsManager?.loadFromFile(false);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   *\n   * Usually, you don't need to override this method. Consider using {@link onloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onload()` first.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onunloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `super.onunload()` first.\n   */\n  public override onunload(): void {\n    super.onunload();\n    invokeAsyncSafely(async () => {\n      try {\n        await this.onunloadImpl();\n      } finally {\n        await this.triggerLifecycleEvent('unload');\n      }\n    });\n  }\n\n  /**\n   * Registers a callback to be executed when a lifecycle event is triggered.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback to execute.\n   */\n  public registerForLifecycleEvent(name: LifecycleEventName, callback: () => Promise<void>): void {\n    invokeAsyncSafely(async () => {\n      await this.waitForLifecycleEvent(name);\n      await callback();\n    });\n  }\n\n  /**\n   * Waits for a lifecycle event to be triggered.\n   *\n   * If you `await` this method during lifecycle event, it might cause a deadlock.\n   *\n   * Consider wrapping this call with {@link invokeAsyncSafely}.\n   *\n   * @param name - The name of the event.\n   * @returns A {@link Promise} that resolves when the event is triggered.\n   */\n  public async waitForLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    if (this.lifecycleEventNames.has(name)) {\n      return;\n    }\n\n    await new Promise<void>((resolve) => {\n      this.events.once(name, () => {\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected createSettingsManager(): ExtractPluginSettingsManager<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Creates a plugin settings tab.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected createSettingsTab(): ExtractPluginSettingsTab<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Executed when the plugin is loaded.\n   *\n   * If this method fails, the plugin will be automatically unloaded.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n    if (this._settingsManager) {\n      registerAsyncEvent(this, this._settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n      registerAsyncEvent(this, this._settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n    }\n\n    await this._settingsManager?.loadFromFile(true);\n    this._settingsTab = this.createSettingsTab();\n    if (this._settingsTab) {\n      this.addSettingTab(this._settingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort(new SilentError(`Plugin ${this.manifest.id} had been unloaded`));\n    });\n  }\n\n  /**\n   * Called when the plugin settings are loaded or reloaded.\n   *\n   * @param _loadedSettings - The loaded settings wrapper.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saved.\n   *\n   * @param _newSettings - The new settings.\n   * @param _oldSettings - The old settings.\n   * @param _context - The context.\n   */\n  protected async onSaveSettings(\n    _newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   */\n  protected async onunloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this._abortSignal.aborted) {\n      return;\n    }\n    await this.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.triggerLifecycleEvent('layoutReady');\n    }\n  }\n\n  private async triggerLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    this.lifecycleEventNames.add(name);\n    await this.events.triggerAsync(name);\n  }\n}\n"],
  "mappings": "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",
  "names": ["ObsidianPlugin"]
}

|
package/dist/lib/esm/Async.mjs
CHANGED
|
@@ -20,7 +20,8 @@ import {
|
|
|
20
20
|
CustomStackTraceError,
|
|
21
21
|
emitAsyncErrorEvent,
|
|
22
22
|
getStackTrace,
|
|
23
|
-
printError
|
|
23
|
+
printError,
|
|
24
|
+
SilentError
|
|
24
25
|
} from "./Error.mjs";
|
|
25
26
|
import { noop } from "./Function.mjs";
|
|
26
27
|
async function addErrorHandler(asyncFn, stackTrace) {
|
|
@@ -28,6 +29,10 @@ async function addErrorHandler(asyncFn, stackTrace) {
|
|
|
28
29
|
try {
|
|
29
30
|
await asyncFn();
|
|
30
31
|
} catch (asyncError) {
|
|
32
|
+
if (asyncError instanceof SilentError) {
|
|
33
|
+
printSilentError(asyncError, stackTrace);
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
31
36
|
emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));
|
|
32
37
|
}
|
|
33
38
|
}
|
|
@@ -114,6 +119,12 @@ async function promiseAllAsyncFnsSequentially(asyncFns) {
|
|
|
114
119
|
async function promiseAllSequentially(promises) {
|
|
115
120
|
return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));
|
|
116
121
|
}
|
|
122
|
+
function printSilentError(error, stackTrace) {
|
|
123
|
+
getLibDebugger("Async:printSilentError")("Silent error", {
|
|
124
|
+
error,
|
|
125
|
+
stackTrace
|
|
126
|
+
});
|
|
127
|
+
}
|
|
117
128
|
const terminateRetryErrors = /* @__PURE__ */ new WeakSet();
|
|
118
129
|
class Result {
|
|
119
130
|
constructor(result) {
|
|
@@ -295,4 +306,4 @@ export {
|
|
|
295
306
|
timeout,
|
|
296
307
|
toArray
|
|
297
308
|
};
|
|
298
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  abortSignalAny,\n  abortSignalNever,\n  abortSignalTimeout,\n  waitForAbort\n} from './AbortController.ts';\nimport {\n  getLibDebugger,\n  printWithStackTrace\n} from './Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError\n} from './Error.ts';\nimport { noop } from './Function.ts';\n\n/**\n * A type representing a function that resolves a {@link Promise}.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryOptions {\n  /**\n   * A abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * A delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * A maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Adds an error handler to a {@link Promise} that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>, stackTrace?: string): Promise<void> {\n  stackTrace ??= getStackTrace(1);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const ans: T[] = [];\n\n  const length = arr.length;\n  for (let i = 0; i < length; i++) {\n    if (!Object.hasOwn(arr, i)) {\n      continue;\n    }\n\n    const item = arr[i] as T;\n    if (await predicate(item, i, arr)) {\n      ans.push(item);\n    }\n  }\n\n  return ans;\n}\n\n/**\n * Filters an array asynchronously in place, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves when the array is filtered.\n */\nexport async function asyncFilterInPlace<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<void> {\n  const length = arr.length;\n  let writeIndex = 0;\n  for (let readIndex = 0; readIndex < length; readIndex++) {\n    if (!Object.hasOwn(arr, readIndex)) {\n      continue;\n    }\n\n    const current = arr[readIndex] as T;\n    if (await predicate(current, readIndex, arr)) {\n      // eslint-disable-next-line require-atomic-updates\n      arr[writeIndex++] = current;\n    }\n  }\n  arr.length = writeIndex;\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await promiseAllSequentially(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @param stackTrace - The stack trace of the source function.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>, stackTrace?: string): (...args: Args) => void {\n  stackTrace ??= getStackTrace(1);\n  return (...args: Args): void => {\n    const innerStackTrace = getStackTrace(1);\n    stackTrace = `${stackTrace ?? ''}\\n    at --- convertAsyncToSync --- (0)\\n${innerStackTrace}`;\n    invokeAsyncSafely(() => asyncFunc(...args), stackTrace);\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a {@link Promise}.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return async (...args: Args): Promise<Result> => {\n    await Promise.resolve();\n    return syncFn(...args);\n  };\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A {@link Promise} that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  const ignoreErrorDebugger = getLibDebugger('Async:ignoreError');\n  const stackTrace = getStackTrace(1);\n  try {\n    return await promise;\n  } catch (e) {\n    ignoreErrorDebugger('Ignored error', new CustomStackTraceError('Ignored error', stackTrace, e));\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a {@link Promise} and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n * @param stackTrace - The stack trace of the source function.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>, stackTrace?: string): void {\n  stackTrace ??= getStackTrace(1);\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn, stackTrace);\n}\n\n/**\n * Invokes an asynchronous function after a delay.\n *\n * @param asyncFn - The asynchronous function to invoke.\n * @param delayInMilliseconds - The delay in milliseconds.\n * @param stackTrace - The stack trace of the source function.\n * @param abortSignal - The abort signal to listen to.\n */\nexport function invokeAsyncSafelyAfterDelay(\n  asyncFn: (abortSignal: AbortSignal) => Promisable<void>,\n  delayInMilliseconds = 0,\n  stackTrace?: string,\n  abortSignal?: AbortSignal\n): void {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  stackTrace ??= getStackTrace(1);\n  invokeAsyncSafely(async () => {\n    await sleep(delayInMilliseconds, abortSignal, true);\n    await asyncFn(abortSignal);\n  }, stackTrace);\n}\n\n/**\n * Executes async functions sequentially.\n *\n * @typeParam T - The type of the value.\n * @param asyncFns - The async functions to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the async functions.\n */\nexport async function promiseAllAsyncFnsSequentially<T>(asyncFns: (() => Promisable<T>)[]): Promise<T[]> {\n  const results: T[] = [];\n  for (const asyncFn of asyncFns) {\n    results.push(await asyncFn());\n  }\n  return results;\n}\n\n/**\n * Executes promises sequentially.\n *\n * @typeParam T - The type of the value.\n * @param promises - The promises to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the promises.\n */\nexport async function promiseAllSequentially<T>(promises: Promisable<T>[]): Promise<T[]> {\n  return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));\n}\n\nconst terminateRetryErrors = new WeakSet<Error>();\n\nclass Result<R> {\n  public constructor(public readonly result: R) {}\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n */\nexport function marksAsTerminateRetry(error: Error): void {\n  terminateRetryErrors.add(error);\n}\n\n/**\n * An async function that never ends.\n *\n * @returns A {@link Promise} that never resolves.\n */\nexport async function neverEnds(): Promise<never> {\n  await new Promise(() => {\n    noop();\n  });\n  throw new Error('Should never happen');\n}\n\n/**\n * Gets the next tick.\n *\n * @returns A promise that resolves when the next tick is available.\n */\nexport async function nextTickAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    process.nextTick(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next queue microtask.\n *\n * @returns A promise that resolves when the next queue microtask is available.\n */\nexport async function queueMicrotaskAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    queueMicrotask(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next request animation frame.\n *\n * @returns A promise that resolves when the next request animation frame is available.\n */\nexport async function requestAnimationFrameAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    requestAnimationFrame(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\n      const combinedAbortSignal = abortSignalAny(fullOptions.abortSignal, abortSignal);\n      combinedAbortSignal.throwIfAborted();\n      let attempt = 0;\n      while (!combinedAbortSignal.aborted) {\n        attempt++;\n        let isSuccess: boolean;\n        try {\n          isSuccess = await fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n          if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error as Error)) {\n            throw new CustomStackTraceError('retryWithTimeout failed', stackTrace, error);\n          }\n          printError(error);\n          isSuccess = false;\n        }\n        if (isSuccess) {\n          printWithStackTrace(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {\n            fn\n          });\n          return;\n        }\n\n        printWithStackTrace(\n          retryWithTimeoutDebugger,\n          stackTrace,\n          `Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,\n          {\n            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\n  );\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<R>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<R> {\n  stackTrace ??= getStackTrace(1);\n  const startTime = performance.now();\n\n  const abortController = new AbortController();\n\n  await Promise.race([run(), innerTimeout()]);\n  if (abortController.signal.reason instanceof Result) {\n    return abortController.signal.reason.result as R;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, abortController.signal.reason as unknown);\n\n  async function run(): Promise<void> {\n    try {\n      const result = await fn(abortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      abortController.abort(new Result(result));\n    } catch (e) {\n      abortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    while (!abortController.signal.aborted) {\n      await sleep(timeoutInMilliseconds, abortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (abortController.signal.aborted) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        abortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\n  }\n}\n\n/**\n * Gets the next set immediate.\n *\n * @returns A promise that resolves when the next set immediate is available.\n */\nexport async function setImmediateAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    setImmediate(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param delay - The time to wait in milliseconds.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function setTimeoutAsync(delay?: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, delay);\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<void> {\n  await waitForAbort(abortSignalAny(abortSignal, abortSignalTimeout(milliseconds)));\n  if (shouldThrowOnAbort) {\n    abortSignal?.throwIfAborted();\n  }\n}\n\n/**\n * Returns a {@link Promise} that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<never> {\n  await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);\n  throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A {@link Promise} that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
309
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport {\n  abortSignalAny,\n  abortSignalNever,\n  abortSignalTimeout,\n  waitForAbort\n} from './AbortController.ts';\nimport {\n  getLibDebugger,\n  printWithStackTrace\n} from './Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError,\n  SilentError\n} from './Error.ts';\nimport { noop } from './Function.ts';\n\n/**\n * A type representing a function that resolves a {@link Promise}.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for {@link retryWithTimeout}.\n */\nexport interface RetryOptions {\n  /**\n   * A abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * A delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * A maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * Adds an error handler to a {@link Promise} that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>, stackTrace?: string): Promise<void> {\n  stackTrace ??= getStackTrace(1);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    if (asyncError instanceof SilentError) {\n      printSilentError(asyncError, stackTrace);\n      return;\n    }\n    emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, asyncError));\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const ans: T[] = [];\n\n  const length = arr.length;\n  for (let i = 0; i < length; i++) {\n    if (!Object.hasOwn(arr, i)) {\n      continue;\n    }\n\n    const item = arr[i] as T;\n    if (await predicate(item, i, arr)) {\n      ans.push(item);\n    }\n  }\n\n  return ans;\n}\n\n/**\n * Filters an array asynchronously in place, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A {@link Promise} that resolves when the array is filtered.\n */\nexport async function asyncFilterInPlace<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<void> {\n  const length = arr.length;\n  let writeIndex = 0;\n  for (let readIndex = 0; readIndex < length; readIndex++) {\n    if (!Object.hasOwn(arr, readIndex)) {\n      continue;\n    }\n\n    const current = arr[readIndex] as T;\n    if (await predicate(current, readIndex, arr)) {\n      // eslint-disable-next-line require-atomic-updates\n      arr[writeIndex++] = current;\n    }\n  }\n  arr.length = writeIndex;\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A {@link Promise} that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await promiseAllSequentially(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @param stackTrace - The stack trace of the source function.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>, stackTrace?: string): (...args: Args) => void {\n  stackTrace ??= getStackTrace(1);\n  return (...args: Args): void => {\n    const innerStackTrace = getStackTrace(1);\n    stackTrace = `${stackTrace ?? ''}\\n    at --- convertAsyncToSync --- (0)\\n${innerStackTrace}`;\n    invokeAsyncSafely(() => asyncFunc(...args), stackTrace);\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a {@link Promise}.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return async (...args: Args): Promise<Result> => {\n    await Promise.resolve();\n    return syncFn(...args);\n  };\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A {@link Promise} that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A {@link Promise} that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  const ignoreErrorDebugger = getLibDebugger('Async:ignoreError');\n  const stackTrace = getStackTrace(1);\n  try {\n    return await promise;\n  } catch (e) {\n    ignoreErrorDebugger('Ignored error', new CustomStackTraceError('Ignored error', stackTrace, e));\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a {@link Promise} and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n * @param stackTrace - The stack trace of the source function.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>, stackTrace?: string): void {\n  stackTrace ??= getStackTrace(1);\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn, stackTrace);\n}\n\n/**\n * Invokes an asynchronous function after a delay.\n *\n * @param asyncFn - The asynchronous function to invoke.\n * @param delayInMilliseconds - The delay in milliseconds.\n * @param stackTrace - The stack trace of the source function.\n * @param abortSignal - The abort signal to listen to.\n */\nexport function invokeAsyncSafelyAfterDelay(\n  asyncFn: (abortSignal: AbortSignal) => Promisable<void>,\n  delayInMilliseconds = 0,\n  stackTrace?: string,\n  abortSignal?: AbortSignal\n): void {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n  stackTrace ??= getStackTrace(1);\n  invokeAsyncSafely(async () => {\n    await sleep(delayInMilliseconds, abortSignal, true);\n    await asyncFn(abortSignal);\n  }, stackTrace);\n}\n\n/**\n * Executes async functions sequentially.\n *\n * @typeParam T - The type of the value.\n * @param asyncFns - The async functions to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the async functions.\n */\nexport async function promiseAllAsyncFnsSequentially<T>(asyncFns: (() => Promisable<T>)[]): Promise<T[]> {\n  const results: T[] = [];\n  for (const asyncFn of asyncFns) {\n    results.push(await asyncFn());\n  }\n  return results;\n}\n\n/**\n * Executes promises sequentially.\n *\n * @typeParam T - The type of the value.\n * @param promises - The promises to execute sequentially.\n * @returns A {@link Promise} that resolves with an array of the results of the promises.\n */\nexport async function promiseAllSequentially<T>(promises: Promisable<T>[]): Promise<T[]> {\n  return await promiseAllAsyncFnsSequentially(promises.map((promise) => () => promise));\n}\n\nfunction printSilentError(error: SilentError, stackTrace: string): void {\n  getLibDebugger('Async:printSilentError')('Silent error', {\n    error,\n    stackTrace\n  });\n}\n\nconst terminateRetryErrors = new WeakSet<Error>();\n\nclass Result<R> {\n  public constructor(public readonly result: R) {}\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n */\nexport function marksAsTerminateRetry(error: Error): void {\n  terminateRetryErrors.add(error);\n}\n\n/**\n * An async function that never ends.\n *\n * @returns A {@link Promise} that never resolves.\n */\nexport async function neverEnds(): Promise<never> {\n  await new Promise(() => {\n    noop();\n  });\n  throw new Error('Should never happen');\n}\n\n/**\n * Gets the next tick.\n *\n * @returns A promise that resolves when the next tick is available.\n */\nexport async function nextTickAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    process.nextTick(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next queue microtask.\n *\n * @returns A promise that resolves when the next queue microtask is available.\n */\nexport async function queueMicrotaskAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    queueMicrotask(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Gets the next request animation frame.\n *\n * @returns A promise that resolves when the next request animation frame is available.\n */\nexport async function requestAnimationFrameAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    requestAnimationFrame(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A {@link Promise} that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: (abortSignal: AbortSignal) => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  fullOptions.abortSignal?.throwIfAborted();\n\n  await runWithTimeout(\n    fullOptions.timeoutInMilliseconds,\n    async (abortSignal: AbortSignal) => {\n      const combinedAbortSignal = abortSignalAny(fullOptions.abortSignal, abortSignal);\n      combinedAbortSignal.throwIfAborted();\n      let attempt = 0;\n      while (!combinedAbortSignal.aborted) {\n        attempt++;\n        let isSuccess: boolean;\n        try {\n          isSuccess = await fn(combinedAbortSignal);\n        } catch (error) {\n          // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n          if (combinedAbortSignal.aborted || !fullOptions.shouldRetryOnError || terminateRetryErrors.has(error as Error)) {\n            throw new CustomStackTraceError('retryWithTimeout failed', stackTrace, error);\n          }\n          printError(error);\n          isSuccess = false;\n        }\n        if (isSuccess) {\n          printWithStackTrace(retryWithTimeoutDebugger, stackTrace, `Retry completed successfully after ${String(attempt)} attempts`, {\n            fn\n          });\n          return;\n        }\n\n        printWithStackTrace(\n          retryWithTimeoutDebugger,\n          stackTrace,\n          `Retry attempt ${String(attempt)} completed unsuccessfully. Trying again in ${String(fullOptions.retryDelayInMilliseconds)} milliseconds`,\n          {\n            fn\n          }\n        );\n\n        await sleep(fullOptions.retryDelayInMilliseconds, abortSignal);\n      }\n    },\n    { retryFn: fn },\n    stackTrace\n  );\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @param context - The context of the function.\n * @param stackTrace - The stack trace of the source function.\n * @returns A {@link Promise} that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(\n  timeoutInMilliseconds: number,\n  fn: (abortSignal: AbortSignal) => Promisable<R>,\n  context?: unknown,\n  stackTrace?: string\n): Promise<R> {\n  stackTrace ??= getStackTrace(1);\n  const startTime = performance.now();\n\n  const abortController = new AbortController();\n\n  await Promise.race([run(), innerTimeout()]);\n  if (abortController.signal.reason instanceof Result) {\n    return abortController.signal.reason.result as R;\n  }\n\n  throw new CustomStackTraceError('Run with timeout failed', stackTrace, abortController.signal.reason as unknown);\n\n  async function run(): Promise<void> {\n    try {\n      const result = await fn(abortController.signal);\n      const duration = performance.now() - startTime;\n      const runWithTimeoutDebugger = getLibDebugger('Async:runWithTimeout');\n      printWithStackTrace(runWithTimeoutDebugger, stackTrace ?? '', `Execution time: ${String(duration)} milliseconds`, { context, fn });\n      abortController.abort(new Result(result));\n    } catch (e) {\n      abortController.abort(e);\n    }\n  }\n\n  async function innerTimeout(): Promise<void> {\n    while (!abortController.signal.aborted) {\n      await sleep(timeoutInMilliseconds, abortController.signal);\n      // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n      if (abortController.signal.aborted) {\n        return;\n      }\n      const duration = performance.now() - startTime;\n      console.warn(`Timed out after ${String(duration)} milliseconds`, { context, fn });\n      const timeoutDebugger = getLibDebugger('Async:runWithTimeout:timeout');\n      if (!timeoutDebugger.enabled) {\n        abortController.abort(new Error(`Timed out after ${String(duration)} milliseconds`));\n        return;\n      }\n\n      timeoutDebugger(\n        `The execution is not terminated because debugger ${timeoutDebugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md for more information.`\n      );\n    }\n  }\n}\n\n/**\n * Gets the next set immediate.\n *\n * @returns A promise that resolves when the next set immediate is available.\n */\nexport async function setImmediateAsync(): Promise<void> {\n  return new Promise((resolve) => {\n    setImmediate(() => {\n      resolve();\n    });\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param delay - The time to wait in milliseconds.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function setTimeoutAsync(delay?: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, delay);\n  });\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<void> {\n  await waitForAbort(abortSignalAny(abortSignal, abortSignalTimeout(milliseconds)));\n  if (shouldThrowOnAbort) {\n    abortSignal?.throwIfAborted();\n  }\n}\n\n/**\n * Returns a {@link Promise} that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @param abortSignal - The abort signal to listen to.\n * @param shouldThrowOnAbort - Whether to throw an error if the abort signal is aborted.\n * @returns A {@link Promise} that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number, abortSignal?: AbortSignal, shouldThrowOnAbort?: boolean): Promise<never> {\n  await sleep(timeoutInMilliseconds, abortSignal, shouldThrowOnAbort);\n  throw new Error(`Timed out in ${String(timeoutInMilliseconds)} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A {@link Promise} that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": ";;;;;;;AAQA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,YAAY;AA0CrB,eAAsB,gBAAgB,SAAiC,YAAoC;AACzG,iBAAe,cAAc,CAAC;AAC9B,MAAI;AACF,UAAM,QAAQ;AAAA,EAChB,SAAS,YAAY;AACnB,QAAI,sBAAsB,aAAa;AACrC,uBAAiB,YAAY,UAAU;AACvC;AAAA,IACF;AACA,wBAAoB,IAAI,sBAAsB,6BAA6B,YAAY,UAAU,CAAC;AAAA,EACpG;AACF;AAUA,eAAsB,YAAe,KAAU,WAAuF;AACpI,QAAM,MAAW,CAAC;AAElB,QAAM,SAAS,IAAI;AACnB,WAAS,IAAI,GAAG,IAAI,QAAQ,KAAK;AAC/B,QAAI,CAAC,OAAO,OAAO,KAAK,CAAC,GAAG;AAC1B;AAAA,IACF;AAEA,UAAM,OAAO,IAAI,CAAC;AAClB,QAAI,MAAM,UAAU,MAAM,GAAG,GAAG,GAAG;AACjC,UAAI,KAAK,IAAI;AAAA,IACf;AAAA,EACF;AAEA,SAAO;AACT;AAUA,eAAsB,mBAAsB,KAAU,WAAwF;AAC5I,QAAM,SAAS,IAAI;AACnB,MAAI,aAAa;AACjB,WAAS,YAAY,GAAG,YAAY,QAAQ,aAAa;AACvD,QAAI,CAAC,OAAO,OAAO,KAAK,SAAS,GAAG;AAClC;AAAA,IACF;AAEA,UAAM,UAAU,IAAI,SAAS;AAC7B,QAAI,MAAM,UAAU,SAAS,WAAW,GAAG,GAAG;AAE5C,UAAI,YAAY,IAAI;AAAA,IACtB;AAAA,EACF;AACA,MAAI,SAAS;AACf;AAWA,eAAsB,aAAmB,KAAU,UAAkF;AACnI,UAAQ,MAAM,SAAS,KAAK,QAAQ,GAAG,KAAK;AAC9C;AAWA,eAAsB,SAAe,KAAU,UAAgF;AAC7H,SAAO,MAAM,uBAAuB,IAAI,IAAI,QAAQ,CAAC;AACvD;AAUO,SAAS,mBAA2C,WAAgD,YAA8C;AACvJ,iBAAe,cAAc,CAAC;AAC9B,SAAO,IAAI,SAAqB;AAC9B,UAAM,kBAAkB,cAAc,CAAC;AACvC,iBAAa,GAAG,cAAc,EAAE;AAAA;AAAA,EAA4C,eAAe;AAC3F,sBAAkB,MAAM,UAAU,GAAG,IAAI,GAAG,UAAU;AAAA,EACxD;AACF;AAUO,SAAS,mBAAmD,QAAuE;AACxI,SAAO,UAAU,SAAgC;AAC/C,UAAM,QAAQ,QAAQ;AACtB,WAAO,OAAO,GAAG,IAAI;AAAA,EACvB;AACF;AAmBA,eAAsB,YAAe,SAAqB,eAA8B;AACtF,QAAM,sBAAsB,eAAe,mBAAmB;AAC9D,QAAM,aAAa,cAAc,CAAC;AAClC,MAAI;AACF,WAAO,MAAM;AAAA,EACf,SAAS,GAAG;AACV,wBAAoB,iBAAiB,IAAI,sBAAsB,iBAAiB,YAAY,CAAC,CAAC;AAC9F,WAAO;AAAA,EACT;AACF;AAQO,SAAS,kBAAkB,SAAiC,YAA2B;AAC5F,iBAAe,cAAc,CAAC;AAE9B,OAAK,gBAAgB,SAAS,UAAU;AAC1C;AAUO,SAAS,4BACd,SACA,sBAAsB,GACtB,YACA,aACM;AACN,kBAAgB,iBAAiB;AACjC,cAAY,eAAe;AAC3B,iBAAe,cAAc,CAAC;AAC9B,oBAAkB,YAAY;AAC5B,UAAM,MAAM,qBAAqB,aAAa,IAAI;AAClD,UAAM,QAAQ,WAAW;AAAA,EAC3B,GAAG,UAAU;AACf;AASA,eAAsB,+BAAkC,UAAiD;AACvG,QAAM,UAAe,CAAC;AACtB,aAAW,WAAW,UAAU;AAC9B,YAAQ,KAAK,MAAM,QAAQ,CAAC;AAAA,EAC9B;AACA,SAAO;AACT;AASA,eAAsB,uBAA0B,UAAyC;AACvF,SAAO,MAAM,+BAA+B,SAAS,IAAI,CAAC,YAAY,MAAM,OAAO,CAAC;AACtF;AAEA,SAAS,iBAAiB,OAAoB,YAA0B;AACtE,iBAAe,wBAAwB,EAAE,gBAAgB;AAAA,IACvD;AAAA,IACA;AAAA,EACF,CAAC;AACH;AAEA,MAAM,uBAAuB,oBAAI,QAAe;AAEhD,MAAM,OAAU;AAAA,EACP,YAA4B,QAAW;AAAX;AAAA,EAAY;AACjD;AAOO,SAAS,sBAAsB,OAAoB;AACxD,uBAAqB,IAAI,KAAK;AAChC;AAOA,eAAsB,YAA4B;AAChD,QAAM,IAAI,QAAQ,MAAM;AACtB,SAAK;AAAA,EACP,CAAC;AACD,QAAM,IAAI,MAAM,qBAAqB;AACvC;AAOA,eAAsB,gBAA+B;AACnD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,YAAQ,SAAS,MAAM;AACrB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,sBAAqC;AACzD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,mBAAe,MAAM;AACnB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAOA,eAAsB,6BAA4C;AAChE,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,0BAAsB,MAAM;AAC1B,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAUA,eAAsB,iBAAiB,IAAuD,cAA6B,YAAoC;AAC7J,QAAM,2BAA2B,eAAe,wBAAwB;AACxE,iBAAe,cAAc,CAAC;AAC9B,QAAM,wBAAwB;AAAA;AAAA,IAE5B,0BAA0B;AAAA,IAC1B,oBAAoB;AAAA;AAAA,IAEpB,uBAAuB;AAAA,EACzB;AACA,QAAM,cAAc,EAAE,GAAG,uBAAuB,GAAG,aAAa;AAChE,cAAY,aAAa,eAAe;AAExC,QAAM;AAAA,IACJ,YAAY;AAAA,IACZ,OAAO,gBAA6B;AAClC,YAAM,sBAAsB,eAAe,YAAY,aAAa,WAAW;AAC/E,0BAAoB,eAAe;AACnC,UAAI,UAAU;AACd,aAAO,CAAC,oBAAoB,SAAS;AACnC;AACA,YAAI;AACJ,YAAI;AACF,sBAAY,MAAM,GAAG,mBAAmB;AAAA,QAC1C,SAAS,OAAO;AAEd,cAAI,oBAAoB,WAAW,CAAC,YAAY,sBAAsB,qBAAqB,IAAI,KAAc,GAAG;AAC9G,kBAAM,IAAI,sBAAsB,2BAA2B,YAAY,KAAK;AAAA,UAC9E;AACA,qBAAW,KAAK;AAChB,sBAAY;AAAA,QACd;AACA,YAAI,WAAW;AACb,8BAAoB,0BAA0B,YAAY,sCAAsC,OAAO,OAAO,CAAC,aAAa;AAAA,YAC1H;AAAA,UACF,CAAC;AACD;AAAA,QACF;AAEA;AAAA,UACE;AAAA,UACA;AAAA,UACA,iBAAiB,OAAO,OAAO,CAAC,8CAA8C,OAAO,YAAY,wBAAwB,CAAC;AAAA,UAC1H;AAAA,YACE;AAAA,UACF;AAAA,QACF;AAEA,cAAM,MAAM,YAAY,0BAA0B,WAAW;AAAA,MAC/D;AAAA,IACF;AAAA,IACA,EAAE,SAAS,GAAG;AAAA,IACd;AAAA,EACF;AACF;AAcA,eAAsB,eACpB,uBACA,IACA,SACA,YACY;AACZ,iBAAe,cAAc,CAAC;AAC9B,QAAM,YAAY,YAAY,IAAI;AAElC,QAAM,kBAAkB,IAAI,gBAAgB;AAE5C,QAAM,QAAQ,KAAK,CAAC,IAAI,GAAG,aAAa,CAAC,CAAC;AAC1C,MAAI,gBAAgB,OAAO,kBAAkB,QAAQ;AACnD,WAAO,gBAAgB,OAAO,OAAO;AAAA,EACvC;AAEA,QAAM,IAAI,sBAAsB,2BAA2B,YAAY,gBAAgB,OAAO,MAAiB;AAE/G,iBAAe,MAAqB;AAClC,QAAI;AACF,YAAM,SAAS,MAAM,GAAG,gBAAgB,MAAM;AAC9C,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,YAAM,yBAAyB,eAAe,sBAAsB;AACpE,0BAAoB,wBAAwB,cAAc,IAAI,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AACjI,sBAAgB,MAAM,IAAI,OAAO,MAAM,CAAC;AAAA,IAC1C,SAAS,GAAG;AACV,sBAAgB,MAAM,CAAC;AAAA,IACzB;AAAA,EACF;AAEA,iBAAe,eAA8B;AAC3C,WAAO,CAAC,gBAAgB,OAAO,SAAS;AACtC,YAAM,MAAM,uBAAuB,gBAAgB,MAAM;AAEzD,UAAI,gBAAgB,OAAO,SAAS;AAClC;AAAA,MACF;AACA,YAAM,WAAW,YAAY,IAAI,IAAI;AACrC,cAAQ,KAAK,mBAAmB,OAAO,QAAQ,CAAC,iBAAiB,EAAE,SAAS,GAAG,CAAC;AAChF,YAAM,kBAAkB,eAAe,8BAA8B;AACrE,UAAI,CAAC,gBAAgB,SAAS;AAC5B,wBAAgB,MAAM,IAAI,MAAM,mBAAmB,OAAO,QAAQ,CAAC,eAAe,CAAC;AACnF;AAAA,MACF;AAEA;AAAA,QACE,oDAAoD,gBAAgB,SAAS;AAAA,MAC/E;AAAA,IACF;AAAA,EACF;AACF;AAOA,eAAsB,oBAAmC;AACvD,SAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,iBAAa,MAAM;AACjB,cAAQ;AAAA,IACV,CAAC;AAAA,EACH,CAAC;AACH;AAQA,eAAsB,gBAAgB,OAA+B;AACnE,QAAM,IAAI,QAAQ,CAAC,YAAY;AAC7B,eAAW,SAAS,KAAK;AAAA,EAC3B,CAAC;AACH;AAUA,eAAsB,MAAM,cAAsB,aAA2B,oBAA6C;AACxH,QAAM,aAAa,eAAe,aAAa,mBAAmB,YAAY,CAAC,CAAC;AAChF,MAAI,oBAAoB;AACtB,iBAAa,eAAe;AAAA,EAC9B;AACF;AAUA,eAAsB,QAAQ,uBAA+B,aAA2B,oBAA8C;AACpI,QAAM,MAAM,uBAAuB,aAAa,kBAAkB;AAClE,QAAM,IAAI,MAAM,gBAAgB,OAAO,qBAAqB,CAAC,eAAe;AAC9E;AASA,eAAsB,QAAW,MAA8C;AAC7E,QAAM,MAAW,CAAC;AAClB,mBAAiB,QAAQ,MAAM;AAC7B,QAAI,KAAK,IAAI;AAAA,EACf;AACA,SAAO;AACT;",
  "names": []
}

|
package/dist/lib/esm/Error.d.mts
CHANGED
|
@@ -20,6 +20,17 @@ export declare class CustomStackTraceError extends Error {
|
|
|
20
20
|
*/
|
|
21
21
|
constructor(message: string, stackTrace: string, cause: unknown);
|
|
22
22
|
}
|
|
23
|
+
/**
|
|
24
|
+
* An error that is not printed to the console.
|
|
25
|
+
*/
|
|
26
|
+
export declare class SilentError extends Error {
|
|
27
|
+
/**
|
|
28
|
+
* Creates a new SilentError.
|
|
29
|
+
*
|
|
30
|
+
* @param message - The message of the error.
|
|
31
|
+
*/
|
|
32
|
+
constructor(message: string);
|
|
33
|
+
}
|
|
23
34
|
/**
|
|
24
35
|
* Emits an asynchronous error event.
|
|
25
36
|
*
|
package/dist/lib/esm/Error.mjs
CHANGED
|
@@ -42,6 +42,18 @@ class CustomStackTraceError extends Error {
|
|
|
42
42
|
this.stack = originalStackLines.join("\n");
|
|
43
43
|
}
|
|
44
44
|
}
|
|
45
|
+
class SilentError extends Error {
|
|
46
|
+
/**
|
|
47
|
+
* Creates a new SilentError.
|
|
48
|
+
*
|
|
49
|
+
* @param message - The message of the error.
|
|
50
|
+
*/
|
|
51
|
+
constructor(message) {
|
|
52
|
+
super(message);
|
|
53
|
+
this.name = "SilentError";
|
|
54
|
+
Error.captureStackTrace(this, SilentError);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
45
57
|
function emitAsyncErrorEvent(asyncError) {
|
|
46
58
|
asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);
|
|
47
59
|
}
|
|
@@ -109,6 +121,7 @@ function parseErrorEntries(error, level = 0, entries = []) {
|
|
|
109
121
|
export {
|
|
110
122
|
ASYNC_WRAPPER_ERROR_MESSAGE,
|
|
111
123
|
CustomStackTraceError,
|
|
124
|
+
SilentError,
|
|
112
125
|
emitAsyncErrorEvent,
|
|
113
126
|
errorToString,
|
|
114
127
|
getStackTrace,
|
|
@@ -116,4 +129,4 @@ export {
|
|
|
116
129
|
registerAsyncErrorEventHandler,
|
|
117
130
|
throwExpression
|
|
118
131
|
};
|
|
119
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Error.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for error handling.\n */\n\nimport { AsyncEvents } from './AsyncEvents.ts';\n\nconst ASYNC_ERROR_EVENT = 'asyncError';\n\nconst ERROR_STACK_PREFIX = 'Error stack:\\n';\n\nconst asyncErrorEventEmitter = new AsyncEvents();\nasyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handleAsyncError);\n\ninterface ErrorEntry {\n  level: number;\n  message: string;\n  shouldClearAnsiSequence?: boolean;\n}\n\n/**\n * A message of the AsyncWrapperError.\n */\nexport const ASYNC_WRAPPER_ERROR_MESSAGE = 'An unhandled error occurred executing async operation';\n\n/**\n * An error that wraps an error with a custom stack trace.\n */\nexport class CustomStackTraceError extends Error {\n  /**\n   * Creates a new CustomStackTraceError.\n   *\n   * @param message - The message of the error.\n   * @param stackTrace - The stack trace of the error.\n   * @param cause - The cause of the error.\n   */\n  public constructor(message: string, stackTrace: string, cause: unknown) {\n    super(message, { cause });\n    this.name = 'CustomStackTraceError';\n    Error.captureStackTrace(this, CustomStackTraceError);\n\n    let rootCause = cause;\n    const parentCauses = new Set<CustomStackTraceError>();\n    while (rootCause instanceof CustomStackTraceError) {\n      if (parentCauses.has(rootCause)) {\n        throw new Error('Circular cause detected');\n      }\n      parentCauses.add(rootCause);\n      rootCause = rootCause.cause;\n    }\n\n    const originalStackLines = (this.stack ?? '').split('\\n');\n    const stackLines = stackTrace.split('\\n');\n    const ERROR_HEADER_REG_EXP = /^\\w*Error(?:: |$)/;\n    if (ERROR_HEADER_REG_EXP.test(stackLines[0] ?? '')) {\n      stackLines.splice(0, 1);\n    }\n    originalStackLines.splice(1, originalStackLines.length - 1, ...stackLines);\n    this.stack = originalStackLines.join('\\n');\n  }\n}\n\n/**\n * Emits an asynchronous error event.\n *\n * @param asyncError - The error to emit as an asynchronous error event.\n */\nexport function emitAsyncErrorEvent(asyncError: unknown): void {\n  asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);\n}\n\n/**\n * Converts an error to a string representation, including nested causes with indentation.\n *\n * @param error - The error to convert to a string.\n * @param shouldEnsureAnsiSequencesCleared - Whether to ensure ANSI sequences in the error message are cleared.\n * @returns The string representation of the error.\n */\nexport function errorToString(error: unknown, shouldEnsureAnsiSequencesCleared = false): string {\n  const ANSI_CLEAR_SEQUENCE = '\\x1b[0m';\n  return parseErrorEntries(error)\n    .map((entry) => {\n      const prefix = '  '.repeat(entry.level);\n      // NOTE: Cannot use `String.trimStart()` function here because of the circular dependency.\n      let message = entry.message.startsWith(ERROR_STACK_PREFIX) ? entry.message.slice(ERROR_STACK_PREFIX.length) : entry.message;\n      if (entry.shouldClearAnsiSequence && shouldEnsureAnsiSequencesCleared) {\n        message = `${ANSI_CLEAR_SEQUENCE}${message}${ANSI_CLEAR_SEQUENCE}`;\n      }\n      return prefix + message;\n    })\n    .join('\\n');\n}\n\n/**\n * Gets the current stack trace as a string, excluding the current function call.\n *\n * @param framesToSkip - The number of frames to skip in the stack trace.\n * @returns A string representation of the current stack trace, excluding the current function call.\n */\nexport function getStackTrace(framesToSkip = 0): string {\n  // Skipping Error prefix and `getStackTrace` function call\n  const ADDITIONAL_FRAMES_TO_SKIP = 2;\n  const stack = new Error().stack ?? '';\n  const lines = stack.split('\\n');\n  return lines.slice(framesToSkip + ADDITIONAL_FRAMES_TO_SKIP).join('\\n');\n}\n\n/**\n * Prints an error to the console, including nested causes and optional ANSI sequence clearing.\n *\n * @param error - The error to print.\n * @param console - The console to print to (default: `globalThis.console`).\n */\nexport function printError(error: unknown, console?: Console): void {\n  console ??= globalThis.console;\n  console.error(errorToString(error, true));\n}\n\n/**\n * Registers an event handler for asynchronous errors.\n *\n * @param handler - The handler function to be called when an asynchronous error event occurs.\n * @returns A function to unregister the handler.\n */\nexport function registerAsyncErrorEventHandler(handler: (asyncError: unknown) => void): () => void {\n  const eventRef = asyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handler);\n  return () => {\n    asyncErrorEventEmitter.offref(eventRef);\n  };\n}\n\n/**\n * Throws an error with the specified message.\n *\n * @param error - The error to throw.\n * @throws\n */\nexport function throwExpression(error: unknown): never {\n  throw error;\n}\n\n/**\n * Handles asynchronous errors by printing them.\n *\n * @param asyncError - The asynchronous error to handle.\n */\nfunction handleAsyncError(asyncError: unknown): void {\n  printError(asyncError);\n}\n\n/**\n * Parses an error into an array of ErrorEntry objects, including nested causes.\n *\n * @param error - The error to parse.\n * @param level - The current indentation level for nested causes.\n * @param entries - The array of ErrorEntry objects to populate.\n * @returns An array of ErrorEntry objects representing the error and its causes.\n */\nfunction parseErrorEntries(error: unknown, level = 0, entries: ErrorEntry[] = []): ErrorEntry[] {\n  if (error === undefined) {\n    return entries;\n  }\n\n  if (!(error instanceof Error)) {\n    let str: string;\n\n    if (error === null) {\n      str = '(null)';\n    } else if (typeof error === 'string') {\n      str = error;\n    } else {\n      str = JSON.stringify(error) ?? 'undefined';\n    }\n\n    entries.push({ level, message: str });\n    return entries;\n  }\n\n  const title = `${error.name}: ${error.message}`;\n  entries.push({ level, message: title, shouldClearAnsiSequence: true });\n\n  if (error.stack) {\n    const restStack = error.stack.startsWith(title) ? error.stack.slice(title.length + 1) : error.stack;\n    entries.push({ level, message: `${ERROR_STACK_PREFIX}${restStack}` });\n  }\n\n  if (error.cause !== undefined) {\n    entries.push({ level, message: 'Caused by:' });\n    parseErrorEntries(error.cause, level + 1, entries);\n  }\n\n  return entries;\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
132
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Error.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for error handling.\n */\n\nimport { AsyncEvents } from './AsyncEvents.ts';\n\nconst ASYNC_ERROR_EVENT = 'asyncError';\n\nconst ERROR_STACK_PREFIX = 'Error stack:\\n';\n\nconst asyncErrorEventEmitter = new AsyncEvents();\nasyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handleAsyncError);\n\ninterface ErrorEntry {\n  level: number;\n  message: string;\n  shouldClearAnsiSequence?: boolean;\n}\n\n/**\n * A message of the AsyncWrapperError.\n */\nexport const ASYNC_WRAPPER_ERROR_MESSAGE = 'An unhandled error occurred executing async operation';\n\n/**\n * An error that wraps an error with a custom stack trace.\n */\nexport class CustomStackTraceError extends Error {\n  /**\n   * Creates a new CustomStackTraceError.\n   *\n   * @param message - The message of the error.\n   * @param stackTrace - The stack trace of the error.\n   * @param cause - The cause of the error.\n   */\n  public constructor(message: string, stackTrace: string, cause: unknown) {\n    super(message, { cause });\n    this.name = 'CustomStackTraceError';\n    Error.captureStackTrace(this, CustomStackTraceError);\n\n    let rootCause = cause;\n    const parentCauses = new Set<CustomStackTraceError>();\n    while (rootCause instanceof CustomStackTraceError) {\n      if (parentCauses.has(rootCause)) {\n        throw new Error('Circular cause detected');\n      }\n      parentCauses.add(rootCause);\n      rootCause = rootCause.cause;\n    }\n\n    const originalStackLines = (this.stack ?? '').split('\\n');\n    const stackLines = stackTrace.split('\\n');\n    const ERROR_HEADER_REG_EXP = /^\\w*Error(?:: |$)/;\n    if (ERROR_HEADER_REG_EXP.test(stackLines[0] ?? '')) {\n      stackLines.splice(0, 1);\n    }\n    originalStackLines.splice(1, originalStackLines.length - 1, ...stackLines);\n    this.stack = originalStackLines.join('\\n');\n  }\n}\n\n/**\n * An error that is not printed to the console.\n */\nexport class SilentError extends Error {\n  /**\n   * Creates a new SilentError.\n   *\n   * @param message - The message of the error.\n   */\n  public constructor(message: string) {\n    super(message);\n    this.name = 'SilentError';\n    Error.captureStackTrace(this, SilentError);\n  }\n}\n\n/**\n * Emits an asynchronous error event.\n *\n * @param asyncError - The error to emit as an asynchronous error event.\n */\nexport function emitAsyncErrorEvent(asyncError: unknown): void {\n  asyncErrorEventEmitter.trigger(ASYNC_ERROR_EVENT, asyncError);\n}\n\n/**\n * Converts an error to a string representation, including nested causes with indentation.\n *\n * @param error - The error to convert to a string.\n * @param shouldEnsureAnsiSequencesCleared - Whether to ensure ANSI sequences in the error message are cleared.\n * @returns The string representation of the error.\n */\nexport function errorToString(error: unknown, shouldEnsureAnsiSequencesCleared = false): string {\n  const ANSI_CLEAR_SEQUENCE = '\\x1b[0m';\n  return parseErrorEntries(error)\n    .map((entry) => {\n      const prefix = '  '.repeat(entry.level);\n      // NOTE: Cannot use `String.trimStart()` function here because of the circular dependency.\n      let message = entry.message.startsWith(ERROR_STACK_PREFIX) ? entry.message.slice(ERROR_STACK_PREFIX.length) : entry.message;\n      if (entry.shouldClearAnsiSequence && shouldEnsureAnsiSequencesCleared) {\n        message = `${ANSI_CLEAR_SEQUENCE}${message}${ANSI_CLEAR_SEQUENCE}`;\n      }\n      return prefix + message;\n    })\n    .join('\\n');\n}\n\n/**\n * Gets the current stack trace as a string, excluding the current function call.\n *\n * @param framesToSkip - The number of frames to skip in the stack trace.\n * @returns A string representation of the current stack trace, excluding the current function call.\n */\nexport function getStackTrace(framesToSkip = 0): string {\n  // Skipping Error prefix and `getStackTrace` function call\n  const ADDITIONAL_FRAMES_TO_SKIP = 2;\n  const stack = new Error().stack ?? '';\n  const lines = stack.split('\\n');\n  return lines.slice(framesToSkip + ADDITIONAL_FRAMES_TO_SKIP).join('\\n');\n}\n\n/**\n * Prints an error to the console, including nested causes and optional ANSI sequence clearing.\n *\n * @param error - The error to print.\n * @param console - The console to print to (default: `globalThis.console`).\n */\nexport function printError(error: unknown, console?: Console): void {\n  console ??= globalThis.console;\n  console.error(errorToString(error, true));\n}\n\n/**\n * Registers an event handler for asynchronous errors.\n *\n * @param handler - The handler function to be called when an asynchronous error event occurs.\n * @returns A function to unregister the handler.\n */\nexport function registerAsyncErrorEventHandler(handler: (asyncError: unknown) => void): () => void {\n  const eventRef = asyncErrorEventEmitter.on(ASYNC_ERROR_EVENT, handler);\n  return () => {\n    asyncErrorEventEmitter.offref(eventRef);\n  };\n}\n\n/**\n * Throws an error with the specified message.\n *\n * @param error - The error to throw.\n * @throws\n */\nexport function throwExpression(error: unknown): never {\n  throw error;\n}\n\n/**\n * Handles asynchronous errors by printing them.\n *\n * @param asyncError - The asynchronous error to handle.\n */\nfunction handleAsyncError(asyncError: unknown): void {\n  printError(asyncError);\n}\n\n/**\n * Parses an error into an array of ErrorEntry objects, including nested causes.\n *\n * @param error - The error to parse.\n * @param level - The current indentation level for nested causes.\n * @param entries - The array of ErrorEntry objects to populate.\n * @returns An array of ErrorEntry objects representing the error and its causes.\n */\nfunction parseErrorEntries(error: unknown, level = 0, entries: ErrorEntry[] = []): ErrorEntry[] {\n  if (error === undefined) {\n    return entries;\n  }\n\n  if (!(error instanceof Error)) {\n    let str: string;\n\n    if (error === null) {\n      str = '(null)';\n    } else if (typeof error === 'string') {\n      str = error;\n    } else {\n      str = JSON.stringify(error) ?? 'undefined';\n    }\n\n    entries.push({ level, message: str });\n    return entries;\n  }\n\n  const title = `${error.name}: ${error.message}`;\n  entries.push({ level, message: title, shouldClearAnsiSequence: true });\n\n  if (error.stack) {\n    const restStack = error.stack.startsWith(title) ? error.stack.slice(title.length + 1) : error.stack;\n    entries.push({ level, message: `${ERROR_STACK_PREFIX}${restStack}` });\n  }\n\n  if (error.cause !== undefined) {\n    entries.push({ level, message: 'Caused by:' });\n    parseErrorEntries(error.cause, level + 1, entries);\n  }\n\n  return entries;\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/dist/lib/esm/Library.mjs
CHANGED
|
@@ -5,7 +5,7 @@ if you want to view the source, please visit the github repository of this plugi
|
|
|
5
5
|
|
|
6
6
|
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd(){return"/"},env:{},platform:"android"};globalThis.process=browserProcess})();
|
|
7
7
|
|
|
8
|
-
const LIBRARY_VERSION = "37.
|
|
8
|
+
const LIBRARY_VERSION = "37.2.0";
|
|
9
9
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
|
10
10
|
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-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.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/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.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;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;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&.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&.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";
|
|
11
11
|
export {
|
|
@@ -16,7 +16,10 @@ import {
|
|
|
16
16
|
} from "../../Async.mjs";
|
|
17
17
|
import { AsyncEvents } from "../../AsyncEvents.mjs";
|
|
18
18
|
import { getDebugger } from "../../Debug.mjs";
|
|
19
|
-
import {
|
|
19
|
+
import {
|
|
20
|
+
registerAsyncErrorEventHandler,
|
|
21
|
+
SilentError
|
|
22
|
+
} from "../../Error.mjs";
|
|
20
23
|
import { noopAsync } from "../../Function.mjs";
|
|
21
24
|
import { registerAsyncEvent } from "../Components/AsyncEventsComponent.mjs";
|
|
22
25
|
import { initPluginContext } from "./PluginContext.mjs";
|
|
@@ -201,7 +204,7 @@ class PluginBase extends ObsidianPlugin {
|
|
|
201
204
|
const abortController = new AbortController();
|
|
202
205
|
this._abortSignal = abortController.signal;
|
|
203
206
|
this.register(() => {
|
|
204
|
-
abortController.abort(new
|
|
207
|
+
abortController.abort(new SilentError(`Plugin ${this.manifest.id} had been unloaded`));
|
|
205
208
|
});
|
|
206
209
|
}
|
|
207
210
|
/**
|
|
@@ -263,4 +266,4 @@ ${message}`);
|
|
|
263
266
|
export {
|
|
264
267
|
PluginBase
|
|
265
268
|
};
|
|
266
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { ReadonlyDeep } from 'type-fest';\n\nimport {\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type {\n  ExtractPluginSettings,\n  ExtractPluginSettingsManager,\n  ExtractPluginSettingsTab,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely,\n  invokeAsyncSafelyAfterDelay\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { getDebugger } from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noopAsync } from '../../Function.ts';\nimport { registerAsyncEvent } from '../Components/AsyncEventsComponent.ts';\nimport { initPluginContext } from './PluginContext.ts';\n\ntype LifecycleEventName = 'layoutReady' | 'load' | 'unload';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginBase<PluginTypes extends PluginTypesBase> extends ObsidianPlugin {\n  public readonly events = new AsyncEvents();\n\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<ExtractPluginSettings<PluginTypes>> {\n    return this.settingsManager.settingsWrapper.safeSettings as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n  }\n\n  /**\n   * Gets the plugin settings manager.\n   *\n   * @returns The plugin settings manager.\n   */\n  public get settingsManager(): ExtractPluginSettingsManager<PluginTypes> {\n    if (!this._settingsManager) {\n      throw new Error('Settings manager not defined');\n    }\n\n    return this._settingsManager;\n  }\n\n  /**\n   * Gets the plugin settings tab.\n   *\n   * @returns The plugin settings tab.\n   */\n  public get settingsTab(): ExtractPluginSettingsTab<PluginTypes> {\n    if (!this._settingsTab) {\n      throw new Error('Settings tab not defined');\n    }\n\n    return this._settingsTab;\n  }\n\n  private _abortSignal!: AbortSignal;\n  private _settingsManager: ExtractPluginSettingsManager<PluginTypes> | null = null;\n  private _settingsTab: ExtractPluginSettingsTab<PluginTypes> | null = null;\n  private lifecycleEventNames = new Set<LifecycleEventName>();\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const pluginDebugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    pluginDebugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onLoadSettings} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this._settingsManager?.loadFromFile(false);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   *\n   * Usually, you don't need to override this method. Consider using {@link onloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onload()` first.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onunloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `super.onunload()` first.\n   */\n  public override onunload(): void {\n    super.onunload();\n    invokeAsyncSafely(async () => {\n      try {\n        await this.onunloadImpl();\n      } finally {\n        await this.triggerLifecycleEvent('unload');\n      }\n    });\n  }\n\n  /**\n   * Registers a callback to be executed when a lifecycle event is triggered.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback to execute.\n   */\n  public registerForLifecycleEvent(name: LifecycleEventName, callback: () => Promise<void>): void {\n    invokeAsyncSafely(async () => {\n      await this.waitForLifecycleEvent(name);\n      await callback();\n    });\n  }\n\n  /**\n   * Waits for a lifecycle event to be triggered.\n   *\n   * If you `await` this method during lifecycle event, it might cause a deadlock.\n   *\n   * Consider wrapping this call with {@link invokeAsyncSafely}.\n   *\n   * @param name - The name of the event.\n   * @returns A {@link Promise} that resolves when the event is triggered.\n   */\n  public async waitForLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    if (this.lifecycleEventNames.has(name)) {\n      return;\n    }\n\n    await new Promise<void>((resolve) => {\n      this.events.once(name, () => {\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected createSettingsManager(): ExtractPluginSettingsManager<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Creates a plugin settings tab.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected createSettingsTab(): ExtractPluginSettingsTab<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Executed when the plugin is loaded.\n   *\n   * If this method fails, the plugin will be automatically unloaded.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n    if (this._settingsManager) {\n      registerAsyncEvent(this, this._settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n      registerAsyncEvent(this, this._settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n    }\n\n    await this._settingsManager?.loadFromFile(true);\n    this._settingsTab = this.createSettingsTab();\n    if (this._settingsTab) {\n      this.addSettingTab(this._settingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort(new Error(`Plugin ${this.manifest.id} had been unloaded`));\n    });\n  }\n\n  /**\n   * Called when the plugin settings are loaded or reloaded.\n   *\n   * @param _loadedSettings - The loaded settings wrapper.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saved.\n   *\n   * @param _newSettings - The new settings.\n   * @param _oldSettings - The old settings.\n   * @param _context - The context.\n   */\n  protected async onSaveSettings(\n    _newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   */\n  protected async onunloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this._abortSignal.aborted) {\n      return;\n    }\n    await this.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.triggerLifecycleEvent('layoutReady');\n    }\n  }\n\n  private async triggerLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    this.lifecycleEventNames.add(name);\n    await this.events.triggerAsync(name);\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
269
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { ReadonlyDeep } from 'type-fest';\n\nimport {\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type {\n  ExtractPluginSettings,\n  ExtractPluginSettingsManager,\n  ExtractPluginSettingsTab,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely,\n  invokeAsyncSafelyAfterDelay\n} from '../../Async.ts';\nimport { AsyncEvents } from '../../AsyncEvents.ts';\nimport { getDebugger } from '../../Debug.ts';\nimport {\n  registerAsyncErrorEventHandler,\n  SilentError\n} from '../../Error.ts';\nimport { noopAsync } from '../../Function.ts';\nimport { registerAsyncEvent } from '../Components/AsyncEventsComponent.ts';\nimport { initPluginContext } from './PluginContext.ts';\n\ntype LifecycleEventName = 'layoutReady' | 'load' | 'unload';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginBase<PluginTypes extends PluginTypesBase> extends ObsidianPlugin {\n  public readonly events = new AsyncEvents();\n\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<ExtractPluginSettings<PluginTypes>> {\n    return this.settingsManager.settingsWrapper.safeSettings as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n  }\n\n  /**\n   * Gets the plugin settings manager.\n   *\n   * @returns The plugin settings manager.\n   */\n  public get settingsManager(): ExtractPluginSettingsManager<PluginTypes> {\n    if (!this._settingsManager) {\n      throw new Error('Settings manager not defined');\n    }\n\n    return this._settingsManager;\n  }\n\n  /**\n   * Gets the plugin settings tab.\n   *\n   * @returns The plugin settings tab.\n   */\n  public get settingsTab(): ExtractPluginSettingsTab<PluginTypes> {\n    if (!this._settingsTab) {\n      throw new Error('Settings tab not defined');\n    }\n\n    return this._settingsTab;\n  }\n\n  private _abortSignal!: AbortSignal;\n  private _settingsManager: ExtractPluginSettingsManager<PluginTypes> | null = null;\n  private _settingsTab: ExtractPluginSettingsTab<PluginTypes> | null = null;\n  private lifecycleEventNames = new Set<LifecycleEventName>();\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/blob/main/docs/debugging.md} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const pluginDebugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    pluginDebugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onLoadSettings} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this._settingsManager?.loadFromFile(false);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   *\n   * Usually, you don't need to override this method. Consider using {@link onloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `await super.onload()` first.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   *\n   * Usually, you don't need to override this method. Consider using {@link onunloadImpl} instead.\n   *\n   * If you still need to override this method, make sure to call `super.onunload()` first.\n   */\n  public override onunload(): void {\n    super.onunload();\n    invokeAsyncSafely(async () => {\n      try {\n        await this.onunloadImpl();\n      } finally {\n        await this.triggerLifecycleEvent('unload');\n      }\n    });\n  }\n\n  /**\n   * Registers a callback to be executed when a lifecycle event is triggered.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback to execute.\n   */\n  public registerForLifecycleEvent(name: LifecycleEventName, callback: () => Promise<void>): void {\n    invokeAsyncSafely(async () => {\n      await this.waitForLifecycleEvent(name);\n      await callback();\n    });\n  }\n\n  /**\n   * Waits for a lifecycle event to be triggered.\n   *\n   * If you `await` this method during lifecycle event, it might cause a deadlock.\n   *\n   * Consider wrapping this call with {@link invokeAsyncSafely}.\n   *\n   * @param name - The name of the event.\n   * @returns A {@link Promise} that resolves when the event is triggered.\n   */\n  public async waitForLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    if (this.lifecycleEventNames.has(name)) {\n      return;\n    }\n\n    await new Promise<void>((resolve) => {\n      this.events.once(name, () => {\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected createSettingsManager(): ExtractPluginSettingsManager<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Creates a plugin settings tab.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected createSettingsTab(): ExtractPluginSettingsTab<PluginTypes> | null {\n    return null;\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Executed when the plugin is loaded.\n   *\n   * If this method fails, the plugin will be automatically unloaded.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n    if (this._settingsManager) {\n      registerAsyncEvent(this, this._settingsManager.on('loadSettings', this.onLoadSettings.bind(this)));\n      registerAsyncEvent(this, this._settingsManager.on('saveSettings', this.onSaveSettings.bind(this)));\n    }\n\n    await this._settingsManager?.loadFromFile(true);\n    this._settingsTab = this.createSettingsTab();\n    if (this._settingsTab) {\n      this.addSettingTab(this._settingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort(new SilentError(`Plugin ${this.manifest.id} had been unloaded`));\n    });\n  }\n\n  /**\n   * Called when the plugin settings are loaded or reloaded.\n   *\n   * @param _loadedSettings - The loaded settings wrapper.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   */\n  protected async onLoadSettings(_loadedSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>, _isInitialLoad: boolean): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saved.\n   *\n   * @param _newSettings - The new settings.\n   * @param _oldSettings - The old settings.\n   * @param _context - The context.\n   */\n  protected async onSaveSettings(\n    _newSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _oldSettings: ExtractReadonlyPluginSettingsWrapper<PluginTypes>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin is unloaded.\n   */\n  protected async onunloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this._abortSignal.aborted) {\n      return;\n    }\n    await this.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.triggerLifecycleEvent('layoutReady');\n    }\n  }\n\n  private async triggerLifecycleEvent(name: LifecycleEventName): Promise<void> {\n    this.lifecycleEventNames.add(name);\n    await this.events.triggerAsync(name);\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|