Content Scripts
When creating content script entrypoints, they are automatically included in the manifest.json
along with any CSS files they import.
Filenames
Input Pattern | Output Path | |
---|---|---|
entrypoints/content.[jt]sx? | /content-scripts/content.js | |
entrypoints/content/index.[jt]sx? | /content-scripts/content.js | |
entrypoints/<name>.content.[jt]sx? | /content-scripts/<name>.js | |
entrypoints/<name>.content/index.[jt]sx? | /content-scripts/<name>.js |
Definition
export default defineContentScript({
// Set manifest options
matches: string[],
excludeMatches: undefined | [],
includeGlobs: undefined | [],
excludeGlobs: undefined | [],
allFrames: undefined | [],
runAt: undefined | 'document_start' | 'document_end' | 'document_idle',
matchAboutBlank: undefined | true | false,
matchOriginAsFallback: undefined | true | false,
world: undefined | 'ISOLATED' | 'MAIN',
// Set include/exclude if the background should be removed from some builds
include: undefined | string[],
exclude: undefined | string[],
// Configure how CSS is injected onto the page
cssInjectionMode: undefined | "manifest" | "manual" | "ui",
// Configure how/when content script will be registered
registration: undefined | "manifest" | "runtime",
main(ctx) {
// Executed when content script is loaded
},
});
All manifest options default to
undefined
.
When defining multiple content scripts, content script entrypoints that have the same set of options will be merged into a single content_script
item in the manifest.
Context
Old content scripts are not automatically stopped when an extension updates and reloads. Often, this leads to "Invalidated context" errors in production when a content script from an old version of your extension tries to use a web extension API (ie, the browser
or chrome
globals).
WXT provides a utility for handling this process: ContentScriptContext
. An instance of this class is provided to you automatically inside the main
function of your content script.
When your extension updates or is uninstalled, the context will become invalidated, and will trigger any ctx.onInvalidated
listeners you add:
export default defineContentScript({
// ...
main(ctx: ContentScriptContext) {
// Add custom listeners for stopping work
ctx.onInvalidated(() => {
// ...
});
// Timeout utilities that are automatically cleared when invalidated
ctx.setTimeout(() => {
// ...
}, 5e3);
ctx.setInterval(() => {
// ...
}, 60e3);
// Or add event listeners that get removed when invalidated
ctx.addEventListener(document, 'visibilitychange', (event) => {
// ...
});
// You can also stop fetch requests
fetch('...url', { signal: ctx.signal });
},
});
The class extends AbortController
and provides other utilities for stopping a content script's logic once it becomes invalidated.
WARNING
When working with content scripts, you should always use the ctx
object to stop any async or future work.
This prevents old content scripts from interfering with new content scripts, and prevents error messages from the console in production.
If you're using a framework like React, Vue, Svelte, etc., make sure you're unmounting your UI properly in the onRemove
option of createShadowRootUi
.
CSS
To include CSS with your content script, import the CSS file at the top of your entrypoint.
<srcDir>/
└─ entrypoints/
└─ overlay.content/
├─ index.ts
└─ style.css
// entrypoints/overlay.content/index.ts
import './style.css';
export default defineContentScript({
matches: ['*://google.com/*', '*://duckduckgo.com/*'],
main(ctx) {
// ...
},
});
Any styles imported in your content script will be added to that content script's css
array in your manifest.json
:
// .output/chrome-mv3/manifest.json
{
"content_scripts": [
{
"matches": ["*://google.com/*", "*://duckduckgo.com/*"],
"js": ["content-scripts/overlay.js"],
"css": ["content-scripts/overlay.css"]
}
]
}
To disable this behavior, set cssInjectionMode
to "manual"
or "ui"
.
export default defineContentScript({
matches: ['*://google.com/*', '*://duckduckgo.com/*'],
cssInjectionMode: 'manual',
main(ctx) {
// ...
},
});
See Content Script UI for more info on creating UIs and including CSS in content scripts.