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
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 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": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AA2CO,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;AASA,eAAsB,eACpB,GACA,UACyB;AACzB,QAAM,MAAM,UAAU,CAAC;AACvB,QAAM,WAAW,GAAG;AACpB,SAAO;AACT;AAUA,eAAsB,cACpB,KACA,GACA,UACmC;AACnC,QAAM,KAAK,SAAS,KAAK,CAAC;AAC1B,QAAM,WAAW,EAAE;AACnB,SAAO;AACT;AAQA,eAAsB,oBAAoB,UAAkF;AAC1H,QAAM,WAAW,eAAe;AAChC,QAAM,WAAW,QAAQ;AACzB,SAAO;AACT;AASA,eAAsB,gBACpB,GACA,UAC0B;AAC1B,QAAM,OAAO,WAAW,CAAC;AACzB,QAAM,WAAW,IAAI;AACrB,SAAO;AACT;AAUA,eAAsB,eACpB,KAEA,GACA,UACkC;AAClC,QAAM,MAAM,UAAU,KAAK,CAAC;AAC5B,QAAM,WAAW,GAAG;AACpB,SAAO;AACT;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,UAAU,IAAqB;AAC7C,MAAI,MAAsB;AAE1B,SAAO,KAAK;AACV,UAAM,YAAY,iBAAiB,GAAG,EAAE;AACxC,UAAM,SAAS,OAAO,SAAS,WAAW,EAAE;AAC5C,QAAI,CAAC,OAAO,MAAM,MAAM,GAAG;AACzB,aAAO;AAAA,IACT;AACA,UAAM,IAAI;AAAA,EACZ;AAEA,SAAO;AACT;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": []
}
