Files
react-native/packages/rn-tester/js/examples/Performance/PerformanceApiExample.js
Rubén Norte 9962add377 Remove redudant fields from ReactNativeStartupTiming (#53711)
Summary:
Pull Request resolved: https://github.com/facebook/react-native/pull/53711

Changelog: [internal]

This removes some fields that contain the same time as `endTime`, which is confusing when documenting them.

Reviewed By: christophpurrer

Differential Revision: D82112473

fbshipit-source-id: 461e2b4b495ae641dcb3233874360a4f7b90dabf
2025-09-11 06:09:29 -07:00

265 lines
7.8 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 strict-local
* @format
*/
import type {RNTesterModuleExample} from '../../types/RNTesterTypes';
import type Performance from 'react-native/src/private/webapis/performance/Performance';
import RNTesterText from '../../components/RNTesterText';
import * as React from 'react';
import {useEffect} from 'react';
import {Button, StyleSheet, View} from 'react-native';
declare var performance: Performance;
const {useState, useCallback} = React;
function MemoryExample(): React.Node {
// Memory API testing
const [memoryInfo, setMemoryInfo] =
useState<?typeof performance.memory>(null);
const onGetMemoryInfo = useCallback(() => {
// performance.memory is not included in bom.js yet.
// Once we release the change in flow this can be removed.
setMemoryInfo(performance.memory);
}, []);
return (
<View style={styles.container}>
<Button onPress={onGetMemoryInfo} title="Click to update memory info" />
<View>
<RNTesterText>
{`jsHeapSizeLimit: ${String(memoryInfo?.jsHeapSizeLimit)} bytes`}
</RNTesterText>
<RNTesterText>
{`totalJSHeapSize: ${String(memoryInfo?.totalJSHeapSize)} bytes`}
</RNTesterText>
<RNTesterText>
{`usedJSHeapSize: ${String(memoryInfo?.usedJSHeapSize)} bytes`}
</RNTesterText>
</View>
</View>
);
}
function StartupTimingExample(): React.Node {
// React Startup Timing API testing
const [startUpTiming, setStartUpTiming] =
useState<?typeof performance.rnStartupTiming>(null);
const onGetStartupTiming = useCallback(() => {
// performance.reactNativeStartupTiming is not included in bom.js yet.
// Once we release the change in flow this can be removed.
setStartUpTiming(performance.rnStartupTiming);
}, []);
return (
<View style={styles.container}>
<Button
onPress={onGetStartupTiming}
title="Click to update React startup timing"
/>
<View>
<RNTesterText>{`startTime: ${String(startUpTiming?.startTime)} ms`}</RNTesterText>
<RNTesterText>{`initializeRuntimeStart: ${String(
startUpTiming?.initializeRuntimeStart,
)} ms`}</RNTesterText>
<RNTesterText>
{`executeJavaScriptBundleEntryPointStart: ${String(
startUpTiming?.executeJavaScriptBundleEntryPointStart,
)} ms`}
</RNTesterText>
<RNTesterText>{`endTime: ${String(startUpTiming?.endTime)} ms`}</RNTesterText>
</View>
</View>
);
}
function PerformanceObserverUserTimingExample(): React.Node {
const [entries, setEntries] = useState<$ReadOnlyArray<PerformanceEntry>>([]);
useEffect(() => {
const observer = new PerformanceObserver(list => {
const newEntries = list
.getEntries()
.filter(entry => entry.name.startsWith('rntester-'));
if (newEntries.length > 0) {
setEntries(newEntries);
}
});
observer.observe({entryTypes: ['mark', 'measure']});
return () => observer.disconnect();
}, []);
const onPress = useCallback(() => {
performance.mark('rntester-mark1');
performance.mark('rntester-mark2');
performance.measure(
'rntester-measure1',
'rntester-mark1',
'rntester-mark2',
);
}, []);
return (
<View style={styles.container}>
<Button onPress={onPress} title="Click to log some marks and measures" />
<View>
{entries.map((entry, index) =>
entry.entryType === 'mark' ? (
<RNTesterText key={index}>
Mark {entry.name}: {entry.startTime.toFixed(2)}
</RNTesterText>
) : (
<RNTesterText key={index}>
Measure {entry.name}: {entry.startTime.toFixed(2)} -{' '}
{(entry.startTime + entry.duration).toFixed(2)} (
{entry.duration.toFixed(2)}ms)
</RNTesterText>
),
)}
</View>
</View>
);
}
function PerformanceObserverEventTimingExample(): React.Node {
const [count, setCount] = useState(0);
const [entries, setEntries] = useState<
$ReadOnlyArray<PerformanceEventTiming>,
>([]);
useEffect(() => {
const observer = new PerformanceObserver(list => {
const newEntries: $ReadOnlyArray<PerformanceEventTiming> =
// $FlowExpectedError[incompatible-type] This is guaranteed because we're only observing `event` entry types.
list.getEntries();
setEntries(newEntries);
});
observer.observe({type: 'event'});
return () => observer.disconnect();
}, []);
const onPress = useCallback(() => {
busyWait(500);
// Force a state update to show how/if we're reporting paint times as well.
setCount(currentCount => currentCount + 1);
}, []);
return (
<View style={styles.container}>
<Button
onPress={onPress}
title={`Click to force a slow event (clicked ${count} times)`}
/>
<View>
{entries.map((entry, index) => (
<RNTesterText key={index}>
Event: {entry.name}
{'\n'}
Start: {entry.startTime.toFixed(2)}
{'\n'}
End: {(entry.startTime + entry.duration).toFixed(2)}
{'\n'}
Duration: {entry.duration.toFixed(2)}ms{'\n'}
Processing start: {entry.processingStart.toFixed(2)} (delay:{' '}
{(entry.processingStart - entry.startTime).toFixed(2)}ms){'\n'}
Processing end: {entry.processingEnd.toFixed(2)} (duration:{' '}
{(entry.processingEnd - entry.processingStart).toFixed(2)}ms){'\n'}
</RNTesterText>
))}
</View>
</View>
);
}
function PerformanceObserverLongtaskExample(): React.Node {
const [entries, setEntries] = useState<$ReadOnlyArray<PerformanceEntry>>([]);
useEffect(() => {
const observer = new PerformanceObserver(list => {
setEntries(list.getEntries());
});
observer.observe({entryTypes: ['longtask']});
return () => observer.disconnect();
}, []);
const onPress = useCallback(() => {
// Wait 1s to force a long task
busyWait(1000);
}, []);
return (
<View style={styles.container}>
<Button onPress={onPress} title="Click to force a long task" />
<View>
{entries.map((entry, index) => (
<RNTesterText key={index}>
Long task {entry.name}: {entry.startTime} -{' '}
{entry.startTime + entry.duration} ({entry.duration}ms)
</RNTesterText>
))}
</View>
</View>
);
}
function busyWait(ms: number): void {
const end = performance.now() + ms;
while (performance.now() < end) {}
}
const styles = StyleSheet.create({
container: {
padding: 10,
},
});
export const title = 'Performance API Examples';
export const category = 'Basic';
export const description = 'Shows the performance API provided in React Native';
export const examples: Array<RNTesterModuleExample> = ([
{
title: 'performance.memory',
render: (): React.Node => {
return <MemoryExample />;
},
},
{
title: 'performance.reactNativeStartupTiming',
render: (): React.Node => {
return <StartupTimingExample />;
},
},
{
title: 'PerformanceObserver (marks and measures)',
render: (): React.Node => {
return <PerformanceObserverUserTimingExample />;
},
},
{
title: 'PerformanceObserver (events)',
render: (): React.Node => {
return <PerformanceObserverEventTimingExample />;
},
},
PerformanceObserver.supportedEntryTypes.includes('longtask')
? {
title: 'PerformanceObserver (long tasks)',
render: (): React.Node => {
return <PerformanceObserverLongtaskExample />;
},
}
: null,
]: Array<?RNTesterModuleExample>).filter(Boolean);