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/README.md +4 -4
- package/dist/cli.cjs +73 -29
- package/dist/client.d.ts +132 -4
- package/dist/client.js +131 -12
- package/dist/index.cjs +72 -28
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +4 -3
- package/dist/index.d.ts +4 -3
- package/dist/index.js +72 -28
- package/dist/index.js.map +1 -1
- package/dist/virtual-modules/content-script-entrypoint.js +12 -7
- package/dist/virtual-modules/content-script-entrypoint.js.map +1 -1
- package/package.json +4 -2
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
|
-
|
|
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
|
|
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/
|
|
14
|
-
|
|
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
|
-
|
|
170
|
-
|
|
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
|
|
2489
|
-
|
|
2490
|
-
|
|
2491
|
-
|
|
2492
|
-
|
|
2493
|
-
|
|
2494
|
-
|
|
2495
|
-
|
|
2496
|
-
|
|
2497
|
-
|
|
2498
|
-
|
|
2499
|
-
|
|
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,
|
|
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,
|
|
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
|
|
2775
|
-
|
|
2776
|
-
|
|
2777
|
-
if (
|
|
2778
|
-
css.push(
|
|
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
|
|
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 ${
|
|
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
|
|
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
|
|
4327
|
-
|
|
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
|
|
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.
|
|
4475
|
+
var version2 = "0.7.1-alpha1";
|
|
4432
4476
|
|
|
4433
4477
|
// src/core/utils/defineConfig.ts
|
|
4434
4478
|
function defineConfig(config) {
|