/** @flow */ import * as React from 'react'; import {forwardRef} from 'react'; import Bridge from 'react-devtools-shared/src/bridge'; import Store from 'react-devtools-shared/src/devtools/store'; import DevTools from 'react-devtools-shared/src/devtools/views/DevTools'; import { getAppendComponentStack, getBreakOnConsoleErrors, getSavedComponentFilters, getShowInlineWarningsAndErrors, } from 'react-devtools-shared/src/utils'; import { MESSAGE_TYPE_GET_SAVED_PREFERENCES, MESSAGE_TYPE_SAVED_PREFERENCES, } from './constants'; import type {Wall} from 'react-devtools-shared/src/types'; import type {FrontendBridge} from 'react-devtools-shared/src/bridge'; import type {Props} from 'react-devtools-shared/src/devtools/views/DevTools'; export function createStore(bridge: FrontendBridge): Store { return new Store(bridge, { checkBridgeProtocolCompatibility: true, supportsTraceUpdates: true, }); } export function createBridge( contentWindow: window, wall?: Wall, ): FrontendBridge { if (wall == null) { wall = { listen(fn) { const onMessage = ({data}) => { fn(data); }; window.addEventListener('message', onMessage); return () => { window.removeEventListener('message', onMessage); }; }, send(event: string, payload: any, transferable?: Array) { contentWindow.postMessage({event, payload}, '*', transferable); }, }; } return (new Bridge(wall): FrontendBridge); } export function initialize( contentWindow: window, { bridge, store, }: {| bridge?: FrontendBridge, store?: Store, |} = {}, ): React.AbstractComponent { const onGetSavedPreferencesMessage = ({data, source}) => { if (source === 'react-devtools-content-script') { // Ignore messages from the DevTools browser extension. } switch (data.type) { case MESSAGE_TYPE_GET_SAVED_PREFERENCES: // This is the only message we're listening for, // so it's safe to cleanup after we've received it. window.removeEventListener('message', onGetSavedPreferencesMessage); // The renderer interface can't read saved preferences directly, // because they are stored in localStorage within the context of the extension. // Instead it relies on the extension to pass them through. contentWindow.postMessage( { type: MESSAGE_TYPE_SAVED_PREFERENCES, appendComponentStack: getAppendComponentStack(), breakOnConsoleErrors: getBreakOnConsoleErrors(), componentFilters: getSavedComponentFilters(), showInlineWarningsAndErrors: getShowInlineWarningsAndErrors(), }, '*', ); break; default: break; } }; window.addEventListener('message', onGetSavedPreferencesMessage); if (bridge == null) { bridge = createBridge(contentWindow); } if (store == null) { store = createStore(bridge); } const ForwardRef = forwardRef((props, ref) => ( )); ForwardRef.displayName = 'DevTools'; return ForwardRef; }