-
-
Notifications
You must be signed in to change notification settings - Fork 10.4k
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Feature]: Take a SetStateAction in useSearchParams setter #8287
Comments
Would be awesome to merge #8344 I'm missing this functionality as well. Coming from v5 and using |
Do you have any news about this feature ? |
Adapting @zmthy snippet, we can make it work using function useSearchParam(key: string) {
const [search, setSearch] = useSearchParams();
const setter = useCallback((value) => {
const params = createSearchParams(search);
params.set(key, value);
setSearch(params);
}, [search, setSearch, key]);
return [createSearchParams(search).get(key), setter];
}; |
@zmthy My solution to this is:
This spreads the object created from the searchParams key-value pairs into the new setSearchParams object. Note that Object.fromEntries is not IE compatible. |
Thanks to #7586, you can now solve this problem with the You can then reference the Assuming the export function useSearchParamsUpdate(): [
URLSearchParams,
Dispatch<(params: URLSearchParams) => URLSearchParamsInit>
] {
const [params, setParams] = useSearchParams();
return [
params,
useCallback(
(update) =>
setParams(update(new URLSearchParams(history.location.search))),
[setParams]
),
];
} This can be used to implement I'm not sure if it's reasonable to close the issue just yet, because it's inconvenient that you have to set up the router with your own managed history instead of using browser router, and the history router interface is still unstable. Ideally there would be a way to at least get a location ref with a stable identity and a mutable reference to the current location across any of the routers. |
I'm going to convert this to a discussion so it can go through our new Open Development process. Please upvote the new Proposal if you'd like to see this considered! |
This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
What is the new or updated feature that you are suggesting?
It would be great if the setter function returned by
useSearchParams
could accept a function that takes the current search params and builds a new one, like the setter returned byuseState
.The input type of the first argument of the setter would be expanded from
URLSearchParamsInit
toSetStateAction<URLSearchParamsInit>
.The resulting function you get from the
useCallback
in the example above now has a stable identity, and does not have to change asparams
changes in order to preserve search params other than"abc"
.Why should this feature be included?
I'm struggling to recreate functionality I had in v5 to implement a hook that manages the value of single search parameter rather than the entire set of search params, so different hooks and components can simultaneously manage parts of the search params without interfering with each other (and without having to reconstruct the rest of the search state when setting its parameter to a new value). A trimmed down implementation of this functionality in v5 looked like this:
The key part of this hook was the appearance of
history.location.search
to construct an up to date view of the search params in the setter. Using this meant two things:search
does not need to be in the deps array ofuseCallback
, so the identity of the setter does not change when the location does.history.location.search
is always up to date.The second point is a crucial one: if a single event calls multiple setters of different
useSearchParam
hooks, using the value ofsearch
fromuseLocation
will mean each change gets reset by each subsequent setter, so only the last one will take effect. Using the mutable value ofhistory.location.search
allows each change to be preserved in the changes that the following setters take.I've managed to implement the same behaviour in v6 by fetching the history out of the context:
This relies on knowledge about the internals of the library that are intentionally not exposed, for good reason.
If
setParams
can provide the current value of the search params when you call it, then both of these problems are solved:params
need not be in the deps array of auseCallback
in order to avoid touching other parameters, and if the given params always reflect previous changes in the same event then they will be preserved in the resulting transformations.The text was updated successfully, but these errors were encountered: