Skip to main content

useLoading()

Helps track loading state of imperative async functions.

tip

useSuspense() or useDLE() are better for GET/read endpoints.

Part of @data-client/hooks

Usage

import { useLoading } from '@data-client/hooks';

function Button({ onClick, children, ...props }) {
const [clickHandler, loading, error] = useLoading(onClick);
return (
<button onClick={clickHandler} {...props}>
{loading ? 'Loading...' : children}
</button>
);
}

Todo toggle example

import { useCallback } from 'react';
import { useController } from '@data-client/react';
import { useLoading } from '@data-client/hooks';

import { TodoResource, Todo } from 'api/Todo';

function TodoListItem({ todo }) {
const ctrl = useController();

const [toggleHandler, loading, error] = useLoading(
(e: ChangeEvent<HTMLInputElement>) =>
ctrl.fetch(
TodoResource.partialUpdate,
{ id },
{ completed: e.currentTarget.checked },
),
[ctrl],
);

return (
<div>
<input
type="checkbox"
checked={todo.completed}
onChange={toggleHandler}
/>
{loading ? <Spinner /> : null}
{error ? <Error>{error}</Error> : null}
{todo.title}
</div>
);
}

interface Props {
todo: Todo;
}

Eslint

Eslint configuration

Since we use the deps list, be sure to add useLoading to the 'additionalHooks' configuration of react-hooks/exhaustive-deps rule if you use it.

{
"rules": {
// ...
"react-hooks/exhaustive-deps": ["warn", {
"additionalHooks": "(useLoading)"
}]
}
}

Types

export default function useLoading<F extends (...args: any) => Promise<any>>(
func: F,
deps: readonly any[] = [],
): [F, boolean];