wxt 0.8.1 → 0.8.2

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
@@ -1,18 +1,20 @@
1
1
  import { Manifest } from 'webextension-polyfill';
2
2
 
3
3
  /**
4
- * Extends [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).
4
+ * Implements [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).
5
5
  * Used to detect and stop content script code when the script is invalidated.
6
6
  *
7
7
  * It also provides several utilities like `ctx.setTimeout` and `ctx.setInterval` that should be used in
8
8
  * content scripts instead of `window.setTimeout` or `window.setInterval`.
9
9
  */
10
- declare class ContentScriptContext extends AbortController {
10
+ declare class ContentScriptContext implements AbortController {
11
11
  #private;
12
12
  private readonly contentScriptName;
13
13
  readonly options?: Omit<ContentScriptDefinition, "main"> | undefined;
14
14
  private static SCRIPT_STARTED_MESSAGE_TYPE;
15
15
  constructor(contentScriptName: string, options?: Omit<ContentScriptDefinition, "main"> | undefined);
16
+ get signal(): AbortSignal;
17
+ abort(reason?: any): void;
16
18
  get isInvalid(): boolean;
17
19
  get isValid(): boolean;
18
20
  /**
package/dist/client.js CHANGED
@@ -153,11 +153,11 @@ async function loadCss() {
153
153
  }
154
154
 
155
155
  // src/client/utils/ContentScriptContext.ts
156
- var ContentScriptContext = class _ContentScriptContext extends AbortController {
156
+ var ContentScriptContext = class _ContentScriptContext {
157
157
  constructor(contentScriptName, options) {
158
- super();
159
158
  this.contentScriptName = contentScriptName;
160
159
  this.options = options;
160
+ this.#abortController = new AbortController();
161
161
  if (this.#isTopFrame) {
162
162
  this.#stopOldScripts();
163
163
  }
@@ -167,6 +167,13 @@ var ContentScriptContext = class _ContentScriptContext extends AbortController {
167
167
  }
168
168
  static SCRIPT_STARTED_MESSAGE_TYPE = "wxt:content-script-started";
169
169
  #isTopFrame = window.self === window.top;
170
+ #abortController;
171
+ get signal() {
172
+ return this.#abortController.signal;
173
+ }
174
+ abort(reason) {
175
+ return this.#abortController.abort(reason);
176
+ }
170
177
  get isInvalid() {
171
178
  if (browser.runtime.id == null) {
172
179
  this.notifyInvalidated();
@@ -306,4 +313,4 @@ export {
306
313
  defineBackground,
307
314
  defineContentScript
308
315
  };
309
- //# 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 { BackgroundDefinition } from '..';\n\nexport function defineBackground(main: () => void): BackgroundDefinition;\nexport function defineBackground(\n  definition: BackgroundDefinition,\n): BackgroundDefinition;\nexport function defineBackground(\n  arg: (() => void) | BackgroundDefinition,\n): BackgroundDefinition {\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/entrypoints/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<TApp>(\n  ctx: ContentScriptContext,\n  options: ContentScriptUiOptions<TApp>,\n): Promise<ContentScriptUi<TApp>> {\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: TApp;\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<TApp> {\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: TApp;\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<TApp> {\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) => TApp;\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: TApp) => 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/entrypoints/content-scripts.html#ui for more info.\n   */\n  css?: string;\n}\n\nexport type OverlayContentScriptUiOptions<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  | OverlayContentScriptUiOptions<TApp>\n  | ModalContentScriptUiOptions<TApp>\n  | InlineContentScriptUiOptions<TApp>;\n", "/**\n * @module wxt/browser\n */\nimport 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  private 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   * @internal\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": "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",
  "names": []
}

