/** * Copyright (c) 2015-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. * * @providesModule YellowBox * @flow */ 'use strict'; const EventEmitter = require('EventEmitter'); const Platform = require('Platform'); const React = require('React'); const StyleSheet = require('StyleSheet'); const symbolicateStackTrace = require('symbolicateStackTrace'); const parseErrorStack = require('parseErrorStack'); import type EmitterSubscription from 'EmitterSubscription'; import type {StackFrame} from 'parseErrorStack'; type WarningInfo = { count: number; stacktrace: Array; }; const _warningEmitter = new EventEmitter(); const _warningMap: Map = new Map(); /** * YellowBox renders warnings at the bottom of the app being developed. * * Warnings help guard against subtle yet significant issues that can impact the * quality of the app. This "in your face" style of warning allows developers to * notice and correct these issues as quickly as possible. * * By default, the warning box is enabled in `__DEV__`. Set the following flag * to disable it (and call `console.warn` to update any rendered ): * * console.disableYellowBox = true; * console.warn('YellowBox is disabled.'); * * Warnings can be ignored programmatically by setting the array: * * console.ignoredYellowBox = ['Warning: ...']; * * Strings in `console.ignoredYellowBox` can be a prefix of the warning that * should be ignored. */ if (__DEV__) { const {error, warn} = console; console.error = function() { error.apply(console, arguments); // Show yellow box for the `warning` module. if (typeof arguments[0] === 'string' && arguments[0].startsWith('Warning: ')) { updateWarningMap.apply(null, arguments); } }; console.warn = function() { warn.apply(console, arguments); updateWarningMap.apply(null, arguments); }; } /** * Simple function for formatting strings. * * Replaces placeholders with values passed as extra arguments * * @param {string} format the base string * @param ...args the values to insert * @return {string} the replaced string */ function sprintf(format, ...args) { var index = 0; return format.replace(/%s/g, match => args[index++]); } function updateWarningMap(format, ...args): void { const stringifySafe = require('stringifySafe'); format = String(format); const argCount = (format.match(/%s/g) || []).length; const warning = [ sprintf(format, ...args.slice(0, argCount)), ...args.slice(argCount).map(stringifySafe), ].join(' '); var warningInfo = _warningMap.get(warning); if (warningInfo) { warningInfo.count += 1; } else { warningInfo = {count: 1, stacktrace: []}; } _warningMap.set(warning, warningInfo); _warningEmitter.emit('warning', _warningMap); var error : any = new Error(); error.framesToPop = 2; symbolicateStackTrace(parseErrorStack(error)).then(stack => { warningInfo = _warningMap.get(warning); if (warningInfo) { warningInfo.stacktrace = stack; _warningMap.set(warning, warningInfo); _warningEmitter.emit('warning', _warningMap); } }, () => { /* Do nothing when can't load source map */ }); } function isWarningIgnored(warning: string): boolean { return ( Array.isArray(console.ignoredYellowBox) && console.ignoredYellowBox.some( ignorePrefix => warning.startsWith(ignorePrefix) ) ); } const WarningRow = ({count, warning, onPress}) => { const Text = require('Text'); const TouchableHighlight = require('TouchableHighlight'); const View = require('View'); const countText = count > 1 ? {'(' + count + ') '} : null; return ( {countText} {warning} ); }; type StackRowProps = { frame: StackFrame }; const StackRow = ({frame}: StackRowProps) => { const Text = require('Text'); const fileParts = frame.file.split('/'); const fileName = fileParts[fileParts.length - 1]; return ( {`${fileName}:${frame.lineNumber}`} ); }; const WarningInspector = ({ warningInfo, warning, stacktraceVisible, onClose, onDismiss, onDismissAll, toggleStacktrace, }) => { const ScrollView = require('ScrollView'); const Text = require('Text'); const TouchableHighlight = require('TouchableHighlight'); const View = require('View'); const {count, stacktrace} = warningInfo || {}; const countSentence = 'Warning encountered ' + count + ' time' + (count - 1 ? 's' : '') + '.'; let stacktraceList; if (stacktraceVisible && stacktrace) { stacktraceList = ( {stacktrace.map((frame, ii) => )} ); } return ( {countSentence} {stacktraceVisible ? 'Hide' : 'Show'} stacktrace {stacktraceList} {warning} Dismiss Dismiss All ); }; class YellowBox extends React.Component { state: { stacktraceVisible: boolean; inspecting: ?string; warningMap: Map; }; _listener: ?EmitterSubscription; dismissWarning: (warning: ?string) => void; constructor(props: mixed, context: mixed) { super(props, context); this.state = { inspecting: null, stacktraceVisible: false, warningMap: _warningMap, }; this.dismissWarning = warning => { const {inspecting, warningMap} = this.state; if (warning) { warningMap.delete(warning); } else { warningMap.clear(); } this.setState({ inspecting: (warning && inspecting !== warning) ? inspecting : null, warningMap, }); }; } componentDidMount() { let scheduled = null; this._listener = _warningEmitter.addListener('warning', warningMap => { // Use `setImmediate` because warnings often happen during render, but // state cannot be set while rendering. scheduled = scheduled || setImmediate(() => { scheduled = null; this.setState({ warningMap, }); }); }); } componentWillUnmount() { if (this._listener) { this._listener.remove(); } } render() { if (console.disableYellowBox || this.state.warningMap.size === 0) { return null; } const ScrollView = require('ScrollView'); const View = require('View'); const {inspecting, stacktraceVisible} = this.state; const inspector = inspecting !== null ? this.setState({inspecting: null})} onDismiss={() => this.dismissWarning(inspecting)} onDismissAll={() => this.dismissWarning(null)} toggleStacktrace={() => this.setState({stacktraceVisible: !stacktraceVisible})} /> : null; const rows = []; this.state.warningMap.forEach((warningInfo, warning) => { if (!isWarningIgnored(warning)) { rows.push( this.setState({inspecting: warning})} onDismiss={() => this.dismissWarning(warning)} /> ); } }); const listStyle = [ styles.list, // Additional `0.4` so the 5th row can peek into view. {height: Math.min(rows.length, 4.4) * (rowGutter + rowHeight)}, ]; return ( {rows} {inspector} ); } } const backgroundColor = opacity => 'rgba(250, 186, 48, ' + opacity + ')'; const textColor = 'white'; const rowGutter = 1; const rowHeight = 46; var styles = StyleSheet.create({ fullScreen: { backgroundColor: 'transparent', position: 'absolute', left: 0, right: 0, top: 0, bottom: 0, }, inspector: { backgroundColor: backgroundColor(0.95), flex: 1, }, inspectorContainer: { flex: 1, }, inspectorButtons: { flexDirection: 'row', position: 'absolute', left: 0, right: 0, bottom: 0, }, inspectorButton: { flex: 1, padding: 22, backgroundColor: backgroundColor(1), }, stacktraceButton: { flex: 1, padding: 5, }, inspectorButtonText: { color: textColor, fontSize: 14, opacity: 0.8, textAlign: 'center', }, inspectorContent: { flex: 1, paddingTop: 5, }, inspectorCount: { padding: 15, paddingBottom: 0, }, inspectorCountText: { color: textColor, fontSize: 14, }, inspectorWarning: { paddingHorizontal: 15, }, inspectorWarningText: { color: textColor, fontSize: 16, fontWeight: '600', }, list: { backgroundColor: 'transparent', position: 'absolute', left: 0, right: 0, bottom: 0, }, listRow: { position: 'relative', backgroundColor: backgroundColor(0.95), flex: 1, height: rowHeight, marginTop: rowGutter, }, listRowContent: { flex: 1, }, listRowCount: { color: 'rgba(255, 255, 255, 0.5)', }, listRowText: { color: textColor, position: 'absolute', left: 0, top: Platform.OS === 'android' ? 5 : 7, marginLeft: 15, marginRight: 15, }, }); module.exports = YellowBox;