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

Allow useNavigate to be called from child component effects #10394

Merged
merged 8 commits into from
Apr 25, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Add tests and switch to useIsomorphicLayoutEffect
  • Loading branch information
brophdawg11 committed Apr 25, 2023
commit 92284dcddd0ae9a4ba2410990c76e7c960cb32e2
5 changes: 5 additions & 0 deletions .changeset/navigate-in-effect.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"react-router": patch
---

Fix detection of `useNavigate` in the render cycle by setting the `activeRef` in a layout effect, allowing the `navigate` function to be passed to child components and called in a `useEffect` there.
273 changes: 202 additions & 71 deletions packages/react-router/__tests__/useNavigate-test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -167,77 +167,6 @@ describe("useNavigate", () => {
]
`);
});

it("transitions to the new location when called immediately", () => {
const Home = React.forwardRef(function Home(_props, ref) {
let navigate = useNavigate();

React.useImperativeHandle(ref, () => ({
navigate: () => navigate("/about")
}))

return null
})

let homeRef;

let renderer: TestRenderer.ReactTestRenderer;
renderer = TestRenderer.create(
<MemoryRouter initialEntries={["/home"]}>
<Routes>
<Route path="home" element={<Home ref={(ref) => homeRef = ref} />} />
<Route path="about" element={<h1>About</h1>} />
</Routes>
</MemoryRouter>
);

TestRenderer.act(() => {
homeRef.navigate();
})

expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
});

it("allows navigation in child useEffects", () => {
function Child({ onChildRendered }) {

React.useEffect(() => {
onChildRendered();
});

return null;
}

function Parent() {
let navigate = useNavigate();

let onChildRendered = React.useCallback(() => navigate("/about"), []);

return <Child onChildRendered={onChildRendered} />;
}

let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(
<MemoryRouter initialEntries={["/home"]}>
<Routes>
<Route path="home" element={<Parent />} />
<Route path="about" element={<h1>About</h1>} />
</Routes>
</MemoryRouter>
);
});

expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
});

it("navigates to the new location with empty query string when no query string is provided", () => {
function Home() {
Expand Down Expand Up @@ -372,6 +301,208 @@ describe("useNavigate", () => {
);
});

describe("navigating in effects versus render", () => {
let warnSpy: jest.SpyInstance;

beforeEach(() => {
warnSpy = jest.spyOn(console, "warn").mockImplementation(() => {});
});

afterEach(() => {
warnSpy.mockRestore();
});

describe("MemoryRouter", () => {
it("does not allow navigation from the render cycle", () => {
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(
<MemoryRouter>
<Routes>
<Route index element={<Home />} />
<Route path="about" element={<h1>About</h1>} />
</Routes>
</MemoryRouter>
);
});

function Home() {
let navigate = useNavigate();
navigate("/about");
return <h1>Home</h1>;
}

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
Home
</h1>
`);
expect(warnSpy).toHaveBeenCalledWith(
"You should call navigate() in a React.useEffect(), not when your component is first rendered."
);
});

it("allows navigation from effects", () => {
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(
<MemoryRouter>
<Routes>
<Route index element={<Home />} />
<Route path="about" element={<h1>About</h1>} />
</Routes>
</MemoryRouter>
);
});

function Home() {
let navigate = useNavigate();
React.useEffect(() => navigate("/about"), []);
return <h1>Home</h1>;
}

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
expect(warnSpy).not.toHaveBeenCalled();
});

it("allows navigation in child useEffects", () => {
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(
<MemoryRouter initialEntries={["/home"]}>
<Routes>
<Route path="home" element={<Parent />} />
<Route path="about" element={<h1>About</h1>} />
</Routes>
</MemoryRouter>
);
});

function Parent() {
let navigate = useNavigate();
let onChildRendered = React.useCallback(() => navigate("/about"), []);
return <Child onChildRendered={onChildRendered} />;
}

function Child({ onChildRendered }) {
React.useEffect(() => onChildRendered());
return null;
}

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
});
});

