bigbuffet-rw/app/soapbox/entity-store/reducer.ts

183 lines
4.6 KiB
TypeScript
Raw Normal View History

2022-12-04 14:58:13 -08:00
import produce, { enableMapSet } from 'immer';
import {
ENTITIES_IMPORT,
2023-03-14 12:13:49 -07:00
ENTITIES_DELETE,
ENTITIES_DISMISS,
ENTITIES_FETCH_REQUEST,
ENTITIES_FETCH_SUCCESS,
ENTITIES_FETCH_FAIL,
EntityAction,
ENTITIES_INVALIDATE_LIST,
2023-03-22 17:45:02 -07:00
ENTITIES_INCREMENT,
} from './actions';
2022-12-04 14:58:13 -08:00
import { createCache, createList, updateStore, updateList } from './utils';
2023-03-15 14:52:09 -07:00
import type { DeleteEntitiesOpts } from './actions';
2022-12-04 15:53:56 -08:00
import type { Entity, EntityCache, EntityListState } from './types';
2022-12-04 14:58:13 -08:00
enableMapSet();
/** Entity reducer state. */
interface State {
[entityType: string]: EntityCache | undefined
}
2022-12-04 14:58:13 -08:00
/** Import entities into the cache. */
const importEntities = (
state: State,
2022-12-04 14:58:13 -08:00
entityType: string,
entities: Entity[],
listKey?: string,
2022-12-04 15:53:56 -08:00
newState?: EntityListState,
overwrite = false,
2022-12-04 14:58:13 -08:00
): State => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
2022-12-04 14:58:13 -08:00
cache.store = updateStore(cache.store, entities);
if (typeof listKey === 'string') {
let list = cache.lists[listKey] ?? createList();
if (overwrite) {
list.ids = new Set();
}
2022-12-04 15:53:56 -08:00
list = updateList(list, entities);
2022-12-04 15:53:56 -08:00
if (newState) {
list.state = newState;
}
cache.lists[listKey] = list;
2022-12-04 14:58:13 -08:00
}
draft[entityType] = cache;
2022-12-04 14:58:13 -08:00
});
};
2023-03-14 12:13:49 -07:00
const deleteEntities = (
state: State,
entityType: string,
ids: Iterable<string>,
2023-03-15 14:52:09 -07:00
opts: DeleteEntitiesOpts,
2023-03-14 12:13:49 -07:00
) => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
for (const id of ids) {
delete cache.store[id];
2023-03-15 14:52:09 -07:00
if (!opts?.preserveLists) {
for (const list of Object.values(cache.lists)) {
if (list) {
list.ids.delete(id);
if (typeof list.state.totalCount === 'number') {
list.state.totalCount--;
}
}
2023-03-15 14:52:09 -07:00
}
2023-03-14 12:13:49 -07:00
}
}
draft[entityType] = cache;
});
};
const dismissEntities = (
state: State,
entityType: string,
ids: Iterable<string>,
listKey: string,
) => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
const list = cache.lists[listKey];
if (list) {
for (const id of ids) {
list.ids.delete(id);
if (typeof list.state.totalCount === 'number') {
list.state.totalCount--;
}
}
draft[entityType] = cache;
}
});
};
2023-03-22 17:45:02 -07:00
const incrementEntities = (
state: State,
entityType: string,
listKey: string,
diff: number,
) => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
const list = cache.lists[listKey];
if (typeof list?.state?.totalCount === 'number') {
list.state.totalCount += diff;
draft[entityType] = cache;
}
});
};
const setFetching = (
state: State,
entityType: string,
listKey: string | undefined,
isFetching: boolean,
error?: any,
) => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
if (typeof listKey === 'string') {
const list = cache.lists[listKey] ?? createList();
list.state.fetching = isFetching;
list.state.error = error;
cache.lists[listKey] = list;
}
draft[entityType] = cache;
});
};
const invalidateEntityList = (state: State, entityType: string, listKey: string) => {
return produce(state, draft => {
const cache = draft[entityType] ?? createCache();
const list = cache.lists[listKey] ?? createList();
list.state.invalid = true;
});
};
2022-12-04 14:58:13 -08:00
/** Stores various entity data and lists in a one reducer. */
function reducer(state: Readonly<State> = {}, action: EntityAction): State {
2022-12-04 14:58:13 -08:00
switch (action.type) {
case ENTITIES_IMPORT:
return importEntities(state, action.entityType, action.entities, action.listKey);
2023-03-14 12:13:49 -07:00
case ENTITIES_DELETE:
2023-03-15 14:52:09 -07:00
return deleteEntities(state, action.entityType, action.ids, action.opts);
case ENTITIES_DISMISS:
return dismissEntities(state, action.entityType, action.ids, action.listKey);
2023-03-22 17:45:02 -07:00
case ENTITIES_INCREMENT:
return incrementEntities(state, action.entityType, action.listKey, action.diff);
2022-12-04 15:53:56 -08:00
case ENTITIES_FETCH_SUCCESS:
return importEntities(state, action.entityType, action.entities, action.listKey, action.newState, action.overwrite);
case ENTITIES_FETCH_REQUEST:
return setFetching(state, action.entityType, action.listKey, true);
case ENTITIES_FETCH_FAIL:
return setFetching(state, action.entityType, action.listKey, false, action.error);
case ENTITIES_INVALIDATE_LIST:
return invalidateEntityList(state, action.entityType, action.listKey);
2022-12-04 14:58:13 -08:00
default:
return state;
}
}
2023-03-14 12:13:49 -07:00
export default reducer;
export type { State };