import { DependencyList, useCallback, useRef, useState } from 'react'; import type { FunctionReturningPromise, PromiseType } from '../types'; import { useMountedState } from './useMountedState'; // Reference: https://github.com/streamich/react-use/blob/master/src/useAsyncFn.ts export type AsyncState = | { loading: boolean; error?: undefined; value?: undefined; } | { loading: true; error?: Error | undefined; value?: T; } | { loading: false; error: Error; value?: undefined; } | { loading: false; error?: undefined; value: T; }; type StateFromFunctionReturningPromise = AsyncState>>; export type AsyncFnReturn< T extends FunctionReturningPromise = FunctionReturningPromise > = [StateFromFunctionReturningPromise, T]; export function useAsyncFn( fn: T, deps: DependencyList = [], initialState: StateFromFunctionReturningPromise = { loading: false } ): AsyncFnReturn { const lastCallId = useRef(0); const isMounted = useMountedState(); const [state, set] = useState>(initialState); const callback = useCallback((...args: Parameters): ReturnType => { const callId = ++lastCallId.current; set((prevState) => ({ ...prevState, loading: true })); return fn(...args).then( (value) => { isMounted() && callId === lastCallId.current && set({ value, loading: false }); return value; }, (error) => { isMounted() && callId === lastCallId.current && set({ error, loading: false }); return error; } ) as ReturnType; }, deps); return [state, callback as unknown as T]; }