/** * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * * @format * @flow */ 'use strict'; const Dimensions = require('Dimensions'); const InspectorOverlay = require('InspectorOverlay'); const InspectorPanel = require('InspectorPanel'); const Platform = require('Platform'); const React = require('React'); const ReactNative = require('ReactNative'); const StyleSheet = require('StyleSheet'); const Touchable = require('Touchable'); const UIManager = require('UIManager'); const View = require('View'); /* $FlowFixMe(>=0.54.0 site=react_native_oss) This comment suppresses an error * found when Flow v0.54 was deployed. To see the error delete this comment and * run Flow. */ const emptyObject = require('fbjs/lib/emptyObject'); const invariant = require('fbjs/lib/invariant'); export type ReactRenderer = { getInspectorDataForViewTag: (viewTag: number) => Object, }; const hook = window.__REACT_DEVTOOLS_GLOBAL_HOOK__; const renderers = findRenderers(); // required for devtools to be able to edit react native styles hook.resolveRNStyle = require('flattenStyle'); function findRenderers(): $ReadOnlyArray { const allRenderers = Object.keys(hook._renderers).map( key => hook._renderers[key], ); invariant( allRenderers.length >= 1, 'Expected to find at least one React Native renderer on DevTools hook.', ); return allRenderers; } function getInspectorDataForViewTag(touchedViewTag: number) { for (let i = 0; i < renderers.length; i++) { const renderer = renderers[i]; const inspectorData = renderer.getInspectorDataForViewTag(touchedViewTag); if (inspectorData.hierarchy.length > 0) { return inspectorData; } } throw new Error('Expected to find at least one React renderer.'); } class Inspector extends React.Component< { inspectedViewTag: ?number, onRequestRerenderApp: (callback: (tag: ?number) => void) => void, }, { devtoolsAgent: ?Object, hierarchy: any, panelPos: string, inspecting: boolean, selection: ?number, perfing: boolean, inspected: any, inspectedViewTag: any, networking: boolean, }, > { _subs: ?Array<() => void>; constructor(props: Object) { super(props); this.state = { devtoolsAgent: null, hierarchy: null, panelPos: 'bottom', inspecting: true, perfing: false, inspected: null, selection: null, inspectedViewTag: this.props.inspectedViewTag, networking: false, }; } componentDidMount() { hook.on('react-devtools', this.attachToDevtools); // if devtools is already started if (hook.reactDevtoolsAgent) { this.attachToDevtools(hook.reactDevtoolsAgent); } } componentWillUnmount() { if (this._subs) { this._subs.map(fn => fn()); } hook.off('react-devtools', this.attachToDevtools); } UNSAFE_componentWillReceiveProps(newProps: Object) { this.setState({inspectedViewTag: newProps.inspectedViewTag}); } attachToDevtools = (agent: Object) => { let _hideWait = null; const hlSub = agent.sub('highlight', ({node, name, props}) => { /* $FlowFixMe(>=0.63.0 site=react_native_fb) This comment suppresses an * error found when Flow v0.63 was deployed. To see the error delete this * comment and run Flow. */ clearTimeout(_hideWait); if (typeof node !== 'number') { // Fiber node = ReactNative.findNodeHandle(node); } UIManager.measure(node, (x, y, width, height, left, top) => { this.setState({ hierarchy: [], inspected: { frame: {left, top, width, height}, style: props ? props.style : emptyObject, }, }); }); }); const hideSub = agent.sub('hideHighlight', () => { if (this.state.inspected === null) { return; } // we wait to actually hide in order to avoid flicker _hideWait = setTimeout(() => { this.setState({ inspected: null, }); }, 100); }); this._subs = [hlSub, hideSub]; agent.on('shutdown', () => { this.setState({devtoolsAgent: null}); this._subs = null; }); this.setState({ devtoolsAgent: agent, }); }; setSelection(i: number) { const hierarchyItem = this.state.hierarchy[i]; // we pass in ReactNative.findNodeHandle as the method is injected const {measure, props, source} = hierarchyItem.getInspectorData( ReactNative.findNodeHandle, ); measure((x, y, width, height, left, top) => { this.setState({ inspected: { frame: {left, top, width, height}, style: props.style, source, }, selection: i, }); }); } onTouchViewTag(touchedViewTag: number, frame: Object, pointerY: number) { // Most likely the touched instance is a native wrapper (like RCTView) // which is not very interesting. Most likely user wants a composite // instance that contains it (like View) const {hierarchy, props, selection, source} = getInspectorDataForViewTag( touchedViewTag, ); if (this.state.devtoolsAgent) { // Skip host leafs const offsetFromLeaf = hierarchy.length - 1 - selection; this.state.devtoolsAgent.selectFromDOMNode( touchedViewTag, true, offsetFromLeaf, ); } this.setState({ panelPos: pointerY > Dimensions.get('window').height / 2 ? 'top' : 'bottom', selection, hierarchy, inspected: { style: props.style, frame, source, }, }); } setPerfing(val: boolean) { this.setState({ perfing: val, inspecting: false, inspected: null, networking: false, }); } setInspecting(val: boolean) { this.setState({ inspecting: val, inspected: null, }); } setTouchTargeting(val: boolean) { Touchable.TOUCH_TARGET_DEBUG = val; this.props.onRequestRerenderApp(inspectedViewTag => { this.setState({inspectedViewTag}); }); } setNetworking(val: boolean) { this.setState({ networking: val, perfing: false, inspecting: false, inspected: null, }); } render() { const panelContainerStyle = this.state.panelPos === 'bottom' ? {bottom: 0} : {top: Platform.OS === 'ios' ? 20 : 0}; return ( {this.state.inspecting && ( )} ); } } const styles = StyleSheet.create({ container: { position: 'absolute', backgroundColor: 'transparent', top: 0, left: 0, right: 0, bottom: 0, }, panelContainer: { position: 'absolute', left: 0, right: 0, }, }); module.exports = Inspector;