316
+ //# 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 { BackgroundDefinition } from '..';\n\nexport function defineBackground(main: () => void): BackgroundDefinition;\nexport function defineBackground(\n  definition: BackgroundDefinition,\n): BackgroundDefinition;\nexport function defineBackground(\n  arg: (() => void) | BackgroundDefinition,\n): BackgroundDefinition {\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/entrypoints/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<TApp>(\n  ctx: ContentScriptContext,\n  options: ContentScriptUiOptions<TApp>,\n): Promise<ContentScriptUi<TApp>> {\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: TApp;\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<TApp> {\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: TApp;\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<TApp> {\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) => TApp;\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: TApp) => 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/entrypoints/content-scripts.html#ui for more info.\n   */\n  css?: string;\n}\n\nexport type OverlayContentScriptUiOptions<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  BaseContentScriptUiOptions<TApp> & {\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<TApp> =\n  | OverlayContentScriptUiOptions<TApp>\n  | ModalContentScriptUiOptions<TApp>\n  | InlineContentScriptUiOptions<TApp>;\n", "/**\n * @module wxt/browser\n */\nimport 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 * Implements [`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 implements AbortController {\n  private static SCRIPT_STARTED_MESSAGE_TYPE = 'wxt:content-script-started';\n\n  #isTopFrame = window.self === window.top;\n  #abortController: AbortController;\n\n  constructor(\n    private readonly contentScriptName: string,\n    public readonly options?: Omit<ContentScriptDefinition, 'main'>,\n  ) {\n    this.#abortController = new AbortController();\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 signal() {\n    return this.#abortController.signal;\n  }\n\n  abort(reason?: any): void {\n    return this.#abortController.abort(reason);\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   * @internal\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": "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",
  "names": []
}

package/dist/index.cjs CHANGED
@@ -1971,17 +1971,15 @@ async function findEntrypoints(config) {
1971
1971
  });
1972
1972
  relativePaths.sort();
1973
1973
  const pathGlobs = Object.keys(PATH_GLOB_TO_TYPE_MAP);
1974
- const existingNames = {};
1975
- const entrypoints = [];
1976
1974
  let hasBackground = false;
