From 5955129455e6f8704f9188617a14bcc788982334 Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Fri, 24 May 2024 17:10:30 -0700 Subject: [PATCH 1/7] add debounce on pagination --- .../src/components/search/SearchPagination.tsx | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/frontend/src/components/search/SearchPagination.tsx b/frontend/src/components/search/SearchPagination.tsx index b8f2964a4..65413f0df 100644 --- a/frontend/src/components/search/SearchPagination.tsx +++ b/frontend/src/components/search/SearchPagination.tsx @@ -1,6 +1,7 @@ "use client"; import { Pagination } from "@trussworks/react-uswds"; +import { useDebouncedCallback } from "use-debounce"; import { useFormStatus } from "react-dom"; export enum PaginationPosition { @@ -19,6 +20,7 @@ interface SearchPaginationProps { } const MAX_SLOTS = 7; +const DEBOUNCE_TIME = 500; export default function SearchPagination({ showHiddenInput, @@ -31,6 +33,10 @@ export default function SearchPagination({ }: SearchPaginationProps) { const { pending } = useFormStatus(); + const debouncedHandlePageChange = useDebouncedCallback((newPage: number) => { + handlePageChange(newPage); + }, DEBOUNCE_TIME); + // If there's no results, don't show pagination if (searchResultsLength < 1) { return null; @@ -59,9 +65,11 @@ export default function SearchPagination({ totalPages={totalPages} currentPage={page} maxSlots={MAX_SLOTS} - onClickNext={() => handlePageChange(page + 1)} - onClickPrevious={() => handlePageChange(page - 1)} - onClickPageNumber={(event, page) => handlePageChange(page)} + onClickNext={() => debouncedHandlePageChange(page + 1)} + onClickPrevious={() => debouncedHandlePageChange(page - 1)} + onClickPageNumber={(event: React.MouseEvent, page: number) => + debouncedHandlePageChange(page) + } /> ); From 908bf26236b8c7d828c06ce2b1fe3c4f17e42717 Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Wed, 29 May 2024 12:43:26 -0700 Subject: [PATCH 2/7] update debounce to 300ms --- frontend/src/components/search/SearchPagination.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/frontend/src/components/search/SearchPagination.tsx b/frontend/src/components/search/SearchPagination.tsx index 65413f0df..7ac27d984 100644 --- a/frontend/src/components/search/SearchPagination.tsx +++ b/frontend/src/components/search/SearchPagination.tsx @@ -20,7 +20,7 @@ interface SearchPaginationProps { } const MAX_SLOTS = 7; -const DEBOUNCE_TIME = 500; +const DEBOUNCE_TIME = 300; export default function SearchPagination({ showHiddenInput, From 8b36108dfa1d95b10845fc739b55d4d9a3c8dda3 Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Fri, 31 May 2024 16:42:44 -0700 Subject: [PATCH 3/7] fix typing on search tests --- frontend/tests/e2e/search/search.spec.ts | 36 ++++++++++++++++++------ 1 file changed, 27 insertions(+), 9 deletions(-) diff --git a/frontend/tests/e2e/search/search.spec.ts b/frontend/tests/e2e/search/search.spec.ts index e6c69f638..7c3e7b56c 100644 --- a/frontend/tests/e2e/search/search.spec.ts +++ b/frontend/tests/e2e/search/search.spec.ts @@ -1,3 +1,4 @@ +import { Page, expect, test } from "@playwright/test"; import { clickAccordionWithTitle, clickLastPaginationPage, @@ -20,9 +21,18 @@ import { toggleCheckboxes, waitForSearchResultsInitialLoad, } from "./searchSpecUtil"; -import { expect, test } from "@playwright/test"; -test("should navigate from index to search page", async ({ page }) => { +import { BrowserContextOptions } from "playwright-core"; + +interface PageProps { + page: Page; + browserName?: string; + contextOptions?: BrowserContextOptions; +} + +test("should navigate from index to search page", async ({ + page, +}: PageProps) => { // Start from the index page with feature flag set await page.goto("/?_ff=showSearchV0:true"); @@ -48,7 +58,7 @@ test("should navigate from index to search page", async ({ page }) => { }); test.describe("Search page tests", () => { - test.beforeEach(async ({ page }) => { + test.beforeEach(async ({ page }: PageProps) => { // Navigate to the search page with the feature flag set await page.goto("/search?_ff=showSearchV0:true"); }); @@ -56,7 +66,7 @@ test.describe("Search page tests", () => { test("should return 0 results when searching for obscure term", async ({ page, browserName, - }) => { + }: PageProps) => { // TODO (Issue #2005): fix test for webkit test.skip( browserName === "webkit", @@ -80,7 +90,10 @@ test.describe("Search page tests", () => { ); }); - test("should show and hide loading state", async ({ page, browserName }) => { + test("should show and hide loading state", async ({ + page, + browserName, + }: PageProps) => { // TODO (Issue #2005): fix test for webkit test.skip( browserName === "webkit", @@ -98,7 +111,10 @@ test.describe("Search page tests", () => { await expect(loadingIndicator).toBeVisible(); await expect(loadingIndicator).toBeHidden(); }); - test("should refresh and retain filters in a new tab", async ({ page }) => { + + test("should refresh and retain filters in a new tab", async ({ + page, + }: PageProps) => { // Set all inputs, then refresh the page. Those same inputs should be // set from query params. const searchTerm = "education"; @@ -175,7 +191,9 @@ test.describe("Search page tests", () => { } }); - test("resets page back to 1 when choosing a filter", async ({ page }) => { + test("resets page back to 1 when choosing a filter", async ({ + page, + }: PageProps) => { await clickPaginationPageNumber(page, 2); // Verify that page 1 is highlighted @@ -201,7 +219,7 @@ test.describe("Search page tests", () => { test("last result becomes first result when flipping sort order", async ({ page, - }) => { + }: PageProps) => { await selectSortBy(page, "opportunityTitleDesc"); await clickLastPaginationPage(page); @@ -219,7 +237,7 @@ test.describe("Search page tests", () => { test("number of results is the same with none or all opportunity status checked", async ({ page, - }) => { + }: PageProps) => { const initialNumberOfOpportunityResults = await getNumberOfOpportunitySearchResults(page); From b8deb59beaa0037d54fc721abf0278e430ad07ac Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Fri, 31 May 2024 16:43:02 -0700 Subject: [PATCH 4/7] add delay for pagination debounce --- frontend/tests/e2e/search/searchSpecUtil.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/frontend/tests/e2e/search/searchSpecUtil.ts b/frontend/tests/e2e/search/searchSpecUtil.ts index f54d32b15..df54d24fb 100644 --- a/frontend/tests/e2e/search/searchSpecUtil.ts +++ b/frontend/tests/e2e/search/searchSpecUtil.ts @@ -146,6 +146,9 @@ export async function clickPaginationPageNumber( `button[data-testid="pagination-page-number"][aria-label="Page ${pageNumber}"]`, ); await paginationButton.first().click(); + + // Delay for pagination debounce + await page.waitForTimeout(400); } export async function clickLastPaginationPage(page: Page) { From 3a4b71fcc0e45e286f4bc249e2f60f9612a40b6e Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Fri, 31 May 2024 16:53:13 -0700 Subject: [PATCH 5/7] add pagination debounce wait for clicking last page --- frontend/tests/e2e/search/searchSpecUtil.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/frontend/tests/e2e/search/searchSpecUtil.ts b/frontend/tests/e2e/search/searchSpecUtil.ts index df54d24fb..63782e670 100644 --- a/frontend/tests/e2e/search/searchSpecUtil.ts +++ b/frontend/tests/e2e/search/searchSpecUtil.ts @@ -159,6 +159,8 @@ export async function clickLastPaginationPage(page: Page) { if (count > 2) { await paginationButtons.nth(count - 2).click(); } + // Delay for pagination debounce + await page.waitForTimeout(400); } export async function getFirstSearchResultTitle(page: Page) { From b6e1e6d1d94b2638807ec2ac4c4d13cb1abd052c Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Mon, 3 Jun 2024 09:37:54 -0700 Subject: [PATCH 6/7] fix page reset test by selecting first --- frontend/tests/e2e/search/search.spec.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/frontend/tests/e2e/search/search.spec.ts b/frontend/tests/e2e/search/search.spec.ts index 7c3e7b56c..53541afdd 100644 --- a/frontend/tests/e2e/search/search.spec.ts +++ b/frontend/tests/e2e/search/search.spec.ts @@ -197,7 +197,7 @@ test.describe("Search page tests", () => { await clickPaginationPageNumber(page, 2); // Verify that page 1 is highlighted - let currentPageButton = page.locator(".usa-pagination__button.usa-current"); + let currentPageButton = page.locator(".usa-pagination__button.usa-current").first(); await expect(currentPageButton).toHaveAttribute("aria-label", "Page 2"); // Select the 'Closed' checkbox under 'Opportunity status' @@ -217,6 +217,7 @@ test.describe("Search page tests", () => { expectURLContainsQueryParam(page, "page", "1", false); }); + test("last result becomes first result when flipping sort order", async ({ page, }: PageProps) => { From 19cd99d6df3a03c14086fd0e478cb6c8d2da3493 Mon Sep 17 00:00:00 2001 From: Ryan Lewis Date: Mon, 3 Jun 2024 10:30:09 -0700 Subject: [PATCH 7/7] format --- frontend/tests/e2e/search/search.spec.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/frontend/tests/e2e/search/search.spec.ts b/frontend/tests/e2e/search/search.spec.ts index 53541afdd..e73cd256c 100644 --- a/frontend/tests/e2e/search/search.spec.ts +++ b/frontend/tests/e2e/search/search.spec.ts @@ -197,7 +197,9 @@ test.describe("Search page tests", () => { await clickPaginationPageNumber(page, 2); // Verify that page 1 is highlighted - let currentPageButton = page.locator(".usa-pagination__button.usa-current").first(); + let currentPageButton = page + .locator(".usa-pagination__button.usa-current") + .first(); await expect(currentPageButton).toHaveAttribute("aria-label", "Page 2"); // Select the 'Closed' checkbox under 'Opportunity status' @@ -217,7 +219,6 @@ test.describe("Search page tests", () => { expectURLContainsQueryParam(page, "page", "1", false); }); - test("last result becomes first result when flipping sort order", async ({ page, }: PageProps) => {