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

v7: When switching routes, components using useNavigate() will be re rendered #12520

Open
yibird opened this issue Dec 11, 2024 · 5 comments
Open
Labels

Comments

@yibird
Copy link

yibird commented Dec 11, 2024

I'm using React Router as a...

library

Reproduction

import React from 'react';
import { createBrowserRouter, RouterProvider, useNavigate } from 'react-router';

function Test() {
  console.log('Test:');
  const navigate = useNavigate();
  return (
    <div>
      <button onClick={() => navigate('/test')}>test</button>
      <button onClick={() => navigate('/demo')}>demo</button>
    </div>
  );
}

function Demo() {
  console.log('Demo:');
  const navigate = useNavigate();
  return <div>demo</div>;
}

const AppLayout = React.memo(function AppLayout() {
  console.log('AppLayout:');

  return (
    <div>
      <Test />
      <Demo />
    </div>
  );
});

When clicking the button in the Test component to switch routes, the Demo component will also be re rendered, and theoretically it should not be re rendered. If the demo does not use useNavigate(), everything will be normal

System Info

window

Used Package Manager

npm

Expected Behavior

v7: When switching routes, components using useNavigate() will be re rendered

Actual Behavior

v7: When switching routes, components using useNavigate() will be re rendered

@yibird yibird added the bug label Dec 11, 2024
@MadMed677
Copy link

Note: the same problem with v6. When route changes - useNavigate will return a new instance

@yibird
Copy link
Author

yibird commented Dec 11, 2024

Is there any good way? This seriously affects performance

@MadMed677
Copy link

@yibird If you want to avoid this for now, you could wrap your useEffect or where you use it intouseRef to save a reference to the function. In that case your useEffect won't be affected by changing navigate

Or in your case, move the button which should trigger navigate() into separate function to re-render only this small component.

I don't know other options to avoid component re-rendering

@yibird
Copy link
Author

yibird commented Dec 12, 2024

This is not the most effective method. You can encapsulate a Hooks and use the history API to simulate the useNavigate() behavior

@asegarra
Copy link

Note: the same problem with v6. When route changes - useNavigate will return a new instance

I believe that would only happen if you are on one of the non data routers. From my testing navigate is stable.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

3 participants