obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
115 lines (112 loc) • 14.1 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 initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd(){return"/"},env:{},platform:"android"};globalThis.process=browserProcess})();
function appendCodeBlock(el, code) {
el.createEl("strong", { cls: "markdown-rendered code" }, (strong) => {
strong.createEl("code", { text: code });
});
}
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 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"));
}
export {
appendCodeBlock,
ensureLoaded,
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\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 * 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 * 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": ";;;;;;;AAyCO,SAAS,gBAAgB,IAAoC,MAAoB;AACtF,KAAG,SAAS,UAAU,EAAE,KAAK,yBAAyB,GAAG,CAAC,WAAW;AACnE,WAAO,SAAS,QAAQ,EAAE,MAAM,KAAK,CAAC;AAAA,EACxC,CAAC;AACH;AAQA,eAAsB,aAAa,IAA4B;AAC7D,MAAI,SAAS,EAAE,GAAG;AAChB;AAAA,EACF;AACA,MACE,cAAc,mBACX,cAAc,oBACd,cAAc,qBACd,cAAc,oBACd,cAAc,mBACd,cAAc,qBACd,cAAc,oBACd,cAAc,kBACjB;AACA,UAAM,IAAI,QAAQ,CAAC,YAAY;AAC7B,SAAG,iBAAiB,QAAQ,OAAO;AACnC,SAAG,iBAAiB,SAAS,OAAO;AAAA,IACtC,CAAC;AACD;AAAA,EACF;AAEA,QAAM,QAAQ,IAAI,oBAAoB,EAAE,EAAE,IAAI,YAAY,CAAC;AAC7D;AAQO,SAAS,+BAA+B,IAA0B;AACvE,QAAM,WAAW,GAAG;AACpB,MAAI,CAAC,UAAU;AACb,WAAO;AAAA,EACT;AAEA,QAAM,SAAS,GAAG,sBAAsB;AACxC,QAAM,eAAe,SAAS,sBAAsB;AAEpD,SACE,aAAa,OAAO,OAAO,OACxB,OAAO,UAAU,aAAa,UAC9B,aAAa,QAAQ,OAAO,QAC5B,OAAO,SAAS,aAAa;AAEpC;AAQO,SAAS,SAAS,IAAsB;AAC7C,MAAI,cAAc,iBAAiB;AACjC,WAAO,SAAS,eAAe,cAAc,SAAS,eAAe;AAAA,EACvE;AAEA,MAAI,cAAc,kBAAkB;AAClC,WAAO,GAAG,YAAY,GAAG,eAAe;AAAA,EAC1C;AAEA,MAAI,cAAc,mBAAmB;AACnC,WAAO,CAAC,CAAC,GAAG;AAAA,EACd;AAEA,MAAI,cAAc,kBAAkB;AAClC,WAAO,CAAC,CAAC,GAAG,eAAe;AAAA,EAC7B;AAEA,MAAI,cAAc,iBAAiB;AACjC,WAAO,GAAG,QAAQ,eAAe,GAAG,UAAU,OAAO;AAAA,EACvD;AAEA,MAAI,cAAc,mBAAmB;AACnC,WAAO,CAAC,CAAC,GAAG,mBAAmB,CAAC,CAAC,GAAG,eAAe;AAAA,EACrD;AAEA,MAAI,cAAc,mBAAmB;AACnC,WAAO;AAAA,EACT;AAEA,MAAI,cAAc,kBAAkB;AAClC,WAAO,CAAC,CAAC,GAAG;AAAA,EACd;AAEA,MAAI,cAAc,kBAAkB;AAClC,UAAM,qBAAqB;AAC3B,WAAO,GAAG,eAAe;AAAA,EAC3B;AAEA,SAAO,oBAAoB,EAAE,EAAE,MAAM,QAAQ;AAC/C;AASO,SAAS,yBAAyB,MAAY,UAAkC;AACrF,QAAM,YAA2B,CAAC;AAClC,YAAU,KAAK,QAAQ;AACvB,YAAU,KAAK,MAAM;AAErB,MAAI,cAA2B;AAE/B,SAAO,aAAa;AAClB,cAAU,KAAK,WAAW;AAC1B,kBAAc,YAAY;AAAA,EAC5B;AAEA,MAAI,mBAAmB;AAEvB,aAAW,YAAY,WAAW;AAChC,aAAS,iBAAiB,UAAU,gBAAgB,EAAE,SAAS,KAAK,CAAC;AACrE,aAAS,iBAAiB,UAAU,gBAAgB,EAAE,SAAS,KAAK,CAAC;AAAA,EACvE;AAEA,SAAO,MAAM;AACX,eAAW,YAAY,WAAW;AAChC,eAAS,oBAAoB,UAAU,gBAAgB,EAAE,SAAS,KAAK,CAAC;AACxE,eAAS,oBAAoB,UAAU,gBAAgB,EAAE,SAAS,KAAK,CAAC;AAAA,IAC1E;AAAA,EACF;AAEA,WAAS,iBAAuB;AAC9B,QAAI,kBAAkB;AACpB;AAAA,IACF;AAEA,uBAAmB;AAEnB,0BAAsB,MAAM;AAC1B,UAAI;AACF,iBAAS;AAAA,MACX,UAAE;AACA,2BAAmB;AAAA,MACrB;AAAA,IACF,CAAC;AAAA,EACH;AACF;AAQO,SAAS,KAAK,OAAuB;AAC1C,SAAO,GAAG,OAAO,KAAK,CAAC;AACzB;AAEA,SAAS,oBAAoB,IAAwB;AACnD,SAAO,MAAM,KAAK,GAAG,iBAAiB,8DAA8D,CAAC;AACvG;",
  "names": []
}
