mirror of
https://github.com/status-im/react-native.git
synced 2025-01-23 16:00:37 +00:00
d0219a0301
Summary: Replaces the existing `YellowBox` with a modern one. Here are the notable changes: - Sort warnings by recency (with most recent on top). - Group warnings by format string if present. - Present stack traces similar to RedBox. - Show status of loading source maps. - Support inspecting each occurrence of a warning. - Fixed a bunch of edge cases and race conditions. Reviewed By: TheSavior Differential Revision: D8345180 fbshipit-source-id: b9e10d526b262c3985bbea639ba2ea0e7cad5081
142 lines
3.1 KiB
JavaScript
142 lines
3.1 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow strict-local
|
|
* @format
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
const YellowBoxWarning = require('YellowBoxWarning');
|
|
|
|
import type {Category} from 'YellowBoxCategory';
|
|
|
|
export type Registry = Map<Category, $ReadOnlyArray<YellowBoxWarning>>;
|
|
|
|
export type Observer = (registry: Registry) => void;
|
|
|
|
export type Subscription = $ReadOnly<{|
|
|
unsubscribe: () => void,
|
|
|}>;
|
|
|
|
const observers: Set<{observer: Observer}> = new Set();
|
|
const ignorePatterns: Set<string> = new Set();
|
|
const registry: Registry = new Map();
|
|
|
|
let disabled = false;
|
|
let projection = new Map();
|
|
let updateTimeout = null;
|
|
|
|
function isWarningIgnored(warning: YellowBoxWarning): boolean {
|
|
for (const pattern of ignorePatterns) {
|
|
if (warning.message.content.includes(pattern)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function handleUpdate(): void {
|
|
projection = new Map();
|
|
if (!disabled) {
|
|
for (const [category, warnings] of registry) {
|
|
const filtered = warnings.filter(warning => !isWarningIgnored(warning));
|
|
if (filtered.length > 0) {
|
|
projection.set(category, filtered);
|
|
}
|
|
}
|
|
}
|
|
if (updateTimeout == null) {
|
|
updateTimeout = setImmediate(() => {
|
|
updateTimeout = null;
|
|
for (const {observer} of observers) {
|
|
observer(projection);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
const YellowBoxRegistry = {
|
|
add({
|
|
args,
|
|
framesToPop,
|
|
}: $ReadOnly<{|
|
|
args: $ReadOnlyArray<mixed>,
|
|
framesToPop: number,
|
|
|}>): void {
|
|
if (typeof args[0] === 'string' && args[0].startsWith('(ADVICE)')) {
|
|
return;
|
|
}
|
|
const {category, message, stack} = YellowBoxWarning.parse({
|
|
args,
|
|
framesToPop: framesToPop + 1,
|
|
});
|
|
|
|
let warnings = registry.get(category);
|
|
if (warnings == null) {
|
|
warnings = [];
|
|
}
|
|
warnings = [...warnings, new YellowBoxWarning(message, stack)];
|
|
|
|
registry.delete(category);
|
|
registry.set(category, warnings);
|
|
|
|
handleUpdate();
|
|
},
|
|
|
|
delete(category: Category): void {
|
|
if (registry.has(category)) {
|
|
registry.delete(category);
|
|
handleUpdate();
|
|
}
|
|
},
|
|
|
|
clear(): void {
|
|
if (registry.size > 0) {
|
|
registry.clear();
|
|
handleUpdate();
|
|
}
|
|
},
|
|
|
|
addIgnorePatterns(patterns: $ReadOnlyArray<string>): void {
|
|
const newPatterns = patterns.filter(
|
|
pattern => !ignorePatterns.has(pattern),
|
|
);
|
|
if (newPatterns.length === 0) {
|
|
return;
|
|
}
|
|
for (const pattern of newPatterns) {
|
|
ignorePatterns.add(pattern);
|
|
}
|
|
handleUpdate();
|
|
},
|
|
|
|
setDisabled(value: boolean): void {
|
|
if (value === disabled) {
|
|
return;
|
|
}
|
|
disabled = value;
|
|
handleUpdate();
|
|
},
|
|
|
|
isDisabled(): boolean {
|
|
return disabled;
|
|
},
|
|
|
|
observe(observer: Observer): Subscription {
|
|
const subscription = {observer};
|
|
observers.add(subscription);
|
|
observer(projection);
|
|
return {
|
|
unsubscribe(): void {
|
|
observers.delete(subscription);
|
|
},
|
|
};
|
|
},
|
|
};
|
|
|
|
module.exports = YellowBoxRegistry;
|