mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
142fd27bf6
The `useOpenResource` hook is now used to open links. Currently, the `<>` icon for the component stacks and the link in the bottom of the components stack. But it'll also be used for many new links like stacks. If this new option is configured, and this is a local file then this is opened directly in the external editor. Otherwise it fallbacks to open in the Sources tab or whatever the standalone or inline is configured to use. <img width="453" height="252" alt="Screenshot 2025-07-24 at 4 09 09 PM" src="https://github.com/user-attachments/assets/04cae170-dd30-4485-a9ee-e8fe1612978e" /> I prominently surface this option in the Source pane to make it discoverable. <img width="588" height="144" alt="Screenshot 2025-07-24 at 4 03 48 PM" src="https://github.com/user-attachments/assets/0f3a7da9-2fae-4b5b-90ec-769c5a9c5361" /> When this is configured, the "Open in Editor" is hidden since that's just the default. I plan on deprecating this button to avoid having the two buttons going forward. Notably there's one exception where this doesn't work. When you click an Action or Event listener it takes you to the Sources tab and you have to open in editor from there. That's because we use the `inspect()` mechanism instead of extracting the source location. That's because we can't do the "throw trick" since these can have side-effects. The Chrome debugger protocol would solve this but it pops up an annoying dialog. We could maybe only attach the debugger only for that case. Especially if the dialog disappears before you focus on the browser again.
415 lines
11 KiB
JavaScript
415 lines
11 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
import {createElement} from 'react';
|
|
import {flushSync} from 'react-dom';
|
|
import {createRoot} from 'react-dom/client';
|
|
import Bridge from 'react-devtools-shared/src/bridge';
|
|
import Store from 'react-devtools-shared/src/devtools/store';
|
|
import {getSavedComponentFilters} from 'react-devtools-shared/src/utils';
|
|
import {registerDevToolsEventLogger} from 'react-devtools-shared/src/registerDevToolsEventLogger';
|
|
import {Server} from 'ws';
|
|
import {join} from 'path';
|
|
import {readFileSync} from 'fs';
|
|
import DevTools from 'react-devtools-shared/src/devtools/views/DevTools';
|
|
import {doesFilePathExist, launchEditor} from './editor';
|
|
import {
|
|
__DEBUG__,
|
|
LOCAL_STORAGE_DEFAULT_TAB_KEY,
|
|
} from 'react-devtools-shared/src/constants';
|
|
import {localStorageSetItem} from 'react-devtools-shared/src/storage';
|
|
|
|
import type {FrontendBridge} from 'react-devtools-shared/src/bridge';
|
|
import type {ReactFunctionLocation, ReactCallSite} from 'shared/ReactTypes';
|
|
|
|
export type StatusTypes = 'server-connected' | 'devtools-connected' | 'error';
|
|
export type StatusListener = (message: string, status: StatusTypes) => void;
|
|
export type OnDisconnectedCallback = () => void;
|
|
|
|
let node: HTMLElement = ((null: any): HTMLElement);
|
|
let nodeWaitingToConnectHTML: string = '';
|
|
let projectRoots: Array<string> = [];
|
|
let statusListener: StatusListener = (
|
|
message: string,
|
|
status?: StatusTypes,
|
|
) => {};
|
|
let disconnectedCallback: OnDisconnectedCallback = () => {};
|
|
|
|
// TODO (Webpack 5) Hopefully we can remove this prop after the Webpack 5 migration.
|
|
function hookNamesModuleLoaderFunction() {
|
|
return import(
|
|
/* webpackChunkName: 'parseHookNames' */ 'react-devtools-shared/src/hooks/parseHookNames'
|
|
);
|
|
}
|
|
|
|
function setContentDOMNode(value: HTMLElement): typeof DevtoolsUI {
|
|
node = value;
|
|
|
|
// Save so we can restore the exact waiting message between sessions.
|
|
nodeWaitingToConnectHTML = node.innerHTML;
|
|
|
|
return DevtoolsUI;
|
|
}
|
|
|
|
function setProjectRoots(value: Array<string>) {
|
|
projectRoots = value;
|
|
}
|
|
|
|
function setStatusListener(value: StatusListener): typeof DevtoolsUI {
|
|
statusListener = value;
|
|
return DevtoolsUI;
|
|
}
|
|
|
|
function setDisconnectedCallback(
|
|
value: OnDisconnectedCallback,
|
|
): typeof DevtoolsUI {
|
|
disconnectedCallback = value;
|
|
return DevtoolsUI;
|
|
}
|
|
|
|
let bridge: FrontendBridge | null = null;
|
|
let store: Store | null = null;
|
|
let root = null;
|
|
|
|
const log = (...args: Array<mixed>) => console.log('[React DevTools]', ...args);
|
|
log.warn = (...args: Array<mixed>) => console.warn('[React DevTools]', ...args);
|
|
log.error = (...args: Array<mixed>) =>
|
|
console.error('[React DevTools]', ...args);
|
|
|
|
function debug(methodName: string, ...args: Array<mixed>) {
|
|
if (__DEBUG__) {
|
|
console.log(
|
|
`%c[core/standalone] %c${methodName}`,
|
|
'color: teal; font-weight: bold;',
|
|
'font-weight: bold;',
|
|
...args,
|
|
);
|
|
}
|
|
}
|
|
|
|
function safeUnmount() {
|
|
flushSync(() => {
|
|
if (root !== null) {
|
|
root.unmount();
|
|
root = null;
|
|
}
|
|
});
|
|
}
|
|
|
|
function reload() {
|
|
safeUnmount();
|
|
|
|
node.innerHTML = '';
|
|
|
|
setTimeout(() => {
|
|
root = createRoot(node);
|
|
root.render(
|
|
createElement(DevTools, {
|
|
bridge: ((bridge: any): FrontendBridge),
|
|
canViewElementSourceFunction,
|
|
hookNamesModuleLoaderFunction,
|
|
showTabBar: true,
|
|
store: ((store: any): Store),
|
|
warnIfLegacyBackendDetected: true,
|
|
viewElementSourceFunction,
|
|
fetchFileWithCaching,
|
|
}),
|
|
);
|
|
}, 100);
|
|
}
|
|
|
|
const resourceCache: Map<string, string> = new Map();
|
|
|
|
// As a potential improvement, this should be done from the backend of RDT.
|
|
// Browser extension is doing this via exchanging messages
|
|
// between devtools_page and dedicated content script for it, see `fetchFileWithCaching.js`.
|
|
async function fetchFileWithCaching(url: string) {
|
|
if (resourceCache.has(url)) {
|
|
return Promise.resolve(resourceCache.get(url));
|
|
}
|
|
|
|
return fetch(url)
|
|
.then(data => data.text())
|
|
.then(content => {
|
|
resourceCache.set(url, content);
|
|
|
|
return content;
|
|
});
|
|
}
|
|
|
|
function canViewElementSourceFunction(
|
|
_source: ReactFunctionLocation | ReactCallSite,
|
|
symbolicatedSource: ReactFunctionLocation | ReactCallSite | null,
|
|
): boolean {
|
|
if (symbolicatedSource == null) {
|
|
return false;
|
|
}
|
|
const [, sourceURL, ,] = symbolicatedSource;
|
|
|
|
return doesFilePathExist(sourceURL, projectRoots);
|
|
}
|
|
|
|
function viewElementSourceFunction(
|
|
_source: ReactFunctionLocation | ReactCallSite,
|
|
symbolicatedSource: ReactFunctionLocation | ReactCallSite | null,
|
|
): void {
|
|
if (symbolicatedSource == null) {
|
|
return;
|
|
}
|
|
|
|
const [, sourceURL, line] = symbolicatedSource;
|
|
launchEditor(sourceURL, line, projectRoots);
|
|
}
|
|
|
|
function onDisconnected() {
|
|
safeUnmount();
|
|
|
|
node.innerHTML = nodeWaitingToConnectHTML;
|
|
|
|
disconnectedCallback();
|
|
}
|
|
|
|
function onError({code, message}: $FlowFixMe) {
|
|
safeUnmount();
|
|
|
|
if (code === 'EADDRINUSE') {
|
|
node.innerHTML = `
|
|
<div class="box">
|
|
<div class="box-header">
|
|
Another instance of DevTools is running.
|
|
</div>
|
|
<div class="box-content">
|
|
Only one copy of DevTools can be used at a time.
|
|
</div>
|
|
</div>
|
|
`;
|
|
} else {
|
|
node.innerHTML = `
|
|
<div class="box">
|
|
<div class="box-header">
|
|
Unknown error
|
|
</div>
|
|
<div class="box-content">
|
|
${message}
|
|
</div>
|
|
</div>
|
|
`;
|
|
}
|
|
}
|
|
|
|
function openProfiler() {
|
|
// Mocked up bridge and store to allow the DevTools to be rendered
|
|
bridge = new Bridge({listen: () => {}, send: () => {}});
|
|
store = new Store(bridge, {});
|
|
|
|
// Ensure the Profiler tab is shown initially.
|
|
localStorageSetItem(
|
|
LOCAL_STORAGE_DEFAULT_TAB_KEY,
|
|
JSON.stringify('profiler'),
|
|
);
|
|
|
|
reload();
|
|
}
|
|
|
|
function initialize(socket: WebSocket) {
|
|
const listeners = [];
|
|
socket.onmessage = event => {
|
|
let data;
|
|
try {
|
|
if (typeof event.data === 'string') {
|
|
data = JSON.parse(event.data);
|
|
|
|
if (__DEBUG__) {
|
|
debug('WebSocket.onmessage', data);
|
|
}
|
|
} else {
|
|
throw Error();
|
|
}
|
|
} catch (e) {
|
|
log.error('Failed to parse JSON', event.data);
|
|
return;
|
|
}
|
|
listeners.forEach(fn => {
|
|
try {
|
|
fn(data);
|
|
} catch (error) {
|
|
log.error('Error calling listener', data);
|
|
throw error;
|
|
}
|
|
});
|
|
};
|
|
|
|
bridge = new Bridge({
|
|
listen(fn) {
|
|
listeners.push(fn);
|
|
return () => {
|
|
const index = listeners.indexOf(fn);
|
|
if (index >= 0) {
|
|
listeners.splice(index, 1);
|
|
}
|
|
};
|
|
},
|
|
send(event: string, payload: any, transferable?: Array<any>) {
|
|
if (socket.readyState === socket.OPEN) {
|
|
socket.send(JSON.stringify({event, payload}));
|
|
}
|
|
},
|
|
});
|
|
((bridge: any): FrontendBridge).addListener('shutdown', () => {
|
|
socket.close();
|
|
});
|
|
|
|
// $FlowFixMe[incompatible-call] found when upgrading Flow
|
|
store = new Store(bridge, {
|
|
checkBridgeProtocolCompatibility: true,
|
|
supportsTraceUpdates: true,
|
|
supportsClickToInspect: true,
|
|
});
|
|
|
|
log('Connected');
|
|
statusListener('DevTools initialized.', 'devtools-connected');
|
|
reload();
|
|
}
|
|
|
|
let startServerTimeoutID: TimeoutID | null = null;
|
|
|
|
function connectToSocket(socket: WebSocket): {close(): void} {
|
|
socket.onerror = err => {
|
|
onDisconnected();
|
|
log.error('Error with websocket connection', err);
|
|
};
|
|
socket.onclose = () => {
|
|
onDisconnected();
|
|
log('Connection to RN closed');
|
|
};
|
|
initialize(socket);
|
|
|
|
return {
|
|
close: function () {
|
|
onDisconnected();
|
|
},
|
|
};
|
|
}
|
|
|
|
type ServerOptions = {
|
|
key?: string,
|
|
cert?: string,
|
|
};
|
|
|
|
type LoggerOptions = {
|
|
surface?: ?string,
|
|
};
|
|
|
|
function startServer(
|
|
port: number = 8097,
|
|
host: string = 'localhost',
|
|
httpsOptions?: ServerOptions,
|
|
loggerOptions?: LoggerOptions,
|
|
): {close(): void} {
|
|
registerDevToolsEventLogger(loggerOptions?.surface ?? 'standalone');
|
|
|
|
const useHttps = !!httpsOptions;
|
|
const httpServer = useHttps
|
|
? require('https').createServer(httpsOptions)
|
|
: require('http').createServer();
|
|
const server = new Server({server: httpServer, maxPayload: 1e9});
|
|
let connected: WebSocket | null = null;
|
|
server.on('connection', (socket: WebSocket) => {
|
|
if (connected !== null) {
|
|
connected.close();
|
|
log.warn(
|
|
'Only one connection allowed at a time.',
|
|
'Closing the previous connection',
|
|
);
|
|
}
|
|
connected = socket;
|
|
socket.onerror = error => {
|
|
connected = null;
|
|
onDisconnected();
|
|
log.error('Error with websocket connection', error);
|
|
};
|
|
socket.onclose = () => {
|
|
connected = null;
|
|
onDisconnected();
|
|
log('Connection to RN closed');
|
|
};
|
|
initialize(socket);
|
|
});
|
|
|
|
server.on('error', (event: $FlowFixMe) => {
|
|
onError(event);
|
|
log.error('Failed to start the DevTools server', event);
|
|
startServerTimeoutID = setTimeout(() => startServer(port), 1000);
|
|
});
|
|
|
|
httpServer.on('request', (request: $FlowFixMe, response: $FlowFixMe) => {
|
|
// Serve a file that immediately sets up the connection.
|
|
const backendFile = readFileSync(join(__dirname, 'backend.js'));
|
|
|
|
// The renderer interface doesn't read saved component filters directly,
|
|
// because they are generally stored in localStorage within the context of the extension.
|
|
// Because of this it relies on the extension to pass filters, so include them wth the response here.
|
|
// This will ensure that saved filters are shared across different web pages.
|
|
const savedPreferencesString = `
|
|
window.__REACT_DEVTOOLS_COMPONENT_FILTERS__ = ${JSON.stringify(
|
|
getSavedComponentFilters(),
|
|
)};`;
|
|
|
|
response.end(
|
|
savedPreferencesString +
|
|
'\n;' +
|
|
backendFile.toString() +
|
|
'\n;' +
|
|
'ReactDevToolsBackend.initialize();' +
|
|
'\n' +
|
|
`ReactDevToolsBackend.connectToDevTools({port: ${port}, host: '${host}', useHttps: ${
|
|
useHttps ? 'true' : 'false'
|
|
}});
|
|
`,
|
|
);
|
|
});
|
|
|
|
httpServer.on('error', (event: $FlowFixMe) => {
|
|
onError(event);
|
|
statusListener('Failed to start the server.', 'error');
|
|
startServerTimeoutID = setTimeout(() => startServer(port), 1000);
|
|
});
|
|
|
|
httpServer.listen(port, () => {
|
|
statusListener(
|
|
'The server is listening on the port ' + port + '.',
|
|
'server-connected',
|
|
);
|
|
});
|
|
|
|
return {
|
|
close: function () {
|
|
connected = null;
|
|
onDisconnected();
|
|
if (startServerTimeoutID !== null) {
|
|
clearTimeout(startServerTimeoutID);
|
|
}
|
|
server.close();
|
|
httpServer.close();
|
|
},
|
|
};
|
|
}
|
|
|
|
const DevtoolsUI = {
|
|
connectToSocket,
|
|
setContentDOMNode,
|
|
setProjectRoots,
|
|
setStatusListener,
|
|
setDisconnectedCallback,
|
|
startServer,
|
|
openProfiler,
|
|
};
|
|
|
|
export default DevtoolsUI;
|