From 6057bf6ede5e07cdc09e33b52d48b3494c2e6fc7 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Mon, 11 Nov 2024 15:12:05 +0100 Subject: [PATCH] Return pageSize and hasMorePages as part of usePaginatedAPIFetch result (#6845) --- lms/static/scripts/frontend_apps/utils/api.ts | 25 ++++++++++++++++- .../frontend_apps/utils/test/api-test.js | 27 +++++++++++++++++++ 2 files changed, 51 insertions(+), 1 deletion(-) diff --git a/lms/static/scripts/frontend_apps/utils/api.ts b/lms/static/scripts/frontend_apps/utils/api.ts index 42991f37ca..38fccb60f4 100644 --- a/lms/static/scripts/frontend_apps/utils/api.ts +++ b/lms/static/scripts/frontend_apps/utils/api.ts @@ -315,6 +315,16 @@ export type PaginatedFetchResult = Omit, 'mutate'> & { /** Determines if currently loading the first page */ isLoadingFirstPage: boolean; loadNextPage: () => void; + + /** + * The size of every page (maximum number of results returned per page). + * If the API indicates there is only one page, this will be undefined. You + * can trust `data.length` as the total number of items in that case. + */ + pageSize?: number; + + /** Whether there are still more pages to load or not */ + hasMorePages: boolean; }; /** @@ -338,6 +348,8 @@ export function usePaginatedAPIFetch< ): PaginatedFetchResult { const [nextPageURL, setNextPageURL] = useState(); const [currentList, setCurrentList] = useState(); + const [pageSize, setPageSize] = useState(); + const [hasMorePages, setHasMorePages] = useState(false); const fetchResult = useAPIFetch( nextPageURL ?? path, @@ -350,8 +362,8 @@ export function usePaginatedAPIFetch< } }, [fetchResult.data?.pagination.next, fetchResult.isLoading]); - // Every time data is loaded, append the result to the list useEffect(() => { + // Every time data is loaded, append the result to the list setCurrentList(prev => { if (!fetchResult.data) { return prev; @@ -359,6 +371,15 @@ export function usePaginatedAPIFetch< return [...(prev ?? []), ...fetchResult.data[prop]] as ListType; }); + + const hasMorePages = !!fetchResult.data?.pagination.next; + setHasMorePages(hasMorePages); + + // Set pageSize if the API indicated there are more pages. + // Once page size has been set, we keep the value, ignoring next page loads + if (hasMorePages) { + setPageSize(prev => prev ?? fetchResult.data?.[prop].length); + } }, [prop, fetchResult.data]); // Every time params change, discard previous pages and start from scratch @@ -382,5 +403,7 @@ export function usePaginatedAPIFetch< error: fetchResult.error, retry: fetchResult.retry, loadNextPage, + pageSize, + hasMorePages, }; } diff --git a/lms/static/scripts/frontend_apps/utils/test/api-test.js b/lms/static/scripts/frontend_apps/utils/test/api-test.js index b22fc1b908..6de1eaab00 100644 --- a/lms/static/scripts/frontend_apps/utils/test/api-test.js +++ b/lms/static/scripts/frontend_apps/utils/test/api-test.js @@ -566,6 +566,10 @@ describe('usePaginatedAPIFetch', () => { {!result.isLoading && (result.data ? result.data.join(',') : 'No content')} +
{result.pageSize ?? 'No page size'}
+
+ {result.hasMorePages ? 'Yes' : 'No'} +