UNPKG

obsidian-dev-utils

Version:

This is the collection of useful functions that you can use for your Obsidian plugin development

289 lines (275 loc) 24.4 kB
/* 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 {}; } })(); "use strict"; 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 HTMLElement_exports = {}; __export(HTMLElement_exports, { appendCodeBlock: () => appendCodeBlock, createDivAsync: () => createDivAsync, createElAsync: () => createElAsync, createFragmentAsync: () => createFragmentAsync, createSpanAsync: () => createSpanAsync, createSvgAsync: () => createSvgAsync, ensureLoaded: () => ensureLoaded, getZIndex: () => getZIndex, isElementVisibleInOffsetParent: () => isElementVisibleInOffsetParent, isLoaded: () => isLoaded, onAncestorScrollOrResize: () => onAncestorScrollOrResize, toPx: () => toPx }); module.exports = __toCommonJS(HTMLElement_exports); function appendCodeBlock(el, code) { el.createEl("strong", { cls: "markdown-rendered code" }, (strong) => { strong.createEl("code", { text: code }); }); } async function createDivAsync(o, callback) { const div = createDiv(o); await callback?.(div); return div; } async function createElAsync(tag, o, callback) { const el = createEl(tag, o); await callback?.(el); return el; } async function createFragmentAsync(callback) { const fragment = createFragment(); await callback?.(fragment); return fragment; } async function createSpanAsync(o, callback) { const span = createSpan(o); await callback?.(span); return span; } async function createSvgAsync(tag, o, callback) { const svg = createSvg(tag, o); await callback?.(svg); return svg; } async function ensureLoaded(el) { if (isLoaded(el)) { return; } if (el instanceof HTMLBodyElement || el instanceof HTMLImageElement || el instanceof HTMLIFrameElement || el instanceof HTMLEmbedElement || el instanceof HTMLLinkElement || el instanceof HTMLObjectElement || el instanceof HTMLStyleElement || el instanceof HTMLTrackElement) { await new Promise((resolve) => { el.addEventListener("load", resolve); el.addEventListener("error", resolve); }); return; } await Promise.all(getLoadableElements(el).map(ensureLoaded)); } function getZIndex(el) { let el2 = el; while (el2) { const zIndexStr = getComputedStyle(el2).zIndex; const zIndex = Number.parseInt(zIndexStr, 10); if (!Number.isNaN(zIndex)) { return zIndex; } el2 = el2.parentElement; } return 0; } function isElementVisibleInOffsetParent(el) { const parentEl = el.offsetParent; if (!parentEl) { return false; } const elRect = el.getBoundingClientRect(); const parentElRect = parentEl.getBoundingClientRect(); return parentElRect.top <= elRect.top && elRect.bottom <= parentElRect.bottom && parentElRect.left <= elRect.left && elRect.right <= parentElRect.right; } function isLoaded(el) { if (el instanceof HTMLBodyElement) { return document.readyState === "complete" || document.readyState === "interactive"; } if (el instanceof HTMLImageElement) { return el.complete && el.naturalWidth > 0; } if (el instanceof HTMLIFrameElement) { return !!el.contentDocument; } if (el instanceof HTMLEmbedElement) { return !!el.getSVGDocument(); } if (el instanceof HTMLLinkElement) { return el.rel === "stylesheet" ? el.sheet !== null : true; } if (el instanceof HTMLObjectElement) { return !!el.contentDocument || !!el.getSVGDocument(); } if (el instanceof HTMLScriptElement) { return true; } if (el instanceof HTMLStyleElement) { return !!el.sheet; } if (el instanceof HTMLTrackElement) { const READY_STATE_LOADED = 2; return el.readyState === READY_STATE_LOADED; } return getLoadableElements(el).every(isLoaded); } function onAncestorScrollOrResize(node, callback) { const ancestors = []; ancestors.push(document); ancestors.push(window); let currentNode = node; while (currentNode) { ancestors.push(currentNode); currentNode = currentNode.parentNode; } let isEventTriggered = false; for (const ancestor of ancestors) { ancestor.addEventListener("scroll", callbackSmooth, { capture: true }); ancestor.addEventListener("resize", callbackSmooth, { capture: true }); } return () => { for (const ancestor of ancestors) { ancestor.removeEventListener("scroll", callbackSmooth, { capture: true }); ancestor.removeEventListener("resize", callbackSmooth, { capture: true }); } }; function callbackSmooth() { if (isEventTriggered) { return; } isEventTriggered = true; requestAnimationFrame(() => { try { callback(); } finally { isEventTriggered = false; } }); } } function toPx(value) { return `${String(value)}px`; } function getLoadableElements(el) { return Array.from(el.querySelectorAll("body, img, iframe, embed, link, object, script, style, track")); } // Annotate the CommonJS export names for ESM import in node: 0 && (module.exports = { appendCodeBlock, createDivAsync, createElAsync, createFragmentAsync, createSpanAsync, createSvgAsync, ensureLoaded, getZIndex, isElementVisibleInOffsetParent, isLoaded, onAncestorScrollOrResize, toPx }); //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/HTMLElement.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Helpers for working with HTML elements.\n */\n\nimport type { Promisable } from 'type-fest';\n\n/**\n * A HTML element that can be validated.\n */\nexport interface ValidatorElement extends HTMLElement {\n  /**\n   * Checks the validity of the element.\n   *\n   * @returns True if the element is valid, false otherwise.\n   */\n  checkValidity(): boolean;\n\n  /**\n   * Reports the validity of the element.\n   */\n  reportValidity(): boolean;\n\n  /**\n   * Sets a custom error message on the element.\n   *\n   * @param error - The error message to set on the element.\n   */\n  setCustomValidity(error: string): void;\n\n  /**\n   * An error message of the element.\n   */\n  readonly validationMessage: string;\n}\n\n/**\n * Appends a code block to the given DocumentFragment or HTMLElement.\n *\n * @param el - The DocumentFragment or HTMLElement to append the code block to.\n * @param code - The code to be displayed in the code block.\n */\nexport function appendCodeBlock(el: DocumentFragment | HTMLElement, code: string): void {\n  el.createEl('strong', { cls: 'markdown-rendered code' }, (strong) => {\n    strong.createEl('code', { text: code });\n  });\n}\n\n/**\n * Creates a div asynchronously.\n *\n * @param o - The element information.\n * @param callback - The callback to call when the div is created.\n * @returns A {@link Promise} that resolves to the div.\n */\nexport async function createDivAsync(\n  o?: DomElementInfo | string,\n  callback?: (el: HTMLDivElement) => Promisable<void>\n): Promise<HTMLDivElement> {\n  const div = createDiv(o);\n  await callback?.(div);\n  return div;\n}\n\n/**\n * Creates an element asynchronously.\n *\n * @param tag - The tag name of the element to create.\n * @param o - The element information.\n * @param callback - The callback to call when the element is created.\n * @returns A {@link Promise} that resolves to the element.\n */\nexport async function createElAsync<K extends keyof HTMLElementTagNameMap>(\n  tag: K,\n  o?: DomElementInfo | string,\n  callback?: (el: HTMLElementTagNameMap[K]) => Promisable<void>\n): Promise<HTMLElementTagNameMap[K]> {\n  const el = createEl(tag, o);\n  await callback?.(el);\n  return el;\n}\n\n/**\n * Creates a DocumentFragment asynchronously.\n *\n * @param callback - The callback to call when the DocumentFragment is created.\n * @returns A {@link Promise} that resolves to the DocumentFragment.\n */\nexport async function createFragmentAsync(callback?: (el: DocumentFragment) => Promisable<void>): Promise<DocumentFragment> {\n  const fragment = createFragment();\n  await callback?.(fragment);\n  return fragment;\n}\n\n/**\n * Creates a span asynchronously.\n *\n * @param o - The element information.\n * @param callback - The callback to call when the span is created.\n * @returns A {@link Promise} that resolves to the span.\n */\nexport async function createSpanAsync(\n  o?: DomElementInfo | string,\n  callback?: (el: HTMLSpanElement) => Promisable<void>\n): Promise<HTMLSpanElement> {\n  const span = createSpan(o);\n  await callback?.(span);\n  return span;\n}\n\n/**\n * Creates a svg asynchronously.\n *\n * @param tag - The tag name of the svg to create.\n * @param o - The svg information.\n * @param callback - The callback to call when the svg is created.\n * @returns A {@link Promise} that resolves to the svg.\n */\nexport async function createSvgAsync<K extends keyof SVGElementTagNameMap>(\n  tag: K,\n  // eslint-disable-next-line no-undef -- Workaround until https://github.com/obsidianmd/eslint-plugin/pull/89 is merged.\n  o?: string | SvgElementInfo,\n  callback?: (el: SVGElementTagNameMap[K]) => Promisable<void>\n): Promise<SVGElementTagNameMap[K]> {\n  const svg = createSvg(tag, o);\n  await callback?.(svg);\n  return svg;\n}\n\n/**\n * Ensures that the given element is loaded.\n *\n * @param el - The element to ensure is loaded.\n * @returns A {@link Promise} that resolves when the element is loaded.\n */\nexport async function ensureLoaded(el: Element): Promise<void> {\n  if (isLoaded(el)) {\n    return;\n  }\n  if (\n    el instanceof HTMLBodyElement\n    || el instanceof HTMLImageElement\n    || el instanceof HTMLIFrameElement\n    || el instanceof HTMLEmbedElement\n    || el instanceof HTMLLinkElement\n    || el instanceof HTMLObjectElement\n    || el instanceof HTMLStyleElement\n    || el instanceof HTMLTrackElement\n  ) {\n    await new Promise((resolve) => {\n      el.addEventListener('load', resolve);\n      el.addEventListener('error', resolve);\n    });\n    return;\n  }\n\n  await Promise.all(getLoadableElements(el).map(ensureLoaded));\n}\n\n/**\n * Gets the z-index of the given element.\n *\n * @param el - The element to get the z-index of.\n * @returns The z-index of the element.\n */\nexport function getZIndex(el: Element): number {\n  let el2: Element | null = el;\n\n  while (el2) {\n    const zIndexStr = getComputedStyle(el2).zIndex;\n    const zIndex = Number.parseInt(zIndexStr, 10);\n    if (!Number.isNaN(zIndex)) {\n      return zIndex;\n    }\n    el2 = el2.parentElement;\n  }\n\n  return 0;\n}\n\n/**\n * Checks if the element is visible in the offset parent.\n *\n * @param el - The element to check.\n * @returns True if the element is visible in the offset parent, false otherwise.\n */\nexport function isElementVisibleInOffsetParent(el: HTMLElement): boolean {\n  const parentEl = el.offsetParent;\n  if (!parentEl) {\n    return false;\n  }\n\n  const elRect = el.getBoundingClientRect();\n  const parentElRect = parentEl.getBoundingClientRect();\n\n  return (\n    parentElRect.top <= elRect.top\n    && elRect.bottom <= parentElRect.bottom\n    && parentElRect.left <= elRect.left\n    && elRect.right <= parentElRect.right\n  );\n}\n\n/**\n * Checks if the element is loaded.\n *\n * @param el - The element to check.\n * @returns True if the element is loaded, false otherwise.\n */\nexport function isLoaded(el: Element): boolean {\n  if (el instanceof HTMLBodyElement) {\n    return document.readyState === 'complete' || document.readyState === 'interactive';\n  }\n\n  if (el instanceof HTMLImageElement) {\n    return el.complete && el.naturalWidth > 0;\n  }\n\n  if (el instanceof HTMLIFrameElement) {\n    return !!el.contentDocument;\n  }\n\n  if (el instanceof HTMLEmbedElement) {\n    return !!el.getSVGDocument();\n  }\n\n  if (el instanceof HTMLLinkElement) {\n    return el.rel === 'stylesheet' ? el.sheet !== null : true;\n  }\n\n  if (el instanceof HTMLObjectElement) {\n    return !!el.contentDocument || !!el.getSVGDocument();\n  }\n\n  if (el instanceof HTMLScriptElement) {\n    return true;\n  }\n\n  if (el instanceof HTMLStyleElement) {\n    return !!el.sheet;\n  }\n\n  if (el instanceof HTMLTrackElement) {\n    const READY_STATE_LOADED = 2;\n    return el.readyState === READY_STATE_LOADED;\n  }\n\n  return getLoadableElements(el).every(isLoaded);\n}\n\n/**\n * Adds an event listener to the ancestor nodes of the given node.\n *\n * @param node - The node to add the event listener to.\n * @param callback - The callback to call when the event is triggered.\n * @returns A function to remove the event listener.\n */\nexport function onAncestorScrollOrResize(node: Node, callback: () => void): () => void {\n  const ancestors: EventTarget[] = [];\n  ancestors.push(document);\n  ancestors.push(window);\n\n  let currentNode: Node | null = node;\n\n  while (currentNode) {\n    ancestors.push(currentNode);\n    currentNode = currentNode.parentNode;\n  }\n\n  let isEventTriggered = false;\n\n  for (const ancestor of ancestors) {\n    ancestor.addEventListener('scroll', callbackSmooth, { capture: true });\n    ancestor.addEventListener('resize', callbackSmooth, { capture: true });\n  }\n\n  return () => {\n    for (const ancestor of ancestors) {\n      ancestor.removeEventListener('scroll', callbackSmooth, { capture: true });\n      ancestor.removeEventListener('resize', callbackSmooth, { capture: true });\n    }\n  };\n\n  function callbackSmooth(): void {\n    if (isEventTriggered) {\n      return;\n    }\n\n    isEventTriggered = true;\n\n    requestAnimationFrame(() => {\n      try {\n        callback();\n      } finally {\n        isEventTriggered = false;\n      }\n    });\n  }\n}\n\n/**\n * Converts a number to a string with 'px' appended.\n *\n * @param value - The number to convert.\n * @returns The number as a string with 'px' appended.\n */\nexport function toPx(value: number): string {\n  return `${String(value)}px`;\n}\n\nfunction getLoadableElements(el: Element): Element[] {\n  return Array.from(el.querySelectorAll('body, img, iframe, embed, link, object, script, style, track'));\n}\n"],
  "mappings": "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",
  "names": []
}