describe("RouterProvider", () => {
it("does not allow navigation from the render cycle", async () => {
let router = createMemoryRouter([
{
index: true,
Component() {
let navigate = useNavigate();
navigate("/about");
return <h1>Home</h1>;
},
},
{
path: "about",
element: <h1>About</h1>,
},
]);
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(<RouterProvider router={router} />);
});

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
Home
</h1>
`);
expect(warnSpy).toHaveBeenCalledWith(
"You should call navigate() in a React.useEffect(), not when your component is first rendered."
);
});

it("allows navigation from effects", () => {
let router = createMemoryRouter([
{
index: true,
Component() {
let navigate = useNavigate();
React.useEffect(() => navigate("/about"), []);
return <h1>Home</h1>;
},
},
{
path: "about",
element: <h1>About</h1>,
},
]);
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(<RouterProvider router={router} />);
});

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
expect(warnSpy).not.toHaveBeenCalled();
});

it("allows navigation in child useEffects", () => {
let router = createMemoryRouter([
{
index: true,
Component() {
let navigate = useNavigate();
let onChildRendered = React.useCallback(
() => navigate("/about"),
[]
);
return <Child onChildRendered={onChildRendered} />;
},
},
{
path: "about",
element: <h1>About</h1>,
},
]);
let renderer: TestRenderer.ReactTestRenderer;
TestRenderer.act(() => {
renderer = TestRenderer.create(<RouterProvider router={router} />);
});

function Child({ onChildRendered }) {
React.useEffect(() => onChildRendered());
return null;
}

// @ts-expect-error
expect(renderer.toJSON()).toMatchInlineSnapshot(`
<h1>
About
</h1>
`);
});
});
});

describe("with state", () => {
it("adds the state to location.state", () => {
function Home() {
Expand Down
38 changes: 32 additions & 6 deletions packages/react-router/lib/hooks.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -150,6 +150,21 @@ export interface NavigateFunction {
(delta: number): void;
}

const navigateEffectWarning =
`You should call navigate() in a React.useEffect(), not when ` +
`your component is first rendered.`;

// Mute warnings for calls to useNavigate in SSR environments
function useIsomorphicLayoutEffect(
cb: Parameters<typeof React.useLayoutEffect>[0]
) {
let isStatic = React.useContext(NavigationContext).static;
if (!isStatic) {
// eslint-disable-next-line react-hooks/rules-of-hooks
React.useLayoutEffect(cb);
}
}

/**
* Returns an imperative method for changing the location. Used by <Link>s, but
* may also be used by other elements to change the location.
Expand Down Expand Up @@ -180,17 +195,17 @@ function useNavigateUnstable(): NavigateFunction {
);

let activeRef = React.useRef(false);
React.useLayoutEffect(() => {
useIsomorphicLayoutEffect(() => {
activeRef.current = true;
});

let navigate: NavigateFunction = React.useCallback(
(to: To | number, options: NavigateOptions = {}) => {
warning(
activeRef.current,
`You should call navigate() in a React.useEffect(), not when ` +
`your component is first rendered.`
);
warning(activeRef.current, navigateEffectWarning);

// Short circuit here since if this happens on first render the navigate
// is useless because we haven't wired up our subscriber yet
if (!activeRef.current) return;

if (typeof to === "number") {
navigator.go(to);
Expand Down Expand Up @@ -929,8 +944,19 @@ function useNavigateStable(): NavigateFunction {
let { router } = useDataRouterContext(DataRouterHook.UseNavigateStable);
let id = useCurrentRouteId(DataRouterStateHook.UseNavigateStable);

let activeRef = React.useRef(false);
useIsomorphicLayoutEffect(() => {
activeRef.current = true;
});

let navigate: NavigateFunction = React.useCallback(
(to: To | number, options: NavigateOptions = {}) => {
warning(activeRef.current, navigateEffectWarning);

// Short circuit here since if this happens on first render the navigate
// is useless because we haven't wired up our subscriber yet
if (!activeRef.current) return;

if (typeof to === "number") {
router.navigate(to);
} else {
Expand Down