-
-
Notifications
You must be signed in to change notification settings - Fork 865
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
fix(react-router): memo of useLinkProps #3365
base: main
Are you sure you want to change the base?
Conversation
View your CI Pipeline Execution ↗ for commit 5025710.
☁️ Nx Cloud last updated this comment at |
More templates
@tanstack/arktype-adapter
@tanstack/create-router
@tanstack/directive-functions-plugin
@tanstack/create-start
@tanstack/history
@tanstack/react-cross-context
@tanstack/eslint-plugin-router
@tanstack/react-router
@tanstack/react-router-with-query
@tanstack/router-cli
@tanstack/router-core
@tanstack/router-devtools
@tanstack/router-generator
@tanstack/router-plugin
@tanstack/router-utils
@tanstack/router-vite-plugin
@tanstack/server-functions-plugin
@tanstack/start
@tanstack/start-api-routes
@tanstack/start-client
@tanstack/start-config
@tanstack/start-plugin
@tanstack/start-router-manifest
@tanstack/start-server
@tanstack/start-server-functions-client
@tanstack/start-server-functions-fetcher
@tanstack/start-server-functions-handler
@tanstack/start-server-functions-server
@tanstack/start-server-functions-ssr
@tanstack/valibot-adapter
@tanstack/virtual-file-routes
@tanstack/zod-adapter
commit: |
what problem does their shape cause? |
These are userland objects, they're not coming from hooks with structural sharing. For example: <Link
to="/foo/$a/$b"
params={{ a: 'a', b: 'b' }}
search={{ hello: 'world' }}
> |
oh right... then they need to memoized by the caller :( |
Yeah, but I doubt most codebases will be this stringent on memoization (until everyone has the |
Important
I'm not saying we should merge this, I just want to demonstrate how it would look to enable memoization in this hook
The issue
Currently, the
useLinkProps
hook is either called from the<Link>
component, or explicitly in userland.<Link>
component (which is equivalent tocreateLink
):In both cases, the
options
object will always be a new object on every render.In addition to this, the
options
object is also internally overwritten which forces it to always be a new object:This means that some of the memoization inside
useLinkProps
is ineffective. For example:or
And this instability cascades down to more things, like
useLayoutEffect
anduseIntersectionObserver
Possible solutions
Because of this "instability" of the
options
object, usinguseMemo
/useCallback
adds cost, and doesn't bring any benefits.There are 2 possible avenues for improving this situation:
stabilize the
options
object to make memoization possible. This is what this PR does, but it also demonstrates that it could potentially be verbose, and maybe pointless (some objects might have a shape that makes them somewhat inappropriate for memoization likesearch
,params
, ... unless we start resorting to ugly patterns of dynamic dependency arrays). It's not ideal, but for some situations it could be better than nothing.remove most of the memoization. Since it is currently pure cost / no benefits, we could reduce the cost. This means that
useLayoutEffect
anduseIntersectionObserver
will keep re-executing on every render for now. This solution is most likely a bet on thereact-compiler
to solve this in the near future. As a reminder, the react-compiler will not compile libraries from the application's pipeline so this would have to be included in tanstack's publishing pipeline