obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
224 lines (210 loc) • 17.7 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initCjs() {
const globalThisRecord = globalThis;
globalThisRecord['__name'] ??= name;
const originalRequire = require;
if (originalRequire && !originalRequire.__isPatched) {
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
require = Object.assign(
(id) => requirePatched(id),
originalRequire,
{
__isPatched: true
}
);
}
const newFuncs = {
__extractDefault() {
return extractDefault;
},
process() {
const browserProcess = {
browser: true,
cwd() {
return '/';
},
env: {},
platform: 'android'
};
return browserProcess;
}
};
for (const key of Object.keys(newFuncs)) {
globalThisRecord[key] ??= newFuncs[key]?.();
}
function name(obj) {
return obj;
}
function extractDefault(module) {
return module && module.__esModule && 'default' in module ? module.default : module;
}
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'obsidian',
'@codemirror/autocomplete',
'@codemirror/collab',
'@codemirror/commands',
'@codemirror/language',
'@codemirror/lint',
'@codemirror/search',
'@codemirror/state',
'@codemirror/text',
'@codemirror/view',
'@lezer/common',
'@lezer/lr',
'@lezer/highlight'];
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
'@codemirror/closebrackets',
'@codemirror/comment',
'@codemirror/fold',
'@codemirror/gutter',
'@codemirror/highlight',
'@codemirror/history',
'@codemirror/matchbrackets',
'@codemirror/panel',
'@codemirror/rangeset',
'@codemirror/rectangular-selection',
'@codemirror/stream-parser',
'@codemirror/tooltip'];
function requirePatched(id) {
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
return originalRequire?.(id);
}
// eslint-disable-next-line @typescript-eslint/no-deprecated, @typescript-eslint/no-unnecessary-condition -- We need access to app here which might not be available yet.
if (globalThis?.app?.isMobile) {
if (id === 'process' || id === 'node:process') {
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
return globalThis.process;
}
} else {
const module = originalRequire?.(id);
if (module) {
return extractDefault(module);
}
}
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
return {};
}
})();
;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var Loop_exports = {};
__export(Loop_exports, {
loop: () => loop
});
module.exports = __toCommonJS(Loop_exports);
var import_obsidian = require('obsidian');
var import_AbortController = require('../AbortController.cjs');
var import_Async = require('../Async.cjs');
var import_Debug = require('../Debug.cjs');
var import_Error = require('../Error.cjs');
var import_Function = require('../Function.cjs');
var import_PluginContext = require('./Plugin/PluginContext.cjs');
async function loop(options) {
const DEFAULT_OPTIONS = {
abortSignal: (0, import_AbortController.abortSignalNever)(),
buildNoticeMessage() {
throw new Error("buildNoticeMessage is required");
},
items: [],
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
noticeBeforeShownTimeoutInMilliseconds: 500,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
noticeMinTimeoutInMilliseconds: 2e3,
processItem: import_Function.noop,
progressBarTitle: "",
shouldContinueOnError: true,
shouldShowNotice: true,
shouldShowProgressBar: true,
// eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.
uiUpdateThresholdInMilliseconds: 100
};
const fullOptions = {
...DEFAULT_OPTIONS,
...options
};
const stackTrace = (0, import_Error.getStackTrace)(1);
const items = fullOptions.items;
let iterationCount = 0;
let notice = null;
let isDone = false;
(0, import_Async.invokeAsyncSafely)(() => showNotice());
const noticeMinTimeoutPromise = sleep(fullOptions.noticeMinTimeoutInMilliseconds);
const progressBarEl = createEl("progress");
(0, import_PluginContext.addPluginCssClasses)(progressBarEl, "loop");
progressBarEl.max = items.length;
let lastUIUpdateTimestamp = performance.now();
for (const item of items) {
if (fullOptions.abortSignal.aborted) {
notice?.hide();
return;
}
iterationCount++;
const iterationStr = `# ${String(iterationCount)} / ${String(items.length)}`;
const message = fullOptions.buildNoticeMessage(item, iterationStr);
if (!fullOptions.shouldShowProgressBar) {
notice?.setMessage(message);
}
(0, import_Debug.getLibDebugger)("Loop")(message);
try {
if (performance.now() - lastUIUpdateTimestamp > fullOptions.uiUpdateThresholdInMilliseconds) {
await (0, import_Async.requestAnimationFrameAsync)();
lastUIUpdateTimestamp = performance.now();
}
await fullOptions.processItem(item);
} catch (error) {
console.error("Error processing item", item);
if (!fullOptions.shouldContinueOnError) {
notice?.hide();
throw new import_Error.CustomStackTraceError("loop failed", stackTrace, error);
}
(0, import_Error.emitAsyncErrorEvent)(new import_Error.CustomStackTraceError(import_Error.ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, error));
}
progressBarEl.value++;
}
if (notice) {
await noticeMinTimeoutPromise;
}
notice?.hide();
isDone = true;
async function showNotice() {
if (!fullOptions.shouldShowNotice) {
return;
}
await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);
if (isDone) {
return;
}
notice = new import_obsidian.Notice("", 0);
if (!fullOptions.shouldShowProgressBar) {
return;
}
const fragment = createFragment();
fragment.createDiv({ text: fullOptions.progressBarTitle });
fragment.appendChild(progressBarEl);
notice.setMessage(fragment);
}
}
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
loop
});
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Loop.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for looping in Obsidian.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  invokeAsyncSafely,\n  requestAnimationFrameAsync\n} from '../Async.ts';\nimport { getLibDebugger } from '../Debug.ts';\nimport {\n  ASYNC_WRAPPER_ERROR_MESSAGE,\n  CustomStackTraceError,\n  emitAsyncErrorEvent,\n  getStackTrace\n} from '../Error.ts';\nimport { noop } from '../Function.ts';\nimport { addPluginCssClasses } from './Plugin/PluginContext.ts';\n\n/**\n * Options for {@link loop}.\n */\nexport interface LoopOptions<T> {\n  /**\n   * An optional abort signal to cancel the loop.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * Build a notice message for each item.\n   *\n   * @param item - The current item.\n   * @param iterationStr - A string representing the current iteration.\n   * @returns A string to display in the notice.\n   */\n  buildNoticeMessage(item: T, iterationStr: string): string;\n\n  /**\n   * Items to loop over.\n   */\n  items: T[];\n\n  /**\n   * A timeout for the notice before it is shown. Default is `500`.\n   */\n  noticeBeforeShownTimeoutInMilliseconds?: number;\n\n  /**\n   * A minimum timeout for the notice. Default is `2000`.\n   */\n  noticeMinTimeoutInMilliseconds?: number;\n\n  /**\n   * Process each item.\n   *\n   * @param item - The current item.\n   */\n  processItem(item: T): Promisable<void>;\n\n  /**\n   * A title of the progress bar. Default is `''`.\n   */\n  progressBarTitle?: string;\n\n  /**\n   * Whether to continue the loop on error. Default is `true`.\n   */\n  shouldContinueOnError?: boolean;\n\n  /**\n   * Whether to show a notice. Default is `true`.\n   */\n  shouldShowNotice?: boolean;\n\n  /**\n   * Whether to show a progress bar. Default is `true`.\n   */\n  shouldShowProgressBar?: boolean;\n\n  /**\n   * A threshold for the UI update. Default is `100`.\n   */\n  uiUpdateThresholdInMilliseconds?: number;\n}\n\n/**\n * Loops over a list of items and processes each item.\n *\n * @param options - The options for the loop.\n */\nexport async function loop<T>(options: LoopOptions<T>): Promise<void> {\n  const DEFAULT_OPTIONS: Required<LoopOptions<T>> = {\n    abortSignal: abortSignalNever(),\n    buildNoticeMessage() {\n      throw new Error('buildNoticeMessage is required');\n    },\n    items: [],\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeBeforeShownTimeoutInMilliseconds: 500,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    noticeMinTimeoutInMilliseconds: 2000,\n    processItem: noop,\n    progressBarTitle: '',\n    shouldContinueOnError: true,\n    shouldShowNotice: true,\n    shouldShowProgressBar: true,\n    // eslint-disable-next-line no-magic-numbers -- Extracting magic number as a constant would be repetitive, as the value is used only once and its name would be the same as the property.\n    uiUpdateThresholdInMilliseconds: 100\n  };\n\n  const fullOptions: Required<LoopOptions<T>> = {\n    ...DEFAULT_OPTIONS,\n    ...options\n  };\n\n  const stackTrace = getStackTrace(1);\n\n  const items = fullOptions.items;\n  let iterationCount = 0;\n  let notice = null as Notice | null;\n  let isDone = false;\n  invokeAsyncSafely(() => showNotice());\n\n  const noticeMinTimeoutPromise = sleep(fullOptions.noticeMinTimeoutInMilliseconds);\n  const progressBarEl = createEl('progress');\n  addPluginCssClasses(progressBarEl, 'loop');\n  progressBarEl.max = items.length;\n\n  let lastUIUpdateTimestamp = performance.now();\n\n  for (const item of items) {\n    if (fullOptions.abortSignal.aborted) {\n      notice?.hide();\n      return;\n    }\n    iterationCount++;\n    const iterationStr = `# ${String(iterationCount)} / ${String(items.length)}`;\n    const message = fullOptions.buildNoticeMessage(item, iterationStr);\n    if (!fullOptions.shouldShowProgressBar) {\n      notice?.setMessage(message);\n    }\n    getLibDebugger('Loop')(message);\n\n    try {\n      if (performance.now() - lastUIUpdateTimestamp > fullOptions.uiUpdateThresholdInMilliseconds) {\n        await requestAnimationFrameAsync();\n        lastUIUpdateTimestamp = performance.now();\n      }\n      await fullOptions.processItem(item);\n    } catch (error) {\n      console.error('Error processing item', item);\n      if (!fullOptions.shouldContinueOnError) {\n        notice?.hide();\n        throw new CustomStackTraceError('loop failed', stackTrace, error);\n      }\n\n      emitAsyncErrorEvent(new CustomStackTraceError(ASYNC_WRAPPER_ERROR_MESSAGE, stackTrace, error));\n    }\n    progressBarEl.value++;\n  }\n  if (notice) {\n    await noticeMinTimeoutPromise;\n  }\n  notice?.hide();\n  isDone = true;\n\n  async function showNotice(): Promise<void> {\n    if (!fullOptions.shouldShowNotice) {\n      return;\n    }\n    await sleep(fullOptions.noticeBeforeShownTimeoutInMilliseconds);\n    if (isDone) {\n      return;\n    }\n    notice = new Notice('', 0);\n    if (!fullOptions.shouldShowProgressBar) {\n      return;\n    }\n    const fragment = createFragment();\n    fragment.createDiv({ text: fullOptions.progressBarTitle });\n    fragment.appendChild(progressBarEl);\n    notice.setMessage(fragment);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,sBAAuB;AAEvB,6BAAiC;AACjC,mBAGO;AACP,mBAA+B;AAC/B,mBAKO;AACP,sBAAqB;AACrB,2BAAoC;AAyEpC,eAAsB,KAAQ,SAAwC;AACpE,QAAM,kBAA4C;AAAA,IAChD,iBAAa,yCAAiB;AAAA,IAC9B,qBAAqB;AACnB,YAAM,IAAI,MAAM,gCAAgC;AAAA,IAClD;AAAA,IACA,OAAO,CAAC;AAAA;AAAA,IAER,wCAAwC;AAAA;AAAA,IAExC,gCAAgC;AAAA,IAChC,aAAa;AAAA,IACb,kBAAkB;AAAA,IAClB,uBAAuB;AAAA,IACvB,kBAAkB;AAAA,IAClB,uBAAuB;AAAA;AAAA,IAEvB,iCAAiC;AAAA,EACnC;AAEA,QAAM,cAAwC;AAAA,IAC5C,GAAG;AAAA,IACH,GAAG;AAAA,EACL;AAEA,QAAM,iBAAa,4BAAc,CAAC;AAElC,QAAM,QAAQ,YAAY;AAC1B,MAAI,iBAAiB;AACrB,MAAI,SAAS;AACb,MAAI,SAAS;AACb,sCAAkB,MAAM,WAAW,CAAC;AAEpC,QAAM,0BAA0B,MAAM,YAAY,8BAA8B;AAChF,QAAM,gBAAgB,SAAS,UAAU;AACzC,gDAAoB,eAAe,MAAM;AACzC,gBAAc,MAAM,MAAM;AAE1B,MAAI,wBAAwB,YAAY,IAAI;AAE5C,aAAW,QAAQ,OAAO;AACxB,QAAI,YAAY,YAAY,SAAS;AACnC,cAAQ,KAAK;AACb;AAAA,IACF;AACA;AACA,UAAM,eAAe,KAAK,OAAO,cAAc,CAAC,MAAM,OAAO,MAAM,MAAM,CAAC;AAC1E,UAAM,UAAU,YAAY,mBAAmB,MAAM,YAAY;AACjE,QAAI,CAAC,YAAY,uBAAuB;AACtC,cAAQ,WAAW,OAAO;AAAA,IAC5B;AACA,qCAAe,MAAM,EAAE,OAAO;AAE9B,QAAI;AACF,UAAI,YAAY,IAAI,IAAI,wBAAwB,YAAY,iCAAiC;AAC3F,kBAAM,yCAA2B;AACjC,gCAAwB,YAAY,IAAI;AAAA,MAC1C;AACA,YAAM,YAAY,YAAY,IAAI;AAAA,IACpC,SAAS,OAAO;AACd,cAAQ,MAAM,yBAAyB,IAAI;AAC3C,UAAI,CAAC,YAAY,uBAAuB;AACtC,gBAAQ,KAAK;AACb,cAAM,IAAI,mCAAsB,eAAe,YAAY,KAAK;AAAA,MAClE;AAEA,4CAAoB,IAAI,mCAAsB,0CAA6B,YAAY,KAAK,CAAC;AAAA,IAC/F;AACA,kBAAc;AAAA,EAChB;AACA,MAAI,QAAQ;AACV,UAAM;AAAA,EACR;AACA,UAAQ,KAAK;AACb,WAAS;AAET,iBAAe,aAA4B;AACzC,QAAI,CAAC,YAAY,kBAAkB;AACjC;AAAA,IACF;AACA,UAAM,MAAM,YAAY,sCAAsC;AAC9D,QAAI,QAAQ;AACV;AAAA,IACF;AACA,aAAS,IAAI,uBAAO,IAAI,CAAC;AACzB,QAAI,CAAC,YAAY,uBAAuB;AACtC;AAAA,IACF;AACA,UAAM,WAAW,eAAe;AAChC,aAAS,UAAU,EAAE,MAAM,YAAY,iBAAiB,CAAC;AACzD,aAAS,YAAY,aAAa;AAClC,WAAO,WAAW,QAAQ;AAAA,EAC5B;AACF;",
  "names": []
}
