2023-03-09 12:20:04 -08:00
|
|
|
import { useEffect } from 'react';
|
2023-03-13 14:23:11 -07:00
|
|
|
import z from 'zod';
|
2023-03-09 12:20:04 -08:00
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
import { getNextLink, getPrevLink } from 'soapbox/api';
|
2022-12-04 14:58:13 -08:00
|
|
|
import { useApi, useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
2023-03-13 14:39:23 -07:00
|
|
|
import { filteredArray } from 'soapbox/schemas/utils';
|
2022-12-04 14:58:13 -08:00
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
import { entitiesFetchFail, entitiesFetchRequest, entitiesFetchSuccess } from '../actions';
|
2022-12-04 14:58:13 -08:00
|
|
|
|
|
|
|
import type { Entity } from '../types';
|
2023-03-09 12:43:09 -08:00
|
|
|
import type { RootState } from 'soapbox/store';
|
2022-12-04 14:58:13 -08:00
|
|
|
|
2023-03-09 10:32:50 -08:00
|
|
|
/** Tells us where to find/store the entity in the cache. */
|
|
|
|
type EntityPath = [
|
|
|
|
/** Name of the entity type for use in the global cache, eg `'Notification'`. */
|
|
|
|
entityType: string,
|
|
|
|
/** Name of a particular index of this entity type. You can use empty-string (`''`) if you don't need separate lists. */
|
|
|
|
listKey: string,
|
|
|
|
]
|
|
|
|
|
|
|
|
/** Additional options for the hook. */
|
2023-03-13 14:23:11 -07:00
|
|
|
interface UseEntitiesOpts<TEntity extends Entity> {
|
|
|
|
/** A zod schema to parse the API entities. */
|
|
|
|
schema?: z.ZodType<TEntity, z.ZodTypeDef, any>
|
2023-03-09 12:20:04 -08:00
|
|
|
/**
|
|
|
|
* Time (milliseconds) until this query becomes stale and should be refetched.
|
|
|
|
* It is 1 minute by default, and can be set to `Infinity` to opt-out of automatic fetching.
|
|
|
|
*/
|
|
|
|
staleTime?: number
|
2023-03-09 10:32:50 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** A hook for fetching and displaying API entities. */
|
|
|
|
function useEntities<TEntity extends Entity>(
|
|
|
|
/** Tells us where to find/store the entity in the cache. */
|
|
|
|
path: EntityPath,
|
2023-03-09 12:43:09 -08:00
|
|
|
/** API route to GET, eg `'/api/v1/notifications'`. If undefined, nothing will be fetched. */
|
|
|
|
endpoint: string | undefined,
|
2023-03-09 10:32:50 -08:00
|
|
|
/** Additional options for the hook. */
|
|
|
|
opts: UseEntitiesOpts<TEntity> = {},
|
|
|
|
) {
|
2022-12-04 14:58:13 -08:00
|
|
|
const api = useApi();
|
|
|
|
const dispatch = useAppDispatch();
|
|
|
|
|
|
|
|
const [entityType, listKey] = path;
|
|
|
|
|
2023-03-13 14:23:11 -07:00
|
|
|
const defaultSchema = z.custom<TEntity>();
|
|
|
|
const schema = opts.schema || defaultSchema;
|
2023-03-09 10:32:50 -08:00
|
|
|
|
2022-12-04 16:54:54 -08:00
|
|
|
const cache = useAppSelector(state => state.entities[entityType]);
|
|
|
|
const list = cache?.lists[listKey];
|
2022-12-04 14:58:13 -08:00
|
|
|
|
|
|
|
const entityIds = list?.ids;
|
|
|
|
|
|
|
|
const entities: readonly TEntity[] = entityIds ? (
|
|
|
|
Array.from(entityIds).reduce<TEntity[]>((result, id) => {
|
2023-03-13 14:39:23 -07:00
|
|
|
const entity = cache?.store[id];
|
|
|
|
if (entity) {
|
|
|
|
result.push(entity as TEntity);
|
2022-12-04 14:58:13 -08:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}, [])
|
|
|
|
) : [];
|
|
|
|
|
|
|
|
const isFetching = Boolean(list?.state.fetching);
|
|
|
|
const isLoading = isFetching && entities.length === 0;
|
|
|
|
const hasNextPage = Boolean(list?.state.next);
|
|
|
|
const hasPreviousPage = Boolean(list?.state.prev);
|
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
const fetchPage = async(url: string): Promise<void> => {
|
2023-03-09 12:43:09 -08:00
|
|
|
// Get `isFetching` state from the store again to prevent race conditions.
|
|
|
|
const isFetching = dispatch((_, getState: () => RootState) => Boolean(getState().entities[entityType]?.lists[listKey]?.state.fetching));
|
|
|
|
if (isFetching) return;
|
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
dispatch(entitiesFetchRequest(entityType, listKey));
|
|
|
|
try {
|
|
|
|
const response = await api.get(url);
|
2023-03-13 14:39:23 -07:00
|
|
|
const entities = filteredArray(schema).parse(response.data);
|
|
|
|
|
|
|
|
dispatch(entitiesFetchSuccess(entities, entityType, listKey, {
|
2022-12-04 15:53:56 -08:00
|
|
|
next: getNextLink(response),
|
|
|
|
prev: getPrevLink(response),
|
|
|
|
fetching: false,
|
|
|
|
error: null,
|
2023-03-09 12:20:04 -08:00
|
|
|
lastFetchedAt: new Date(),
|
2022-12-04 15:53:56 -08:00
|
|
|
}));
|
|
|
|
} catch (error) {
|
|
|
|
dispatch(entitiesFetchFail(entityType, listKey, error));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const fetchEntities = async(): Promise<void> => {
|
2023-03-09 12:43:09 -08:00
|
|
|
if (endpoint) {
|
|
|
|
await fetchPage(endpoint);
|
|
|
|
}
|
2022-12-04 14:58:13 -08:00
|
|
|
};
|
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
const fetchNextPage = async(): Promise<void> => {
|
2022-12-04 14:58:13 -08:00
|
|
|
const next = list?.state.next;
|
|
|
|
|
|
|
|
if (next) {
|
2022-12-04 15:53:56 -08:00
|
|
|
await fetchPage(next);
|
2022-12-04 14:58:13 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-12-04 15:53:56 -08:00
|
|
|
const fetchPreviousPage = async(): Promise<void> => {
|
2022-12-04 14:58:13 -08:00
|
|
|
const prev = list?.state.prev;
|
|
|
|
|
|
|
|
if (prev) {
|
2022-12-04 15:53:56 -08:00
|
|
|
await fetchPage(prev);
|
2022-12-04 14:58:13 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-03-09 12:20:04 -08:00
|
|
|
const staleTime = opts.staleTime ?? 60000;
|
|
|
|
const lastFetchedAt = list?.state.lastFetchedAt;
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!isFetching && (!lastFetchedAt || lastFetchedAt.getTime() + staleTime <= Date.now())) {
|
|
|
|
fetchEntities();
|
|
|
|
}
|
|
|
|
}, [endpoint]);
|
|
|
|
|
2022-12-04 14:58:13 -08:00
|
|
|
return {
|
|
|
|
entities,
|
|
|
|
fetchEntities,
|
|
|
|
isFetching,
|
|
|
|
isLoading,
|
|
|
|
hasNextPage,
|
|
|
|
hasPreviousPage,
|
|
|
|
fetchNextPage,
|
|
|
|
fetchPreviousPage,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
|
|
|
useEntities,
|
|
|
|
};
|