From b4e213a21f74e0ddc37987557758b0164717095f Mon Sep 17 00:00:00 2001 From: Remko Date: Mon, 16 Oct 2023 11:37:46 +0200 Subject: [PATCH 1/7] Updated conduction-theme package --- pwa/package-lock.json | 8 ++++---- pwa/package.json | 6 ++---- 2 files changed, 6 insertions(+), 8 deletions(-) diff --git a/pwa/package-lock.json b/pwa/package-lock.json index 965dd77d..d151a2de 100644 --- a/pwa/package-lock.json +++ b/pwa/package-lock.json @@ -9,7 +9,7 @@ "version": "1.0.0", "dependencies": { "@conduction/components": "2.2.16", - "@conduction/theme": "1.0.48", + "@conduction/theme": "1.0.49", "@fortawesome/fontawesome-svg-core": "^6.1.1", "@fortawesome/free-solid-svg-icons": "^6.4.0", "@fortawesome/react-fontawesome": "^0.1.18", @@ -2052,9 +2052,9 @@ } }, "node_modules/@conduction/theme": { - "version": "1.0.48", - "resolved": "https://registry.npmjs.org/@conduction/theme/-/theme-1.0.48.tgz", - "integrity": "sha512-5XRYHl9W0NvBxVJBZ1/SYyhNYV4hHpm/9taDEw1hjuGzzzyz+XhpN2UMkbtiqFwgN/U6pcZMdxfJIcsw49iI0Q==", + "version": "1.0.49", + "resolved": "https://registry.npmjs.org/@conduction/theme/-/theme-1.0.49.tgz", + "integrity": "sha512-uyEwfKH+5x1QsKlOlfG32HGrfAelkDh0g0C9ddWvaiii245nZXDntIkdtFv57j7LEF3rI0v4veoU6IkMX24q6g==", "dependencies": { "@nl-design-system-unstable/rotterdam-design-tokens": "^1.0.0-alpha.100" } diff --git a/pwa/package.json b/pwa/package.json index 25e129f6..5e389ad8 100644 --- a/pwa/package.json +++ b/pwa/package.json @@ -4,9 +4,7 @@ "private": true, "description": "Product Website Template", "author": "Conduction", - "keywords": [ - "gatsby" - ], + "keywords": ["gatsby"], "scripts": { "develop": "gatsby develop", "start": "gatsby develop", @@ -25,7 +23,7 @@ }, "dependencies": { "@conduction/components": "2.2.16", - "@conduction/theme": "1.0.48", + "@conduction/theme": "1.0.49", "@fortawesome/fontawesome-svg-core": "^6.1.1", "@fortawesome/free-solid-svg-icons": "^6.4.0", "@fortawesome/react-fontawesome": "^0.1.18", From f946269fef23f141ba1f3719ac080f7aeaa79b9a Mon Sep 17 00:00:00 2001 From: Remko Date: Tue, 17 Oct 2023 13:30:35 +0200 Subject: [PATCH 2/7] added pagination limit selector --- pwa/package.json | 6 +- pwa/src/apiService/resources/openWoo.ts | 6 +- .../PaginationLimitSelect.module.css | 12 ++++ .../PaginationLimitSelect.tsx | 72 +++++++++++++++++++ pwa/src/context/global.ts | 3 + pwa/src/context/queryLimit.ts | 24 +++++++ pwa/src/hooks/openWoo.ts | 4 +- pwa/src/templates/landing/LandingTemplate.tsx | 8 ++- pwa/src/translations/en.ts | 3 + pwa/src/translations/nl.ts | 3 + 10 files changed, 133 insertions(+), 8 deletions(-) create mode 100644 pwa/src/components/paginationLimitSelect/PaginationLimitSelect.module.css create mode 100644 pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx create mode 100644 pwa/src/context/queryLimit.ts diff --git a/pwa/package.json b/pwa/package.json index 6753018e..ea587d63 100644 --- a/pwa/package.json +++ b/pwa/package.json @@ -22,8 +22,8 @@ "prepare": "cd .. && husky install" }, "dependencies": { - "@conduction/components": "2.2.17", - "@conduction/theme": "1.0.49", + "@conduction/components": "2.2.18", + "@conduction/theme": "1.0.50", "@fortawesome/fontawesome-svg-core": "^6.1.1", "@fortawesome/free-solid-svg-icons": "^6.4.0", "@fortawesome/react-fontawesome": "^0.1.18", @@ -57,7 +57,7 @@ "@tabler/icons-react": "2.21.0", "@types/react": "^18.0.28", "@types/react-dom": "^18.0.11", - "@utrecht/component-library-react": "^1.0.0-alpha.355", + "@utrecht/component-library-react": "^1.0.0-alpha.394", "@utrecht/design-tokens": "^1.0.0-alpha.524", "axios": "^0.25.0", "clsx": "^1.1.1", diff --git a/pwa/src/apiService/resources/openWoo.ts b/pwa/src/apiService/resources/openWoo.ts index 91174196..4a2f178f 100644 --- a/pwa/src/apiService/resources/openWoo.ts +++ b/pwa/src/apiService/resources/openWoo.ts @@ -14,8 +14,10 @@ export default class OpenWoo { this._send = send; } - public getAll = async (filters: IFiltersContext, currentPage: number): Promise => { - let endpoint = `/openWOO?extend[]=all${filtersToQueryParams(filters)}&_order[Publicatiedatum]=desc&_limit=${OPEN_WOO_LIMIT}&_page=${currentPage}`; + public getAll = async (filters: IFiltersContext, currentPage: number, limit: number): Promise => { + let endpoint = `/openWOO?extend[]=all${filtersToQueryParams( + filters, + )}&_order[Publicatiedatum]=desc&_limit=${limit}&_page=${currentPage}`; if (process.env.GATSBY_OIDN_NUMBER) { endpoint += `&oidn=${process.env.GATSBY_OIDN_NUMBER}`; diff --git a/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.module.css b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.module.css new file mode 100644 index 00000000..4357e4cd --- /dev/null +++ b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.module.css @@ -0,0 +1,12 @@ +.container { + display: flex; + flex-wrap: wrap; + gap: 8px; + align-items: center; + list-style-type: none; + + padding: 0; + margin: 0; + + user-select: none; +} diff --git a/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx new file mode 100644 index 00000000..0a516e0d --- /dev/null +++ b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx @@ -0,0 +1,72 @@ +import * as React from "react"; +import * as styles from "./PaginationLimitSelect.module.css"; +import clsx from "clsx"; +import { useForm } from "react-hook-form"; +import { SelectSingle } from "@conduction/components"; +import { IQueryLimitContext, queryLimitDefault, useQueryLimitContext } from "../../context/queryLimit"; +import { useTranslation } from "react-i18next"; + +interface PaginationLimitSelectProps { + queryLimitName: string; + layoutClassName?: string; +} + +export const PaginationLimitSelectComponent: React.FC = ({ + queryLimitName, + layoutClassName, +}) => { + const { + watch, + register, + control, + setValue, + formState: { errors }, + } = useForm(); + const { queryLimit, setQueryLimit } = useQueryLimitContext(); + const { t } = useTranslation(); + + const watchLimit = watch("limit"); + + const value = queryLimit[queryLimitName as keyof IQueryLimitContext]; + + React.useEffect(() => { + if (!watchLimit) return; + if (parseInt(watchLimit.value) === value) return; + + const selectedLimit = limitSelectOptions.find((LimitOption) => LimitOption.value === watchLimit.value); + + selectedLimit !== undefined && setQueryLimit({ ...queryLimit, [queryLimitName]: parseInt(selectedLimit.value) }); + }, [watchLimit]); + + React.useEffect(() => { + setValue( + "limit", + limitSelectOptions.find((LimitOption) => LimitOption.value === (value !== undefined && value.toString())), + ); + }, []); + + return ( +
+ {`${t("Results per page")}:`} + +
+ ); +}; + +const limitSelectOptions = [ + { label: "6", value: "6" }, + { label: "9", value: "9" }, + { label: "12", value: "12" }, + { label: "21", value: "21" }, + { label: "30", value: "30" }, + { label: "60", value: "60" }, + { label: "120", value: "120" }, +]; diff --git a/pwa/src/context/global.ts b/pwa/src/context/global.ts index c9c3f4b8..189ac629 100644 --- a/pwa/src/context/global.ts +++ b/pwa/src/context/global.ts @@ -3,6 +3,7 @@ import { defaultGatsbyContext, IGatsbyContext } from "./gatsby"; import { defaultFiltersContext, IFiltersContext } from "./filters"; import { defaultDisplayContext, IDisplayContext } from "./displays"; import { defaultPaginationContext, IPaginationContext } from "./pagination"; +import { defaultQueryLimitContext, IQueryLimitContext } from "./queryLimit"; export interface IGlobalContext { initiated: boolean; @@ -10,6 +11,7 @@ export interface IGlobalContext { filters: IFiltersContext; displays: IDisplayContext; pagination: IPaginationContext; + queryLimit: IQueryLimitContext; } export const defaultGlobalContext: IGlobalContext = { @@ -18,6 +20,7 @@ export const defaultGlobalContext: IGlobalContext = { filters: defaultFiltersContext, displays: defaultDisplayContext, pagination: defaultPaginationContext, + queryLimit: defaultQueryLimitContext, }; export const GlobalContext = React.createContext< diff --git a/pwa/src/context/queryLimit.ts b/pwa/src/context/queryLimit.ts new file mode 100644 index 00000000..2a7c021c --- /dev/null +++ b/pwa/src/context/queryLimit.ts @@ -0,0 +1,24 @@ +import * as React from "react"; +import { GlobalContext } from "./global"; + +export const queryLimitDefault = 12; + +export interface IQueryLimitContext { + objectsQueryLimit: number; +} + +export const defaultQueryLimitContext: IQueryLimitContext = { + objectsQueryLimit: queryLimitDefault, +}; + +export const useQueryLimitContext = () => { + const [globalContext, setGlobalContext] = React.useContext(GlobalContext); + + const queryLimit: IQueryLimitContext = globalContext.queryLimit; + + const setQueryLimit = (query: IQueryLimitContext) => { + setGlobalContext((context) => ({ ...context, queryLimit: { ...globalContext.queryLimit, ...query } })); + }; + + return { setQueryLimit, queryLimit }; +}; diff --git a/pwa/src/hooks/openWoo.ts b/pwa/src/hooks/openWoo.ts index e3f7d642..11d6bb03 100644 --- a/pwa/src/hooks/openWoo.ts +++ b/pwa/src/hooks/openWoo.ts @@ -7,8 +7,8 @@ import { IFiltersContext } from "../context/filters"; export const useOpenWoo = (queryClient: QueryClient) => { const API: APIService | null = React.useContext(APIContext); - const getAll = (filters: IFiltersContext, currentPage: number) => - useQuery(["OpenWoo", filters, currentPage], () => API?.OpenWoo.getAll(filters, currentPage), { + const getAll = (filters: IFiltersContext, currentPage: number, limit: number) => + useQuery(["OpenWoo", filters, currentPage, limit], () => API?.OpenWoo.getAll(filters, currentPage, limit), { onError: (error) => { console.warn(error.message); }, diff --git a/pwa/src/templates/landing/LandingTemplate.tsx b/pwa/src/templates/landing/LandingTemplate.tsx index fd8e27be..410e463b 100644 --- a/pwa/src/templates/landing/LandingTemplate.tsx +++ b/pwa/src/templates/landing/LandingTemplate.tsx @@ -11,18 +11,22 @@ import { QueryClient } from "react-query"; import { Pagination } from "@conduction/components"; import { usePaginationContext } from "../../context/pagination"; import { useTranslation } from "react-i18next"; +import { useQueryLimitContext } from "../../context/queryLimit"; +import { PaginationLimitSelectComponent } from "../../components/paginationLimitSelect/PaginationLimitSelect"; export const LandingTemplate: React.FC = () => { const { currentPage, setCurrentPage } = usePaginationContext(); const { filters } = useFiltersContext(); const { t } = useTranslation(); + const { queryLimit } = useQueryLimitContext(); const queryClient = new QueryClient(); - const getItems = useOpenWoo(queryClient).getAll(filters, currentPage); + const getItems = useOpenWoo(queryClient).getAll(filters, currentPage, queryLimit.objectsQueryLimit); return ( <>

+ @@ -39,6 +43,8 @@ export const LandingTemplate: React.FC = () => { totalPages={getItems.data.pages} {...{ currentPage, setCurrentPage }} /> + + )} {getItems.isLoading && } diff --git a/pwa/src/translations/en.ts b/pwa/src/translations/en.ts index 1570bfbd..6189a248 100644 --- a/pwa/src/translations/en.ts +++ b/pwa/src/translations/en.ts @@ -14,6 +14,7 @@ export const en = { Address: "Address", Page: "Page", Jumbotron: "Jumbotron", + Limit: "Limit", "Jumbotron card": "Jumbotron card", "N/A": "N/A", "Details page": "Details page", @@ -51,4 +52,6 @@ export const en = { "Can open a new window": "Can open a new window", "No results found": "No results found", "Enter search query": "Enter search query", + "Results per page": "Results per page", + "Select result limit": "Select result limit", }; diff --git a/pwa/src/translations/nl.ts b/pwa/src/translations/nl.ts index 49eae986..1d862fdd 100644 --- a/pwa/src/translations/nl.ts +++ b/pwa/src/translations/nl.ts @@ -14,6 +14,7 @@ export const nl = { Address: "Adres", Page: "Pagina", Jumbotron: "Jumbotron", + Limit: "Limiet", "Jumbotron card": "Jumbotron tegel", "N/A": "N.v.t", "Details page": "Detailpagina", @@ -51,4 +52,6 @@ export const nl = { "Can open a new window": "Kan een nieuw venster openen", "No results found": "Geen resultaten gevonden", "Enter search query": "Voer zoekopdracht in", + "Results per page": "Resultaten per pagina", + "Select result limit": "Selecteer resultaten limiet", }; From 3356e0b5def86060009ca10e8beb4f7a18092fd4 Mon Sep 17 00:00:00 2001 From: Remko Date: Tue, 17 Oct 2023 14:54:04 +0200 Subject: [PATCH 3/7] updated package.lock and added HorizontalOverflowWrapper to TableResultsTemplate --- pwa/package-lock.json | 40 +++++++---- .../TableResultsTemplate.tsx | 66 +++++++++++-------- pwa/src/translations/en.ts | 2 + pwa/src/translations/nl.ts | 2 + 4 files changed, 68 insertions(+), 42 deletions(-) diff --git a/pwa/package-lock.json b/pwa/package-lock.json index 8dbe3670..66a84a4c 100644 --- a/pwa/package-lock.json +++ b/pwa/package-lock.json @@ -8,8 +8,8 @@ "name": "product-website-template", "version": "1.0.0", "dependencies": { - "@conduction/components": "2.2.17", - "@conduction/theme": "1.0.49", + "@conduction/components": "2.2.18", + "@conduction/theme": "1.0.50", "@fortawesome/fontawesome-svg-core": "^6.1.1", "@fortawesome/free-solid-svg-icons": "^6.4.0", "@fortawesome/react-fontawesome": "^0.1.18", @@ -43,7 +43,7 @@ "@tabler/icons-react": "2.21.0", "@types/react": "^18.0.28", "@types/react-dom": "^18.0.11", - "@utrecht/component-library-react": "^1.0.0-alpha.355", + "@utrecht/component-library-react": "^1.0.0-alpha.394", "@utrecht/design-tokens": "^1.0.0-alpha.524", "axios": "^0.25.0", "clsx": "^1.1.1", @@ -2055,9 +2055,9 @@ } }, "node_modules/@conduction/components": { - "version": "2.2.17", - "resolved": "https://registry.npmjs.org/@conduction/components/-/components-2.2.17.tgz", - "integrity": "sha512-xSrAuuWbaXQsdFiUQa/ldjovX7FbPseudz1Hc7GsjD5iFf8Nly+rNTXxkQp9QxOhTn5c8UdZlOrJ/V3qshafUA==", + "version": "2.2.18", + "resolved": "https://registry.npmjs.org/@conduction/components/-/components-2.2.18.tgz", + "integrity": "sha512-dXuDeA4BVGtNzQ0w2t2J3tDoqEHEOehwd0YYm68NWCDYMy/EtlX8tJpTrQSbXpkKt/pQV6dGt5WuBknWHOPHkA==", "dependencies": { "@fortawesome/fontawesome-svg-core": "^6.2.0", "@fortawesome/free-solid-svg-icons": "^6.2.0", @@ -2110,9 +2110,9 @@ } }, "node_modules/@conduction/theme": { - "version": "1.0.49", - "resolved": "https://registry.npmjs.org/@conduction/theme/-/theme-1.0.49.tgz", - "integrity": "sha512-uyEwfKH+5x1QsKlOlfG32HGrfAelkDh0g0C9ddWvaiii245nZXDntIkdtFv57j7LEF3rI0v4veoU6IkMX24q6g==", + "version": "1.0.50", + "resolved": "https://registry.npmjs.org/@conduction/theme/-/theme-1.0.50.tgz", + "integrity": "sha512-VdkN12+r12WORjVtl2W1kSLMaeZLtqRdrwtHJKuuUFGZmUnqDoxrdjPnX9c5in+MROLgIbBN1Xp/8/Ucy/Y6Lg==", "dependencies": { "@nl-design-system-unstable/rotterdam-design-tokens": "^1.0.0-alpha.100" } @@ -4351,16 +4351,30 @@ } }, "node_modules/@utrecht/component-library-react": { - "version": "1.0.0-alpha.356", - "license": "EUPL-1.2", + "version": "1.0.0-alpha.406", + "resolved": "https://registry.npmjs.org/@utrecht/component-library-react/-/component-library-react-1.0.0-alpha.406.tgz", + "integrity": "sha512-lZoV/O4EFDIYghVF8okdXBNlmal7bcvZ1jqNk+b1gac54TwpvTmmbd4dlwB2+kulN1vhwiFN+oUII3zOMuYCuQ==", "dependencies": { "clsx": "1.2.1", - "date-fns": "2.30.0", "lodash.chunk": "4.2.0" }, "peerDependencies": { + "date-fns": "^2.30.0", "react": "18", - "react-dom": "18" + "react-dom": "18", + "react-vega": "^7.6.0", + "vega": "^5.25.0" + }, + "peerDependenciesMeta": { + "date-fns": { + "optional": true + }, + "react-vega": { + "optional": true + }, + "vega": { + "optional": true + } } }, "node_modules/@utrecht/design-tokens": { diff --git a/pwa/src/templates/templateParts/tableResultsTemplate/TableResultsTemplate.tsx b/pwa/src/templates/templateParts/tableResultsTemplate/TableResultsTemplate.tsx index 990c0164..13e2a5e9 100644 --- a/pwa/src/templates/templateParts/tableResultsTemplate/TableResultsTemplate.tsx +++ b/pwa/src/templates/templateParts/tableResultsTemplate/TableResultsTemplate.tsx @@ -11,6 +11,7 @@ import { import { navigate } from "gatsby"; import { translateDate } from "../../../services/dateFormat"; import { useTranslation } from "react-i18next"; +import { HorizontalOverflowWrapper } from "@conduction/components"; interface TableResultsTemplateProps { requests: any[]; @@ -20,35 +21,42 @@ export const TableResultsTemplate: React.FC = ({ requ const { t, i18n } = useTranslation(); return ( - - - - {t("Subject")} - {t("Publication date")} - {t("Summary")} - - - - {requests.map((request) => ( - navigate(request.id)} - tabIndex={0} - aria-label={`${request.Titel}, ${ - request.Publicatiedatum ? translateDate(i18n.language, request.Publicatiedatum) : t("N/A") - }, ${request.Samenvatting}`} - > - {request.Titel ?? t("No subject available")} - - {request.Publicatiedatum - ? translateDate(i18n.language, request.Publicatiedatum) - : t("No publication date available")} - - {request.Samenvatting ?? t("No summary available")} + +
+ + + {t("Subject")} + {t("Publication date")} + {t("Summary")} - ))} - -
+ + + {requests.map((request) => ( + navigate(request.id)} + tabIndex={0} + aria-label={`${request.Titel}, ${ + request.Publicatiedatum ? translateDate(i18n.language, request.Publicatiedatum) : t("N/A") + }, ${request.Samenvatting}`} + > + {request.Titel ?? t("No subject available")} + + {request.Publicatiedatum + ? translateDate(i18n.language, request.Publicatiedatum) + : t("No publication date available")} + + {request.Samenvatting ?? t("No summary available")} + + ))} + + + ); }; diff --git a/pwa/src/translations/en.ts b/pwa/src/translations/en.ts index 6189a248..9f4620f1 100644 --- a/pwa/src/translations/en.ts +++ b/pwa/src/translations/en.ts @@ -54,4 +54,6 @@ export const en = { "Enter search query": "Enter search query", "Results per page": "Results per page", "Select result limit": "Select result limit", + "Scroll table to the left": "Scroll table to the left", + "Scroll table to the right": "Scroll table to the right", }; diff --git a/pwa/src/translations/nl.ts b/pwa/src/translations/nl.ts index 1d862fdd..1df1e1cb 100644 --- a/pwa/src/translations/nl.ts +++ b/pwa/src/translations/nl.ts @@ -54,4 +54,6 @@ export const nl = { "Enter search query": "Voer zoekopdracht in", "Results per page": "Resultaten per pagina", "Select result limit": "Selecteer resultaten limiet", + "Scroll table to the left": "Scroll tabel naar links", + "Scroll table to the right": "Scroll tabel naar rechts", }; From c094ceeb8532a34b54be577e13c900e253bbfa56 Mon Sep 17 00:00:00 2001 From: Remko Date: Tue, 17 Oct 2023 15:12:03 +0200 Subject: [PATCH 4/7] added feedback --- .../landing/LandingTemplate.module.css | 6 +++--- pwa/src/templates/landing/LandingTemplate.tsx | 18 ++++++++---------- 2 files changed, 11 insertions(+), 13 deletions(-) diff --git a/pwa/src/templates/landing/LandingTemplate.module.css b/pwa/src/templates/landing/LandingTemplate.module.css index 48669c37..ddb02a7a 100644 --- a/pwa/src/templates/landing/LandingTemplate.module.css +++ b/pwa/src/templates/landing/LandingTemplate.module.css @@ -6,7 +6,7 @@ margin-block-end: var(--utrecht-space-block-lg); } -.header1 { - margin-block-start: 0px; - margin-block-end: 0px; +.pagination { + display: flex; + justify-content: space-between; } diff --git a/pwa/src/templates/landing/LandingTemplate.tsx b/pwa/src/templates/landing/LandingTemplate.tsx index 410e463b..f7b1ac3c 100644 --- a/pwa/src/templates/landing/LandingTemplate.tsx +++ b/pwa/src/templates/landing/LandingTemplate.tsx @@ -25,8 +25,6 @@ export const LandingTemplate: React.FC = () => { return ( <> -

- @@ -37,14 +35,14 @@ export const LandingTemplate: React.FC = () => { {getItems.data?.results && getItems.data?.results?.length > 0 && (
- - - - +
+ + +
)} {getItems.isLoading && } From ef4af2cffc288c1125191ec4ddf06099be1bcd7b Mon Sep 17 00:00:00 2001 From: Remko Date: Wed, 18 Oct 2023 10:49:05 +0200 Subject: [PATCH 5/7] Wrong page bugfix --- pwa/src/templates/landing/LandingTemplate.tsx | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/pwa/src/templates/landing/LandingTemplate.tsx b/pwa/src/templates/landing/LandingTemplate.tsx index f7b1ac3c..01a3d29f 100644 --- a/pwa/src/templates/landing/LandingTemplate.tsx +++ b/pwa/src/templates/landing/LandingTemplate.tsx @@ -23,6 +23,10 @@ export const LandingTemplate: React.FC = () => { const queryClient = new QueryClient(); const getItems = useOpenWoo(queryClient).getAll(filters, currentPage, queryLimit.objectsQueryLimit); + React.useEffect(() => { + setCurrentPage(1); + }, [queryLimit.objectsQueryLimit]); + return ( <> From 95476669c8e61412c1bccd2c28ac6011dae1c534 Mon Sep 17 00:00:00 2001 From: Remko Date: Thu, 19 Oct 2023 15:40:37 +0200 Subject: [PATCH 6/7] added working query params --- pwa/package-lock.json | 40 ++++++++++++- pwa/package.json | 6 +- pwa/src/services/filtersToQueryParams.ts | 51 +++++++++++++++- .../templateParts/filters/FiltersTemplate.tsx | 58 +++++++++++++++++-- 4 files changed, 147 insertions(+), 8 deletions(-) diff --git a/pwa/package-lock.json b/pwa/package-lock.json index 66a84a4c..b5b625ed 100644 --- a/pwa/package-lock.json +++ b/pwa/package-lock.json @@ -41,6 +41,7 @@ "@nl-design-system-unstable/zwolle-design-tokens": "^1.0.0-alpha.100", "@parcel/watcher": "^2.3.0", "@tabler/icons-react": "2.21.0", + "@types/qs": "^6.9.9", "@types/react": "^18.0.28", "@types/react-dom": "^18.0.11", "@utrecht/component-library-react": "^1.0.0-alpha.394", @@ -56,6 +57,7 @@ "i18next": "^21.6.16", "jwt-decode": "^3.1.2", "lodash": "^4.17.21", + "qs": "^6.11.2", "react": "^18.2.0", "react-collapsible": "^2.10.0", "react-dom": "^18.2.0", @@ -3998,6 +4000,11 @@ "version": "15.7.5", "license": "MIT" }, + "node_modules/@types/qs": { + "version": "6.9.9", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.9.tgz", + "integrity": "sha512-wYLxw35euwqGvTDx6zfY1vokBFnsK0HNrzc6xNHchxfO2hpuRg74GbkEW7e3sSmPvj0TjCDT1VCa6OtHXnubsg==" + }, "node_modules/@types/reach__router": { "version": "1.3.11", "license": "MIT", @@ -5381,6 +5388,20 @@ "version": "2.0.0", "license": "MIT" }, + "node_modules/body-parser/node_modules/qs": { + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz", + "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==", + "dependencies": { + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/boolbase": { "version": "1.0.0", "license": "ISC" @@ -7903,6 +7924,20 @@ "version": "2.0.0", "license": "MIT" }, + "node_modules/express/node_modules/qs": { + "version": "6.11.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz", + "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==", + "dependencies": { + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/ext": { "version": "1.7.0", "license": "ISC", @@ -13528,8 +13563,9 @@ } }, "node_modules/qs": { - "version": "6.11.0", - "license": "BSD-3-Clause", + "version": "6.11.2", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.2.tgz", + "integrity": "sha512-tDNIz22aBzCDxLtVH++VnTfzxlfeK5CbqohpSqpJgj1Wg/cQbStNAz3NuqCs5vV+pjBsK4x4pN9HlVh7rcYRiA==", "dependencies": { "side-channel": "^1.0.4" }, diff --git a/pwa/package.json b/pwa/package.json index ea587d63..6d6751db 100644 --- a/pwa/package.json +++ b/pwa/package.json @@ -4,7 +4,9 @@ "private": true, "description": "Product Website Template", "author": "Conduction", - "keywords": ["gatsby"], + "keywords": [ + "gatsby" + ], "scripts": { "develop": "gatsby develop", "start": "gatsby develop", @@ -55,6 +57,7 @@ "@nl-design-system-unstable/zwolle-design-tokens": "^1.0.0-alpha.100", "@parcel/watcher": "^2.3.0", "@tabler/icons-react": "2.21.0", + "@types/qs": "^6.9.9", "@types/react": "^18.0.28", "@types/react-dom": "^18.0.11", "@utrecht/component-library-react": "^1.0.0-alpha.394", @@ -70,6 +73,7 @@ "i18next": "^21.6.16", "jwt-decode": "^3.1.2", "lodash": "^4.17.21", + "qs": "^6.11.2", "react": "^18.2.0", "react-collapsible": "^2.10.0", "react-dom": "^18.2.0", diff --git a/pwa/src/services/filtersToQueryParams.ts b/pwa/src/services/filtersToQueryParams.ts index 72df8ae5..c87a1407 100644 --- a/pwa/src/services/filtersToQueryParams.ts +++ b/pwa/src/services/filtersToQueryParams.ts @@ -5,7 +5,7 @@ export const filtersToQueryParams = (filters: any): string => { delete filters[key]; }); - let params = ""; + let params: string = ""; for (const [key, value] of Object.entries(filters)) { if (!value) continue; @@ -28,4 +28,53 @@ export const filtersToQueryParams = (filters: any): string => { return params; }; +export const filtersToUrlQueryParams = (filters: any): string => { + Object.keys(filters) + .filter((key) => filterKeysToRemove.includes(key)) + .forEach((key) => { + delete filters[key]; + }); + + let params: string = ""; + + var first_iteration = true; + for (const [key, value] of Object.entries(filters)) { + if (!value) continue; + + if (first_iteration) { + if (typeof value === "string") { + params += `?${key}=${value.replace(/\s+/g, "_")}`; + } + + if (Array.isArray(value)) { + let arrayParams = ""; + + value.forEach((value) => { + arrayParams += `?${key}[]=${value.replace(/\s+/g, "_")}`; + }); + + params += arrayParams; + } + + first_iteration = false; + } else { + if (typeof value === "string") { + params += `&${key}=${value.replace(/\s+/g, "_")}`; + } + + if (Array.isArray(value)) { + let arrayParams = ""; + + value.forEach((value) => { + arrayParams += `&${key}[]=${value.replace(/\s+/g, "_")}`; + }); + + params += arrayParams; + } + } + } + + return params; +}; + const filterKeysToRemove: string[] = []; diff --git a/pwa/src/templates/templateParts/filters/FiltersTemplate.tsx b/pwa/src/templates/templateParts/filters/FiltersTemplate.tsx index 0cba6fd1..34dcc340 100644 --- a/pwa/src/templates/templateParts/filters/FiltersTemplate.tsx +++ b/pwa/src/templates/templateParts/filters/FiltersTemplate.tsx @@ -1,23 +1,30 @@ import * as React from "react"; import * as styles from "./FiltersTemplate.module.css"; import ResultsDisplaySwitch from "../../../components/resultsDisplaySwitch/ResultsDisplaySwitch"; +import _ from "lodash"; +import qs from "qs"; import { useForm } from "react-hook-form"; import { InputText, SelectSingle } from "@conduction/components"; -import { useFiltersContext } from "../../../context/filters"; +import { IFiltersContext, defaultFiltersContext, useFiltersContext } from "../../../context/filters"; import { Button } from "@utrecht/component-library-react/dist/css-module"; import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; import { faMagnifyingGlass, faSpinner } from "@fortawesome/free-solid-svg-icons"; import { generateYearsArray } from "../../../data/years"; import { TEMP_PUBLICATION_TYPES } from "../../../data/PublicationType"; import { useTranslation } from "react-i18next"; +import { filtersToUrlQueryParams } from "../../../services/filtersToQueryParams"; +import { navigate } from "gatsby"; +import { useGatsbyContext } from "../../../context/gatsby"; interface FiltersTemplateProps { isLoading: boolean; } export const FiltersTemplate: React.FC = ({ isLoading }) => { - const { filters, setFilters } = useFiltersContext(); const { t } = useTranslation(); + const { filters, setFilters } = useFiltersContext(); + const { gatsbyContext } = useGatsbyContext(); + const [queryParams, setQueryParams] = React.useState(defaultFiltersContext); const filterTimeout = React.useRef(null); const { @@ -25,6 +32,7 @@ export const FiltersTemplate: React.FC = ({ isLoading }) = register, handleSubmit, watch, + setValue, formState: { errors }, } = useForm(); @@ -33,9 +41,30 @@ export const FiltersTemplate: React.FC = ({ isLoading }) = const today = new Date(); const currentYear = today.getFullYear(); + const url = gatsbyContext.location.search; + const [, params] = url.split("?"); + const parsedParams = qs.parse(params); + + const handleSetFormValues = (params: any): void => { + const basicFields: string[] = ["_search", "category"]; + basicFields.forEach((field) => setValue(field, params[field])); + + setValue( + "year", + generateYearsArray(currentYear - 1995).find((year: any) => { + return year.after === params.Publicatiedatum?.after && year.before === params.Publicatiedatum?.before; + }), + ); + + setValue( + "category", + TEMP_PUBLICATION_TYPES.find((option) => option.value === params.Categorie?.replace(/_/g, " ")), + ); + }; + const onSubmit = (data: any) => { setFilters({ - _search: data.title, + _search: data._search, "Publicatiedatum[after]": data.year?.after, "Publicatiedatum[before]": data.year?.before, Categorie: data.category?.value, @@ -48,24 +77,45 @@ export const FiltersTemplate: React.FC = ({ isLoading }) = filterTimeout.current = setTimeout(() => onSubmit(watcher), 500); }, [watcher]); + React.useEffect(() => { + if (_.isEmpty(parsedParams)) return; + + handleSetFormValues(parsedParams); + }, []); + + React.useEffect(() => { + //Prevents loop that puts user at top of page after scroll + if (_.isEqual(filters, queryParams)) return; + + setQueryParams(filters); + navigate(`/${filtersToUrlQueryParams(filters)}`); + }, [filters]); + return (
+ { + return ( + year.after === filters["Publicatiedatum[after]"] && year.before === filters["Publicatiedatum[before]"] + ); + })} {...{ register, errors, control }} ariaLabel={t("Select year")} /> + Date: Mon, 23 Oct 2023 10:51:52 +0200 Subject: [PATCH 7/7] added requested changes --- .../PaginationLimitSelect.tsx | 14 +-- pwa/src/context/queryLimit.ts | 6 +- pwa/src/services/filtersToQueryParams.ts | 95 ++++++------------- pwa/src/templates/landing/LandingTemplate.tsx | 6 +- 4 files changed, 43 insertions(+), 78 deletions(-) diff --git a/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx index 0a516e0d..5e10b88e 100644 --- a/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx +++ b/pwa/src/components/paginationLimitSelect/PaginationLimitSelect.tsx @@ -3,7 +3,7 @@ import * as styles from "./PaginationLimitSelect.module.css"; import clsx from "clsx"; import { useForm } from "react-hook-form"; import { SelectSingle } from "@conduction/components"; -import { IQueryLimitContext, queryLimitDefault, useQueryLimitContext } from "../../context/queryLimit"; +import { IQueryLimitContext, QUERY_LIMIT_DEFAULT, useQueryLimitContext } from "../../context/queryLimit"; import { useTranslation } from "react-i18next"; interface PaginationLimitSelectProps { @@ -33,25 +33,27 @@ export const PaginationLimitSelectComponent: React.FC LimitOption.value === watchLimit.value); + const selectedLimit = limitSelectOptions.find((limitOption) => limitOption.value === watchLimit.value); - selectedLimit !== undefined && setQueryLimit({ ...queryLimit, [queryLimitName]: parseInt(selectedLimit.value) }); + if (selectedLimit) { + setQueryLimit({ ...queryLimit, [queryLimitName]: parseInt(selectedLimit.value) }); + } }, [watchLimit]); React.useEffect(() => { setValue( "limit", - limitSelectOptions.find((LimitOption) => LimitOption.value === (value !== undefined && value.toString())), + limitSelectOptions.find((limitOption) => limitOption.value === (value !== undefined && value.toString())), ); }, []); return (
- {`${t("Results per page")}:`} + {t("Results per page")}: { diff --git a/pwa/src/services/filtersToQueryParams.ts b/pwa/src/services/filtersToQueryParams.ts index c87a1407..8a15c6f3 100644 --- a/pwa/src/services/filtersToQueryParams.ts +++ b/pwa/src/services/filtersToQueryParams.ts @@ -1,80 +1,43 @@ export const filtersToQueryParams = (filters: any): string => { - Object.keys(filters) - .filter((key) => filterKeysToRemove.includes(key)) - .forEach((key) => { - delete filters[key]; - }); + const cleanedFilters = Object.fromEntries( + Object.entries(filters).filter(([key]) => !filterKeysToRemove.includes(key)), + ); - let params: string = ""; + const params = Object.entries(cleanedFilters) + .map(([key, value]) => { + if (!value) return null; - for (const [key, value] of Object.entries(filters)) { - if (!value) continue; + const formattedValue = Array.isArray(value) + ? value.map((v: string) => v.replace(/\s+/g, "_")).join(`&${key}[]=`) + : (value as string).replace(/\s+/g, "_"); - if (typeof value === "string") { - params += `&${key}=${value}`; - } + return `${Array.isArray(value) ? `${key}[]` : key}=${formattedValue}`; + }) + .filter(Boolean) + .join("&"); - if (Array.isArray(value)) { - let arrayParams = ""; - - value.forEach((value) => { - arrayParams += `&${key}[]=${value}`; - }); - - params += arrayParams; - } - } - - return params; + return params ? `&${params}` : ""; }; -export const filtersToUrlQueryParams = (filters: any): string => { - Object.keys(filters) - .filter((key) => filterKeysToRemove.includes(key)) - .forEach((key) => { - delete filters[key]; - }); - - let params: string = ""; - - var first_iteration = true; - for (const [key, value] of Object.entries(filters)) { - if (!value) continue; - - if (first_iteration) { - if (typeof value === "string") { - params += `?${key}=${value.replace(/\s+/g, "_")}`; - } - - if (Array.isArray(value)) { - let arrayParams = ""; - - value.forEach((value) => { - arrayParams += `?${key}[]=${value.replace(/\s+/g, "_")}`; - }); - - params += arrayParams; - } - - first_iteration = false; - } else { - if (typeof value === "string") { - params += `&${key}=${value.replace(/\s+/g, "_")}`; - } +export const filtersToUrlQueryParams = (filters: Record): string => { + const cleanedFilters = Object.fromEntries( + Object.entries(filters).filter(([key]) => !filterKeysToRemove.includes(key)), + ); - if (Array.isArray(value)) { - let arrayParams = ""; + const params = Object.entries(cleanedFilters) + .map(([key, value]) => { + if (!value) return null; - value.forEach((value) => { - arrayParams += `&${key}[]=${value.replace(/\s+/g, "_")}`; - }); + const formattedValue = Array.isArray(value) + ? value.map((v: string) => v.replace(/\s+/g, "_")).join(`&${key}[]=`) + : (value as string).replace(/\s+/g, "_"); - params += arrayParams; - } - } - } + return `${Array.isArray(value) ? `${key}[]` : key}=${formattedValue}`; + }) + .filter(Boolean) + .join("&"); - return params; + return params ? `?${params}` : ""; }; const filterKeysToRemove: string[] = []; diff --git a/pwa/src/templates/landing/LandingTemplate.tsx b/pwa/src/templates/landing/LandingTemplate.tsx index 01a3d29f..195fb473 100644 --- a/pwa/src/templates/landing/LandingTemplate.tsx +++ b/pwa/src/templates/landing/LandingTemplate.tsx @@ -21,11 +21,11 @@ export const LandingTemplate: React.FC = () => { const { queryLimit } = useQueryLimitContext(); const queryClient = new QueryClient(); - const getItems = useOpenWoo(queryClient).getAll(filters, currentPage, queryLimit.objectsQueryLimit); + const getItems = useOpenWoo(queryClient).getAll(filters, currentPage, queryLimit.openWooObjectsQueryLimit); React.useEffect(() => { setCurrentPage(1); - }, [queryLimit.objectsQueryLimit]); + }, [queryLimit.openWooObjectsQueryLimit]); return ( <> @@ -45,7 +45,7 @@ export const LandingTemplate: React.FC = () => { totalPages={getItems.data.pages} {...{ currentPage, setCurrentPage }} /> - +
)}