diff --git a/docs/guide/essentials/extension-apis.md b/docs/guide/essentials/extension-apis.md index 529558d83..d014f98fd 100644 --- a/docs/guide/essentials/extension-apis.md +++ b/docs/guide/essentials/extension-apis.md @@ -4,52 +4,29 @@ Different browsers provide different global variables for accessing the extension APIs (chrome provides `chrome`, firefox provides `browser`, etc). -WXT simplifies this - always use `browser`: +WXT merges these two into a unified API accessed through the `browser` variable. ```ts +import { browser } from 'wxt/browser'; + browser.action.onClicked.addListener(() => { // ... }); ``` -Other than that, refer to Chrome and Mozilla's documentation for how to use specific APIs. Everything a normal extension can do, WXT can do as well, just via `browser` instead of `chrome`. - -## Webextension Polyfill - -> Since `v0.1.0` - -By default, WXT uses the [`webextension-polyfill` by Mozilla](https://www.npmjs.com/package/webextension-polyfill) to make the extension API consistent between browsers. - -To access types, you should import the relevant namespace from `wxt/browser`: - -```ts -import { Runtime } from 'wxt/browser'; - -function handleMessage(message: any, sender: Runtime.Sender) { - // ... -} -``` - -### Disabling the polyfill - -> Since `v0.19.0` - -After the release of MV3 and Chrome's official deprecation of MV2 in June 2024, the polyfill isn't really doing anything useful anymore. +:::tip +With auto-imports enabled, you don't even need to import this variable from `wxt/browser`! +::: -You can disable it with a single line: +The `browser` variable WXT provides is a simple export of the `browser` or `chrome` globals provided by the browser at runtime: -```ts -// wxt.config.ts -export default defineConfig({ - extensionApi: 'chrome', -}); -``` +<<< @/../packages/wxt/src/browser.ts#snippet -This will change `wxt/browser` to simply export the `browser` or `chrome` globals based on browser at runtime: +This means you can use the promise-style API for both MV2 and MV3, and it will work across all browsers (Chromium, Firefox, Safari, etc). -<<< @/../packages/wxt/src/browser/chrome.ts#snippet +## Accessing Types -Accessing types is a little different with the polyfill disabled. They do not need to be imported; they're available on the `browser` object itself: +All types can be accessed via WXT's `browser` object: ```ts function handleMessage(message: any, sender: browser.runtime.Sender) { @@ -59,7 +36,7 @@ function handleMessage(message: any, sender: browser.runtime.Sender) { ## Feature Detection -Depending on the manifest version and browser, some APIs are not available at runtime. If an API is not available, it will be `undefined`. +Depending on the manifest version, browser, and permissions, some APIs are not available at runtime. If an API is not available, it will be `undefined`. :::warning Types will not help you here. The types WXT provides for `browser` assume all APIs exist. You are responsible for knowing whether an API is available or not. diff --git a/packages/wxt-demo/src/entrypoints/__tests__/background.test.ts b/packages/wxt-demo/src/entrypoints/__tests__/background.test.ts index 62f79bed1..0383bac5a 100644 --- a/packages/wxt-demo/src/entrypoints/__tests__/background.test.ts +++ b/packages/wxt-demo/src/entrypoints/__tests__/background.test.ts @@ -11,7 +11,7 @@ describe('Background Entrypoint', () => { fakeBrowser.reset(); }); - it("should log the extenion's runtime ID", () => { + it("should log the extension's runtime ID", () => { const id = 'some-id'; fakeBrowser.runtime.id = id; diff --git a/packages/wxt-demo/wxt.config.ts b/packages/wxt-demo/wxt.config.ts index 57f8d5204..f92b5ae63 100644 --- a/packages/wxt-demo/wxt.config.ts +++ b/packages/wxt-demo/wxt.config.ts @@ -2,7 +2,6 @@ import { defineConfig } from 'wxt'; export default defineConfig({ srcDir: 'src', - extensionApi: 'chrome', manifest: { permissions: ['storage'], default_locale: 'en', diff --git a/packages/wxt/e2e/tests/auto-imports.test.ts b/packages/wxt/e2e/tests/auto-imports.test.ts index 167b33a44..118c52e2b 100644 --- a/packages/wxt/e2e/tests/auto-imports.test.ts +++ b/packages/wxt/e2e/tests/auto-imports.test.ts @@ -54,6 +54,7 @@ describe('Auto Imports', () => { /// /// /// + /// /// " `); @@ -93,6 +94,7 @@ describe('Auto Imports', () => { /// /// /// + /// " `, ); diff --git a/packages/wxt/e2e/tests/modules.test.ts b/packages/wxt/e2e/tests/modules.test.ts index 367099b52..1108ddbe6 100644 --- a/packages/wxt/e2e/tests/modules.test.ts +++ b/packages/wxt/e2e/tests/modules.test.ts @@ -191,10 +191,7 @@ describe('Module Helpers', () => { ); const expectedText = addPluginModule(project); - await project.build({ - // reduce build output when comparing test failures - extensionApi: 'chrome', - }); + await project.build(); await expect(project.serializeOutput()).resolves.toContain(expectedText); }); @@ -211,10 +208,7 @@ describe('Module Helpers', () => { ); const expectedText = addPluginModule(project); - await project.build({ - // reduce build output when comparing test failures - extensionApi: 'chrome', - }); + await project.build(); await expect(project.serializeOutput()).resolves.toContain(expectedText); }); @@ -232,10 +226,7 @@ describe('Module Helpers', () => { ); const expectedText = addPluginModule(project); - await project.build({ - // reduce build output when comparing test failures - extensionApi: 'chrome', - }); + await project.build(); await expect(project.serializeOutput()).resolves.toContain(expectedText); }); @@ -248,10 +239,7 @@ describe('Module Helpers', () => { ); const expectedText = addPluginModule(project); - await project.build({ - // reduce build output when comparing test failures - extensionApi: 'chrome', - }); + await project.build(); await expect(project.serializeOutput()).resolves.toContain(expectedText); }); diff --git a/packages/wxt/e2e/tests/output-structure.test.ts b/packages/wxt/e2e/tests/output-structure.test.ts index 260157ae7..f882b8595 100644 --- a/packages/wxt/e2e/tests/output-structure.test.ts +++ b/packages/wxt/e2e/tests/output-structure.test.ts @@ -262,9 +262,6 @@ describe('Output Directory Structure', () => { project.addFile('entrypoints/popup/main.ts', `logHello('popup')`); await project.build({ - // Simplify the build output for comparison - extensionApi: 'chrome', - vite: () => ({ build: { // Make output for snapshot readible @@ -347,9 +344,6 @@ describe('Output Directory Structure', () => { project.addFile('entrypoints/popup/main.ts', `logHello('popup')`); await project.build({ - // Simplify the build output for comparison - extensionApi: 'chrome', - vite: () => ({ build: { // Make output for snapshot readible diff --git a/packages/wxt/e2e/tests/typescript-project.test.ts b/packages/wxt/e2e/tests/typescript-project.test.ts index 34c372af6..20e8e8db0 100644 --- a/packages/wxt/e2e/tests/typescript-project.test.ts +++ b/packages/wxt/e2e/tests/typescript-project.test.ts @@ -238,6 +238,7 @@ describe('TypeScript Project', () => { /// /// /// + /// /// " `); diff --git a/packages/wxt/e2e/tests/user-config.test.ts b/packages/wxt/e2e/tests/user-config.test.ts index c0b497508..b444b07b8 100644 --- a/packages/wxt/e2e/tests/user-config.test.ts +++ b/packages/wxt/e2e/tests/user-config.test.ts @@ -1,6 +1,5 @@ import { describe, it, expect } from 'vitest'; import { TestProject } from '../utils'; -import { InlineConfig } from '../../src/types'; describe('User Config', () => { // Root directory is tested with all tests. @@ -88,24 +87,6 @@ describe('User Config', () => { `); }); - it('should exclude the polyfill when extensionApi="chrome"', async () => { - const buildBackground = async (config?: InlineConfig) => { - const background = `export default defineBackground(() => console.log(browser.runtime.id));`; - const projectWithPolyfill = new TestProject(); - projectWithPolyfill.addFile('entrypoints/background.ts', background); - await projectWithPolyfill.build(config); - return await projectWithPolyfill.serializeFile( - '.output/chrome-mv3/background.js', - ); - }; - - const withPolyfill = await buildBackground(); - const withoutPolyfill = await buildBackground({ - extensionApi: 'chrome', - }); - expect(withoutPolyfill).not.toBe(withPolyfill); - }); - it('should respect changing config files', async () => { const project = new TestProject(); project.addFile( diff --git a/packages/wxt/package.json b/packages/wxt/package.json index eb44a03c3..ab9b4f5c0 100644 --- a/packages/wxt/package.json +++ b/packages/wxt/package.json @@ -46,12 +46,8 @@ "default": "./dist/sandbox/index.mjs" }, "./browser": { - "types": "./dist/browser/index.d.ts", - "default": "./dist/browser/index.mjs" - }, - "./browser/chrome": { - "types": "./dist/browser/chrome.d.ts", - "import": "./dist/browser/chrome.mjs" + "types": "./dist/browser.d.ts", + "default": "./dist/browser.mjs" }, "./testing": { "types": "./dist/testing/index.d.ts", @@ -83,7 +79,6 @@ "dependencies": { "@aklinker1/rollup-plugin-visualizer": "5.12.0", "@types/chrome": "^0.0.269", - "@types/webextension-polyfill": "^0.10.7", "@webext-core/fake-browser": "^1.3.1", "@webext-core/isolated-element": "^1.1.2", "@webext-core/match-patterns": "^1.0.3", @@ -124,8 +119,7 @@ "unimport": "^3.13.1", "vite": "^5.4.8", "vite-node": "^2.1.2", - "web-ext-run": "^0.2.1", - "webextension-polyfill": "^0.12.0" + "web-ext-run": "^0.2.1" }, "devDependencies": { "@aklinker1/check": "^1.4.5", diff --git a/packages/wxt/src/browser/chrome.ts b/packages/wxt/src/browser.ts similarity index 56% rename from packages/wxt/src/browser/chrome.ts rename to packages/wxt/src/browser.ts index d1dad85db..10cd6e8b9 100644 --- a/packages/wxt/src/browser/chrome.ts +++ b/packages/wxt/src/browser.ts @@ -1,12 +1,19 @@ /// /** - * EXPERIMENTAL - * - * Includes the `chrome` API and types when using `extensionApi: 'chrome'`. - * - * @module wxt/browser/chrome + * @module wxt/browser */ -import type { WxtRuntime, WxtI18n } from './index'; + +/** + * This interface is empty because it is generated per-project when running `wxt prepare`. See: + * - `.wxt/types/paths.d.ts` + */ +export interface WxtRuntime {} + +/** + * This interface is empty because it is generated per-project when running `wxt prepare`. See: + * - `.wxt/types/i18n.d.ts` + */ +export interface WxtI18n {} export type WxtBrowser = Omit & { runtime: WxtRuntime & Omit<(typeof chrome)['runtime'], 'getURL'>; diff --git a/packages/wxt/src/browser/index.ts b/packages/wxt/src/browser/index.ts deleted file mode 100644 index 3eb44e110..000000000 --- a/packages/wxt/src/browser/index.ts +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Includes the `browser` API and types when using `extensionApi: 'webextension-polyfill'` (the default). - * - * @module wxt/browser - */ - -import originalBrowser, { Browser } from 'webextension-polyfill'; - -export type AugmentedBrowser = Omit & { - runtime: WxtRuntime & Omit; - i18n: WxtI18n & Omit; -}; - -/** - * This interface is empty because it is generated per-project when running `wxt prepare`. See: - * - `.wxt/types/paths.d.ts` - */ -export interface WxtRuntime {} - -/** - * This interface is empty because it is generated per-project when running `wxt prepare`. See: - * - `.wxt/types/i18n.d.ts` - */ -export interface WxtI18n {} - -export const browser: AugmentedBrowser = originalBrowser; - -// re-export all the types from webextension-polyfill -// Because webextension-polyfill uses a weird namespace with "import export", there isn't a good way -// to get these types without re-listing them. -/** @ignore */ -export type { - ActivityLog, - Alarms, - Bookmarks, - Action, - BrowserAction, - BrowserSettings, - BrowsingData, - CaptivePortal, - Clipboard, - Commands, - ContentScripts, - ContextualIdentities, - Cookies, - DeclarativeNetRequest, - Devtools, - Dns, - Downloads, - Events, - Experiments, - Extension, - ExtensionTypes, - Find, - GeckoProfiler, - History, - I18n, - Identity, - Idle, - Management, - Manifest, // TODO: Export custom manifest types that are valid for both Chrome and Firefox. - ContextMenus, - Menus, - NetworkStatus, - NormandyAddonStudy, - Notifications, - Omnibox, - PageAction, - Permissions, - Pkcs11, - Privacy, - Proxy, - Runtime, - Scripting, - Search, - Sessions, - SidebarAction, - Storage, - Tabs, - Theme, - TopSites, - Types, - Urlbar, - UserScripts, - WebNavigation, - WebRequest, - Windows, -} from 'webextension-polyfill'; diff --git a/packages/wxt/src/client/content-scripts/__tests__/content-script-context.test.ts b/packages/wxt/src/client/content-scripts/__tests__/content-script-context.test.ts index 6f7d14c18..099c9453f 100644 --- a/packages/wxt/src/client/content-scripts/__tests__/content-script-context.test.ts +++ b/packages/wxt/src/client/content-scripts/__tests__/content-script-context.test.ts @@ -16,7 +16,7 @@ describe('Content Script Context', () => { const onInvalidated = vi.fn(); ctx.onInvalidated(onInvalidated); - // @ts-expect-error + // @ts-ignore delete fakeBrowser.runtime.id; const isValid = ctx.isValid; diff --git a/packages/wxt/src/core/builders/vite/index.ts b/packages/wxt/src/core/builders/vite/index.ts index 6ab15ab4a..5fa87581c 100644 --- a/packages/wxt/src/core/builders/vite/index.ts +++ b/packages/wxt/src/core/builders/vite/index.ts @@ -74,7 +74,6 @@ export async function createViteBuilder( wxtPlugins.tsconfigPaths(wxtConfig), wxtPlugins.noopBackground(), wxtPlugins.globals(wxtConfig), - wxtPlugins.resolveExtensionApi(wxtConfig), wxtPlugins.defineImportMeta(), wxtPlugins.wxtPluginLoader(wxtConfig), wxtPlugins.resolveAppConfig(wxtConfig), diff --git a/packages/wxt/src/core/builders/vite/plugins/extensionApiMock.ts b/packages/wxt/src/core/builders/vite/plugins/extensionApiMock.ts index 7724219ec..0e2b18f3a 100644 --- a/packages/wxt/src/core/builders/vite/plugins/extensionApiMock.ts +++ b/packages/wxt/src/core/builders/vite/plugins/extensionApiMock.ts @@ -3,12 +3,7 @@ import type * as vite from 'vite'; import { ResolvedConfig } from '../../../../types'; /** - * Mock `webextension-polyfill`, `wxt/browser`, and `wxt/browser/*` by inlining - * all dependencies that import them and adding a custom alias so that Vite - * resolves to a mocked version of the module. - * - * TODO: Detect non-wxt dependencies (like `@webext-core/*`) that import `webextension-polyfill` via - * `npm list` and inline them automatically. + * Mock `wxt/browser` and stub the global `browser`/`chrome` types with a fake version of the extension APIs */ export function extensionApiMock(config: ResolvedConfig): vite.PluginOption { const virtualSetupModule = 'virtual:wxt-setup'; @@ -27,14 +22,12 @@ export function extensionApiMock(config: ResolvedConfig): vite.PluginOption { }, resolve: { alias: [ - { find: 'webextension-polyfill', replacement }, // wxt/browser, wxt/browser/... - { find: /^wxt\/browser.*/, replacement }, + { find: 'wxt/browser', replacement }, ], }, ssr: { - // Inline all WXT modules so vite processes them so the aliases can - // be resolved + // Inline all WXT modules subdependencies can be mocked noExternal: ['wxt'], }, }; diff --git a/packages/wxt/src/core/builders/vite/plugins/index.ts b/packages/wxt/src/core/builders/vite/plugins/index.ts index f70dde240..5ed13ff81 100644 --- a/packages/wxt/src/core/builders/vite/plugins/index.ts +++ b/packages/wxt/src/core/builders/vite/plugins/index.ts @@ -9,7 +9,6 @@ export * from './cssEntrypoints'; export * from './bundleAnalysis'; export * from './globals'; export * from './extensionApiMock'; -export * from './resolveExtensionApi'; export * from './entrypointGroupGlobals'; export * from './defineImportMeta'; export * from './removeEntrypointMainFunction'; diff --git a/packages/wxt/src/core/builders/vite/plugins/resolveExtensionApi.ts b/packages/wxt/src/core/builders/vite/plugins/resolveExtensionApi.ts deleted file mode 100644 index 17ec02064..000000000 --- a/packages/wxt/src/core/builders/vite/plugins/resolveExtensionApi.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { ResolvedConfig } from '../../../../types'; -import type * as vite from 'vite'; - -/** - * Apply the experimental config for which extension API is used. This only - * effects the extension API included at RUNTIME - during development, types - * depend on the import. - * - * NOTE: this only works if we import `wxt/browser` instead of using the relative path. - */ -export function resolveExtensionApi(config: ResolvedConfig): vite.Plugin { - return { - name: 'wxt:resolve-extension-api', - config() { - // Only apply the config if we're disabling the polyfill - if (config.extensionApi === 'webextension-polyfill') return; - - return { - resolve: { - alias: [ - { find: /^wxt\/browser$/, replacement: 'wxt/browser/chrome' }, - ], - }, - }; - }, - }; -} diff --git a/packages/wxt/src/core/generate-wxt-dir.ts b/packages/wxt/src/core/generate-wxt-dir.ts index cf9980c7f..dfa3149b9 100644 --- a/packages/wxt/src/core/generate-wxt-dir.ts +++ b/packages/wxt/src/core/generate-wxt-dir.ts @@ -37,10 +37,8 @@ export async function generateWxtDir(entrypoints: Entrypoint[]): Promise { // import.meta.env.* entries.push(await getGlobalsDeclarationEntry()); - // @types/chrome - if (wxt.config.extensionApi === 'chrome') { - entries.push({ module: '@types/chrome' }); - } + // wxt/browser types + entries.push({ module: '@types/chrome' }); // tsconfig.json entries.push(await getTsConfigEntry()); diff --git a/packages/wxt/src/core/resolve-config.ts b/packages/wxt/src/core/resolve-config.ts index e99dbb10b..4209cb823 100644 --- a/packages/wxt/src/core/resolve-config.ts +++ b/packages/wxt/src/core/resolve-config.ts @@ -162,8 +162,6 @@ export async function resolveConfig( {}, ); - const extensionApi = mergedConfig.extensionApi ?? 'webextension-polyfill'; - return { browser, command, @@ -173,13 +171,7 @@ export async function resolveConfig( filterEntrypoints, env, fsCache: createFsCache(wxtDir), - imports: await getUnimportOptions( - wxtDir, - srcDir, - logger, - extensionApi, - mergedConfig, - ), + imports: await getUnimportOptions(wxtDir, srcDir, logger, mergedConfig), logger, manifest: await resolveManifestConfig(env, mergedConfig.manifest), manifestVersion, @@ -198,7 +190,6 @@ export async function resolveConfig( analysis: resolveAnalysisConfig(root, mergedConfig), userConfigMetadata: userConfigMetadata ?? {}, alias, - extensionApi, entrypointLoader: mergedConfig.entrypointLoader ?? 'vite-node', experimental: defu(mergedConfig.experimental, {}), dev: { @@ -333,7 +324,6 @@ async function getUnimportOptions( wxtDir: string, srcDir: string, logger: Logger, - extensionApi: ResolvedConfig['extensionApi'], config: InlineConfig, ): Promise { if (config.imports === false) return false; @@ -352,10 +342,7 @@ async function getUnimportOptions( // ignored. ignore: ['options'], }, - { - package: - extensionApi === 'chrome' ? 'wxt/browser/chrome' : 'wxt/browser', - }, + { package: 'wxt/browser' }, { package: 'wxt/sandbox' }, { package: 'wxt/storage' }, ], diff --git a/packages/wxt/src/core/utils/__tests__/manifest.test.ts b/packages/wxt/src/core/utils/__tests__/manifest.test.ts index 1af508dc5..56fdd1efe 100644 --- a/packages/wxt/src/core/utils/__tests__/manifest.test.ts +++ b/packages/wxt/src/core/utils/__tests__/manifest.test.ts @@ -13,7 +13,6 @@ import { fakeWxtDevServer, setFakeWxt, } from '../testing/fake-objects'; -import { Manifest } from 'webextension-polyfill'; import { BuildOutput, ContentScriptEntrypoint, @@ -56,7 +55,7 @@ describe('Manifest Utils', () => { outDir, }, }); - const expected: Partial = { + const expected: Partial = { action: { default_icon: popup.options.defaultIcon, default_title: popup.options.defaultTitle, diff --git a/packages/wxt/src/core/utils/building/rebuild.ts b/packages/wxt/src/core/utils/building/rebuild.ts index 7a17e3ff1..4e7a6c7a5 100644 --- a/packages/wxt/src/core/utils/building/rebuild.ts +++ b/packages/wxt/src/core/utils/building/rebuild.ts @@ -1,4 +1,3 @@ -import type { Manifest } from 'wxt/browser'; import { BuildOutput, Entrypoint, EntrypointGroup } from '../../../types'; import { generateWxtDir } from '../../generate-wxt-dir'; import { buildEntrypoints } from './build-entrypoints'; @@ -30,7 +29,7 @@ export async function rebuild( }, ): Promise<{ output: BuildOutput; - manifest: Manifest.WebExtensionManifest; + manifest: chrome.runtime.Manifest; warnings: any[][]; }> { const { default: ora } = await import('ora'); diff --git a/packages/wxt/src/core/utils/content-scripts.ts b/packages/wxt/src/core/utils/content-scripts.ts index d9b1425dd..cfb6db3e6 100644 --- a/packages/wxt/src/core/utils/content-scripts.ts +++ b/packages/wxt/src/core/utils/content-scripts.ts @@ -1,6 +1,6 @@ -import type { Manifest, Scripting } from 'wxt/browser'; import { ContentScriptEntrypoint, ResolvedConfig } from '../../types'; import { getEntrypointBundlePath } from './entrypoints'; +import { ManifestContentScript } from './types'; /** * Returns a unique and consistent string hash based on a content scripts options. @@ -22,7 +22,7 @@ export function hashContentScriptOptions( if (simplifiedOptions[key] == null) delete simplifiedOptions[key]; }); - const withDefaults: Manifest.ContentScript = { + const withDefaults: ManifestContentScript = { exclude_globs: [], exclude_matches: [], include_globs: [], @@ -50,7 +50,7 @@ export function mapWxtOptionsToContentScript( options: ContentScriptEntrypoint['options'], js: string[] | undefined, css: string[] | undefined, -): Manifest.ContentScript { +): ManifestContentScript { return { matches: options.matches, all_frames: options.allFrames, @@ -72,7 +72,7 @@ export function mapWxtOptionsToRegisteredContentScript( options: ContentScriptEntrypoint['options'], js: string[] | undefined, css: string[] | undefined, -): Omit { +): Omit { return { allFrames: options.allFrames, excludeMatches: options.excludeMatches, @@ -80,7 +80,6 @@ export function mapWxtOptionsToRegisteredContentScript( runAt: options.runAt, js, css, - // @ts-expect-error: Chrome accepts this, not typed in webextension-polyfill (https://developer.chrome.com/docs/extensions/reference/scripting/#type-RegisteredContentScript) world: options.world, }; } diff --git a/packages/wxt/src/core/utils/manifest.ts b/packages/wxt/src/core/utils/manifest.ts index 17a9feb1f..e04efedd3 100644 --- a/packages/wxt/src/core/utils/manifest.ts +++ b/packages/wxt/src/core/utils/manifest.ts @@ -1,4 +1,3 @@ -import type { Manifest } from 'wxt/browser'; import { Entrypoint, BackgroundEntrypoint, @@ -21,12 +20,13 @@ import { normalizePath } from './paths'; import { writeFileIfDifferent } from './fs'; import defu from 'defu'; import { wxt } from '../wxt'; +import { ManifestV3WebAccessibleResource } from './types'; /** * Writes the manifest to the output directory and the build output. */ export async function writeManifest( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, output: BuildOutput, ): Promise { const str = @@ -49,7 +49,7 @@ export async function writeManifest( export async function generateManifest( entrypoints: Entrypoint[], buildOutput: Omit, -): Promise<{ manifest: Manifest.WebExtensionManifest; warnings: any[][] }> { +): Promise<{ manifest: chrome.runtime.Manifest; warnings: any[][] }> { const warnings: any[][] = []; const pkg = await getPackageJson(); @@ -65,7 +65,7 @@ export async function generateManifest( } const version = wxt.config.manifest.version ?? simplifyVersion(versionName); - const baseManifest: Manifest.WebExtensionManifest = { + const baseManifest: chrome.runtime.Manifest = { manifest_version: wxt.config.manifestVersion, name: pkg?.name, description: pkg?.description, @@ -75,10 +75,7 @@ export async function generateManifest( }; const userManifest = wxt.config.manifest; - let manifest = defu( - userManifest, - baseManifest, - ) as Manifest.WebExtensionManifest; + let manifest = defu(userManifest, baseManifest) as chrome.runtime.Manifest; // Add reload command in dev mode if (wxt.config.command === 'serve' && wxt.config.dev.reloadCommand) { @@ -121,7 +118,7 @@ export async function generateManifest( } if (wxt.config.manifestVersion === 3) { - validateMv3WebAccessbileResources(manifest); + validateMv3WebAccessibleResources(manifest); } stripKeys(manifest); @@ -143,7 +140,7 @@ export async function generateManifest( } /** - * Removes suffixes from the version, like X.Y.Z-alpha1 (which brosers don't allow), so it's a + * Removes suffixes from the version, like X.Y.Z-alpha1 (which browsers don't allow), so it's a * simple version number, like X or X.Y or X.Y.Z, which browsers allow. */ function simplifyVersion(versionName: string): string { @@ -161,7 +158,7 @@ function simplifyVersion(versionName: string): string { } function addEntrypoints( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, entrypoints: Entrypoint[], buildOutput: Omit, ): void { @@ -223,7 +220,6 @@ function addEntrypoints( ); } else { manifest.chrome_url_overrides ??= {}; - // @ts-expect-error: bookmarks is untyped in webextension-polyfill, but supported by chrome manifest.chrome_url_overrides.bookmarks = getEntrypointBundlePath( bookmarks, wxt.config.outDir, @@ -239,7 +235,6 @@ function addEntrypoints( ); } else { manifest.chrome_url_overrides ??= {}; - // @ts-expect-error: history is untyped in webextension-polyfill, but supported by chrome manifest.chrome_url_overrides.history = getEntrypointBundlePath( history, wxt.config.outDir, @@ -263,12 +258,13 @@ function addEntrypoints( wxt.config.outDir, '.html', ); - const options: Manifest.ActionManifest = {}; + const options: chrome.runtime.ManifestAction = {}; if (popup.options.defaultIcon) options.default_icon = popup.options.defaultIcon; if (popup.options.defaultTitle) options.default_title = popup.options.defaultTitle; if (popup.options.browserStyle) + // @ts-expect-error: Not typed by @types/chrome, but supported by Firefox options.browser_style = popup.options.browserStyle; if (manifest.manifest_version === 3) { manifest.action = { @@ -298,6 +294,7 @@ function addEntrypoints( const page = getEntrypointBundlePath(options, wxt.config.outDir, '.html'); manifest.options_ui = { open_in_tab: options.options.openInTab, + // @ts-expect-error: Not typed by @types/chrome, but supported by Firefox browser_style: wxt.config.browser === 'firefox' ? options.options.browserStyle @@ -316,7 +313,6 @@ function addEntrypoints( 'Sandboxed pages not supported by Firefox. sandbox.pages was not added to the manifest', ); } else { - // @ts-expect-error: sandbox not typed manifest.sandbox = { pages: sandboxes.map((entry) => getEntrypointBundlePath(entry, wxt.config.outDir, '.html'), @@ -343,7 +339,6 @@ function addEntrypoints( open_at_install: defaultSidepanel.options.openAtInstall, }; } else if (wxt.config.manifestVersion === 3) { - // @ts-expect-error: Untyped manifest.side_panel = { default_path: page, }; @@ -424,7 +419,7 @@ function addEntrypoints( function discoverIcons( buildOutput: Omit, -): Manifest.WebExtensionManifest['icons'] { +): chrome.runtime.Manifest['icons'] { const icons: [string, string][] = []; // prettier-ignore // #region snippet @@ -456,7 +451,7 @@ function discoverIcons( return icons.length > 0 ? Object.fromEntries(icons) : undefined; } -function addDevModeCsp(manifest: Manifest.WebExtensionManifest): void { +function addDevModeCsp(manifest: chrome.runtime.Manifest): void { const permission = `http://${wxt.server?.hostname ?? ''}/*`; const allowedCsp = wxt.server?.origin ?? 'http://localhost:*'; @@ -468,8 +463,7 @@ function addDevModeCsp(manifest: Manifest.WebExtensionManifest): void { const extensionPagesCsp = new ContentSecurityPolicy( manifest.manifest_version === 3 - ? // @ts-expect-error: extension_pages is not typed - (manifest.content_security_policy?.extension_pages ?? + ? (manifest.content_security_policy?.extension_pages ?? "script-src 'self' 'wasm-unsafe-eval'; object-src 'self';") // default extension_pages CSP for MV3 : (manifest.content_security_policy ?? "script-src 'self'; object-src 'self';"), // default CSP for MV2 @@ -487,17 +481,15 @@ function addDevModeCsp(manifest: Manifest.WebExtensionManifest): void { if (manifest.manifest_version === 3) { manifest.content_security_policy ??= {}; - // @ts-expect-error: extension_pages is not typed manifest.content_security_policy.extension_pages = extensionPagesCsp.toString(); - // @ts-expect-error: sandbox is not typed manifest.content_security_policy.sandbox = sandboxCsp.toString(); } else { manifest.content_security_policy = extensionPagesCsp.toString(); } } -function addDevModePermissions(manifest: Manifest.WebExtensionManifest) { +function addDevModePermissions(manifest: chrome.runtime.Manifest) { // For reloading the page addPermission(manifest, 'tabs'); @@ -541,8 +533,7 @@ export function getContentScriptCssWebAccessibleResources( contentScripts: ContentScriptEntrypoint[], contentScriptCssMap: Record, ): any[] { - const resources: Manifest.WebExtensionManifestWebAccessibleResourcesC2ItemType[] = - []; + const resources: ManifestV3WebAccessibleResource[] = []; contentScripts.forEach((script) => { if (script.options.cssInjectionMode !== 'ui') return; @@ -581,16 +572,18 @@ export function getContentScriptsCssMap( } function addPermission( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, permission: string, ): void { manifest.permissions ??= []; + // @ts-expect-error: Allow using strings for permissions for MV2 support if (manifest.permissions.includes(permission)) return; + // @ts-expect-error: Allow using strings for permissions for MV2 support manifest.permissions.push(permission); } function addHostPermission( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, hostPermission: string, ): void { manifest.host_permissions ??= []; @@ -613,10 +606,10 @@ export function stripPathFromMatchPattern(pattern: string) { /** * Converts all MV3 web accessible resources to their MV2 forms. MV3 web accessible resources are * generated in this file, and may be defined by the user in their manifest. In both cases, when - * targetting MV2, automatically convert their definitions down to the basic MV2 array. + * targeting MV2, automatically convert their definitions down to the basic MV2 array. */ export function convertWebAccessibleResourcesToMv2( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, ): void { if (manifest.web_accessible_resources == null) return; @@ -631,17 +624,17 @@ export function convertWebAccessibleResourcesToMv2( } function moveHostPermissionsToPermissions( - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, ): void { if (!manifest.host_permissions?.length) return; - manifest.host_permissions.forEach((permission) => + manifest.host_permissions.forEach((permission: string) => addPermission(manifest, permission), ); delete manifest.host_permissions; } -function convertActionToMv2(manifest: Manifest.WebExtensionManifest): void { +function convertActionToMv2(manifest: chrome.runtime.Manifest): void { if ( manifest.action == null || manifest.browser_action != null || @@ -655,8 +648,8 @@ function convertActionToMv2(manifest: Manifest.WebExtensionManifest): void { /** * Make sure all resources are in MV3 format. If not, add a wanring */ -export function validateMv3WebAccessbileResources( - manifest: Manifest.WebExtensionManifest, +function validateMv3WebAccessibleResources( + manifest: chrome.runtime.Manifest, ): void { if (manifest.web_accessible_resources == null) return; @@ -675,7 +668,7 @@ export function validateMv3WebAccessbileResources( /** * Remove keys from the manifest based on the build target. */ -function stripKeys(manifest: Manifest.WebExtensionManifest): void { +function stripKeys(manifest: chrome.runtime.Manifest): void { let keysToRemove: string[] = []; if (wxt.config.manifestVersion === 2) { keysToRemove.push(...mv3OnlyKeys); @@ -686,7 +679,7 @@ function stripKeys(manifest: Manifest.WebExtensionManifest): void { } keysToRemove.forEach((key) => { - delete manifest[key as keyof Manifest.WebExtensionManifest]; + delete manifest[key as keyof chrome.runtime.Manifest]; }); } diff --git a/packages/wxt/src/core/utils/testing/fake-objects.ts b/packages/wxt/src/core/utils/testing/fake-objects.ts index c097aeab7..db48364d2 100644 --- a/packages/wxt/src/core/utils/testing/fake-objects.ts +++ b/packages/wxt/src/core/utils/testing/fake-objects.ts @@ -4,7 +4,6 @@ import { resolve } from 'path'; import { faker } from '@faker-js/faker'; import merge from 'lodash.merge'; -import { Commands, type Manifest } from 'wxt/browser'; import { FsCache, ResolvedConfig, @@ -207,13 +206,11 @@ export function fakeOutputFile(): OutputFile { return faker.helpers.arrayElement([fakeOutputAsset(), fakeOutputChunk()]); } -export const fakeManifest = fakeObjectCreator( - () => ({ - manifest_version: faker.helpers.arrayElement([2, 3]), - name: faker.string.alphanumeric(), - version: `${faker.number.int()}.${faker.number.int()}.${faker.number.int()}`, - }), -); +export const fakeManifest = fakeObjectCreator(() => ({ + manifest_version: faker.helpers.arrayElement([2, 3]), + name: faker.string.alphanumeric(), + version: `${faker.number.int()}.${faker.number.int()}.${faker.number.int()}`, +})); export const fakeUserManifest = fakeObjectCreator(() => ({ name: faker.string.alphanumeric(), @@ -298,7 +295,6 @@ export const fakeResolvedConfig = fakeObjectCreator(() => { transformManifest: () => {}, userConfigMetadata: {}, alias: {}, - extensionApi: 'webextension-polyfill', entrypointLoader: 'vite-node', experimental: {}, dev: { @@ -355,13 +351,17 @@ export const fakeBuildStepOutput = fakeObjectCreator(() => ({ entrypoints: fakeArray(fakeEntrypoint), })); -export const fakeManifestCommand = fakeObjectCreator(() => ({ - description: faker.string.sample(), - shortcut: `${faker.helpers.arrayElement(['ctrl', 'alt'])}+${faker.number.int({ - min: 0, - max: 9, - })}`, -})); +export const fakeManifestCommand = fakeObjectCreator( + () => ({ + description: faker.string.sample(), + shortcut: `${faker.helpers.arrayElement(['ctrl', 'alt'])}+${faker.number.int( + { + min: 0, + max: 9, + }, + )}`, + }), +); export const fakeDevServer = fakeObjectCreator(() => ({ hostname: 'localhost', diff --git a/packages/wxt/src/core/utils/transform.ts b/packages/wxt/src/core/utils/transform.ts index f6951c614..857fdcff6 100644 --- a/packages/wxt/src/core/utils/transform.ts +++ b/packages/wxt/src/core/utils/transform.ts @@ -5,7 +5,7 @@ import { ProxifiedModule, parseModule } from 'magicast'; * 1. Removes or clears out `main` function from returned object * 2. Removes any unused functions/variables outside the definition that aren't being called/used * 3. Removes unused imports - * 3. Removes value-less, side-effect only imports (like `import "./styles.css"` or `import "webextension-polyfill"`) + * 3. Removes value-less, side-effect only imports (like `import "./styles.css"` or `import "polyfill"`) */ export function removeMainFunctionCode(code: string): { code: string; diff --git a/packages/wxt/src/core/utils/types.ts b/packages/wxt/src/core/utils/types.ts index ed6ff2000..45d68dda9 100644 --- a/packages/wxt/src/core/utils/types.ts +++ b/packages/wxt/src/core/utils/types.ts @@ -6,3 +6,11 @@ * // type Test = {a: string | undefined, b: number} */ export type NullablyRequired = { [K in keyof Required]: T[K] }; + +export type ManifestContentScript = NonNullable< + chrome.runtime.Manifest['content_scripts'] +>[number]; + +export type ManifestV3WebAccessibleResource = NonNullable< + chrome.runtime.ManifestV3['web_accessible_resources'] +>[number]; diff --git a/packages/wxt/src/storage.ts b/packages/wxt/src/storage.ts index 55795d4a7..94c774b21 100644 --- a/packages/wxt/src/storage.ts +++ b/packages/wxt/src/storage.ts @@ -5,7 +5,7 @@ * * @module wxt/storage */ -import { Storage, browser } from 'wxt/browser'; +import { browser } from 'wxt/browser'; import { dequal } from 'dequal/lite'; import { logger } from './sandbox/utils/logger'; import { toArray } from './core/utils/arrays'; @@ -391,7 +391,7 @@ function createDriver(storageArea: StorageArea): WxtStorageDriver { return area; }; const watchListeners = new Set< - (changes: Storage.StorageAreaOnChangedChangesType) => void + (changes: Record) => void >(); return { getItem: async (key) => { @@ -432,7 +432,9 @@ function createDriver(storageArea: StorageArea): WxtStorageDriver { await getStorageArea().set(data); }, watch(key, cb) { - const listener = (changes: Storage.StorageAreaOnChangedChangesType) => { + const listener = ( + changes: Record, + ) => { const change = changes[key]; if (change == null) return; if (dequal(change.newValue, change.oldValue)) return; @@ -646,7 +648,7 @@ export interface WxtStorageItem< migrate(): Promise; } -export type StorageArea = 'local' | 'session' | 'sync' | 'managed'; +export type StorageArea = chrome.storage.AreaName; export type StorageItemKey = `${StorageArea}:${string}`; export interface GetItemOptions { diff --git a/packages/wxt/src/types.ts b/packages/wxt/src/types.ts index 3ad48acfe..8a258f0f5 100644 --- a/packages/wxt/src/types.ts +++ b/packages/wxt/src/types.ts @@ -1,5 +1,4 @@ import type * as vite from 'vite'; -import type { Manifest, Scripting } from 'wxt/browser'; import { UnimportOptions, Import } from 'unimport'; import { LogLevel } from 'consola'; import type { ContentScriptContext } from './client/content-scripts/content-script-context'; @@ -8,6 +7,7 @@ import type { FSWatcher } from 'chokidar'; import { ResolvedConfig as C12ResolvedConfig } from 'c12'; import { Hookable, NestedHooks } from 'hookable'; import type * as Nypm from 'nypm'; +import { ManifestContentScript } from './core/utils/types'; export interface InlineConfig { /** @@ -267,7 +267,7 @@ export interface InlineConfig { * } * }) */ - transformManifest?: (manifest: Manifest.WebExtensionManifest) => void; + transformManifest?: (manifest: chrome.runtime.Manifest) => void; analysis?: { /** * Explicitly include bundle analysis when running `wxt build`. This can be overridden by the @@ -325,16 +325,6 @@ export interface InlineConfig { * } */ alias?: Record; - /** - * Which extension API to use. - * - * - `"webextension-polyfill"`: Use `browser` and types from [`webextension-polyfill`](https://www.npmjs.com/package/webextension-polyfill). - * - `"chrome"`: Use the regular `chrome` (or `browser` for Firefox/Safari) globals provided by the browser. Types provided by [`@types/chrome`](https://www.npmjs.com/package/@types/chrome). - * - * @default "webextension-polyfill" - * @since 0.19.0 - */ - extensionApi?: 'webextension-polyfill' | 'chrome'; /** * @deprecated Will be removed in v0.20.0, please migrate to using `vite-node`, the new default. * @@ -444,7 +434,7 @@ export interface WxtHooks { } export interface BuildOutput { - manifest: Manifest.WebExtensionManifest; + manifest: chrome.runtime.Manifest; publicAssets: OutputAsset[]; steps: BuildStepOutput[]; } @@ -537,7 +527,7 @@ export interface WxtDevServer export interface ReloadContentScriptPayload { registration?: BaseContentScriptEntrypointOptions['registration']; - contentScript: Omit; + contentScript: Omit; } export type TargetBrowser = string; @@ -588,39 +578,39 @@ export interface BackgroundEntrypointOptions extends BaseEntrypointOptions { export interface BaseContentScriptEntrypointOptions extends BaseEntrypointOptions { - matches: PerBrowserOption; + matches: PerBrowserOption>; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default "documentIdle" */ - runAt?: PerBrowserOption; + runAt?: PerBrowserOption; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default false */ matchAboutBlank?: PerBrowserOption< - Manifest.ContentScript['match_about_blank'] + ManifestContentScript['match_about_blank'] >; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default [] */ - excludeMatches?: PerBrowserOption; + excludeMatches?: PerBrowserOption; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default [] */ - includeGlobs?: PerBrowserOption; + includeGlobs?: PerBrowserOption; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default [] */ - excludeGlobs?: PerBrowserOption; + excludeGlobs?: PerBrowserOption; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default false */ - allFrames?: PerBrowserOption; + allFrames?: PerBrowserOption; /** * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/ * @default false @@ -1163,7 +1153,7 @@ export interface WxtHooks { */ 'build:manifestGenerated': ( wxt: Wxt, - manifest: Manifest.WebExtensionManifest, + manifest: chrome.runtime.Manifest, ) => HookResult; /** * Called once all entrypoints have been loaded from the `entrypointsDir`. @@ -1316,7 +1306,7 @@ export interface ResolvedConfig { /** * @deprecated Use `build:manifestGenerated` hook instead. */ - transformManifest?: (manifest: Manifest.WebExtensionManifest) => void; + transformManifest?: (manifest: chrome.runtime.Manifest) => void; analysis: { enabled: boolean; open: boolean; @@ -1334,7 +1324,6 @@ export interface ResolvedConfig { * Import aliases to absolute paths. */ alias: Record; - extensionApi: 'webextension-polyfill' | 'chrome'; entrypointLoader: 'vite-node' | 'jiti'; experimental: {}; dev: { diff --git a/packages/wxt/src/virtual/tsconfig.json b/packages/wxt/src/virtual/tsconfig.json index 6f74c3f88..0820153ad 100644 --- a/packages/wxt/src/virtual/tsconfig.json +++ b/packages/wxt/src/virtual/tsconfig.json @@ -1,7 +1,7 @@ { "extends": "../../../../tsconfig.base.json", "compilerOptions": { - "types": ["vite/client", "../@types/globals.d.ts"] + "types": ["vite/client", "../@types/globals.d.ts", "chrome"] }, "include": ["./*"] } diff --git a/packages/wxt/src/virtual/utils/reload-content-scripts.ts b/packages/wxt/src/virtual/utils/reload-content-scripts.ts index 11b5e0c87..3b82417ae 100644 --- a/packages/wxt/src/virtual/utils/reload-content-scripts.ts +++ b/packages/wxt/src/virtual/utils/reload-content-scripts.ts @@ -84,7 +84,7 @@ async function reloadTabsForContentScript(contentScript: ContentScript) { await Promise.all( matchingTabs.map(async (tab) => { try { - await browser.tabs.reload(tab.id); + await browser.tabs.reload(tab.id!); } catch (err) { logger.warn('Failed to reload tab:', err); } diff --git a/packages/wxt/src/virtual/virtual-module-globals.d.ts b/packages/wxt/src/virtual/virtual-module-globals.d.ts index 5199a83a9..fa95ef866 100644 --- a/packages/wxt/src/virtual/virtual-module-globals.d.ts +++ b/packages/wxt/src/virtual/virtual-module-globals.d.ts @@ -21,7 +21,7 @@ declare module 'virtual:user-unlisted-script-entrypoint' { } declare module 'wxt/browser' { - export const browser: import('webextension-polyfill').Browser; + export { chrome as browser }; } declare module 'virtual:wxt-plugins' { diff --git a/packages/wxt/typedoc.json b/packages/wxt/typedoc.json index 312472814..b5472db17 100644 --- a/packages/wxt/typedoc.json +++ b/packages/wxt/typedoc.json @@ -3,8 +3,7 @@ "src/client/index.ts", "src/testing/index.ts", "src/sandbox/index.ts", - "src/browser/index.ts", - "src/browser/chrome.ts", + "src/browser.ts", "src/index.ts", "src/storage.ts", "src/modules.ts" diff --git a/packages/wxt/vitest.config.ts b/packages/wxt/vitest.config.ts index 8bb1d0dab..0c25b9629 100644 --- a/packages/wxt/vitest.config.ts +++ b/packages/wxt/vitest.config.ts @@ -11,6 +11,7 @@ export default defineConfig({ include: ['src/**'], exclude: ['**/dist', '**/__tests__', 'src/utils/testing'], }, + setupFiles: ['./vitest.setup.ts'], globalSetup: ['./vitest.globalSetup.ts'], }, server: { @@ -22,7 +23,6 @@ export default defineConfig({ resolve: { alias: { 'wxt/testing': path.resolve('src/testing'), - 'webextension-polyfill': path.resolve('src/virtual/mock-browser'), }, }, }); diff --git a/packages/wxt/vitest.setup.ts b/packages/wxt/vitest.setup.ts new file mode 100644 index 000000000..a828762f2 --- /dev/null +++ b/packages/wxt/vitest.setup.ts @@ -0,0 +1,5 @@ +import { fakeBrowser } from '@webext-core/fake-browser'; +import { vi } from 'vitest'; + +vi.stubGlobal('chrome', fakeBrowser); +vi.stubGlobal('browser', fakeBrowser); diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index d94a395f8..a6d891145 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -257,9 +257,6 @@ importers: '@types/chrome': specifier: ^0.0.269 version: 0.0.269 - '@types/webextension-polyfill': - specifier: ^0.10.7 - version: 0.10.7 '@webext-core/fake-browser': specifier: ^1.3.1 version: 1.3.1 @@ -383,9 +380,6 @@ importers: web-ext-run: specifier: ^0.2.1 version: 0.2.1 - webextension-polyfill: - specifier: ^0.12.0 - version: 0.12.0 devDependencies: '@aklinker1/check': specifier: ^1.4.5 @@ -1731,9 +1725,6 @@ packages: '@types/web-bluetooth@0.0.20': resolution: {integrity: sha512-g9gZnnXVq7gM7v3tJCWV/qw7w+KeOlSHAhgF9RytFyifW6AF61hdT2ucrYhPq9hLs5JIryeupHV3qGk95dH9ow==} - '@types/webextension-polyfill@0.10.7': - resolution: {integrity: sha512-10ql7A0qzBmFB+F+qAke/nP1PIonS0TXZAOMVOxEUsm+lGSW6uwVcISFNa0I4Oyj0884TZVWGGMIWeXOVSNFHw==} - '@types/yauzl@2.10.3': resolution: {integrity: sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q==} @@ -4541,9 +4532,6 @@ packages: resolution: {integrity: sha512-5D11VcjdGkA1/xax5UWL0YeAbDySKHzWFe6EpsoPNUMw5Uk9tKk9p6GUOfcaI5N7sINKfBMZYNsTBiu5dzJB9A==} engines: {node: '>=18.0.0', npm: '>=8.0.0'} - webextension-polyfill@0.12.0: - resolution: {integrity: sha512-97TBmpoWJEE+3nFBQ4VocyCdLKfw54rFaJ6EVQYLBCXqCIpLSZkwGgASpv4oPt9gdKCJ80RJlcmNzNn008Ag6Q==} - webidl-conversions@7.0.0: resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==} engines: {node: '>=12'} @@ -5780,8 +5768,6 @@ snapshots: '@types/web-bluetooth@0.0.20': {} - '@types/webextension-polyfill@0.10.7': {} - '@types/yauzl@2.10.3': dependencies: '@types/node': 20.16.10 @@ -8879,8 +8865,6 @@ snapshots: - supports-color - utf-8-validate - webextension-polyfill@0.12.0: {} - webidl-conversions@7.0.0: {} webpack-sources@3.2.3: