bigbuffet-rw/app/soapbox/utils/queries.ts
2022-12-13 15:08:25 -06:00

132 lines
3.7 KiB
TypeScript

import { queryClient } from 'soapbox/queries/client';
import type { InfiniteData, QueryKey } from '@tanstack/react-query';
export interface PaginatedResult<T> {
result: T[],
hasMore: boolean,
link?: string,
}
interface Entity {
id: string,
}
const isEntity = <T = unknown>(object: T): object is T & Entity => {
return object && typeof object === 'object' && 'id' in object;
};
/** Deduplicate an array of entities by their ID. */
const deduplicateById = <T extends Entity>(entities: T[]): T[] => {
const map = entities.reduce<Map<string, T>>((result, entity) => {
return result.set(entity.id, entity);
}, new Map());
return Array.from(map.values());
};
export type SortOrder = 'reverse' | 'default'
/** Flatten paginated results into a single array. */
const flattenPages = <T>(
queryData: InfiniteData<PaginatedResult<T>> | undefined,
order: SortOrder = 'reverse',
) => {
const data = queryData?.pages.reduce<T[]>(
(prev: T[], curr) => {
if (order === 'reverse') {
return [...curr.result, ...prev];
} else {
return [...prev, ...curr.result];
}
},
[],
);
if (data && data.every(isEntity)) {
return deduplicateById(data);
} else if (data) {
return data;
}
};
/** Traverse pages and update the item inside if found. */
const updatePageItem = <T>(queryKey: QueryKey, newItem: T, isItem: (item: T, newItem: T) => boolean) => {
queryClient.setQueriesData<InfiniteData<PaginatedResult<T>>>(queryKey, (data) => {
if (data) {
const pages = data.pages.map(page => {
const result = page.result.map(item => isItem(item, newItem) ? newItem : item);
return { ...page, result };
});
return { ...data, pages };
}
});
};
/** Insert the new item at the beginning of the first page. */
const appendPageItem = <T>(queryKey: QueryKey, newItem: T) => {
queryClient.setQueryData<InfiniteData<PaginatedResult<T>>>(queryKey, (data) => {
if (data) {
const pages = [...data.pages];
pages[0] = { ...pages[0], result: [newItem, ...pages[0].result] };
return { ...data, pages };
}
});
};
/** Remove an item inside if found. */
const removePageItem = <T>(queryKey: QueryKey, itemToRemove: T, isItem: (item: T, newItem: T) => boolean) => {
queryClient.setQueriesData<InfiniteData<PaginatedResult<T>>>(queryKey, (data) => {
if (data) {
const pages = data.pages.map(page => {
const result = page.result.filter(item => !isItem(item, itemToRemove));
return { ...page, result };
});
return { ...data, pages };
}
});
};
const paginateQueryData = <T>(array: T[] | undefined) => {
return array?.reduce((resultArray: any, item: any, index: any) => {
const chunkIndex = Math.floor(index / 20);
if (!resultArray[chunkIndex]) {
resultArray[chunkIndex] = []; // start a new chunk
}
resultArray[chunkIndex].push(item);
return resultArray;
}, []);
};
const sortQueryData = <T>(
queryKey: QueryKey,
comparator: (a: T, b: T) => number,
order: SortOrder = 'reverse',
) => {
queryClient.setQueryData<InfiniteData<PaginatedResult<T>>>(queryKey, (prevResult) => {
if (prevResult) {
const nextResult = { ...prevResult };
const flattenedQueryData = flattenPages(nextResult, order);
const sortedQueryData = flattenedQueryData?.sort(comparator);
const paginatedPages = paginateQueryData(sortedQueryData);
const newPages = paginatedPages.map((page: T, idx: number) => ({
...prevResult.pages[idx],
result: page,
}));
nextResult.pages = newPages;
return nextResult;
}
});
};
export {
flattenPages,
updatePageItem,
appendPageItem,
removePageItem,
sortQueryData,
};