bigbuffet-rw/app/soapbox/entity-store/hooks/useEntityLookup.ts

61 lines
1.7 KiB
TypeScript
Raw Normal View History

import { AxiosError } from 'axios';
import { useEffect, useState } from 'react';
2023-04-12 13:39:41 -07:00
import { z } from 'zod';
import { useAppDispatch, useAppSelector, useLoading } from 'soapbox/hooks';
import { importEntities } from '../actions';
import { findEntity } from '../selectors';
2023-04-12 13:39:41 -07:00
import { Entity } from '../types';
import { EntityFn } from './types';
import { type UseEntityOpts } from './useEntity';
/** Entities will be filtered through this function until it returns true. */
type LookupFn<TEntity extends Entity> = (entity: TEntity) => boolean
function useEntityLookup<TEntity extends Entity>(
entityType: string,
lookupFn: LookupFn<TEntity>,
entityFn: EntityFn<void>,
opts: UseEntityOpts<TEntity> = {},
) {
const { schema = z.custom<TEntity>() } = opts;
const dispatch = useAppDispatch();
const [isFetching, setPromise] = useLoading(true);
const [error, setError] = useState<unknown>();
2023-04-12 13:39:41 -07:00
const entity = useAppSelector(state => findEntity(state, entityType, lookupFn));
2023-06-02 07:28:10 -07:00
const isEnabled = opts.enabled ?? true;
2023-04-12 13:39:41 -07:00
const isLoading = isFetching && !entity;
const fetchEntity = async () => {
try {
const response = await setPromise(entityFn());
const entity = schema.parse(response.data);
dispatch(importEntities([entity], entityType));
} catch (e) {
setError(e);
2023-04-12 13:39:41 -07:00
}
};
useEffect(() => {
2023-06-02 07:28:10 -07:00
if (!isEnabled) return;
2023-04-12 13:39:41 -07:00
if (!entity || opts.refetch) {
fetchEntity();
}
2023-06-02 07:28:10 -07:00
}, [isEnabled]);
2023-04-12 13:39:41 -07:00
return {
entity,
fetchEntity,
isFetching,
isLoading,
isUnauthorized: error instanceof AxiosError && error.response?.status === 401,
isForbidden: error instanceof AxiosError && error.response?.status === 403,
2023-04-12 13:39:41 -07:00
};
}
2023-04-12 13:51:30 -07:00
export { useEntityLookup };