import { DependencyList, useCallback, useState, useRef } from 'react'; import useMountedState from './useMountedState'; import { FnReturningPromise, PromiseType } from './util'; 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 StateFromFnReturningPromise = AsyncState>>; export type AsyncFnReturn = [StateFromFnReturningPromise, T]; export default function useAsyncFn( fn: T, deps: DependencyList = [], initialState: StateFromFnReturningPromise = { 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]; }