wxt 0.6.6 → 0.7.1-alpha1

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/dist/client.d.ts CHANGED
@@ -10,8 +10,9 @@ import { Manifest } from 'webextension-polyfill';
10
10
  declare class ContentScriptContext extends AbortController {
11
11
  #private;
12
12
  private readonly contentScriptName;
13
+ readonly options?: Omit<ContentScriptDefinition, "main"> | undefined;
13
14
  static SCRIPT_STARTED_MESSAGE_TYPE: string;
14
- constructor(contentScriptName: string);
15
+ constructor(contentScriptName: string, options?: Omit<ContentScriptDefinition, "main"> | undefined);
15
16
  get isInvalid(): boolean;
16
17
  get isValid(): boolean;
17
18
  /**
@@ -119,6 +120,18 @@ interface ContentScriptDefinition extends ExcludableEntrypoint {
119
120
  * @default "ISOLATED"
120
121
  */
121
122
  world?: 'ISOLATED' | 'MAIN';
123
+ /**
124
+ * Customize how imported/generated styles are injected with the content script. Regardless of the
125
+ * mode selected, CSS will always be built and included in the output directory.
126
+ *
127
+ * - `"manifest"` - Include the CSS in the manifest, under the content script's `css` array.
128
+ * - `"manual"` - Exclude the CSS from the manifest. You are responsible for manually loading it
129
+ * onto the page. Use `browser.runtime.getURL("content-scripts/<name>.css")` to get the file's
130
+ * URL
131
+ *
132
+ * @default "manifest"
133
+ */
134
+ cssInjectionMode?: 'manifest' | 'manual' | 'ui';
122
135
  /**
123
136
  * Main function executed when the content script is loaded.
124
137
  */
@@ -150,6 +163,133 @@ declare function defineContentScript(definition: ContentScriptDefinition): Conte
150
163
  declare function defineBackground(main: () => void): BackgroundScriptDefintition;
151
164
  declare function defineBackground(definition: BackgroundScriptDefintition): BackgroundScriptDefintition;
152
165
 
153
- declare function mountContentScriptUi(): void;
166
+ /**
167
+ * Utility for mounting content script UI's with isolated styles. Automatically removed from the DOM
168
+ * when the content script's context is invalidated.
169
+ *
170
+ * See <https://wxt.dev/guide/content-scripts.html#ui> for full documentation.
171
+ *
172
+ * @example
173
+ * // entrypoints/example-ui.content/index.ts
174
+ * import "./style.css"
175
+ *
176
+ * export default defineContentScript({
177
+ * matches: ["*://*.google.com/*"],
178
+ * cssInjectionMode: "ui",
179
+ *
180
+ * async main(ctx) {
181
+ * const ui = await createContentScriptUi(ctx, {
182
+ * name: "example-overlay",
183
+ * type: "modal",
184
+ * mount(container) {
185
+ * const app = document.createElement("div");
186
+ * app.textContent = "Content Script UI";
187
+ * container.append(app);
188
+ * }
189
+ * })
190
+ * ui.mount();
191
+ * }
192
+ * })
193
+ */
194
+ declare function createContentScriptUi<T>(ctx: ContentScriptContext, options: ContentScriptUiOptions<T>): Promise<ContentScriptUi<T>>;
195
+ interface ContentScriptUi<T> {
196
+ /**
197
+ * The `HTMLElement` hosting the shadow root used to isolate the UI's styles. This is the element
198
+ * that get's added to the DOM. This element's style is not isolated from the webpage.
199
+ */
200
+ shadowHost: HTMLElement;
201
+ /**
202
+ * The container element inside the `ShadowRoot` whose styles are isolated. The UI is mounted
203
+ * inside this `HTMLElement`.
204
+ */
205
+ uiContainer: HTMLElement;
206
+ /**
207
+ * The shadow root performing the isolation.
208
+ */
209
+ shadow: ShadowRoot;
210
+ /**
211
+ * Custom data returned from the `options.mount` function.
212
+ */
213
+ mounted: T;
214
+ /**
215
+ * Function that mounts or remounts the UI on the page.
216
+ */
217
+ mount: () => void;
218
+ /**
219
+ * Function that removes the UI from the webpage.
220
+ */
221
+ remove: () => void;
222
+ }
223
+ interface BaseContentScriptUiOptions<T> {
224
+ /**
225
+ * The name of the custom component used to host the ShadowRoot. Must be kebab-case.
226
+ */
227
+ name: string;
228
+ /**
229
+ * In combination with `anchor`, decide how to add the UI to the DOM.
230
+ *
231
+ * - `"last"` (default) - Add the UI as the last child of the `anchor` element
232
+ * - `"first"` - Add the UI as the last child of the `anchor` element
233
+ * - `"replace"` - Replace the `anchor` element with the UI.
234
+ * - `"before"` - Add the UI as the sibling before the `anchor` element
235
+ * - `"after"` - Add the UI as the sibling after the `anchor` element
236
+ * - `(anchor, ui) => void` - Customizable function that let's you add the UI to the DOM
237
+ */
238
+ append?: ContentScriptAppendMode | ((anchor: Element, ui: Element) => void);
239
+ /**
240
+ * A CSS selector, element, or function that returns one of the two. Along with `append`, the
241
+ * `anchor` dictates where in the page the UI will be added.
242
+ */
243
+ anchor?: string | Element | null | undefined | (() => string | Element | null | undefined);
244
+ /**
245
+ * Callback executed when mounting the UI. This function should create and append the UI to the
246
+ * `container` element. It is called every time `ui.mount()` is called
247
+ *
248
+ * Optionally return a value that can be accessed at `ui.mounted` or in the `onRemove` callback.
249
+ */
250
+ mount: (container: Element) => T;
251
+ /**
252
+ * Callback called when the UI is removed from the webpage. Use to cleanup your UI, like
253
+ * unmounting your vue or react apps.
254
+ */
255
+ onRemove?: (mounted: T) => void;
256
+ /**
257
+ * Custom CSS text to apply to the UI. If your content script imports/generates CSS and you've
258
+ * set `cssInjectionMode: "ui"`, the imported CSS will be included automatically. You do not need
259
+ * to pass those styles in here. This is for any additional styles not in the imported CSS.
260
+ *
261
+ * See <https://wxt.dev/guide/content-scripts.html#ui> for more info.
262
+ */
263
+ css?: string;
264
+ }
265
+ type OverlayContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {
266
+ type: 'overlay';
267
+ /**
268
+ * When using `type: "overlay"`, the mounted element is 0px by 0px in size. Alignment specifies
269
+ * which corner is aligned with that 0x0 pixel space.
270
+ *
271
+ * @default "top-left"
272
+ */
273
+ alignment?: ContentScriptUiOverlayAlignment;
274
+ /**
275
+ * The `z-index` used on the `shadowHost`. Set to a positive number to show your UI over website
276
+ * content.
277
+ */
278
+ zIndex?: number;
279
+ };
280
+ type ModalContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {
281
+ type: 'modal';
282
+ /**
283
+ * The `z-index` used on the `shadowHost`. Set to a positive number to show your UI over website
284
+ * content.
285
+ */
286
+ zIndex?: number;
287
+ };
288
+ type InlineContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {
289
+ type: 'inline';
290
+ };
291
+ type ContentScriptUiOverlayAlignment = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';
292
+ type ContentScriptAppendMode = 'last' | 'first' | 'replace' | 'before' | 'after';
293
+ type ContentScriptUiOptions<T> = OverlayContentScriptUiOptions<T> | ModalContentScriptUiOptions<T> | InlineContentScriptUiOptions<T>;
154
294
 
155
- export { ContentScriptContext, defineBackground, defineContentScript, mountContentScriptUi };
295
+ export { ContentScriptAppendMode, ContentScriptContext, ContentScriptUi, ContentScriptUiOptions, ContentScriptUiOverlayAlignment, InlineContentScriptUiOptions, ModalContentScriptUiOptions, OverlayContentScriptUiOptions, createContentScriptUi, defineBackground, defineContentScript };
package/dist/client.js CHANGED
@@ -10,10 +10,8 @@ function defineBackground(arg) {
10
10
  return arg;
11
11
  }
12
12
 
13
- // src/client/mountContentScriptUi.ts
14
- function mountContentScriptUi() {
15
- throw Error("Not implemented: mountContentScriptUi");
16
- }
13
+ // src/client/createContentScriptUi.ts
14
+ import { createIsolatedElement } from "@webext-core/isolated-element";
17
15
 
18
16
  // src/client/browser.ts
19
17
  import originalBrowser from "webextension-polyfill";
@@ -37,11 +35,129 @@ var logger = {
37
35
  error: (...args) => print(console.error, ...args)
38
36
  };
39
37
 
38
+ // src/client/createContentScriptUi.ts
39
+ async function createContentScriptUi(ctx, options) {
40
+ const css = [options.css ?? ""];
41
+ if (ctx.options?.cssInjectionMode === "ui") {
42
+ css.push(await loadCss());
43
+ }
44
+ const {
45
+ isolatedElement: uiContainer,
46
+ parentElement: shadowHost,
47
+ shadow
48
+ } = await createIsolatedElement({
49
+ name: options.name,
50
+ css: {
51
+ textContent: css.join("\n").trim()
52
+ },
53
+ mode: "open"
54
+ });
55
+ const getAnchor = () => {
56
+ if (options.anchor == null)
57
+ return document.body;
58
+ let resolved = typeof options.anchor === "function" ? options.anchor() : options.anchor;
59
+ if (typeof resolved === "string")
60
+ return document.querySelector(resolved) ?? void 0;
61
+ return resolved ?? void 0;
62
+ };
63
+ let mounted;
64
+ const mount = () => {
65
+ const anchor = getAnchor();
66
+ if (anchor == null)
67
+ throw Error(
68
+ "Failed to mount content script ui: could not find anchor element"
69
+ );
70
+ mounted = options.mount(uiContainer);
71
+ switch (options.append) {
72
+ case void 0:
73
+ case "last":
74
+ anchor.append(shadowHost);
75
+ break;
76
+ case "first":
77
+ if (anchor.firstChild) {
78
+ anchor.insertBefore(shadowHost, anchor.firstChild);
79
+ } else {
80
+ anchor.append(shadowHost);
81
+ }
82
+ break;
83
+ case "replace":
84
+ anchor.replaceWith(shadowHost);
85
+ break;
86
+ case "after":
87
+ anchor.replaceWith(anchor, shadowHost);
88
+ break;
89
+ case "before":
90
+ anchor.replaceWith(shadowHost, anchor);
91
+ break;
92
+ default:
93
+ options.append(anchor, shadowHost);
94
+ break;
95
+ }
96
+ if (options.type !== "inline") {
97
+ if (options.zIndex != null)
98
+ shadowHost.style.zIndex = String(options.zIndex);
99
+ shadowHost.style.overflow = "visible";
100
+ shadowHost.style.position = "relative";
101
+ shadowHost.style.width = "0";
102
+ shadowHost.style.height = "0";
103
+ shadowHost.style.display = "block";
104
+ const html = shadow.querySelector("html");
105
+ if (options.type === "overlay") {
106
+ html.style.position = "absolute";
107
+ if (options.alignment?.startsWith("bottom-"))
108
+ html.style.bottom = "0";
109
+ else
110
+ html.style.top = "0";
111
+ if (options.alignment?.endsWith("-right"))
112
+ html.style.right = "0";
113
+ else
114
+ html.style.left = "0";
115
+ } else {
116
+ html.style.position = "fixed";
117
+ html.style.top = "0";
118
+ html.style.bottom = "0";
119
+ html.style.left = "0";
120
+ html.style.right = "0";
121
+ }
122
+ }
123
+ };
124
+ const remove = () => {
125
+ shadowHost.remove();
126
+ options.onRemove?.(mounted);
127
+ while (uiContainer.lastChild)
128
+ uiContainer.removeChild(uiContainer.lastChild);
129
+ };
130
+ ctx.onInvalidated(remove);
131
+ return {
132
+ shadow,
133
+ shadowHost,
134
+ uiContainer,
135
+ mount,
136
+ remove,
137
+ mounted
138
+ };
139
+ }
140
+ async function loadCss() {
141
+ const url = browser.runtime.getURL(`/content-scripts/${__ENTRYPOINT__}.css`);
142
+ try {
143
+ const res = await fetch(url);
144
+ const css = await res.text();
145
+ return css.replaceAll(":root", ":host");
146
+ } catch (err) {
147
+ logger.warn(
148
+ `Failed to load styles @ ${url}. Did you forget to import the stylesheet in your entrypoint?`,
149
+ err
150
+ );
151
+ return "";
152
+ }
153
+ }
154
+
40
155
  // src/client/utils/ContentScriptContext.ts
41
156
  var ContentScriptContext = class _ContentScriptContext extends AbortController {
42
- constructor(contentScriptName) {
157
+ constructor(contentScriptName, options) {
43
158
  super();
44
159
  this.contentScriptName = contentScriptName;
160
+ this.options = options;
45
161
  if (this.#isTopFrame) {
46
162
  this.#stopOldScripts();
47
163
  }
@@ -165,10 +281,13 @@ var ContentScriptContext = class _ContentScriptContext extends AbortController {
165
281
  );
166
282
  }
167
283
  #stopOldScripts() {
168
- window.postMessage({
169
- event: _ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE,
170
- contentScriptName: this.contentScriptName
171
- });
284
+ window.postMessage(
285
+ {
286
+ event: _ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE,
287
+ contentScriptName: this.contentScriptName
288
+ },
289
+ "*"
290
+ );
172
291
  }
173
292
  #listenForNewerScripts() {
174
293
  const cb = (event) => {
@@ -182,8 +301,8 @@ var ContentScriptContext = class _ContentScriptContext extends AbortController {
182
301
  };
183
302
  export {
184
303
  ContentScriptContext,
304
+ createContentScriptUi,
185
305
  defineBackground,
186
- defineContentScript,
187
- mountContentScriptUi
306
+ defineContentScript
188
307
  };
189
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/client/defineContentScript.ts", "../src/client/defineBackground.ts", "../src/client/mountContentScriptUi.ts", "../src/client/browser.ts", "../src/client/utils/logger.ts", "../src/client/utils/ContentScriptContext.ts"],
  "sourcesContent": ["import { ContentScriptDefinition } from '../core/types';\n\nexport function defineContentScript(\n  definition: ContentScriptDefinition,\n): ContentScriptDefinition {\n  return definition;\n}\n", "import { BackgroundScriptDefintition } from '..';\n\nexport function defineBackground(main: () => void): BackgroundScriptDefintition;\nexport function defineBackground(\n  definition: BackgroundScriptDefintition,\n): BackgroundScriptDefintition;\nexport function defineBackground(\n  arg: (() => void) | BackgroundScriptDefintition,\n): BackgroundScriptDefintition {\n  if (typeof arg === 'function') return { main: arg };\n  return arg;\n}\n", "export function mountContentScriptUi() {\n  throw Error('Not implemented: mountContentScriptUi');\n}\n", "import originalBrowser, { Browser, Runtime, I18n } from 'webextension-polyfill';\n\nexport interface AugmentedBrowser extends Browser {\n  runtime: WxtRuntime;\n  i18n: WxtI18n;\n}\n\nexport interface WxtRuntime extends Runtime.Static {\n  // Overriden per-project\n}\n\nexport interface WxtI18n extends I18n.Static {\n  // Overriden per-project\n}\n\nexport const browser: AugmentedBrowser = originalBrowser;\n", "function print(method: (...args: any[]) => void, ...args: any[]) {\n  if (import.meta.env.MODE === 'production') return;\n\n  if (typeof args[0] === 'string') {\n    const message = args.shift();\n    method(`[wxt] ${message}`, ...args);\n  } else {\n    method('[wxt]', ...args);\n  }\n}\n\n/**\n * Wrapper around `console` with a \"[wxt]\" prefix\n */\nexport const logger = {\n  debug: (...args: any[]) => print(console.debug, ...args),\n  log: (...args: any[]) => print(console.log, ...args),\n  warn: (...args: any[]) => print(console.warn, ...args),\n  error: (...args: any[]) => print(console.error, ...args),\n};\n", "import { browser } from '../browser';\nimport { logger } from './logger';\n\n/**\n * Extends [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).\n * Used to detect and stop content script code when the script is invalidated.\n *\n * It also provides several utilities like `ctx.setTimeout` and `ctx.setInterval` that should be used in\n * content scripts instead of `window.setTimeout` or `window.setInterval`.\n */\nexport class ContentScriptContext extends AbortController {\n  static SCRIPT_STARTED_MESSAGE_TYPE = 'wxt:content-script-started';\n\n  #isTopFrame = window.self === window.top;\n\n  constructor(private readonly contentScriptName: string) {\n    super();\n\n    if (this.#isTopFrame) {\n      this.#stopOldScripts();\n    }\n    this.setTimeout(() => {\n      // Run on next tick so the listener it adds isn't triggered by stopOldScript\n      this.#listenForNewerScripts();\n    });\n  }\n\n  get isInvalid(): boolean {\n    if (browser.runtime.id == null) {\n      this.notifyInvalidated(); // Sets `signal.aborted` to true\n    }\n    return this.signal.aborted;\n  }\n\n  get isValid(): boolean {\n    return !this.isInvalid;\n  }\n\n  /**\n   * Add a listener that is called when the content script's context is invalidated.\n   *\n   * @returns A function to remove the listener.\n   *\n   * @example\n   * browser.runtime.onMessage.addListener(cb);\n   * const removeInvalidatedListener = ctx.onInvalidated(() => {\n   *   browser.runtime.onMessage.removeListener(cb);\n   * })\n   * // ...\n   * removeInvalidatedListener();\n   */\n  onInvalidated(cb: () => void): () => void {\n    this.signal.addEventListener('abort', cb);\n    return () => this.signal.removeEventListener('abort', cb);\n  }\n\n  /**\n   * Return a promise that never resolves. Useful if you have an async function that shouldn't run\n   * after the context is expired.\n   *\n   * @example\n   * const getValueFromStorage = async () => {\n   *   if (ctx.isInvalid) return ctx.block();\n   *\n   *   // ...\n   * }\n   */\n  block<T>(): Promise<T> {\n    return new Promise(() => {\n      // noop\n    });\n  }\n\n  /**\n   * Wrapper around `window.setInterval` that automatically clears the interval when invalidated.\n   */\n  setInterval(handler: () => void, timeout?: number): number {\n    const id = setInterval(() => {\n      if (this.isValid) handler();\n    }, timeout) as unknown as number;\n    this.onInvalidated(() => clearInterval(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.setTimeout` that automatically clears the interval when invalidated.\n   */\n  setTimeout(handler: () => void, timeout?: number): number {\n    const id = setTimeout(() => {\n      if (this.isValid) handler();\n    }, timeout) as unknown as number;\n    this.onInvalidated(() => clearTimeout(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.requestAnimationFrame` that automatically cancels the request when\n   * invalidated.\n   */\n  requestAnimationFrame(callback: FrameRequestCallback): number {\n    const id = requestAnimationFrame((...args) => {\n      if (this.isValid) callback(...args);\n    });\n\n    this.onInvalidated(() => cancelAnimationFrame(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.requestIdleCallback` that automatically cancels the request when\n   * invalidated.\n   */\n  requestIdleCallback(\n    callback: IdleRequestCallback,\n    options?: IdleRequestOptions,\n  ): number {\n    const id = requestIdleCallback((...args) => {\n      if (!this.signal.aborted) callback(...args);\n    }, options);\n\n    this.onInvalidated(() => cancelIdleCallback(id));\n    return id;\n  }\n\n  /**\n   * Call `target.addEventListener` and remove the event listener when the context is invalidated.\n   *\n   * @example\n   * ctx.addEventListener(window, \"mousemove\", () => {\n   *   // ...\n   * });\n   * ctx.addEventListener(document, \"visibilitychange\", () => {\n   *   // ...\n   * });\n   */\n  addEventListener(\n    target: any,\n    type: string,\n    handler: (event: Event) => void,\n    options?: AddEventListenerOptions,\n  ) {\n    target.addEventListener?.(type, handler, options);\n    this.onInvalidated(\n      () => target.removeEventListener?.(type, handler, options),\n    );\n  }\n\n  /**\n   * Abort the abort controller and execute all `onInvalidated` listeners.\n   */\n  notifyInvalidated() {\n    this.abort('Content script context invalidated');\n    logger.debug(\n      `Content script \"${this.contentScriptName}\" context invalidated`,\n    );\n  }\n\n  #stopOldScripts() {\n    // Use postMessage so it get's sent to all the frames of the page.\n    window.postMessage({\n      event: ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE,\n      contentScriptName: this.contentScriptName,\n    });\n  }\n\n  #listenForNewerScripts() {\n    const cb = (event: MessageEvent) => {\n      if (\n        event.data?.type === ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE &&\n        event.data?.contentScriptName === this.contentScriptName\n      ) {\n        this.notifyInvalidated();\n      }\n    };\n\n    addEventListener('message', cb);\n    this.onInvalidated(() => removeEventListener('message', cb));\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

308
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/client/defineContentScript.ts", "../src/client/defineBackground.ts", "../src/client/createContentScriptUi.ts", "../src/client/browser.ts", "../src/client/utils/logger.ts", "../src/client/utils/ContentScriptContext.ts"],
  "sourcesContent": ["import { ContentScriptDefinition } from '../core/types';\n\nexport function defineContentScript(\n  definition: ContentScriptDefinition,\n): ContentScriptDefinition {\n  return definition;\n}\n", "import { BackgroundScriptDefintition } from '..';\n\nexport function defineBackground(main: () => void): BackgroundScriptDefintition;\nexport function defineBackground(\n  definition: BackgroundScriptDefintition,\n): BackgroundScriptDefintition;\nexport function defineBackground(\n  arg: (() => void) | BackgroundScriptDefintition,\n): BackgroundScriptDefintition {\n  if (typeof arg === 'function') return { main: arg };\n  return arg;\n}\n", "import { createIsolatedElement } from '@webext-core/isolated-element';\nimport { browser } from './browser';\nimport { logger } from './utils/logger';\nimport { ContentScriptContext } from '.';\n\n/**\n * Utility for mounting content script UI's with isolated styles. Automatically removed from the DOM\n * when the content script's context is invalidated.\n *\n * See <https://wxt.dev/guide/content-scripts.html#ui> for full documentation.\n *\n * @example\n * // entrypoints/example-ui.content/index.ts\n * import \"./style.css\"\n *\n * export default defineContentScript({\n *   matches: [\"*://*.google.com/*\"],\n *   cssInjectionMode: \"ui\",\n *\n *   async main(ctx) {\n *     const ui = await createContentScriptUi(ctx, {\n *       name: \"example-overlay\",\n *       type: \"modal\",\n *       mount(container) {\n *         const app = document.createElement(\"div\");\n *         app.textContent = \"Content Script UI\";\n *         container.append(app);\n *       }\n *     })\n *     ui.mount();\n *   }\n * })\n */\nexport async function createContentScriptUi<T>(\n  ctx: ContentScriptContext,\n  options: ContentScriptUiOptions<T>,\n): Promise<ContentScriptUi<T>> {\n  const css = [options.css ?? ''];\n  if (ctx.options?.cssInjectionMode === 'ui') {\n    css.push(await loadCss());\n  }\n\n  const {\n    isolatedElement: uiContainer,\n    parentElement: shadowHost,\n    shadow,\n  } = await createIsolatedElement({\n    name: options.name,\n    css: {\n      textContent: css.join('\\n').trim(),\n    },\n    mode: 'open',\n  });\n\n  const getAnchor = (): Element | undefined => {\n    if (options.anchor == null) return document.body;\n\n    let resolved =\n      typeof options.anchor === 'function' ? options.anchor() : options.anchor;\n    if (typeof resolved === 'string')\n      return document.querySelector<Element>(resolved) ?? undefined;\n    return resolved ?? undefined;\n  };\n\n  let mounted: T;\n\n  const mount = () => {\n    const anchor = getAnchor();\n    if (anchor == null)\n      throw Error(\n        'Failed to mount content script ui: could not find anchor element',\n      );\n\n    // Mount UI inside shadow root\n    mounted = options.mount(uiContainer);\n\n    // Add shadow root element to DOM\n    switch (options.append) {\n      case undefined:\n      case 'last':\n        anchor.append(shadowHost);\n        break;\n      case 'first':\n        if (anchor.firstChild) {\n          anchor.insertBefore(shadowHost, anchor.firstChild);\n        } else {\n          anchor.append(shadowHost);\n        }\n        break;\n      case 'replace':\n        anchor.replaceWith(shadowHost);\n        break;\n      case 'after':\n        anchor.replaceWith(anchor, shadowHost);\n        break;\n      case 'before':\n        anchor.replaceWith(shadowHost, anchor);\n        break;\n      default:\n        options.append(anchor, shadowHost);\n        break;\n    }\n\n    // Apply types\n    if (options.type !== 'inline') {\n      if (options.zIndex != null)\n        shadowHost.style.zIndex = String(options.zIndex);\n\n      shadowHost.style.overflow = 'visible';\n      shadowHost.style.position = 'relative';\n      shadowHost.style.width = '0';\n      shadowHost.style.height = '0';\n      shadowHost.style.display = 'block';\n\n      const html = shadow.querySelector('html')!;\n      // HTML doesn't exist in tests\n      if (options.type === 'overlay') {\n        html.style.position = 'absolute';\n        if (options.alignment?.startsWith('bottom-')) html.style.bottom = '0';\n        else html.style.top = '0';\n\n        if (options.alignment?.endsWith('-right')) html.style.right = '0';\n        else html.style.left = '0';\n      } else {\n        html.style.position = 'fixed';\n        html.style.top = '0';\n        html.style.bottom = '0';\n        html.style.left = '0';\n        html.style.right = '0';\n      }\n    }\n  };\n\n  const remove = () => {\n    // Detatch shadow root from DOM\n    shadowHost.remove();\n    // Cleanup mounted state\n    options.onRemove?.(mounted);\n    // Remove children from uiContainer\n    while (uiContainer.lastChild)\n      uiContainer.removeChild(uiContainer.lastChild);\n  };\n\n  ctx.onInvalidated(remove);\n\n  return {\n    shadow,\n    shadowHost,\n    uiContainer,\n    mount,\n    remove,\n    mounted: mounted!,\n  };\n}\n\n/**\n * Load the CSS for the current entrypoint.\n */\nasync function loadCss(): Promise<string> {\n  const url = browser.runtime.getURL(`/content-scripts/${__ENTRYPOINT__}.css`);\n  try {\n    const res = await fetch(url);\n    const css = await res.text();\n\n    // Replace :root selectors with :host since we're in a shadow root\n    return css.replaceAll(':root', ':host');\n  } catch (err) {\n    logger.warn(\n      `Failed to load styles @ ${url}. Did you forget to import the stylesheet in your entrypoint?`,\n      err,\n    );\n    return '';\n  }\n}\n\nexport interface ContentScriptUi<T> {\n  /**\n   * The `HTMLElement` hosting the shadow root used to isolate the UI's styles. This is the element\n   * that get's added to the DOM. This element's style is not isolated from the webpage.\n   */\n  shadowHost: HTMLElement;\n  /**\n   * The container element inside the `ShadowRoot` whose styles are isolated. The UI is mounted\n   * inside this `HTMLElement`.\n   */\n  uiContainer: HTMLElement;\n  /**\n   * The shadow root performing the isolation.\n   */\n  shadow: ShadowRoot;\n  /**\n   * Custom data returned from the `options.mount` function.\n   */\n  mounted: T;\n  /**\n   * Function that mounts or remounts the UI on the page.\n   */\n  mount: () => void;\n  /**\n   * Function that removes the UI from the webpage.\n   */\n  remove: () => void;\n}\n\ninterface BaseContentScriptUiOptions<T> {\n  /**\n   * The name of the custom component used to host the ShadowRoot. Must be kebab-case.\n   */\n  name: string;\n  /**\n   * In combination with `anchor`, decide how to add the UI to the DOM.\n   *\n   * - `\"last\"` (default) - Add the UI as the last child of the `anchor` element\n   * - `\"first\"` - Add the UI as the last child of the `anchor` element\n   * - `\"replace\"` - Replace the `anchor` element with the UI.\n   * - `\"before\"` - Add the UI as the sibling before the `anchor` element\n   * - `\"after\"` - Add the UI as the sibling after the `anchor` element\n   * - `(anchor, ui) => void` - Customizable function that let's you add the UI to the DOM\n   */\n  append?: ContentScriptAppendMode | ((anchor: Element, ui: Element) => void);\n  /**\n   * A CSS selector, element, or function that returns one of the two. Along with `append`, the\n   * `anchor` dictates where in the page the UI will be added.\n   */\n  anchor?:\n    | string\n    | Element\n    | null\n    | undefined\n    | (() => string | Element | null | undefined);\n  /**\n   * Callback executed when mounting the UI. This function should create and append the UI to the\n   * `container` element. It is called every time `ui.mount()` is called\n   *\n   * Optionally return a value that can be accessed at `ui.mounted` or in the `onRemove` callback.\n   */\n  mount: (container: Element) => T;\n  /**\n   * Callback called when the UI is removed from the webpage. Use to cleanup your UI, like\n   * unmounting your vue or react apps.\n   */\n  onRemove?: (mounted: T) => void;\n  /**\n   * Custom CSS text to apply to the UI. If your content script imports/generates CSS and you've\n   * set `cssInjectionMode: \"ui\"`, the imported CSS will be included automatically. You do not need\n   * to pass those styles in here. This is for any additional styles not in the imported CSS.\n   *\n   * See <https://wxt.dev/guide/content-scripts.html#ui> for more info.\n   */\n  css?: string;\n}\n\nexport type OverlayContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {\n  type: 'overlay';\n  /**\n   * When using `type: \"overlay\"`, the mounted element is 0px by 0px in size. Alignment specifies\n   * which corner is aligned with that 0x0 pixel space.\n   *\n   * @default \"top-left\"\n   */\n  alignment?: ContentScriptUiOverlayAlignment;\n  /**\n   * The `z-index` used on the `shadowHost`. Set to a positive number to show your UI over website\n   * content.\n   */\n  zIndex?: number;\n};\n\nexport type ModalContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {\n  type: 'modal';\n  /**\n   * The `z-index` used on the `shadowHost`. Set to a positive number to show your UI over website\n   * content.\n   */\n  zIndex?: number;\n};\n\nexport type InlineContentScriptUiOptions<T> = BaseContentScriptUiOptions<T> & {\n  type: 'inline';\n};\n\nexport type ContentScriptUiOverlayAlignment =\n  | 'top-left'\n  | 'top-right'\n  | 'bottom-left'\n  | 'bottom-right';\n\nexport type ContentScriptAppendMode =\n  | 'last'\n  | 'first'\n  | 'replace'\n  | 'before'\n  | 'after';\n\nexport type ContentScriptUiOptions<T> =\n  | OverlayContentScriptUiOptions<T>\n  | ModalContentScriptUiOptions<T>\n  | InlineContentScriptUiOptions<T>;\n", "import originalBrowser, { Browser, Runtime, I18n } from 'webextension-polyfill';\n\nexport interface AugmentedBrowser extends Browser {\n  runtime: WxtRuntime;\n  i18n: WxtI18n;\n}\n\nexport interface WxtRuntime extends Runtime.Static {\n  // Overriden per-project\n}\n\nexport interface WxtI18n extends I18n.Static {\n  // Overriden per-project\n}\n\nexport const browser: AugmentedBrowser = originalBrowser;\n", "function print(method: (...args: any[]) => void, ...args: any[]) {\n  if (import.meta.env.MODE === 'production') return;\n\n  if (typeof args[0] === 'string') {\n    const message = args.shift();\n    method(`[wxt] ${message}`, ...args);\n  } else {\n    method('[wxt]', ...args);\n  }\n}\n\n/**\n * Wrapper around `console` with a \"[wxt]\" prefix\n */\nexport const logger = {\n  debug: (...args: any[]) => print(console.debug, ...args),\n  log: (...args: any[]) => print(console.log, ...args),\n  warn: (...args: any[]) => print(console.warn, ...args),\n  error: (...args: any[]) => print(console.error, ...args),\n};\n", "import { ContentScriptDefinition } from '../../core/types';\nimport { browser } from '../browser';\nimport { logger } from './logger';\n\n/**\n * Extends [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).\n * Used to detect and stop content script code when the script is invalidated.\n *\n * It also provides several utilities like `ctx.setTimeout` and `ctx.setInterval` that should be used in\n * content scripts instead of `window.setTimeout` or `window.setInterval`.\n */\nexport class ContentScriptContext extends AbortController {\n  static SCRIPT_STARTED_MESSAGE_TYPE = 'wxt:content-script-started';\n\n  #isTopFrame = window.self === window.top;\n\n  constructor(\n    private readonly contentScriptName: string,\n    public readonly options?: Omit<ContentScriptDefinition, 'main'>,\n  ) {\n    super();\n\n    if (this.#isTopFrame) {\n      this.#stopOldScripts();\n    }\n    this.setTimeout(() => {\n      // Run on next tick so the listener it adds isn't triggered by stopOldScript\n      this.#listenForNewerScripts();\n    });\n  }\n\n  get isInvalid(): boolean {\n    if (browser.runtime.id == null) {\n      this.notifyInvalidated(); // Sets `signal.aborted` to true\n    }\n    return this.signal.aborted;\n  }\n\n  get isValid(): boolean {\n    return !this.isInvalid;\n  }\n\n  /**\n   * Add a listener that is called when the content script's context is invalidated.\n   *\n   * @returns A function to remove the listener.\n   *\n   * @example\n   * browser.runtime.onMessage.addListener(cb);\n   * const removeInvalidatedListener = ctx.onInvalidated(() => {\n   *   browser.runtime.onMessage.removeListener(cb);\n   * })\n   * // ...\n   * removeInvalidatedListener();\n   */\n  onInvalidated(cb: () => void): () => void {\n    this.signal.addEventListener('abort', cb);\n    return () => this.signal.removeEventListener('abort', cb);\n  }\n\n  /**\n   * Return a promise that never resolves. Useful if you have an async function that shouldn't run\n   * after the context is expired.\n   *\n   * @example\n   * const getValueFromStorage = async () => {\n   *   if (ctx.isInvalid) return ctx.block();\n   *\n   *   // ...\n   * }\n   */\n  block<T>(): Promise<T> {\n    return new Promise(() => {\n      // noop\n    });\n  }\n\n  /**\n   * Wrapper around `window.setInterval` that automatically clears the interval when invalidated.\n   */\n  setInterval(handler: () => void, timeout?: number): number {\n    const id = setInterval(() => {\n      if (this.isValid) handler();\n    }, timeout) as unknown as number;\n    this.onInvalidated(() => clearInterval(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.setTimeout` that automatically clears the interval when invalidated.\n   */\n  setTimeout(handler: () => void, timeout?: number): number {\n    const id = setTimeout(() => {\n      if (this.isValid) handler();\n    }, timeout) as unknown as number;\n    this.onInvalidated(() => clearTimeout(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.requestAnimationFrame` that automatically cancels the request when\n   * invalidated.\n   */\n  requestAnimationFrame(callback: FrameRequestCallback): number {\n    const id = requestAnimationFrame((...args) => {\n      if (this.isValid) callback(...args);\n    });\n\n    this.onInvalidated(() => cancelAnimationFrame(id));\n    return id;\n  }\n\n  /**\n   * Wrapper around `window.requestIdleCallback` that automatically cancels the request when\n   * invalidated.\n   */\n  requestIdleCallback(\n    callback: IdleRequestCallback,\n    options?: IdleRequestOptions,\n  ): number {\n    const id = requestIdleCallback((...args) => {\n      if (!this.signal.aborted) callback(...args);\n    }, options);\n\n    this.onInvalidated(() => cancelIdleCallback(id));\n    return id;\n  }\n\n  /**\n   * Call `target.addEventListener` and remove the event listener when the context is invalidated.\n   *\n   * @example\n   * ctx.addEventListener(window, \"mousemove\", () => {\n   *   // ...\n   * });\n   * ctx.addEventListener(document, \"visibilitychange\", () => {\n   *   // ...\n   * });\n   */\n  addEventListener(\n    target: any,\n    type: string,\n    handler: (event: Event) => void,\n    options?: AddEventListenerOptions,\n  ) {\n    target.addEventListener?.(type, handler, options);\n    this.onInvalidated(\n      () => target.removeEventListener?.(type, handler, options),\n    );\n  }\n\n  /**\n   * Abort the abort controller and execute all `onInvalidated` listeners.\n   */\n  notifyInvalidated() {\n    this.abort('Content script context invalidated');\n    logger.debug(\n      `Content script \"${this.contentScriptName}\" context invalidated`,\n    );\n  }\n\n  #stopOldScripts() {\n    // Use postMessage so it get's sent to all the frames of the page.\n    window.postMessage(\n      {\n        event: ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE,\n        contentScriptName: this.contentScriptName,\n      },\n      '*',\n    );\n  }\n\n  #listenForNewerScripts() {\n    const cb = (event: MessageEvent) => {\n      if (\n        event.data?.type === ContentScriptContext.SCRIPT_STARTED_MESSAGE_TYPE &&\n        event.data?.contentScriptName === this.contentScriptName\n      ) {\n        this.notifyInvalidated();\n      }\n    };\n\n    addEventListener('message', cb);\n    this.onInvalidated(() => removeEventListener('message', cb));\n  }\n}\n"],
  "mappings": ";AAEO,SAAS,oBACd,YACyB;AACzB,SAAO;AACT;;;ACAO,SAAS,iBACd,KAC6B;AAC7B,MAAI,OAAO,QAAQ;AAAY,WAAO,EAAE,MAAM,IAAI;AAClD,SAAO;AACT;;;ACXA,SAAS,6BAA6B;;;ACAtC,OAAO,qBAAiD;AAejD,IAAM,UAA4B;;;ACfzC,SAAS,MAAM,WAAqC,MAAa;AAC/D,MAAI,YAAY,IAAI,SAAS;AAAc;AAE3C,MAAI,OAAO,KAAK,CAAC,MAAM,UAAU;AAC/B,UAAM,UAAU,KAAK,MAAM;AAC3B,WAAO,SAAS,OAAO,IAAI,GAAG,IAAI;AAAA,EACpC,OAAO;AACL,WAAO,SAAS,GAAG,IAAI;AAAA,EACzB;AACF;AAKO,IAAM,SAAS;AAAA,EACpB,OAAO,IAAI,SAAgB,MAAM,QAAQ,OAAO,GAAG,IAAI;AAAA,EACvD,KAAK,IAAI,SAAgB,MAAM,QAAQ,KAAK,GAAG,IAAI;AAAA,EACnD,MAAM,IAAI,SAAgB,MAAM,QAAQ,MAAM,GAAG,IAAI;AAAA,EACrD,OAAO,IAAI,SAAgB,MAAM,QAAQ,OAAO,GAAG,IAAI;AACzD;;;AFcA,eAAsB,sBACpB,KACA,SAC6B;AAC7B,QAAM,MAAM,CAAC,QAAQ,OAAO,EAAE;AAC9B,MAAI,IAAI,SAAS,qBAAqB,MAAM;AAC1C,QAAI,KAAK,MAAM,QAAQ,CAAC;AAAA,EAC1B;AAEA,QAAM;AAAA,IACJ,iBAAiB;AAAA,IACjB,eAAe;AAAA,IACf;AAAA,EACF,IAAI,MAAM,sBAAsB;AAAA,IAC9B,MAAM,QAAQ;AAAA,IACd,KAAK;AAAA,MACH,aAAa,IAAI,KAAK,IAAI,EAAE,KAAK;AAAA,IACnC;AAAA,IACA,MAAM;AAAA,EACR,CAAC;AAED,QAAM,YAAY,MAA2B;AAC3C,QAAI,QAAQ,UAAU;AAAM,aAAO,SAAS;AAE5C,QAAI,WACF,OAAO,QAAQ,WAAW,aAAa,QAAQ,OAAO,IAAI,QAAQ;AACpE,QAAI,OAAO,aAAa;AACtB,aAAO,SAAS,cAAuB,QAAQ,KAAK;AACtD,WAAO,YAAY;AAAA,EACrB;AAEA,MAAI;AAEJ,QAAM,QAAQ,MAAM;AAClB,UAAM,SAAS,UAAU;AACzB,QAAI,UAAU;AACZ,YAAM;AAAA,QACJ;AAAA,MACF;AAGF,cAAU,QAAQ,MAAM,WAAW;AAGnC,YAAQ,QAAQ,QAAQ;AAAA,MACtB,KAAK;AAAA,MACL,KAAK;AACH,eAAO,OAAO,UAAU;AACxB;AAAA,MACF,KAAK;AACH,YAAI,OAAO,YAAY;AACrB,iBAAO,aAAa,YAAY,OAAO,UAAU;AAAA,QACnD,OAAO;AACL,iBAAO,OAAO,UAAU;AAAA,QAC1B;AACA;AAAA,MACF,KAAK;AACH,eAAO,YAAY,UAAU;AAC7B;AAAA,MACF,KAAK;AACH,eAAO,YAAY,QAAQ,UAAU;AACrC;AAAA,MACF,KAAK;AACH,eAAO,YAAY,YAAY,MAAM;AACrC;AAAA,MACF;AACE,gBAAQ,OAAO,QAAQ,UAAU;AACjC;AAAA,IACJ;AAGA,QAAI,QAAQ,SAAS,UAAU;AAC7B,UAAI,QAAQ,UAAU;AACpB,mBAAW,MAAM,SAAS,OAAO,QAAQ,MAAM;AAEjD,iBAAW,MAAM,WAAW;AAC5B,iBAAW,MAAM,WAAW;AAC5B,iBAAW,MAAM,QAAQ;AACzB,iBAAW,MAAM,SAAS;AAC1B,iBAAW,MAAM,UAAU;AAE3B,YAAM,OAAO,OAAO,cAAc,MAAM;AAExC,UAAI,QAAQ,SAAS,WAAW;AAC9B,aAAK,MAAM,WAAW;AACtB,YAAI,QAAQ,WAAW,WAAW,SAAS;AAAG,eAAK,MAAM,SAAS;AAAA;AAC7D,eAAK,MAAM,MAAM;AAEtB,YAAI,QAAQ,WAAW,SAAS,QAAQ;AAAG,eAAK,MAAM,QAAQ;AAAA;AACzD,eAAK,MAAM,OAAO;AAAA,MACzB,OAAO;AACL,aAAK,MAAM,WAAW;AACtB,aAAK,MAAM,MAAM;AACjB,aAAK,MAAM,SAAS;AACpB,aAAK,MAAM,OAAO;AAClB,aAAK,MAAM,QAAQ;AAAA,MACrB;AAAA,IACF;AAAA,EACF;AAEA,QAAM,SAAS,MAAM;AAEnB,eAAW,OAAO;AAElB,YAAQ,WAAW,OAAO;AAE1B,WAAO,YAAY;AACjB,kBAAY,YAAY,YAAY,SAAS;AAAA,EACjD;AAEA,MAAI,cAAc,MAAM;AAExB,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACF;AAKA,eAAe,UAA2B;AACxC,QAAM,MAAM,QAAQ,QAAQ,OAAO,oBAAoB,cAAc,MAAM;AAC3E,MAAI;AACF,UAAM,MAAM,MAAM,MAAM,GAAG;AAC3B,UAAM,MAAM,MAAM,IAAI,KAAK;AAG3B,WAAO,IAAI,WAAW,SAAS,OAAO;AAAA,EACxC,SAAS,KAAK;AACZ,WAAO;AAAA,MACL,2BAA2B,GAAG;AAAA,MAC9B;AAAA,IACF;AACA,WAAO;AAAA,EACT;AACF;;;AGlKO,IAAM,uBAAN,MAAM,8BAA6B,gBAAgB;AAAA,EAKxD,YACmB,mBACD,SAChB;AACA,UAAM;AAHW;AACD;AAIhB,QAAI,KAAK,aAAa;AACpB,WAAK,gBAAgB;AAAA,IACvB;AACA,SAAK,WAAW,MAAM;AAEpB,WAAK,uBAAuB;AAAA,IAC9B,CAAC;AAAA,EACH;AAAA,EAjBA,OAAO,8BAA8B;AAAA,EAErC,cAAc,OAAO,SAAS,OAAO;AAAA,EAiBrC,IAAI,YAAqB;AACvB,QAAI,QAAQ,QAAQ,MAAM,MAAM;AAC9B,WAAK,kBAAkB;AAAA,IACzB;AACA,WAAO,KAAK,OAAO;AAAA,EACrB;AAAA,EAEA,IAAI,UAAmB;AACrB,WAAO,CAAC,KAAK;AAAA,EACf;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,cAAc,IAA4B;AACxC,SAAK,OAAO,iBAAiB,SAAS,EAAE;AACxC,WAAO,MAAM,KAAK,OAAO,oBAAoB,SAAS,EAAE;AAAA,EAC1D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAaA,QAAuB;AACrB,WAAO,IAAI,QAAQ,MAAM;AAAA,IAEzB,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,YAAY,SAAqB,SAA0B;AACzD,UAAM,KAAK,YAAY,MAAM;AAC3B,UAAI,KAAK;AAAS,gBAAQ;AAAA,IAC5B,GAAG,OAAO;AACV,SAAK,cAAc,MAAM,cAAc,EAAE,CAAC;AAC1C,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,WAAW,SAAqB,SAA0B;AACxD,UAAM,KAAK,WAAW,MAAM;AAC1B,UAAI,KAAK;AAAS,gBAAQ;AAAA,IAC5B,GAAG,OAAO;AACV,SAAK,cAAc,MAAM,aAAa,EAAE,CAAC;AACzC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,sBAAsB,UAAwC;AAC5D,UAAM,KAAK,sBAAsB,IAAI,SAAS;AAC5C,UAAI,KAAK;AAAS,iBAAS,GAAG,IAAI;AAAA,IACpC,CAAC;AAED,SAAK,cAAc,MAAM,qBAAqB,EAAE,CAAC;AACjD,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,oBACE,UACA,SACQ;AACR,UAAM,KAAK,oBAAoB,IAAI,SAAS;AAC1C,UAAI,CAAC,KAAK,OAAO;AAAS,iBAAS,GAAG,IAAI;AAAA,IAC5C,GAAG,OAAO;AAEV,SAAK,cAAc,MAAM,mBAAmB,EAAE,CAAC;AAC/C,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAaA,iBACE,QACA,MACA,SACA,SACA;AACA,WAAO,mBAAmB,MAAM,SAAS,OAAO;AAChD,SAAK;AAAA,MACH,MAAM,OAAO,sBAAsB,MAAM,SAAS,OAAO;AAAA,IAC3D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,oBAAoB;AAClB,SAAK,MAAM,oCAAoC;AAC/C,WAAO;AAAA,MACL,mBAAmB,KAAK,iBAAiB;AAAA,IAC3C;AAAA,EACF;AAAA,EAEA,kBAAkB;AAEhB,WAAO;AAAA,MACL;AAAA,QACE,OAAO,sBAAqB;AAAA,QAC5B,mBAAmB,KAAK;AAAA,MAC1B;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAAA,EAEA,yBAAyB;AACvB,UAAM,KAAK,CAAC,UAAwB;AAClC,UACE,MAAM,MAAM,SAAS,sBAAqB,+BAC1C,MAAM,MAAM,sBAAsB,KAAK,mBACvC;AACA,aAAK,kBAAkB;AAAA,MACzB;AAAA,IACF;AAEA,qBAAiB,WAAW,EAAE;AAC9B,SAAK,cAAc,MAAM,oBAAoB,WAAW,EAAE,CAAC;AAAA,EAC7D;AACF;",
  "names": []
}
