UNPKG

react-use-query-params

Version:

Strongly typed, routing-library agnostic react hook to use and manipulate query params

156 lines 21.6 kB
import { useCallback, useEffect, useMemo, useReducer, useRef } from 'react'; const listeners = new Set(); function runListeners() { listeners.forEach((listener) => { listener(); }); } (function (history) { function intercept(func) { return function (...args) { const returnedValue = func.apply(history, args); runListeners(); return returnedValue; }; } history.pushState = intercept(history.pushState); history.replaceState = intercept(history.replaceState); // to handle browser's native backward and forward functionality window.addEventListener('popstate', () => { runListeners(); }); })(window.history); export function applyQueryParams(target, queryParams, removeExtras = false) { const params = target instanceof URL ? target.searchParams : target; if (removeExtras) { params.forEach((value, key) => { if (!(key in queryParams)) { params.delete(key); } }); } for (const [key, values] of Object.entries(queryParams)) { const usableValues = Array.isArray(values) ? values : [values]; params.delete(key); for (const value of usableValues) { params.append(key, value !== null && value !== void 0 ? value : ''); } } return target; } export function useQueryParams() { const currentLocation = window.location.href; const urlSearchParams = useMemo(() => { return new URLSearchParams(window.location.search); }, [currentLocation]); const watching = useRef({}); const watchingLength = useRef(false); const pauseWatch = useRef(false); // stores both the key and the values as an array // of the params that are being watched const watch = useCallback((key) => { if (key in watching.current || pauseWatch.current || watchingLength.current) { return; } watching.current[key] = urlSearchParams.getAll(String(key)); }, [urlSearchParams, watching]); const clearWatch = useCallback(() => { watching.current = {}; watchingLength.current = false; }, [watching]); // React's officially recommended way of forcing a rerender const [, increment] = useReducer((state) => { return state + 1; }, 0); const rerender = useCallback(() => { increment(); }, [increment]); // Handles location changes via listener mechanism above. const handle = useCallback(() => { const currentParams = new URLSearchParams(window.location.search); let shouldRerender = false; let length = 0; for (const [key, values] of Object.entries(watching.current)) { const currentValues = currentParams.getAll(key); if (currentValues.length !== (values === null || values === void 0 ? void 0 : values.length)) { shouldRerender = true; break; } // the first mismatched value means we need to rerender for (let i = 0; i < currentValues.length; i++) { if (currentValues[i] !== values[i]) { shouldRerender = true; break; } } length += 1; } shouldRerender = shouldRerender || (watchingLength.current !== false && watchingLength.current !== length); if (shouldRerender) { clearWatch(); rerender(); return; } }, [watching, clearWatch, rerender]); useEffect(() => { // listening on the global window object // via interceptions and listeners. listeners.add(handle); return () => { listeners.delete(handle); }; }, [handle]); const params = useMemo(() => { return new Proxy({}, { get(target, key) { watch(key); return urlSearchParams.getAll(key); }, ownKeys(target) { const keys = new Set(); urlSearchParams.forEach((value, key) => { keys.add(key); }); if (!pauseWatch.current) { watchingLength.current = keys.size; } return [...keys]; }, getOwnPropertyDescriptor(target, prop) { return { configurable: true, enumerable: true, writable: false }; }, has(target, key) { watch(key); return urlSearchParams.has(key); }, }); }, [urlSearchParams]); const setParams = useCallback((nextParams, replace = false) => { try { const nextURL = new URL(window.location.href); pauseWatch.current = true; const nextParamsObject = nextParams instanceof Function ? nextParams(params) : nextParams; pauseWatch.current = false; applyQueryParams(nextURL, nextParamsObject, true); if (replace) { window.history.replaceState(null, '', nextURL); } else { window.history.pushState(null, '', nextURL); } } catch (error) { console.error('Error while setting query params', error); } pauseWatch.current = false; }, [params]); return [params, setParams]; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"useQueryParams.js","sourceRoot":"./src/","sources":["useQueryParams.ts"],"names":[],"mappings":"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","sourcesContent":["import {useCallback, useEffect, useMemo, useReducer, useRef} from 'react';\n\nconst listeners = new Set<() => void>();\n\nfunction runListeners() {\n    listeners.forEach((listener) => {\n        listener();\n    });\n}\n\n(function (history) {\n    function intercept(func: any) {\n        return function (...args: any[]) {\n            const returnedValue = func.apply(history, args);\n            runListeners();\n\n            return returnedValue;\n        };\n    }\n\n    history.pushState = intercept(history.pushState);\n    history.replaceState = intercept(history.replaceState);\n\n    // to handle browser's native backward and forward functionality\n    window.addEventListener('popstate', () => {\n        runListeners();\n    });\n})(window.history);\n\nexport type TDefaultParamsObject = Record<string, any>;\n\nexport type TAllParams<PARAMS extends TDefaultParamsObject> = {\n    [key in keyof PARAMS]: string[];\n};\n\nexport function applyQueryParams<PARAMS extends TDefaultParamsObject>(\n    target: URL | URLSearchParams,\n    queryParams: Partial<TAllParams<PARAMS>>,\n    removeExtras: boolean = false,\n) {\n    const params = target instanceof URL ? target.searchParams : target;\n\n    if (removeExtras) {\n        params.forEach((value, key) => {\n            if (!(key in queryParams)) {\n                params.delete(key);\n            }\n        });\n    }\n\n    for (const [key, values] of Object.entries(queryParams)) {\n        const usableValues = Array.isArray(values) ? values : [values];\n        params.delete(key);\n\n        for (const value of usableValues) {\n            params.append(key, value ?? '');\n        }\n    }\n\n    return target;\n}\n\nexport function useQueryParams<\n    PARAMS extends TDefaultParamsObject = TDefaultParamsObject,\n>() {\n    const currentLocation = window.location.href;\n\n    const urlSearchParams = useMemo(() => {\n        return new URLSearchParams(window.location.search);\n    }, [currentLocation]);\n\n    const watching = useRef<{\n        [key in keyof PARAMS]?: string[];\n    }>({});\n\n    const watchingLength = useRef<number | boolean>(false);\n    const pauseWatch = useRef<boolean>(false);\n\n    // stores both the key and the values as an array\n    // of the params that are being watched\n    const watch = useCallback(\n        (key: keyof PARAMS) => {\n            if (\n                key in watching.current ||\n                pauseWatch.current ||\n                watchingLength.current\n            ) {\n                return;\n            }\n\n            watching.current[key] = urlSearchParams.getAll(String(key));\n        },\n        [urlSearchParams, watching],\n    );\n\n    const clearWatch = useCallback(() => {\n        watching.current = {};\n        watchingLength.current = false;\n    }, [watching]);\n\n    // React's officially recommended way of forcing a rerender\n    const [, increment] = useReducer((state) => {\n        return state + 1;\n    }, 0);\n\n    const rerender = useCallback(() => {\n        increment();\n    }, [increment]);\n\n    // Handles location changes via listener mechanism above.\n    const handle = useCallback(() => {\n        const currentParams = new URLSearchParams(window.location.search);\n\n        let shouldRerender = false;\n        let length = 0;\n\n        for (const [key, values] of Object.entries(watching.current)) {\n            const currentValues = currentParams.getAll(key);\n\n            if (currentValues.length !== values?.length) {\n                shouldRerender = true;\n                break;\n            }\n\n            // the first mismatched value means we need to rerender\n            for (let i = 0; i < currentValues.length; i++) {\n                if (currentValues[i] !== values[i]) {\n                    shouldRerender = true;\n                    break;\n                }\n            }\n\n            length += 1;\n        }\n\n        shouldRerender =\n            shouldRerender ||\n            (watchingLength.current !== false &&\n                watchingLength.current !== length);\n\n        if (shouldRerender) {\n            clearWatch();\n            rerender();\n            return;\n        }\n    }, [watching, clearWatch, rerender]);\n\n    useEffect(() => {\n        // listening on the global window object\n        // via interceptions and listeners.\n        listeners.add(handle);\n\n        return () => {\n            listeners.delete(handle);\n        };\n    }, [handle]);\n\n    const params = useMemo(() => {\n        return new Proxy({} as TAllParams<PARAMS>, {\n            get(target, key: string): string[] {\n                watch(key);\n                return urlSearchParams.getAll(key);\n            },\n            ownKeys(target) {\n                const keys = new Set<string>();\n\n                urlSearchParams.forEach((value, key) => {\n                    keys.add(key);\n                });\n\n                if (!pauseWatch.current) {\n                    watchingLength.current = keys.size;\n                }\n\n                return [...keys];\n            },\n            getOwnPropertyDescriptor(target, prop) {\n                return {configurable: true, enumerable: true, writable: false};\n            },\n            has(target, key: string) {\n                watch(key);\n                return urlSearchParams.has(key);\n            },\n        });\n    }, [urlSearchParams]);\n\n    const setParams = useCallback(\n        (\n            nextParams:\n                | TAllParams<PARAMS>\n                | ((\n                      current: TAllParams<PARAMS>,\n                  ) => Partial<TAllParams<PARAMS>>),\n            replace: boolean = false,\n        ) => {\n            try {\n                const nextURL = new URL(window.location.href);\n\n                pauseWatch.current = true;\n\n                const nextParamsObject =\n                    nextParams instanceof Function\n                        ? nextParams(params)\n                        : nextParams;\n\n                pauseWatch.current = false;\n\n                applyQueryParams(nextURL, nextParamsObject, true);\n\n                if (replace) {\n                    window.history.replaceState(null, '', nextURL);\n                } else {\n                    window.history.pushState(null, '', nextURL);\n                }\n            } catch (error) {\n                console.error('Error while setting query params', error);\n            }\n\n            pauseWatch.current = false;\n        },\n        [params],\n    );\n\n    return [params, setParams] as const;\n}\n"]}