メインコンテンツまでスキップ

バージョン 15

14 から 15 へのアップグレード

Next.js バージョン 15 にアップデートするには、upgrade codemod を使用できます:

Terminal
npx @next/codemod@canary upgrade latest

手動で行いたい場合は、最新の Next と React RC をインストールしていることを確認してください。例:

Terminal
npm i next@latest react@rc react-dom@rc eslint-config-next@latest

Good to know:

  • ピア依存関係の警告が表示された場合は、reactreact-dom を提案されたバージョンに更新するか、警告を無視するために --force または --legacy-peer-deps フラグを使用する必要があるかもしれません。これは、Next.js 15 と React 19 の両方が安定版になったときには必要ありません。

React 19

  • reactreact-dom の最小バージョンは 19 になりました。
  • useFormStateuseActionState に置き換えられました。useFormState フックは React 19 でも利用可能ですが、非推奨であり、将来のリリースで削除される予定です。useActionState が推奨され、pending 状態を直接読み取るなどの追加プロパティが含まれています。詳細はこちら
  • useFormStatus には、datamethodaction などの追加キーが含まれるようになりました。React 19 を使用していない場合は、pending キーのみが利用可能です。詳細はこちら
  • React 19 アップグレードガイドで詳細を確認してください。

Good to know: TypeScript を使用している場合は、@types/react@types/react-dom も最新バージョンにアップグレードしてください。

非同期リクエスト API(破壊的変更)

以前は同期的だったランタイム情報に依存する動的 API は、現在 非同期 です:

移行の負担を軽減するために、プロセスを自動化する codemod が利用可能 であり、API は一時的に同期的にアクセスできます。

cookies

import { cookies } from 'next/headers'

// 以前
const cookieStore = cookies()
const token = cookieStore.get('token')

// 以後
const cookieStore = await cookies()
const token = cookieStore.get('token')

一時的な同期使用法

app/page.tsx
import { cookies, type UnsafeUnwrappedCookies } from 'next/headers'

// 以前
const cookieStore = cookies()
const token = cookieStore.get('token')

// 以後
const cookieStore = cookies() as unknown as UnsafeUnwrappedCookies
// 開発中に警告が表示されます
const token = cookieStore.get('token')

headers

import { headers } from 'next/headers'

// 以前
const headersList = headers()
const userAgent = headersList.get('user-agent')

// 以後
const headersList = await headers()
const userAgent = headersList.get('user-agent')

一時的な同期使用法

app/page.tsx
import { headers, type UnsafeUnwrappedHeaders } from 'next/headers'

// 以前
const headersList = headers()
const userAgent = headersList.get('user-agent')

// 以後
const headersList = headers() as unknown as UnsafeUnwrappedHeaders
// 開発中に警告が表示されます
const userAgent = headersList.get('user-agent')

draftMode

import { draftMode } from 'next/headers'

// 以前
const { isEnabled } = draftMode()

// 以後
const { isEnabled } = await draftMode()

一時的な同期使用法

app/page.tsx
import { draftMode, type UnsafeUnwrappedDraftMode } from 'next/headers'

// 以前
const { isEnabled } = draftMode()

// 以後
// 開発中に警告が表示されます
const { isEnabled } = draftMode() as unknown as UnsafeUnwrappedDraftMode

params & searchParams

非同期レイアウト

app/layout.tsx
// 以前
type Params = { slug: string }

export function generateMetadata({ params }: { params: Params }) {
const { slug } = params
}

export default async function Layout({
children,
params,
}: {
children: React.ReactNode
params: Params
}) {
const { slug } = params
}

// 以後
type Params = Promise<{ slug: string }>

export async function generateMetadata({ params }: { params: Params }) {
const { slug } = await params
}

export default async function Layout({
children,
params,
}: {
children: React.ReactNode
params: Params
}) {
const { slug } = await params
}

同期レイアウト

app/layout.tsx
// 以前
type Params = { slug: string }

export default function Layout({
children,
params,
}: {
children: React.ReactNode
params: Params
}) {
const { slug } = params
}

// 以後
import { use } from 'react'

type Params = Promise<{ slug: string }>

export default function Layout(props: {
children: React.ReactNode
params: Params
}) {
const params = use(props.params)
const slug = params.slug
}

非同期ページ

app/page.tsx
// 以前
type Params = { slug: string }
type SearchParams = { [key: string]: string | string[] | undefined }

export function generateMetadata({
params,
searchParams,
}: {
params: Params
searchParams: SearchParams
}) {
const { slug } = params
const { query } = searchParams
}

export default async function Page({
params,
searchParams,
}: {
params: Params
searchParams: SearchParams
}) {
const { slug } = params
const { query } = searchParams
}

// 以後
type Params = Promise<{ slug: string }>
type SearchParams = Promise<{ [key: string]: string | string[] | undefined }>

export async function generateMetadata(props: {
params: Params
searchParams: SearchParams
}) {
const params = await props.params
const searchParams = await props.searchParams
const slug = params.slug
const query = searchParams.query
}

export default async function Page(props: {
params: Params
searchParams: SearchParams
}) {
const params = await props.params
const searchParams = await props.searchParams
const slug = params.slug
const query = searchParams.query
}

同期ページ

'use client'

// 以前
type Params = { slug: string }
type SearchParams = { [key: string]: string | string[] | undefined }

export default function Page({
params,
searchParams,
}: {
params: Params
searchParams: SearchParams
}) {
const { slug } = params
const { query } = searchParams
}

// 以後
import { use } from 'react'

type Params = Promise<{ slug: string }>
type SearchParams = Promise<{ [key: string]: string | string[] | undefined }>

export default function Page(props: {
params: Params
searchParams: SearchParams
}) {
const params = use(props.params)
const searchParams = use(props.searchParams)
const slug = params.slug
const query = searchParams.query
}
// 以前
export default function Page({ params, searchParams }) {
const { slug } = params
const { query } = searchParams
}

// 以後
import { use } from "react"

export default function Page(props) {
const params = use(props.params)
const searchParams = use(props.searchParams)
const slug = params.slug
const query = searchParams.query
}

ルートハンドラー

app/api/route.ts
// 以前
type Params = { slug: string }

export async function GET(request: Request, segmentData: { params: Params }) {
const params = segmentData.params
const slug = params.slug
}

// 以後
type Params = Promise<{ slug: string }>

export async function GET(request: Request, segmentData: { params: Params }) {
const params = await segmentData.params
const slug = params.slug
}
app/api/route.js
// 以前
export async function GET(request, segmentData) {
const params = segmentData.params
const slug = params.slug
}

// 以後
export async function GET(request, segmentData) {
const params = await segmentData.params
const slug = params.slug
}

runtime 設定(破壊的変更)

以前は experimental-edge の値をサポートしていた runtime セグメント設定 は、edge と同じものを指していました。オプションを簡素化するために、experimental-edge が使用されている場合はエラーが発生します。これを修正するには、runtime 設定を edge に更新してください。これを自動的に行うための codemod が利用可能です。

fetch リクエスト

fetch リクエスト はデフォルトでキャッシュされなくなりました。

特定の fetch リクエストをキャッシュにオプトインするには、cache: 'force-cache' オプションを渡すことができます。

app/layout.js
export default async function RootLayout() {
const a = await fetch('https://...') // キャッシュされない
const b = await fetch('https://...', { cache: 'force-cache' }) // キャッシュされる

// ...
}

レイアウトまたはページ内のすべての fetch リクエストをキャッシュにオプトインするには、export const fetchCache = 'default-cache' セグメント設定オプション を使用できます。個々の fetch リクエストが cache オプションを指定している場合は、それが使用されます。

app/layout.js
// これは root レイアウトなので、アプリ内のすべての fetch リクエスト
// が独自のキャッシュオプションを設定しない限り、キャッシュされます。
export const fetchCache = 'default-cache'

export default async function RootLayout() {
const a = await fetch('https://...') // キャッシュされる
const b = await fetch('https://...', { cache: 'no-store' }) // キャッシュされない

// ...
}

ルートハンドラー

ルートハンドラーGET 関数はデフォルトでキャッシュされなくなりました。GET メソッドをキャッシュにオプトインするには、export const dynamic = 'force-static' などの ルート設定オプション をルートハンドラーファイルに使用できます。

app/api/route.js
export const dynamic = 'force-static'

export async function GET() {}

クライアントサイドルーターキャッシュ

<Link> または useRouter を介してページ間を移動する際、ページ セグメントはクライアントサイドルーターキャッシュから再利用されなくなりました。ただし、ブラウザの戻るおよび進むナビゲーション中や共有レイアウトでは引き続き再利用されます。

ページセグメントをキャッシュにオプトインするには、staleTimes 設定オプションを使用できます:

next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
experimental: {
staleTimes: {
dynamic: 30,
static: 180,
},
},
}

module.exports = nextConfig

レイアウトローディング状態 は引き続きキャッシュされ、ナビゲーション時に再利用されます。

next/font

@next/font パッケージは、組み込みの next/font に置き換えられました。インポートを安全かつ自動的にリネームするための codemod が利用可能 です。

app/layout.js
// 以前
import { Inter } from '@next/font/google'

// 以後
import { Inter } from 'next/font/google'

bundlePagesRouterDependencies

experimental.bundlePagesExternals は安定版となり、bundlePagesRouterDependencies に名前が変更されました。

next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
// 以前
experimental: {
bundlePagesExternals: true,
},

// 以後
bundlePagesRouterDependencies: true,
}

module.exports = nextConfig

serverExternalPackages

experimental.serverComponentsExternalPackages は安定版となり、serverExternalPackages に名前が変更されました。

next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
// 以前
experimental: {
serverComponentsExternalPackages: ['package-name'],
},

// 以後
serverExternalPackages: ['package-name'],
}

module.exports = nextConfig

Speed Insights

Speed Insights の自動インストルメンテーションは Next.js 15 で削除されました。

Speed Insights を引き続き使用するには、Vercel Speed Insights クイックスタート ガイドに従ってください。

NextRequest ジオロケーション

NextRequestgeo および ip プロパティは、これらの値がホスティングプロバイダーによって提供されるため削除されました。これを自動化するための codemod が利用可能です。

Vercel を使用している場合は、代わりに @vercel/functions から geolocation および ipAddress 関数を使用できます:

middleware.ts
import { geolocation } from '@vercel/functions'
import type { NextRequest } from 'next/server'

export function middleware(request: NextRequest) {
const { city } = geolocation(request)

// ...
}
middleware.ts
import { ipAddress } from '@vercel/functions'
import type { NextRequest } from 'next/server'

export function middleware(request: NextRequest) {
const ip = ipAddress(request)

// ...
}