Skip to content

Commit

Permalink
Resolve #934: useAsyncFn: keeping the previous state when start runni…
Browse files Browse the repository at this point in the history
…ng the async function
  • Loading branch information
echoulen committed Jan 30, 2020
1 parent 90eca90 commit 062e606
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 2 deletions.
2 changes: 1 addition & 1 deletion docs/useAsyncFn.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,5 +32,5 @@ const Demo = ({url}) => {
## Reference

```ts
useAsyncFn(fn, deps?: any[]);
useAsyncFn<Result, Args>(fn, deps?: any[], initialState?: AsyncState<Result>);
```
7 changes: 6 additions & 1 deletion src/useAsyncFn.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,11 @@ export type AsyncState<T> =
error?: undefined;
value?: undefined;
}
| {
loading: true;
error?: Error | undefined;
value?: T;
}
| {
loading: false;
error: Error;
Expand Down Expand Up @@ -35,7 +40,7 @@ export default function useAsyncFn<Result = any, Args extends any[] = any[]>(

const callback = useCallback((...args: Args | []) => {
const callId = ++lastCallId.current;
set({ loading: true });
set(prevState => ({ ...prevState, loading: true }));

return fn(...args).then(
value => {
Expand Down
27 changes: 27 additions & 0 deletions tests/useAsyncFn.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -126,4 +126,31 @@ describe('useAsyncFn', () => {
await hook.waitForNextUpdate();
expect(hook.result.current[0]).toEqual({ loading: false, value: 2 });
});

it('should keeping value of initialState when loading', async () => {
const fetch = async () => 'new state';
const initialState = { loading: false, value: 'init state' };

const hook = renderHook<{ fn: () => Promise<string> }, [AsyncState<string>, () => Promise<string>]>(
({ fn }) => useAsyncFn(fn, [fn], initialState),
{
initialProps: { fn: fetch },
}
);

const [state, callback] = hook.result.current;
expect(state.loading).toBe(false);
expect(state.value).toBe('init state');

act(() => {
callback();
});

expect(hook.result.current[0].loading).toBe(true);
expect(hook.result.current[0].value).toBe('init state');

await hook.waitForNextUpdate();
expect(hook.result.current[0].loading).toBe(false);
expect(hook.result.current[0].value).toBe('new state');
});
});

0 comments on commit 062e606

Please sign in to comment.