Skip to content
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

Nuqs rerender count in NextJS #822

Open
alberto-htl opened this issue Dec 22, 2024 · 8 comments · Fixed by #849
Open

Nuqs rerender count in NextJS #822

alberto-htl opened this issue Dec 22, 2024 · 8 comments · Fixed by #849
Labels
adapters/next/app Uses the Next.js app router next.js Issue is related to Next.js internals rather than this library
Milestone

Comments

@alberto-htl
Copy link

Context

What's your version of nuqs?

2.2.3

What framework are you using?

  • ✅ Next.js (app router)

Which version of your framework are you using?

Operating System:
  Platform: darwin
  Arch: arm64
  Version: Darwin Kernel Version 23.4.0: Fri Mar 15 00:12:49 PDT 2024; root:xnu-10063.101.17~1/RELEASE_ARM64_T6020
  Available memory (MB): 16384
  Available CPU cores: 10
Binaries:
  Node: 18.19.0
  npm: 10.2.3
  Yarn: 1.22.21
  pnpm: 8.11.0
Relevant Packages:
  next: 15.1.2 // Latest available version is detected (15.1.2).
  eslint-config-next: 15.1.2
  react: 19.0.0
  react-dom: 19.0.0
  typescript: 5.7.2
Next.js Config:
  output: N/A

Description

The component is rerender 8 times while the component with the default hooks of nextjs is rerendered only 2 times.

Expected behavior is to both component rerender the same amount of times.

The demo offer two components one using the nextjs hooks directly, and another using useQueryState from nuqs.

To test it, use the button. Test the nextjs component and check the console logs. Then use the button on the component that use nuqs and check the console logs.

Reproduction

https://github.com/dev2xl/nuqs-rerender-demo

@alberto-htl alberto-htl added the bug Something isn't working label Dec 22, 2024
@franky47
Copy link
Member

franky47 commented Dec 22, 2024

Updating the URL is an asynchronous operation, and nuqs does optimistic state updates to keep the UI responsive, so having more renders than with a simple synchronous, local useState, is expected. 8 times feels a little high though.

Your reproduction repo is just a stock create-next-app, did you forget to push a commit?

See also: #516.

@franky47 franky47 added next.js Issue is related to Next.js internals rather than this library cannot-reproduce Either no reproduction provided, or cannot reproduce with a minimal setup and removed bug Something isn't working labels Dec 22, 2024
@alberto-htl
Copy link
Author

@franky47 Sorry for the confusion earlier. I've updated the repo; could you please take a look?

Just to clarify, the comparison here is not with useState, but rather with the behavior we're seeing when using useRouter, useParams, and URLSearchParams (similar to how Nuqs handles routing). With these hooks, the component re-renders only twice per update. However, with Nuqs, we’re seeing around 8 renders per update.

Can you check the console logs during the tests to help track this?

When we set shallow to false, the number of renders drops to 6, but ideally, I’d expect the render count to be similar to the two renders we'd see in a typical server-side rendering setup, right?

My primary use case here is a data table where I’d prefer not to update optimistically, since the data is coming from the server. My goal is to minimize these costly re-renders.

Let me know what you think, and thanks for your help!

@franky47
Copy link
Member

Hey, I managed to take a look at your repro, and I can only see 4 renders (the doubles you see are due to StrictMode). You can see only four when building for production.

There might be something I can do to reduce it to 3, see #755 (comment).

@franky47 franky47 added adapters/next/app Uses the Next.js app router and removed cannot-reproduce Either no reproduction provided, or cannot reproduce with a minimal setup labels Dec 27, 2024
@franky47 franky47 changed the title Nuqs rerender problem in NextJS Nuqs rerender count in NextJS Jan 3, 2025
@franky47 franky47 added this to the 🪵 Backlog milestone Jan 3, 2025
@franky47
Copy link
Member

franky47 commented Jan 8, 2025

@alberto-htl I did some render counts in #849, and fixed a couple of issues that could have caused extra renders. It should be more stable at 2 renders on shallow: true and 3 renders on shallow: false in the app router.

Feel free to try [email protected] and let me know if it helps.

@alberto-htl
Copy link
Author

@franky47 That's great! I will take a look. Thank you for your support

@franky47 franky47 linked a pull request Jan 8, 2025 that will close this issue
4 tasks
@Kavan72
Copy link

Kavan72 commented Jan 10, 2025

@franky47 Just wow! I can now see only two renders with [email protected]. I have one question: is there any way we can merge these renders into one?

@franky47
Copy link
Member

I've been wondering about this, giving users control over the update strategy, by enabling/disabling optimistic updates.

If binding useQueryState to a non-controlled input for example, the first render caused by the optimistic local state update might make little sense.

The various steps are (not all implemented in all adapters):

  • Immediate internal state update, with cross-hook sync based on the key.
  • Optimistic shallow URL update (after a throttle timeout)
  • If shallow: false, deep navigation using the router (which causes another render when navigation finishes)

@dlindahl
Copy link

dlindahl commented Jan 10, 2025

FYI I was seeing an infinite rendering loop when using the back button to navigate to a component that uses nuqs through a custom hook. I was able to track it down to state object referential inequality (which is understandable) as well as a referential inequality on the updater callback. Not sure why the updater was changing so frequently when the page mounted, but I was able to work around it by using a deep compare on the state and using useRef for the updater.

However, bumping to 2.3.1-beta.2 solved this issue entirely and now the whole transition works just as I'd expect it to without any additional work on my end. 👍

EDIT: Was previously on 2.3.0

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
adapters/next/app Uses the Next.js app router next.js Issue is related to Next.js internals rather than this library
Projects
None yet
Development

Successfully merging a pull request may close this issue.

4 participants