Skip to content

Commit 83b2c19

Browse files
authored
Upgrade to Manifest V3 (#1714)
* Update Chrome manifest.json * Remove use of window in background * Test devpanel * Inject pageScript using new API * Keep connection from devpanel to background alive * Keep connection from content script to background alive * Replace page action with action * Cleanup syncOptions * Update options to not rely on background page access * Start work on updating popup * Updates * Remove window * Get opening in a separate window working * Remove pageScriptWrap * Add socket to panelStore * Fix tests * Try to use MV3 for Firefox * Fix path * Fix Chrome E2E tests * Revert unintentional change * Skip Electron tests for now Looks like they're still working through stuff in electron/electron#41613 * Better image centering The Firefox popup did not like the old CSS. This is still not perfect, but it's better than it was. * Create shaggy-taxis-cross.md
1 parent 61ec00f commit 83b2c19

33 files changed

+341
-760
lines changed

.changeset/shaggy-taxis-cross.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'remotedev-redux-devtools-extension': minor
3+
---
4+
5+
Upgrade to Manifest V3

extension/build.mjs

+2-18
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import pug from 'pug';
55
const args = process.argv.slice(2);
66
const prod = !args.includes('--dev');
77

8-
const commonEsbuildOptions = {
8+
await esbuild.build({
99
bundle: true,
1010
logLevel: 'info',
1111
outdir: 'dist',
@@ -15,40 +15,24 @@ const commonEsbuildOptions = {
1515
'process.env.NODE_ENV': prod ? '"production"' : '"development"',
1616
'process.env.BABEL_ENV': prod ? '"production"' : '"development"',
1717
},
18-
};
19-
20-
await esbuild.build({
21-
...commonEsbuildOptions,
2218
entryPoints: [
2319
{ out: 'background.bundle', in: 'src/background/index.ts' },
2420
{ out: 'options.bundle', in: 'src/options/index.tsx' },
25-
{ out: 'window.bundle', in: 'src/window/index.tsx' },
2621
{ out: 'remote.bundle', in: 'src/remote/index.tsx' },
2722
{ out: 'devpanel.bundle', in: 'src/devpanel/index.tsx' },
2823
{ out: 'devtools.bundle', in: 'src/devtools/index.ts' },
2924
{ out: 'content.bundle', in: 'src/contentScript/index.ts' },
3025
{ out: 'page.bundle', in: 'src/pageScript/index.ts' },
31-
...(prod ? [] : [{ out: 'pagewrap.bundle', in: 'src/pageScriptWrap.ts' }]),
3226
],
3327
loader: {
3428
'.woff2': 'file',
3529
},
3630
});
3731

38-
if (prod) {
39-
await esbuild.build({
40-
...commonEsbuildOptions,
41-
entryPoints: [{ out: 'pagewrap.bundle', in: 'src/pageScriptWrap.ts' }],
42-
loader: {
43-
'.js': 'text',
44-
},
45-
});
46-
}
47-
4832
console.log();
4933

5034
console.log('Creating HTML files...');
51-
const htmlFiles = ['devpanel', 'devtools', 'options', 'remote', 'window'];
35+
const htmlFiles = ['devpanel', 'devtools', 'options', 'remote'];
5236
for (const htmlFile of htmlFiles) {
5337
fs.writeFileSync(
5438
`dist/${htmlFile}.html`,

extension/chrome/manifest.json

+22-27
Original file line numberDiff line numberDiff line change
@@ -3,26 +3,20 @@
33
"name": "Redux DevTools",
44
"description": "Redux DevTools for debugging application's state changes.",
55
"homepage_url": "https://github.com/reduxjs/redux-devtools",
6-
"manifest_version": 2,
7-
"page_action": {
6+
"manifest_version": 3,
7+
"action": {
88
"default_icon": "img/logo/gray.png",
99
"default_title": "Redux DevTools",
10-
"default_popup": "window.html#popup"
10+
"default_popup": "devpanel.html#popup"
1111
},
1212
"commands": {
13-
"devtools-left": {
14-
"description": "DevTools window to left"
15-
},
16-
"devtools-right": {
17-
"description": "DevTools window to right"
18-
},
19-
"devtools-bottom": {
20-
"description": "DevTools window to bottom"
13+
"devtools-window": {
14+
"description": "DevTools window"
2115
},
2216
"devtools-remote": {
2317
"description": "Remote DevTools"
2418
},
25-
"_execute_page_action": {
19+
"_execute_action": {
2620
"suggested_key": {
2721
"default": "Ctrl+Shift+E"
2822
}
@@ -34,36 +28,37 @@
3428
"128": "img/logo/128x128.png"
3529
},
3630
"options_ui": {
37-
"page": "options.html",
38-
"chrome_style": true
31+
"page": "options.html"
3932
},
4033
"background": {
41-
"scripts": ["background.bundle.js"],
42-
"persistent": false
34+
"service_worker": "background.bundle.js"
4335
},
4436
"content_scripts": [
4537
{
4638
"matches": ["<all_urls>"],
4739
"exclude_globs": ["https://www.google*"],
48-
"js": ["content.bundle.js", "pagewrap.bundle.js"],
40+
"js": ["content.bundle.js"],
4941
"run_at": "document_start",
5042
"all_frames": true
43+
},
44+
{
45+
"matches": ["<all_urls>"],
46+
"exclude_globs": ["https://www.google*"],
47+
"js": ["page.bundle.js"],
48+
"run_at": "document_start",
49+
"all_frames": true,
50+
"world": "MAIN"
5151
}
5252
],
5353
"devtools_page": "devtools.html",
54-
"web_accessible_resources": ["page.bundle.js"],
5554
"externally_connectable": {
5655
"ids": ["*"]
5756
},
58-
"permissions": [
59-
"notifications",
60-
"contextMenus",
61-
"storage",
62-
"file:///*",
63-
"http://*/*",
64-
"https://*/*"
65-
],
66-
"content_security_policy": "script-src 'self' 'unsafe-eval'; object-src 'self'; style-src * 'unsafe-inline'; img-src 'self' data:;",
57+
"permissions": ["notifications", "contextMenus", "storage"],
58+
"host_permissions": ["file:///*", "http://*/*", "https://*/*"],
59+
"content_security_policy": {
60+
"extension_pages": "script-src 'self'; object-src 'self'; style-src * 'unsafe-inline'; img-src 'self' data:;"
61+
},
6762
"update_url": "https://clients2.google.com/service/update2/crx",
6863
"key": "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAsdJEPwY92xUACA9CcDBDBmbdbp8Ap3cKQ0DJTUuVQvqb4FQAv8RtKY3iUjGvdwuAcSJQIZwHXcP2aNDH3TiFik/NhRK2GRW8X3OZyTdkuDueABGP2KEX8q1WQDgjX/rPIinGYztUrvoICw/UerMPwNW62jwGoVU3YhAGf+15CgX2Y6a4tppnf/+1mPedKPidh0RsM+aJY98rX+r1SPAHPcGzMjocLkqcT75DZBXer8VQN14tOOzRCd6T6oy7qm7eWru8lJwcY66qMQvhk0osqEod2G3nA7aTWpmqPFS66VEiecP9PgZlp8gQdgZ3dFhA62exydlD55JuRhiMIR63yQIDAQAB"
6964
}

extension/firefox/manifest.json

+19-24
Original file line numberDiff line numberDiff line change
@@ -1,28 +1,22 @@
11
{
22
"version": "3.1.10",
33
"name": "Redux DevTools",
4-
"manifest_version": 2,
4+
"manifest_version": 3,
55
"description": "Redux Developer Tools for debugging application state changes.",
66
"homepage_url": "https://github.com/reduxjs/redux-devtools",
7-
"applications": {
7+
"browser_specific_settings": {
88
"gecko": {
99
1010
}
1111
},
12-
"page_action": {
12+
"action": {
1313
"default_icon": "img/logo/38x38.png",
1414
"default_title": "Redux DevTools",
15-
"default_popup": "window.html#popup"
15+
"default_popup": "devpanel.html#popup"
1616
},
1717
"commands": {
18-
"devtools-left": {
19-
"description": "DevTools window to left"
20-
},
21-
"devtools-right": {
22-
"description": "DevTools window to right"
23-
},
24-
"devtools-bottom": {
25-
"description": "DevTools window to bottom"
18+
"devtools-window": {
19+
"description": "DevTools window"
2620
},
2721
"devtools-remote": {
2822
"description": "Remote DevTools"
@@ -42,21 +36,22 @@
4236
"content_scripts": [
4337
{
4438
"matches": ["<all_urls>"],
45-
"js": ["content.bundle.js", "pagewrap.bundle.js"],
39+
"js": ["content.bundle.js"],
4640
"run_at": "document_start",
4741
"all_frames": true
42+
},
43+
{
44+
"matches": ["<all_urls>"],
45+
"js": ["page.bundle.js"],
46+
"run_at": "document_start",
47+
"all_frames": true,
48+
"world": "MAIN"
4849
}
4950
],
5051
"devtools_page": "devtools.html",
51-
"web_accessible_resources": ["page.bundle.js"],
52-
"permissions": [
53-
"notifications",
54-
"contextMenus",
55-
"tabs",
56-
"storage",
57-
"file:///*",
58-
"http://*/*",
59-
"https://*/*"
60-
],
61-
"content_security_policy": "script-src 'self'; object-src 'self'; img-src 'self' data:;"
52+
"permissions": ["notifications", "contextMenus", "tabs", "storage"],
53+
"host_permissions": ["file:///*", "http://*/*", "https://*/*"],
54+
"content_security_policy": {
55+
"extension_pages": "script-src 'self'; object-src 'self'; img-src 'self' data:;"
56+
}
6257
}

extension/src/app/Actions.tsx

+4-22
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ import {
1818
TopButtons,
1919
} from '@redux-devtools/app';
2020
import { GoBroadcast } from 'react-icons/go';
21-
import { MdBorderBottom, MdBorderLeft, MdBorderRight } from 'react-icons/md';
21+
import { MdOutlineWindow } from 'react-icons/md';
2222
import type { Position } from '../pageScript/api/openWindow';
2323
import type { SingleMessage } from '../background/store/apiMiddleware';
2424

@@ -98,31 +98,13 @@ class Actions extends Component<Props> {
9898
<DispatcherButton dispatcherIsOpen={this.props.dispatcherIsOpen} />
9999
)}
100100
<Divider />
101-
{!window.isElectron && position !== '#left' && (
102-
<Button
103-
onClick={() => {
104-
this.openWindow('left');
105-
}}
106-
>
107-
<MdBorderLeft />
108-
</Button>
109-
)}
110-
{!window.isElectron && position !== '#right' && (
111-
<Button
112-
onClick={() => {
113-
this.openWindow('right');
114-
}}
115-
>
116-
<MdBorderRight />
117-
</Button>
118-
)}
119-
{!window.isElectron && position !== '#bottom' && (
101+
{!window.isElectron && (
120102
<Button
121103
onClick={() => {
122-
this.openWindow('bottom');
104+
this.openWindow('window');
123105
}}
124106
>
125-
<MdBorderBottom />
107+
<MdOutlineWindow />
126108
</Button>
127109
)}
128110
{!window.isElectron && (

extension/src/background/contextMenus.ts

+5-11
Original file line numberDiff line numberDiff line change
@@ -2,29 +2,23 @@ import openDevToolsWindow, { DevToolsPosition } from './openWindow';
22

33
export function createMenu() {
44
const menus = [
5-
{ id: 'devtools-left', title: 'To left' },
6-
{ id: 'devtools-right', title: 'To right' },
7-
{ id: 'devtools-bottom', title: 'To bottom' },
8-
{
9-
id: 'devtools-panel',
10-
title: 'Open in a panel (enable in browser settings)',
11-
},
5+
{ id: 'devtools-window', title: 'Open in a window' },
126
{ id: 'devtools-remote', title: 'Open Remote DevTools' },
137
];
148

159
let shortcuts: { [commandName: string]: string | undefined } = {};
1610
chrome.commands.getAll((commands) => {
17-
commands.forEach(({ name, shortcut }) => {
11+
for (const { name, shortcut } of commands) {
1812
shortcuts[name!] = shortcut;
19-
});
13+
}
2014

21-
menus.forEach(({ id, title }) => {
15+
for (const { id, title } of menus) {
2216
chrome.contextMenus.create({
2317
id: id,
2418
title: title + (shortcuts[id] ? ' (' + shortcuts[id] + ')' : ''),
2519
contexts: ['all'],
2620
});
27-
});
21+
}
2822
});
2923
}
3024

extension/src/background/index.ts

+7-14
Original file line numberDiff line numberDiff line change
@@ -1,29 +1,22 @@
1-
import '../chromeApiMock';
2-
import { Store } from 'redux';
3-
import configureStore, { BackgroundAction } from './store/backgroundStore';
1+
import configureStore from './store/backgroundStore';
42
import openDevToolsWindow, { DevToolsPosition } from './openWindow';
53
import { createMenu, removeMenu } from './contextMenus';
6-
import syncOptions from '../options/syncOptions';
7-
import { BackgroundState } from './store/backgroundReducer';
8-
9-
declare global {
10-
interface Window {
11-
store: Store<BackgroundState, BackgroundAction>;
12-
}
13-
}
4+
import { getOptions } from '../options/syncOptions';
145

156
// Expose the extension's store globally to access it from the windows
167
// via chrome.runtime.getBackgroundPage
17-
window.store = configureStore();
8+
export const store = configureStore();
189

1910
// Listen for keyboard shortcuts
2011
chrome.commands.onCommand.addListener((shortcut) => {
2112
openDevToolsWindow(shortcut as DevToolsPosition);
2213
});
2314

24-
// Create the context menu when installed
15+
// Disable the action by default and create the context menu when installed
2516
chrome.runtime.onInstalled.addListener(() => {
26-
syncOptions().get((option) => {
17+
chrome.action.disable();
18+
19+
getOptions((option) => {
2720
if (option.showContextMenus) createMenu();
2821
});
2922
});

extension/src/background/logging.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import { LIFTED_ACTION } from '@redux-devtools/app';
2+
import { store } from './index';
23

34
export function getReport(
45
reportId: string,
@@ -24,7 +25,7 @@ export function getReport(
2425
.then((json) => {
2526
const { payload, preloadedState } = json;
2627
if (!payload) return;
27-
window.store.dispatch({
28+
store.dispatch({
2829
type: LIFTED_ACTION,
2930
message: 'IMPORT',
3031
state: JSON.stringify({ payload, preloadedState }),

0 commit comments

Comments
 (0)