UNPKG

@shopify/flash-list

Version:

FlashList is a more performant FlatList replacement

108 lines (100 loc) 2.94 kB
import { useEffect } from "react"; import { FlatList } from "react-native"; import { autoScroll, Cancellable } from "./AutoScrollHelper"; import { JSFPSMonitor } from "./JSFPSMonitor"; import { BenchmarkParams, BenchmarkResult, getFormattedString, } from "./useBenchmark"; export interface FlatListBenchmarkParams extends BenchmarkParams { targetOffset: number; } /** * Runs the benchmark on FlatList and calls the callback method with the result. * Target offset is mandatory in params. * It's recommended to remove pagination while running the benchmark. Removing the onEndReached callback is the easiest way to do that. */ export function useFlatListBenchmark( flatListRef: React.RefObject<FlatList<any>>, callback: (benchmarkResult: BenchmarkResult) => void, params: FlatListBenchmarkParams ) { useEffect(() => { const cancellable = new Cancellable(); if (flatListRef.current) { if (!(Number(flatListRef.current.props.data?.length) > 0)) { throw new Error("Data is empty, cannot run benchmark"); } } const cancelTimeout = setTimeout(async () => { const jsFPSMonitor = new JSFPSMonitor(); jsFPSMonitor.startTracking(); for (let i = 0; i < (params.repeatCount || 1); i++) { await runScrollBenchmark( flatListRef, params.targetOffset, cancellable, params.speedMultiplier || 1 ); } const jsProfilerResponse = jsFPSMonitor.stopAndGetData(); const result: BenchmarkResult = { js: jsProfilerResponse, suggestions: [], interrupted: cancellable.isCancelled(), }; if (!cancellable.isCancelled()) { result.formattedString = getFormattedString(result); } callback(result); }, params.startDelayInMs || 3000); return () => { clearTimeout(cancelTimeout); cancellable.cancel(); }; // eslint-disable-next-line react-hooks/exhaustive-deps }, []); return []; } /** * Scrolls to the target offset and then back to 0 */ async function runScrollBenchmark( flatListRef: React.RefObject<FlatList<any> | null | undefined>, targetOffset: number, cancellable: Cancellable, scrollSpeedMultiplier: number ): Promise<void> { if (flatListRef.current) { const horizontal = flatListRef.current.props.horizontal; const fromX = 0; const fromY = 0; const toX = horizontal ? targetOffset : 0; const toY = horizontal ? 0 : targetOffset; const scrollNow = (x: number, y: number) => { flatListRef.current?.scrollToOffset({ offset: horizontal ? x : y, animated: false, }); }; await autoScroll( scrollNow, fromX, fromY, toX, toY, scrollSpeedMultiplier, cancellable ); await autoScroll( scrollNow, toX, toY, fromX, fromY, scrollSpeedMultiplier, cancellable ); } }