import { useEffect, useState } from 'react'; import z from 'zod'; import { useApi, useAppDispatch, useAppSelector } from 'soapbox/hooks'; import { importEntities } from '../actions'; import type { Entity } from '../types'; type EntityPath = [entityType: string, entityId: string] /** Additional options for the hook. */ interface UseEntityOpts { /** A zod schema to parse the API entity. */ schema?: z.ZodType /** Whether to refetch this entity every time the hook mounts, even if it's already in the store. */ refetch?: boolean } function useEntity( path: EntityPath, endpoint: string, opts: UseEntityOpts = {}, ) { const api = useApi(); const dispatch = useAppDispatch(); const [entityType, entityId] = path; const defaultSchema = z.custom(); const schema = opts.schema || defaultSchema; const entity = useAppSelector(state => { // TODO: parse after fetch, not during render. const result = schema.safeParse(state.entities[entityType]?.store[entityId]); return result.success ? result.data : undefined; }); const [isFetching, setIsFetching] = useState(false); const isLoading = isFetching && !entity; const fetchEntity = () => { setIsFetching(true); api.get(endpoint).then(({ data }) => { dispatch(importEntities([data], entityType)); setIsFetching(false); }).catch(() => { setIsFetching(false); }); }; useEffect(() => { if (!entity || opts.refetch) { fetchEntity(); } }, []); return { entity, fetchEntity, isFetching, isLoading, }; } export { useEntity, };