From 87d2297d490ea2de2723e46b3ee11af1b07e9030 Mon Sep 17 00:00:00 2001 From: Kevin Jackson <30411845+KevinJJackson@users.noreply.github.com> Date: Mon, 2 Dec 2024 16:04:49 -0500 Subject: [PATCH] enhancement/cwms-timeseries-list (#246) * working inf scroll on CWMS Timeseries * bugfix: autocomplete default failing if rendered before domain group finished fetching --- package-lock.json | 40 ++++++++- package.json | 3 +- src/app-components/domain-select.jsx | 29 +++--- .../cwms-timeseries/newCwmsTimeseries.jsx | 89 ++++++++++++++++--- .../collections/cwms-timeseries.ts | 21 +++-- src/app-services/fetch-helpers.ts | 2 +- 6 files changed, 149 insertions(+), 35 deletions(-) diff --git a/package-lock.json b/package-lock.json index dc039670..3988352a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "hhd-ui", - "version": "0.18.0", + "version": "0.18.1", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "hhd-ui", - "version": "0.18.0", + "version": "0.18.1", "dependencies": { "@ag-grid-community/client-side-row-model": "^30.0.3", "@ag-grid-community/core": "^30.0.3", @@ -46,6 +46,7 @@ "react-csv": "^2.2.2", "react-datepicker": "^4.16.0", "react-dom": "^18.2.0", + "react-infinite-scroll-hook": "^5.0.1", "react-papaparse": "^4.1.0", "react-plotly.js": "^2.6.0", "react-resizable-panels": "^2.0.7", @@ -6800,6 +6801,27 @@ "version": "3.2.0", "license": "MIT" }, + "node_modules/react-infinite-scroll-hook": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/react-infinite-scroll-hook/-/react-infinite-scroll-hook-5.0.1.tgz", + "integrity": "sha512-fn6+8BAZLQ9C1fvO5kPicGjDR2WHxK7rP4aaSWuaJkvtoJjYuudGJ9wjgPox7dghKm5Xj9cpKFycM86/wAJ3ig==", + "dependencies": { + "react-intersection-observer-hook": "^3.0.0" + }, + "peerDependencies": { + "react": ">=16.8.0", + "react-dom": ">=16.8.0" + } + }, + "node_modules/react-intersection-observer-hook": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/react-intersection-observer-hook/-/react-intersection-observer-hook-3.0.1.tgz", + "integrity": "sha512-nehzFUV+jVY49tc9FOm4yJHrmFFtCmsBZKiN1rUUB2675CMBf6kKu1ekuTqvCnmh4XG7RFP3bMyAevfqfV94xw==", + "peerDependencies": { + "react": ">=16.8.0", + "react-dom": ">=16.8.0" + } + }, "node_modules/react-is": { "version": "17.0.2", "license": "MIT" @@ -13678,6 +13700,20 @@ "react-fast-compare": { "version": "3.2.0" }, + "react-infinite-scroll-hook": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/react-infinite-scroll-hook/-/react-infinite-scroll-hook-5.0.1.tgz", + "integrity": "sha512-fn6+8BAZLQ9C1fvO5kPicGjDR2WHxK7rP4aaSWuaJkvtoJjYuudGJ9wjgPox7dghKm5Xj9cpKFycM86/wAJ3ig==", + "requires": { + "react-intersection-observer-hook": "^3.0.0" + } + }, + "react-intersection-observer-hook": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/react-intersection-observer-hook/-/react-intersection-observer-hook-3.0.1.tgz", + "integrity": "sha512-nehzFUV+jVY49tc9FOm4yJHrmFFtCmsBZKiN1rUUB2675CMBf6kKu1ekuTqvCnmh4XG7RFP3bMyAevfqfV94xw==", + "requires": {} + }, "react-is": { "version": "17.0.2" }, diff --git a/package.json b/package.json index 83fce011..54af6bbb 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "hhd-ui", - "version": "0.18.1", + "version": "0.18.2", "private": true, "dependencies": { "@ag-grid-community/client-side-row-model": "^30.0.3", @@ -41,6 +41,7 @@ "react-csv": "^2.2.2", "react-datepicker": "^4.16.0", "react-dom": "^18.2.0", + "react-infinite-scroll-hook": "^5.0.1", "react-papaparse": "^4.1.0", "react-plotly.js": "^2.6.0", "react-resizable-panels": "^2.0.7", diff --git a/src/app-components/domain-select.jsx b/src/app-components/domain-select.jsx index cec8690b..e5e4d0e5 100644 --- a/src/app-components/domain-select.jsx +++ b/src/app-components/domain-select.jsx @@ -29,20 +29,21 @@ const DomainSelect = ({ return (
- el[useLabelAsDefault ? 'label' : 'value'] === defaultValue)} - isOptionEqualToValue={(opt, val) => opt.value === val.value} - onChange={(_e, value) => { - const item = domainsByGroup[domain]?.find(el => el.value === value?.label); - onChange(item); - }} - renderInput={(params) => } - options={options} - fullWidth - /> + {!isLoading && ( + el[useLabelAsDefault ? 'label' : 'value'] === defaultValue)} + isOptionEqualToValue={(opt, val) => opt.value === val.value} + onChange={(_e, value) => { + const item = domainsByGroup[domain]?.find(el => el.value === value?.label); + onChange(item); + }} + renderInput={(params) => } + options={options} + fullWidth + /> + )}
); }; diff --git a/src/app-pages/instrument/cwms-timeseries/newCwmsTimeseries.jsx b/src/app-pages/instrument/cwms-timeseries/newCwmsTimeseries.jsx index 1c969791..a6ed5a70 100644 --- a/src/app-pages/instrument/cwms-timeseries/newCwmsTimeseries.jsx +++ b/src/app-pages/instrument/cwms-timeseries/newCwmsTimeseries.jsx @@ -1,7 +1,9 @@ -import React, { useMemo, useState } from 'react'; -import { Autocomplete, TextField } from '@mui/material'; +import React, { useCallback, useMemo, useState } from 'react'; +import { Autocomplete, Box, createFilterOptions, TextField } from '@mui/material'; import { connect } from 'redux-bundler-react'; import { useQueryClient } from '@tanstack/react-query'; +import { useDebounce } from 'react-use'; +import useInfiniteScroll from 'react-infinite-scroll-hook'; import * as Modal from '../../../app-components/modal'; import DomainSelect from '../../../app-components/domain-select.jsx'; @@ -17,17 +19,40 @@ const generateOfficesOptions = cwmsOffices => { }; const generateCwmsTimeseriesOptions = cwmsTimeseries => { - const { total, entries } = cwmsTimeseries || {}; + const { pages } = cwmsTimeseries || {}; - if (!total || !entries) return []; + if (!pages?.length) return []; - return entries.map(entry => ({ + const entries = pages.map(p => p.entries).flat(); + + const elements = entries.map((entry, i) => ({ _extents: entry?.extents?.length ? entry.extents[0] : {}, label: entry?.name, value: entry?.name, + isLast: pages.at(-1)['next-page'] && i === entries.length - 1, })); + + return elements; +}; + +const renderListItem = (props, option, _state, ownerState, sentryRef) => { + const { key, ...optionProps } = props; + const { isLast } = option; + + return ( + + {ownerState.getOptionLabel(option)} + + ); }; +const _filterOptions = createFilterOptions(); + const NewCwmsTimeseriesModal = connect( 'doModalClose', 'doInstrumentTimeseriesDelete', @@ -45,6 +70,8 @@ const NewCwmsTimeseriesModal = connect( const [selectedOffice, setSelectedOffice] = useState(isEdit ? item?.cwms_office_id : ''); const [selectedTimeseries, setSelectedTimeseries] = useState(isEdit ? item?.cwms_timeseries_id : ''); + const [input, setInput] = useState(''); + const [likeQuery, setLikeQuery] = useState(''); const [name, setName] = useState(isEdit ? item?.name : ''); const [parameterId, setParameterId] = useState(isEdit ? item?.parameter_id : ''); const [unitId, setUnitId] = useState(isEdit ? item?.unit_id : ''); @@ -52,11 +79,40 @@ const NewCwmsTimeseriesModal = connect( const { data: cwmsOffices } = useGetCwmsOffices({}); const cwmsOfficesOptions = useMemo(() => generateOfficesOptions(cwmsOffices), [cwmsOffices]); - const { data: cwmsTimeseries } = useGetCwmsTimeseries({ office: selectedOffice?.value || selectedOffice }, { - enabled: !!selectedOffice?.value || !!selectedOffice, - }); + const { data: cwmsTimeseries, error, isFetching, fetchNextPage } = useGetCwmsTimeseries( + { + office: selectedOffice?.value || selectedOffice, + like: likeQuery, + }, { enabled: !!selectedOffice?.value || !!selectedOffice } + ); + const cwmsTimeseriesOptions = useMemo(() => generateCwmsTimeseriesOptions(cwmsTimeseries), [cwmsTimeseries]); + const [optionCount, setOptionCount] = useState(0); + const filterOptions = useCallback((options, state) => { + const results = _filterOptions(options, state); + + if (optionCount !== results.length) { + setOptionCount(results.length); + } + + return results; + }); + + const [,] = useDebounce(() => { + setLikeQuery(input); + }, 300, [input]); + + const [sentryRef] = useInfiniteScroll({ + loading: isFetching, + hasNextPage: !!(cwmsTimeseries?.pages?.at(-1) || {})['next-page'], + disabled: !!error, + onLoadMore: () => { + fetchNextPage(); + }, + rootMargin: '0px 0px 100px 0px', + }); + return ( @@ -66,6 +122,7 @@ const NewCwmsTimeseriesModal = connect( opt.value === val || opt.value === val.value} onChange={(_e, value) => setSelectedOffice(value)} renderInput={(params) => } @@ -74,14 +131,26 @@ const NewCwmsTimeseriesModal = connect( /> {selectedOffice && ( { + setInput(newInputValue); + }} isOptionEqualToValue={(opt, val) => opt.value === val || opt.value === val.value} onChange={(_e, value) => setSelectedTimeseries(value)} - renderInput={(params) => } + renderOption={(props, option, state, ownerState) => renderListItem(props, option, state, ownerState, sentryRef)} + renderInput={(params) => ( + + )} options={cwmsTimeseriesOptions} - fullWidth /> )} {selectedTimeseries && ( diff --git a/src/app-services/collections/cwms-timeseries.ts b/src/app-services/collections/cwms-timeseries.ts index 4f6f2d57..25d57f8b 100644 --- a/src/app-services/collections/cwms-timeseries.ts +++ b/src/app-services/collections/cwms-timeseries.ts @@ -1,4 +1,4 @@ -import { QueryClient, useMutation, useQuery } from '@tanstack/react-query'; +import { QueryClient, useInfiniteQuery, useMutation, useQuery } from '@tanstack/react-query'; import { apiGet, apiPost, apiPut, buildQueryParams } from '../fetch-helpers'; @@ -19,6 +19,8 @@ interface OfficeParams { interface CwmsTimeseriesParams { office: string; + like?: string; + pageParam?: string; } interface CwmsTimeseriesMeasurementParams { @@ -52,12 +54,17 @@ export const useGetCwmsOffices = ({ hasData = true }: OfficeParams, opts: Client }); }; -export const useGetCwmsTimeseries = ({ office }: CwmsTimeseriesParams, opts: ClientQueryOptions) => { - const uri = `/catalog/TIMESERIES?office=${office}&page-size=5000`; - - return useQuery({ - queryKey: [`cwmsTimeseries`, office], - queryFn: () => apiGet(uri, 'CWMS'), +export const useGetCwmsTimeseries = ({ office, like }: CwmsTimeseriesParams, opts: ClientQueryOptions) => { + const buildUri = (pageParam?: string) => { + const query = buildQueryParams({ office, like, page: pageParam, 'page-size': 500 }); + return `/catalog/TIMESERIES${query}`; + }; + + return useInfiniteQuery({ + queryKey: [`cwmsTimeseries`, office, like], + queryFn: ({ pageParam }) => apiGet(buildUri(pageParam), 'CWMS'), + initialPageParam: undefined, + getNextPageParam: (lastPage) => lastPage['next-page'], ...opts, }); }; diff --git a/src/app-services/fetch-helpers.ts b/src/app-services/fetch-helpers.ts index 1ac7c726..90a8af3a 100644 --- a/src/app-services/fetch-helpers.ts +++ b/src/app-services/fetch-helpers.ts @@ -21,7 +21,7 @@ interface CommonItems { export const buildQueryParams = (params: Record) => { const keys = Object.keys(params); const mapped = keys.map(key => { - if (!key || params[key] === undefined) return null; + if (!key || params[key] === undefined || params[key] === '') return null; return `${key}=${params[key]}`; }).filter(e => e);