wxt 0.7.0 → 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
  /**
@@ -130,7 +131,7 @@ interface ContentScriptDefinition extends ExcludableEntrypoint {
130
131
  *
131
132
  * @default "manifest"
132
133
  */
133
- cssInjectionMode?: 'manifest' | 'manual';
134
+ cssInjectionMode?: 'manifest' | 'manual' | 'ui';
134
135
  /**
135
136
  * Main function executed when the content script is loaded.
136
137
  */
@@ -162,6 +163,133 @@ declare function defineContentScript(definition: ContentScriptDefinition): Conte
162
163
  declare function defineBackground(main: () => void): BackgroundScriptDefintition;
163
164
  declare function defineBackground(definition: BackgroundScriptDefintition): BackgroundScriptDefintition;
164
165
 
165
- 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>;
166
294
 
167
- 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": "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",
  "names": []
}

package/dist/index.cjs CHANGED
@@ -2471,6 +2471,7 @@ async function getPackageJson(config) {
2471
2471
 
2472
2472
  // src/core/utils/manifest.ts
2473
2473
  var import_immer = require("immer");
2474
+ var vite4 = __toESM(require("vite"), 1);
2474
2475
  async function writeManifest(manifest, output, config) {
2475
2476
  const str = config.mode === "production" ? JSON.stringify(manifest) : JSON.stringify(manifest, null, 2);
2476
2477
  await import_fs_extra12.default.ensureDir(config.outDir);
@@ -2485,18 +2486,20 @@ async function writeManifest(manifest, output, config) {
2485
2486
  }
2486
2487
  async function generateMainfest(entrypoints, buildOutput, config) {
2487
2488
  const pkg = await getPackageJson(config);
2488
- const manifest = Object.assign(
2489
- {
2490
- manifest_version: config.manifestVersion,
2491
- name: pkg?.name,
2492
- description: pkg?.description,
2493
- version: pkg?.version && simplifyVersion(pkg.version),
2494
- // Only add the version name to chromium and if the user hasn't specified a custom version.
2495
- version_name: config.browser !== "firefox" && !config.manifest.version ? pkg?.version : void 0,
2496
- short_name: pkg?.shortName,
2497
- icons: discoverIcons(buildOutput)
2498
- },
2499
- config.manifest
2489
+ const baseManifest = {
2490
+ manifest_version: config.manifestVersion,
2491
+ name: pkg?.name,
2492
+ description: pkg?.description,
2493
+ version: pkg?.version && simplifyVersion(pkg.version),
2494
+ // Only add the version name to chromium and if the user hasn't specified a custom version.
2495
+ version_name: config.browser !== "firefox" && !config.manifest.version ? pkg?.version : void 0,
2496
+ short_name: pkg?.shortName,
2497
+ icons: discoverIcons(buildOutput)
2498
+ };
2499
+ const userManifest = config.manifest;
2500
+ const manifest = vite4.mergeConfig(
2501
+ baseManifest,
2502
+ userManifest
2500
2503
  );
2501
2504
  addEntrypoints(manifest, entrypoints, buildOutput, config);
2502
2505
  if (config.command === "serve")
@@ -2666,6 +2669,7 @@ function addEntrypoints(manifest, entrypoints, buildOutput, config) {
2666
2669
  }
2667
2670
  }
2668
2671
  if (contentScripts?.length) {
2672
+ const cssMap = getContentScriptsCssMap(buildOutput, contentScripts);
2669
2673
  if (config.command === "serve" && config.manifestVersion === 3) {
2670
2674
  const hostPermissions = new Set(manifest.host_permissions ?? []);
2671
2675
  contentScripts.forEach((script) => {
@@ -2690,7 +2694,7 @@ function addEntrypoints(manifest, entrypoints, buildOutput, config) {
2690
2694
  ...mapWxtOptionsToContentScript(scripts[0].options),
2691
2695
  // TOOD: Sorting css and js arrays here so we get consistent test results... but we
2692
2696
  // shouldn't have to. Where is the inconsistency coming from?
2693
- css: getContentScriptCssFiles(scripts, buildOutput)?.sort(),
2697
+ css: getContentScriptCssFiles(scripts, cssMap)?.sort(),
2694
2698
  js: scripts.map(
2695
2699
  (entry) => getEntrypointBundlePath(entry, config.outDir, ".js")
2696
2700
  ).sort()
@@ -2701,6 +2705,15 @@ function addEntrypoints(manifest, entrypoints, buildOutput, config) {
2701
2705
  manifest.content_scripts.push(...newContentScripts);
2702
2706
  }
2703
2707
  }
2708
+ const contentScriptCssResources = getContentScriptCssWebAccessibleResources(
2709
+ config,
2710
+ contentScripts,
2711
+ cssMap
2712
+ );
2713
+ if (contentScriptCssResources.length > 0) {
2714
+ manifest.web_accessible_resources ??= [];
2715
+ manifest.web_accessible_resources.push(...contentScriptCssResources);
2716
+ }
2704
2717
  }
2705
2718
  }
2706
2719
  function discoverIcons(buildOutput) {
@@ -2765,22 +2778,52 @@ function addDevModePermissions(manifest, config) {
2765
2778
  if (config.manifestVersion === 3)
2766
2779
  addPermission(manifest, "scripting");
2767
2780
  }
2768
- function getContentScriptCssFiles(contentScripts, buildOutput) {
2781
+ function getContentScriptCssFiles(contentScripts, contentScriptCssMap) {
2769
2782
  const css = [];
2770
- const allChunks = buildOutput.steps.flatMap((step) => step.chunks);
2771
2783
  contentScripts.forEach((script) => {
2772
- if (script.options.cssInjectionMode === "manual")
2784
+ if (script.options.cssInjectionMode === "manual" || script.options.cssInjectionMode === "ui")
2773
2785
  return;
2774
- const relatedCss = allChunks.find(
2775
- (chunk) => chunk.fileName === `content-scripts/${script.name}.css`
2776
- );
2777
- if (relatedCss)
2778
- css.push(relatedCss.fileName);
2786
+ const cssFile = contentScriptCssMap[script.name];
2787
+ if (cssFile == null)
2788
+ return;
2789
+ if (cssFile)
2790
+ css.push(cssFile);
2779
2791
  });
2780
2792
  if (css.length > 0)
2781
2793
  return css;
2782
2794
  return void 0;
2783
2795
  }
2796
+ function getContentScriptCssWebAccessibleResources(config, contentScripts, contentScriptCssMap) {
2797
+ const resources = [];
2798
+ contentScripts.forEach((script) => {
2799
+ if (script.options.cssInjectionMode !== "ui")
2800
+ return;
2801
+ const cssFile = contentScriptCssMap[script.name];
2802
+ if (cssFile == null)
2803
+ return;
2804
+ if (config.manifestVersion === 2) {
2805
+ resources.push(cssFile);
2806
+ } else {
2807
+ resources.push({
2808
+ resources: [cssFile],
2809
+ matches: script.options.matches
2810
+ });
2811
+ }
2812
+ });
2813
+ return resources;
2814
+ }
2815
+ function getContentScriptsCssMap(buildOutput, scripts) {
2816
+ const map = {};
2817
+ const allChunks = buildOutput.steps.flatMap((step) => step.chunks);
2818
+ scripts.forEach((script) => {
2819
+ const relatedCss = allChunks.find(
2820
+ (chunk) => chunk.fileName === `content-scripts/${script.name}.css`
2821
+ );
2822
+ if (relatedCss != null)
2823
+ map[script.name] = relatedCss.fileName;
2824
+ });
2825
+ return map;
2826
+ }
2784
2827
  function addPermission(manifest, permission) {
2785
2828
  manifest.permissions ??= [];
2786
2829
  if (manifest.permissions.includes(permission))
@@ -2796,7 +2839,7 @@ function addHostPermission(manifest, hostPermission) {
2796
2839
 
2797
2840
  // src/core/build.ts
2798
2841
  var import_picocolors3 = __toESM(require("picocolors"), 1);
2799
- var vite4 = __toESM(require("vite"), 1);
2842
+ var vite5 = __toESM(require("vite"), 1);
2800
2843
  var import_fs_extra14 = __toESM(require("fs-extra"), 1);
2801
2844
 
2802
2845
  // src/core/utils/groupEntrypoints.ts
@@ -4100,7 +4143,7 @@ async function buildInternal(config) {
4100
4143
  const target = `${config.browser}-mv${config.manifestVersion}`;
4101
4144
  config.logger.info(
4102
4145
  `${verb} ${import_picocolors3.default.cyan(target)} for ${import_picocolors3.default.cyan(config.mode)} with ${import_picocolors3.default.green(
4103
- `Vite ${vite4.version}`
4146
+ `Vite ${vite5.version}`
4104
4147
  )}`
4105
4148
  );
4106
4149
  const startTime = Date.now();
@@ -4178,7 +4221,7 @@ async function combineAnalysisStats(config) {
4178
4221
  }
4179
4222
 
4180
4223
  // src/core/server.ts
4181
- var vite5 = __toESM(require("vite"), 1);
4224
+ var vite6 = __toESM(require("vite"), 1);
4182
4225
 
4183
4226
  // src/core/runners/wsl.ts
4184
4227
  var import_node_path10 = require("path");
@@ -4323,8 +4366,8 @@ async function getServerInfo() {
4323
4366
  }
4324
4367
  async function setupServer(serverInfo, config) {
4325
4368
  const runner = await createExtensionRunner(config);
4326
- const viteServer = await vite5.createServer(
4327
- vite5.mergeConfig(serverInfo, await config.vite(config.env))
4369
+ const viteServer = await vite6.createServer(
4370
+ vite6.mergeConfig(serverInfo, await config.vite(config.env))
4328
4371
  );
4329
4372
  const start = async () => {
4330
4373
  await viteServer.listen(server.port);
@@ -4369,7 +4412,8 @@ function reloadContentScripts(steps, config, server) {
4369
4412
  if (Array.isArray(entry) || entry.type !== "content-script")
4370
4413
  return;
4371
4414
  const js = [getEntrypointBundlePath(entry, config.outDir, ".js")];
4372
- const css = getContentScriptCssFiles([entry], server.currentOutput);
4415
+ const cssMap = getContentScriptsCssMap(server.currentOutput, [entry]);
4416
+ const css = getContentScriptCssFiles([entry], cssMap);
4373
4417
  server.reloadContentScript({
4374
4418
  allFrames: entry.options.allFrames,
4375
4419
  excludeMatches: entry.options.excludeMatches,
@@ -4428,7 +4472,7 @@ async function clean(root = process.cwd()) {
4428
4472
  }
4429
4473
 
4430
4474
  // package.json
4431
- var version2 = "0.7.0";
4475
+ var version2 = "0.7.1-alpha1";
4432
4476
 
4433
4477
  // src/core/utils/defineConfig.ts
4434
4478
  function defineConfig(config) {