pleroma/app/soapbox/entity-store/hooks/useEntity.ts

75 lines
2 KiB
TypeScript
Raw Normal View History

import { AxiosError } from 'axios';
import { useEffect, useState } from 'react';
import z from 'zod';
2022-12-04 14:58:13 -08:00
import { useAppDispatch, useAppSelector, useLoading } from 'soapbox/hooks';
2022-12-04 14:58:13 -08:00
import { importEntities } from '../actions';
import type { Entity } from '../types';
import type { EntitySchema, EntityPath, EntityFn } from './types';
2022-12-04 14:58:13 -08:00
/** Additional options for the hook. */
2023-03-14 12:24:11 -07:00
interface UseEntityOpts<TEntity extends Entity> {
/** A zod schema to parse the API entity. */
2023-03-14 12:24:11 -07:00
schema?: EntitySchema<TEntity>
/** Whether to refetch this entity every time the hook mounts, even if it's already in the store. */
refetch?: boolean
/** A flag to potentially disable sending requests to the API. */
enabled?: boolean
}
function useEntity<TEntity extends Entity>(
path: EntityPath,
entityFn: EntityFn<void>,
opts: UseEntityOpts<TEntity> = {},
) {
const [isFetching, setPromise] = useLoading(true);
const [error, setError] = useState<unknown>();
2022-12-04 14:58:13 -08:00
const dispatch = useAppDispatch();
const [entityType, entityId] = path;
const defaultSchema = z.custom<TEntity>();
const schema = opts.schema || defaultSchema;
const entity = useAppSelector(state => state.entities[entityType]?.store[entityId] as TEntity | undefined);
2022-12-04 14:58:13 -08:00
const isEnabled = opts.enabled ?? true;
2022-12-04 14:58:13 -08:00
const isLoading = isFetching && !entity;
const isLoaded = !isFetching && !!entity;
2022-12-04 14:58:13 -08:00
const fetchEntity = async () => {
try {
const response = await setPromise(entityFn());
const entity = schema.parse(response.data);
dispatch(importEntities([entity], entityType));
} catch (e) {
setError(e);
}
2022-12-04 14:58:13 -08:00
};
useEffect(() => {
if (!isEnabled) return;
if (!entity || opts.refetch) {
fetchEntity();
}
}, [isEnabled]);
2022-12-04 14:58:13 -08:00
return {
entity,
fetchEntity,
isFetching,
isLoading,
isLoaded,
error,
isUnauthorized: error instanceof AxiosError && error.response?.status === 401,
isForbidden: error instanceof AxiosError && error.response?.status === 403,
2022-12-04 14:58:13 -08:00
};
}
export {
useEntity,
2023-04-12 13:39:41 -07:00
type UseEntityOpts,
2022-12-04 14:58:13 -08:00
};