1977
- await Promise.all(
1975
+ const possibleEntrypoints = await Promise.all(
1978
1976
  relativePaths.map(async (relativePath) => {
1979
1977
  const path9 = (0, import_path8.resolve)(config.entrypointsDir, relativePath);
1980
1978
  const matchingGlob = pathGlobs.find(
1981
1979
  (glob5) => (0, import_minimatch.minimatch)(relativePath, glob5)
1982
1980
  );
1983
1981
  if (matchingGlob == null) {
1984
- return config.logger.warn(
1982
+ config.logger.warn(
1985
1983
  `${relativePath} does not match any known entrypoint. Known entrypoints:
1986
1984
  ${JSON.stringify(
1987
1985
  PATH_GLOB_TO_TYPE_MAP,
@@ -1989,33 +1987,27 @@ ${JSON.stringify(
1989
1987
  2
1990
1988
  )}`
1991
1989
  );
1990
+ return;
1992
1991
  }
1993
1992
  const type = PATH_GLOB_TO_TYPE_MAP[matchingGlob];
1994
1993
  if (type === "ignored")
1995
1994
  return;
1996
- let entrypoint;
1997
1995
  switch (type) {
1998
1996
  case "popup":
1999
- entrypoint = await getPopupEntrypoint(config, path9);
2000
- break;
1997
+ return await getPopupEntrypoint(config, path9);
2001
1998
  case "options":
2002
- entrypoint = await getOptionsEntrypoint(config, path9);
2003
- break;
1999
+ return await getOptionsEntrypoint(config, path9);
2004
2000
  case "background":
2005
- entrypoint = await getBackgroundEntrypoint(config, path9);
2006
2001
  hasBackground = true;
2007
- break;
2002
+ return await getBackgroundEntrypoint(config, path9);
2008
2003
  case "content-script":
2009
- entrypoint = await getContentScriptEntrypoint(config, path9);
2010
- break;
2004
+ return await getContentScriptEntrypoint(config, path9);
2011
2005
  case "unlisted-page":
2012
- entrypoint = await getUnlistedPageEntrypoint(config, path9);
2013
- break;
2006
+ return await getUnlistedPageEntrypoint(config, path9);
2014
2007
  case "unlisted-script":
2015
- entrypoint = await getUnlistedScriptEntrypoint(config, path9);
2016
- break;
2008
+ return await getUnlistedScriptEntrypoint(config, path9);
2017
2009
  case "content-script-style":
2018
- entrypoint = {
2010
+ return {
2019
2011
  type,
2020
2012
  name: getEntrypointName(config.entrypointsDir, path9),
2021
2013
  inputPath: path9,
@@ -2025,9 +2017,8 @@ ${JSON.stringify(
2025
2017
  exclude: void 0
2026
2018
  }
2027
2019
  };
2028
- break;
2029
2020
  default:
2030
- entrypoint = {
2021
+ return {
2031
2022
  type,
2032
2023
  name: getEntrypointName(config.entrypointsDir, path9),
2033
2024
  inputPath: path9,
@@ -2038,19 +2029,24 @@ ${JSON.stringify(
2038
2029
  }
2039
2030
  };
2040
2031
  }
2041
- const withSameName = existingNames[entrypoint.name];
2042
- if (withSameName) {
2043
- throw Error(
2044
- `Multiple entrypoints with the name "${entrypoint.name}" detected, but only one is allowed: ${[
2045
- (0, import_path8.relative)(config.root, withSameName.inputPath),
2046
- (0, import_path8.relative)(config.root, entrypoint.inputPath)
2047
- ].join(", ")}`
2048
- );
2049
- }
2050
- entrypoints.push(entrypoint);
2051
- existingNames[entrypoint.name] = entrypoint;
2052
2032
  })
2053
2033
  );
2034
+ const entrypoints = possibleEntrypoints.filter(
2035
+ (entry) => !!entry
2036
+ );
2037
+ const existingNames = {};
2038
+ entrypoints.forEach((entrypoint) => {
2039
+ const withSameName = existingNames[entrypoint.name];
2040
+ if (withSameName) {
2041
+ throw Error(
2042
+ `Multiple entrypoints with the name "${entrypoint.name}" detected, but only one is allowed: ${[
2043
+ (0, import_path8.relative)(config.root, withSameName.inputPath),
2044
+ (0, import_path8.relative)(config.root, entrypoint.inputPath)
2045
+ ].join(", ")}`
2046
+ );
2047
+ }
2048
+ existingNames[entrypoint.name] = entrypoint;
2049
+ });
2054
2050
  if (config.command === "serve" && !hasBackground) {
2055
2051
  entrypoints.push(
2056
2052
  await getBackgroundEntrypoint(config, VIRTUAL_NOOP_BACKGROUND_MODULE_ID)
@@ -2833,12 +2829,10 @@ function addEntrypoints(manifest, entrypoints, buildOutput, config) {
2833
2829
  const newContentScripts = Array.from(hashToEntrypointsMap.entries()).map(
2834
2830
  ([, scripts]) => ({
2835
2831
  ...mapWxtOptionsToContentScript(scripts[0].options, config),
2836
- // TOOD: Sorting css and js arrays here so we get consistent test results... but we
2837
- // shouldn't have to. Where is the inconsistency coming from?
2838
- css: getContentScriptCssFiles(scripts, cssMap)?.sort(),
2832
+ css: getContentScriptCssFiles(scripts, cssMap),
2839
2833
  js: scripts.map(
2840
2834
  (entry) => getEntrypointBundlePath(entry, config.outDir, ".js")
2841
- ).sort()
2835
+ )
2842
2836
  })
2843
2837
  );
2844
2838
  if (newContentScripts.length >= 0) {
@@ -4619,7 +4613,7 @@ async function clean(root = process.cwd()) {
4619
4613
  }
4620
4614
 
4621
4615
  // package.json
4622
- var version2 = "0.8.1";
4616
+ var version2 = "0.8.2";
4623
4617
 
4624
4618
  // src/core/utils/defineConfig.ts
4625
4619
  function defineConfig(config) {