import { queryClient } from 'soapbox/queries/client'; import type { InfiniteData, QueryKey } from '@tanstack/react-query'; export interface PaginatedResult { result: T[], hasMore: boolean, link?: string, } /** Flatten paginated results into a single array. */ const flattenPages = (queryData: InfiniteData> | undefined) => { return queryData?.pages.reduce( (prev: T[], curr) => [...curr.result, ...prev], [], ); }; /** Traverse pages and update the item inside if found. */ const updatePageItem = (queryKey: QueryKey, newItem: T, isItem: (item: T, newItem: T) => boolean) => { queryClient.setQueriesData>>(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 = (queryKey: QueryKey, newItem: T) => { queryClient.setQueryData>>(queryKey, (data) => { if (data) { const pages = [...data.pages]; pages[0] = { ...pages[0], result: [...pages[0].result, newItem] }; return { ...data, pages }; } }); }; /** Remove an item inside if found. */ const removePageItem = (queryKey: QueryKey, itemToRemove: T, isItem: (item: T, newItem: T) => boolean) => { queryClient.setQueriesData>>(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 = (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 = (queryKey: QueryKey, comparator: (a: T, b: T) => number) => { queryClient.setQueryData>>(queryKey, (prevResult) => { if (prevResult) { const nextResult = { ...prevResult }; const flattenedQueryData = flattenPages(nextResult); 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, };