{"version":3,"file":"searchParams-BHYM9cZ7.js","sources":["../../src/utils/searchParams.ts"],"sourcesContent":["import { useCallback, useMemo } from 'react';\nimport { useLocation, useNavigate } from 'react-router-dom';\n\nimport { isUndefined } from 'typeDeclarations/typeGuards';\n\nexport type SearchParameterValue = string | string[];\nexport type SearchParametersDictionary = Record;\n\nexport type UncleanSearchParameterValue = SearchParameterValue | undefined;\nexport type UncleanSearchParametersDictionary = Record;\n\n/**\n * sanitizes an unclean search param value into an expected string value.\n * if there are multiple entries for the param, will default to the first entry found.\n * @param value the unclean value\n * @param defaultValue the default to be used if the search param no longer exists\n */\nexport function sanitizeToSingleValue(value: UncleanSearchParameterValue, defaultValue: string): string;\nexport function sanitizeToSingleValue(value: UncleanSearchParameterValue, defaultValue?: string): string | undefined;\nexport function sanitizeToSingleValue(\n value: UncleanSearchParameterValue,\n defaultValue?: string | undefined,\n): string | undefined {\n if (isUndefined(value)) {\n return defaultValue;\n }\n\n if (Array.isArray(value)) {\n return value[0];\n }\n\n return value;\n}\n\n/**\n * sanitizes an unclean search param value into an expected string array value.\n * if there is only one entry for the param, will default to an array of a single element,\n * which will be the value of the entry.\n * @param value the unclean value\n * @param defaultValue the default to be used if the search param no longer exists\n */\nexport function sanitizeToArrayValue(value: UncleanSearchParameterValue, defaultValue: string[]): string[];\nexport function sanitizeToArrayValue(value: UncleanSearchParameterValue, defaultValue?: string[]): string[] | undefined;\nexport function sanitizeToArrayValue(\n value: UncleanSearchParameterValue,\n defaultValue: string[] | undefined,\n): string[] | undefined {\n if (isUndefined(value)) {\n return defaultValue;\n }\n\n if (!Array.isArray(value)) {\n return [value];\n }\n\n return value;\n}\n\n/**\n * returns a URLSearchParams instance from the current search string\n */\nexport function useSearchParams(): URLSearchParams {\n const { search } = useLocation();\n\n // only create a new instance if the search string changes\n const memoizedSearchParams = useMemo(() => new URLSearchParams(search), [search]);\n\n return memoizedSearchParams;\n}\n\n/**\n * helper function that consolidates the values from a given list of pairs into a dictionary.\n * if the entry doesn't exist in the dictionary, it will be created with the value.\n * if the entry already exists, a new list will be created with the new value appended.\n * @param pairs the `key: value` pairs to consolidate\n */\nfunction consolidateValues(pairs: Array<[string, string]>): UncleanSearchParametersDictionary {\n const dict: UncleanSearchParametersDictionary = {};\n\n pairs.forEach(([key, value]) => {\n const dictValue = dict[key];\n if (isUndefined(dictValue)) {\n dict[key] = value;\n } else if (Array.isArray(dictValue)) {\n dict[key] = [...dictValue, value];\n } else {\n dict[key] = [dictValue, value];\n }\n });\n\n return dict;\n}\n\n/**\n * Generates a dictionary from a given `URLSearchParams` instance.\n *\n * Collapses all entries with the same key into `key: value[]` pairs.\n *\n * If there are no entries for a search param, the value will be `undefined`\n *\n * @param searchParams the `URLSearchParams` instance\n */\nexport function generateSearchParamsDictionary(searchParams: URLSearchParams): UncleanSearchParametersDictionary {\n const pairs = Array.from(searchParams);\n return consolidateValues(pairs);\n}\n\n/**\n * Returns a dictionary of the current search params.\n */\nexport function useSearchParamsDictionary(): UncleanSearchParametersDictionary {\n const searchParams = useSearchParams();\n\n // only generate a new dictionary if the searchParams change\n return useMemo(() => generateSearchParamsDictionary(searchParams), [searchParams]);\n}\n\nexport type SearchParameterSetter = (searchParamsDictionary: SearchParametersDictionary) => void;\n\n/**\n * Returns a callback that converts a dictionary of search params into a search string\n * and then redirects.\n */\nexport function useSearchParamsSetter(): SearchParameterSetter {\n const navigate = useNavigate();\n\n return useCallback(\n (searchParamsDictionary) => {\n // can't use these from the router hooks\n // they retain old values when calling this function in rapid succession\n const { search, pathname } = window.location;\n\n // grab current search params\n const searchParams = new URLSearchParams(search);\n\n // set new params or replace existing param values\n Object.entries(searchParamsDictionary).forEach(([key, value]) => {\n // doesn't matter if it's a string or array.\n // if it doesn't have a length, delete the entry.\n if (!value.length) {\n searchParams.delete(key);\n return;\n }\n\n if (Array.isArray(value)) {\n // delete all current entries for that key\n searchParams.delete(key);\n // for each item in the list, append an entry for the same key.\n value.forEach((item) => searchParams.append(key, item));\n } else {\n // if it's not an array, replace the entry\n searchParams.set(key, value);\n }\n });\n\n // let the platform automatically encode the search string\n const searchString = searchParams.toString();\n\n // redirect to current path with a new search string\n navigate(`${pathname}?${searchString}`);\n },\n [navigate],\n );\n}\n\n/**\n * Utility that aggregates both a search params dictionary and setter\n * to mimic state hooks for ease of use. Does not use initial state (intentional).\n * Returns a tuple of `[currentSearchParamsDictionary, searchParamsSetter]`\n */\nexport function useSearchParamsTuple(): [UncleanSearchParametersDictionary, SearchParameterSetter] {\n const searchParamsDictionary = useSearchParamsDictionary();\n const searchParamsSetter = useSearchParamsSetter();\n\n return useMemo(() => [searchParamsDictionary, searchParamsSetter], [searchParamsDictionary, searchParamsSetter]);\n}\n"],"names":["sanitizeToSingleValue","value","defaultValue","isUndefined","useSearchParams","search","useLocation","useMemo","consolidateValues","pairs","dict","key","dictValue","generateSearchParamsDictionary","searchParams","useSearchParamsDictionary"],"mappings":"iDAmBgB,SAAAA,EACdC,EACAC,EACoB,CAChB,OAAAC,EAAYF,CAAK,EACZC,EAGL,MAAM,QAAQD,CAAK,EACdA,EAAM,CAAC,EAGTA,CACT,CA6BO,SAASG,GAAmC,CAC3C,KAAA,CAAE,OAAAC,GAAWC,IAKZ,OAFsBC,UAAQ,IAAM,IAAI,gBAAgBF,CAAM,EAAG,CAACA,CAAM,CAAC,CAGlF,CAQA,SAASG,EAAkBC,EAAmE,CAC5F,MAAMC,EAA0C,CAAA,EAEhD,OAAAD,EAAM,QAAQ,CAAC,CAACE,EAAKV,CAAK,IAAM,CACxB,MAAAW,EAAYF,EAAKC,CAAG,EACtBR,EAAYS,CAAS,EACvBF,EAAKC,CAAG,EAAIV,EACH,MAAM,QAAQW,CAAS,EAChCF,EAAKC,CAAG,EAAI,CAAC,GAAGC,EAAWX,CAAK,EAEhCS,EAAKC,CAAG,EAAI,CAACC,EAAWX,CAAK,CAC/B,CACD,EAEMS,CACT,CAWO,SAASG,EAA+BC,EAAkE,CACzG,MAAAL,EAAQ,MAAM,KAAKK,CAAY,EACrC,OAAON,EAAkBC,CAAK,CAChC,CAKO,SAASM,GAA+D,CAC7E,MAAMD,EAAeV,IAGrB,OAAOG,EAAAA,QAAQ,IAAMM,EAA+BC,CAAY,EAAG,CAACA,CAAY,CAAC,CACnF"}