diff --git a/src/hooks.ts b/src/hooks.ts index facaf65ef..7a3341d86 100644 --- a/src/hooks.ts +++ b/src/hooks.ts @@ -3,6 +3,7 @@ import { type SetStateAction, useCallback, useEffect, + useRef, useState, } from 'react'; import { history } from '@edx/frontend-platform'; @@ -85,10 +86,13 @@ export const useLoadOnScroll = ( }; /** - * Hook which stores state variables in the URL search parameters. - * - * It wraps useState with functions that get/set a query string - * search parameter when returning/setting the state variable. + * Types used by the useListHelpers and useStateWithUrlSearchParam hooks. + */ +export type FromStringFn = (value: string | null) => Type | undefined; +export type ToStringFn = (value: Type | undefined) => string | undefined; + +/** + * Hook that stores/retrieves state variables using the URL search parameters. * * @param defaultValue: Type * Returned when no valid value is found in the url search parameter. @@ -101,26 +105,103 @@ export const useLoadOnScroll = ( export function useStateWithUrlSearchParam( defaultValue: Type, paramName: string, - fromString: (value: string | null) => Type | undefined, - toString: (value: Type) => string | undefined, + fromString: FromStringFn, + toString: ToStringFn, ): [value: Type, setter: Dispatch>] { + // STATE WORKAROUND: + // If we use this hook to control multiple state parameters on the same + // page, we can run into state update issues. Because our state variables + // are actually stored in setSearchParams, and not in separate variables like + // useState would do, the searchParams "previous" state may not be updated + // for sequential calls to returnSetter in the same render loop (like in + // SearchManager's clearFilters). + // + // One workaround could be to use window.location.search as the "previous" + // value when returnSetter constructs the new URLSearchParams. This works + // fine with BrowserRouter, but our test suite uses MemoryRouter, and that + // router doesn't store URL search params, cf + // https://github.com/remix-run/react-router/issues/9757 + // + // So instead, we maintain a reference to the current useLocation() + // object, and use its search params as the "previous" value when + // initializing URLSearchParams. + const location = useLocation(); + const locationRef = useRef(location); const [searchParams, setSearchParams] = useSearchParams(); + const returnValue: Type = fromString(searchParams.get(paramName)) ?? defaultValue; - // Function to update the url search parameter - const returnSetter: Dispatch> = useCallback((value: Type) => { - setSearchParams((prevParams) => { - const paramValue: string = toString(value) ?? ''; - const newSearchParams = new URLSearchParams(prevParams); - // If using the default paramValue, remove it from the search params. - if (paramValue === defaultValue) { + // Update the url search parameter using: + type ReturnSetterParams = ( + // a Type value + value?: Type + // or a function that returns a Type from the previous returnValue + | ((value: Type) => Type) + ) => void; + const returnSetter: Dispatch> = useCallback((value) => { + setSearchParams((/* prev */) => { + const useValue = value instanceof Function ? value(returnValue) : value; + const paramValue = toString(useValue); + const newSearchParams = new URLSearchParams(locationRef.current.search); + // If the provided value was invalid (toString returned undefined) + // or the same as the defaultValue, remove it from the search params. + if (paramValue === undefined || paramValue === defaultValue) { newSearchParams.delete(paramName); } else { newSearchParams.set(paramName, paramValue); } + + // Update locationRef + locationRef.current.search = newSearchParams.toString(); + return newSearchParams; }, { replace: true }); - }, [setSearchParams]); + }, [returnValue, setSearchParams]); // Return the computed value and wrapped set state function return [returnValue, returnSetter]; } + +/** + * Helper hook for useStateWithUrlSearchParam. + * + * useListHelpers provides toString and fromString handlers that can: + * - split/join a list of values using a separator string, and + * - validate each value using the provided functions, omitting any invalid values. + * + * @param fromString + * Serialize a string to a Type, or undefined if not valid. + * @param toString + * Deserialize a Type to a string. + * @param separator : string to use when splitting/joining the types. + * Defaults value is ','. + */ +export function useListHelpers({ + fromString, + toString, + separator = ',', +}: { + fromString: FromStringFn, + toString: ToStringFn, + separator?: string; +}): [ FromStringFn, ToStringFn ] { + const isType = (item: Type | undefined): item is Type => item !== undefined; + + // Split the given string with separator, + // and convert the parts to a list of Types, omiting any invalid Types. + const fromStringToList : FromStringFn = (value: string) => ( + value + ? value.split(separator).map(fromString).filter(isType) + : [] + ); + // Convert an array of Types to strings and join with separator. + // Returns undefined if the given list contains no valid Types. + const fromListToString : ToStringFn = (value: Type[]) => { + const stringValues = value.map(toString).filter((val) => val !== undefined); + return ( + stringValues && stringValues.length + ? stringValues.join(separator) + : undefined + ); + }; + return [fromStringToList, fromListToString]; +} diff --git a/src/search-manager/SearchManager.ts b/src/search-manager/SearchManager.ts index acf5a0519..5a44db4ea 100644 --- a/src/search-manager/SearchManager.ts +++ b/src/search-manager/SearchManager.ts @@ -12,7 +12,35 @@ import { CollectionHit, ContentHit, SearchSortOption, forceArray, } from './data/api'; import { useContentSearchConnection, useContentSearchResults } from './data/apiHooks'; -import { useStateWithUrlSearchParam } from '../hooks'; +import { + type FromStringFn, + type ToStringFn, + useListHelpers, + useStateWithUrlSearchParam, +} from '../hooks'; + +/** + * Typed hook that returns useState if skipUrlUpdate, + * or useStateWithUrlSearchParam if it's not. + * + * Provided here to reduce some code overhead in SearchManager. + */ +function useStateOrUrlSearchParam( + defaultValue: Type, + paramName: string, + fromString: FromStringFn, + toString: ToStringFn, + skipUrlUpdate?: boolean, +): [value: Type, setter: React.Dispatch>] { + const useStateManager = React.useState(defaultValue); + const urlStateManager = useStateWithUrlSearchParam( + defaultValue, + paramName, + fromString, + toString, + ); + return skipUrlUpdate ? useStateManager : urlStateManager; +} export interface SearchContextData { client?: MeiliSearch; @@ -58,29 +86,47 @@ export const SearchContextProvider: React.FC<{ overrideSearchSortOrder, skipBlockTypeFetch, skipUrlUpdate, ...props }) => { // Search parameters can be set via the query string - // E.g. q=draft+text - // TODO -- how to scrub search terms? - const keywordStateManager = React.useState(''); - const keywordUrlStateManager = useStateWithUrlSearchParam( + // E.g. ?q=draft+text + // TODO -- how to sanitize search terms? + const [searchKeywords, setSearchKeywords] = useStateOrUrlSearchParam( '', 'q', (value: string) => value || '', (value: string) => value || '', - ); - const [searchKeywords, setSearchKeywords] = ( - skipUrlUpdate - ? keywordStateManager - : keywordUrlStateManager + skipUrlUpdate, ); const [blockTypesFilter, setBlockTypesFilter] = React.useState([]); const [problemTypesFilter, setProblemTypesFilter] = React.useState([]); - const [tagsFilter, setTagsFilter] = React.useState([]); - const [usageKey, setUsageKey] = useStateWithUrlSearchParam( + // Tags can be almost any string value, except our separator (|) + // TODO how to sanitize tags? + // E.g ?tags=Skills+>+Abilities|Skills+>+Knowledge + const sanitizeTag = (value: string | null | undefined): string | undefined => ( + (value && /^[^|]+$/.test(value)) + ? value + : undefined + ); + const [tagToList, listToTag] = useListHelpers({ + toString: sanitizeTag, + fromString: sanitizeTag, + separator: '|', + }); + const [tagsFilter, setTagsFilter] = useStateOrUrlSearchParam( + [], + 'tags', + tagToList, + listToTag, + skipUrlUpdate, + ); + + // E.g ?usageKey=lb:OpenCraft:libA:problem:5714eb65-7c36-4eee-8ab9-a54ed5a95849 + const [usageKey, setUsageKey] = useStateOrUrlSearchParam( '', 'usageKey', + // TODO should sanitize usageKeys too. (value: string) => value, (value: string) => value, + skipUrlUpdate, ); let extraFilter: string[] = forceArray(props.extraFilter); @@ -88,21 +134,15 @@ export const SearchContextProvider: React.FC<{ extraFilter = union(extraFilter, [`usage_key = "${usageKey}"`]); } - // The search sort order can be set via the query string - // E.g. ?sort=display_name:desc maps to SearchSortOption.TITLE_ZA. // Default sort by Most Relevant if there's search keyword(s), else by Recently Modified. const defaultSearchSortOrder = searchKeywords ? SearchSortOption.RELEVANCE : SearchSortOption.RECENTLY_MODIFIED; - let sortStateManager = React.useState(defaultSearchSortOrder); - const sortUrlStateManager = useStateWithUrlSearchParam( + const [searchSortOrder, setSearchSortOrder] = useStateOrUrlSearchParam( defaultSearchSortOrder, 'sort', (value: string) => Object.values(SearchSortOption).find((enumValue) => value === enumValue), (value: SearchSortOption) => value.toString(), + skipUrlUpdate, ); - if (!skipUrlUpdate) { - sortStateManager = sortUrlStateManager; - } - const [searchSortOrder, setSearchSortOrder] = sortStateManager; // SearchSortOption.RELEVANCE is special, it means "no custom sorting", so we // send it to useContentSearchResults as an empty array. const searchSortOrderToUse = overrideSearchSortOrder ?? searchSortOrder;