( callback: AsyncFunction<T, U>, deps: React.DependencyList, )
| 18 | type AsyncFunction<T extends unknown[], U> = (...args: T) => Promise<U>; |
| 19 | |
| 20 | export function useHandledAsyncCallback<T extends unknown[], U>( |
| 21 | callback: AsyncFunction<T, U>, |
| 22 | deps: React.DependencyList, |
| 23 | ) { |
| 24 | const [loading, setLoading] = useState(0); |
| 25 | const [error, setError] = useState<Error | null>(null); |
| 26 | |
| 27 | const wrappedCallback = useCallback((...args: T) => { |
| 28 | setLoading((loading) => loading + 1); |
| 29 | setError(null); |
| 30 | |
| 31 | callback(...args) |
| 32 | .catch((error) => { |
| 33 | setError(error as Error); |
| 34 | console.error(error); |
| 35 | }) |
| 36 | .finally(() => { |
| 37 | setLoading((loading) => loading - 1); |
| 38 | }); |
| 39 | |
| 40 | // eslint-disable-next-line react-hooks/exhaustive-deps |
| 41 | }, deps); |
| 42 | |
| 43 | return [wrappedCallback, loading > 0, error] as const; |
| 44 | } |
| 45 | |
| 46 | interface Dimensions { |
| 47 | left: number; |
no test coverage detected