# View Transitions in Next.js ## Setup `` works out of the box for `startTransition`/`Suspense` updates. To also animate `` navigations: ```js // next.config.js const nextConfig = { experimental: { viewTransition: true }, }; module.exports = nextConfig; ``` This wraps every `` navigation in `document.startViewTransition`. Any VT with `default="auto"` fires on **every** link click — use `default="none"` to prevent competing animations. Do **not** install `react@canary` — see SKILL.md "Availability" for details. --- ## Next.js Implementation Additions When following `implementation.md`, apply these additions: **After Step 2:** Enable the experimental flag above. **Step 4:** Use `transitionTypes` on `` — see "The `transitionTypes` Prop" section below for usage and availability. **After Step 6:** For same-route dynamic segments (e.g., `/collection/[slug]`), use the `key` + `name` + `share` pattern — see Same-Route Dynamic Segment Transitions below. --- ## Layout-Level ViewTransition **Do NOT add a layout-level VT wrapping `{children}` if pages have their own VTs.** Nested VTs never fire enter/exit when inside a parent VT — page-level enter/exit will silently not work. Remove the layout VT entirely. A bare `` in layout works only if pages have **no** VTs of their own. **Layouts persist across navigations** — `enter`/`exit` only fire on initial mount, not on route changes. Don't use type-keyed maps in layouts. --- ## The `transitionTypes` Prop on `next/link` No wrapper component needed, works in Server Components: ```tsx View Product ``` Replaces the manual pattern of `onNavigate` + `startTransition` + `addTransitionType` + `router.push()`. Reserve manual `startTransition` for non-link interactions (buttons, forms). **Availability:** `transitionTypes` requires `experimental.viewTransition: true` and is available in Next.js 15+ canary builds and Next.js 16+. If unavailable, use `startTransition` + `addTransitionType` + `router.push()` (see Programmatic Navigation below). To check: `grep -r "transitionTypes" node_modules/next/dist/` — if no results, fall back to programmatic navigation. --- ## Programmatic Navigation ```tsx 'use client'; import { useRouter } from 'next/navigation'; import { startTransition, addTransitionType } from 'react'; function handleNavigate(href: string) { const router = useRouter(); startTransition(() => { addTransitionType('nav-forward'); router.push(href); }); } ``` --- ## Server-Side Filtering with `router.replace` For search/sort/filter that re-renders on the server (via URL params), use `startTransition` + `router.replace`. VTs activate because the state update is inside `startTransition`: ```tsx 'use client'; import { useRouter } from 'next/navigation'; import { startTransition } from 'react'; function handleSort(sort: string) { const router = useRouter(); startTransition(() => { router.replace(`?sort=${sort}`); }); } ``` List items wrapped in `` will animate reorder. This is the server-component alternative to the client-side `useDeferredValue` pattern in `patterns.md`. --- ## Two-Layer Pattern (Directional + Suspense) Directional slides + Suspense reveals coexist because they fire at different moments. Place the directional VT in the **page component** (not layout): ```tsx
}>
``` --- ## `loading.tsx` as Suspense Boundary Next.js `loading.tsx` is an implicit `` boundary. Wrap the skeleton in `` in `loading.tsx`, and the content in `` in the page: ```tsx // loading.tsx // page.tsx ``` Same rules as explicit ``: use simple string props (not type maps) since Suspense reveals fire without transition types. --- ## Shared Elements Across Routes ```tsx // List page {products.map((product) => ( {product.name} ))} // Detail page — same name {product.name} ``` --- ## Same-Route Dynamic Segment Transitions When navigating between dynamic segments of the same route (e.g., `/collection/[slug]`), the page stays mounted — enter/exit never fire. Use `key` + `name` + `share`: ```tsx }> ``` - `key={slug}` forces unmount/remount on change - `name` + `share="auto"` creates a shared element crossfade - VT inside `` (without keying Suspense) keeps old content visible during loading --- ## Server Components - `` works in both Server and Client Components - `` works in Server Components — no `'use client'` needed - `addTransitionType` and `startTransition` for programmatic nav require Client Components