はじめに
コンポーネントからロジックを分離するためにカスタムフックがよく使われますね。
今回はAPIからデータをfetchするカスタムフックの書き方について備忘録的にまとめます。
fetchしたデータを使ってコンポーネントをrenderしたいとき
fetchしたデータをもとにrenderするとき、SWR
がとても便利です。
カスタムフックの実装
SWR
を使うと、fetchしたデータをキャッシュしてrenderに使うことができます。
また、ブラウザのタブを切り替えたときには自動でデータの再検証をしてくれたり、あるいはmutate
をinvokeすることで任意のタイミングでデータを再検証することができます。
import useSWR, { mutate } from 'swr';
import { useCallback } from 'react';
export const useFetchData = (
startFetch: boolean,
) => {
const key = 'some_unique_key';
const { data, error } = useSWR(startFetch ? key : null, () =>
axios.get('your_endpoint_here')
);
const refetchData = useCallback(() => {
mutate(key);
}, [key]);
return {
data,
isLoading: !data && !error,
isError: !!error,
refetchData
};
};
コンポーネントの実装
startFetch === true
になればfetchが開始されます。
データをfetchしている最中はisLoading === true
となり、これを使ってロード中の表示ができます。
fetchはもちろん非同期的に行われますが、{ data, isLoading, refetchData }
ではresolveされた値が返ってくるので、コンポーネント側では非同期を意識する必要はありません。
また、refetchData
をinvokeすればデータを再検証することができます。
これがあればuseEffect
を使ったデータの再検証を代替できますね。
import { useFetchData } from './useFetchData';
export const SomeComponent: React.FC<{}> = ({}) => {
const [startFetch, setStartFetch] = useState(false);
const { data, isLoading, refetchData } = useFetchData(startFetch);
const handleStartFetch = useCallback(() => {
setStartFetch(true);
}, []);
return (
<>
<Button onClick={handleStartFetch}>フェッチ開始</Button>
<Button onClick={refetchData}>データ再検証</Button>
{isLoading && <LoadingComponent />}
{!isLoading && data && <ComponentToShowData data={data}/>}
</>
);
};
任意のイベントが起きた時にデータをfetchして処理したいとき
たとえば、ボタンを押した時にfetchを開始して何らかの処理をする場合を考えます。
上記のSWR
を使ったユースケースと異なるのは、「ボタンを押した時だけ」APIをcallする、という点です。
SWR
はとても便利ですが、fetchしたデータはキャッシュされ、さらにブラウザのタブを切り替えた時などに自動でデータ再検証(APIの再call)が行われます。
上記が不要なケースでは下記のようにカスタムフックを書くことができます。
(*SWRでも自動再検証のキャンセルができるようです)
カスタムフックの実装
import { useCallback, useState } from 'react';
interface IData {
something: string;
}
export const useFetchData = () => {
const [dataState, setDataState] = useState<{
data: IData | null;
isLoading: boolean;
}>({
data: null,
isLoading: false
});
const fetchData = useCallback(async () => {
setDataState({
data: null,
isLoading: true
});
const data = await axios.get('your_endpoint_here')
setDataState({ data, isLoading: false });
return data;
}, []);
const fetchDataAndDoSomething = useCallback(async () => {
const data = await fetchData();
// do something here
}, []);
return { ...dataState, fetchDataAndDoSomething };
};
コンポーネントの実装
import { useFetchData } from './useFetchData';
export const SomeComponent: React.FC<{}> = ({}) => {
const [startFetch, setStartFetch] = useState(false);
const { data, isLoading, fetchDataAndDoSomething } = useFetchData();
return (
<>
<Button onClick={fetchDataAndDoSomething}>
'Fetch data and do something'
{isLoading && <LoadingComponent />}
</Button>
</>
);
};
参考にさせていただいたもの