A Comprehensive Guide to Internationalization in Next.js
6 min read ·
Published · 6 months ago
NextJSVercelAppDirectoryReactTypeScript
Understanding Internationalization Terminology
Locale
A locale is an identifier for a set of language and formatting preferences. It includes the user's preferred language and may also indicate their geographic region. For example:
'en-US'
'en-US'
: English as spoken in the United States'nl-NL'
'nl-NL'
: Dutch as spoken in the Netherlands'en'
'en'
: English language without a specific region
Setting Up Next.js for Internationalization
Create a New Next.js Application
If you haven't already, create a new Next.js project using the following command:
$ npx create-next-app@latest my-i18n-app
$ npx create-next-app@latest my-i18n-app
Setting Up Routing for Internationalization
To provide a seamless experience for users, it's crucial to adapt your application based on their preferred language. This can be achieved by adjusting the routing mechanism.
Utilizing User's Language Preferences
By using libraries like
@formatjs/intl-localematcher
@formatjs/intl-localematcher
and negotiator
negotiator
, you can determine the user's preferred locale based on their headers and your supported locales. This helps you ensure that users are directed to the correct language version of your site.$ npm install @formatjs/intl-localematcher negotiator
$ npm install @formatjs/intl-localematcher negotiator
Implementing Locale-Based Routing
Next.js allows you to internationalize routing either through sub-paths
(/fr/products)
(/fr/products)
or domains (my-site.fr/products)
(my-site.fr/products)
. This information enables you to redirect users based on their preferred locale within the middleware.and inside
src/
src/
directory create middleware.ts
middleware.ts
// middleware.ts
import { match } from '@formatjs/intl-localematcher'
import Negotiator from 'negotiator'
import { NextRequest, NextResponse } from 'next/server'
let defaultLocale = 'en'
let locales = ['cn', 'de', 'en']
// Get the preferred locale, similar to above or using a library
function getLocale(request: NextRequest) {
const acceptedLanguage = request.headers.get('accept-language') ?? undefined
let headers = { 'accept-language': acceptedLanguage }
let languages = new Negotiator({ headers }).languages()
return match(languages, locales, defaultLocale) // -> 'en-US'
}
export function middleware(request: NextRequest) {
// Check if there is any supported locale in the pathname
const pathname = request.nextUrl.pathname
const pathnameIsMissingLocale = locales.every(
(locale) => !pathname.startsWith(`/${locale}/`) && pathname !== `/${locale}`
)
// Redirect if there is no locale
if (pathnameIsMissingLocale) {
const locale = getLocale(request)
// e.g. incoming request is /products
// The new URL is now /en-US/products
return NextResponse.redirect(new URL(`/${locale}/${pathname}`, request.url))
}
}
export const config = {
matcher: [
// Skip all internal paths (_next, assets, api)
'/((?!api|assets|.*\\..*|_next).*)',
// Optional: only run on root (/) URL
// '/'
],
}
// middleware.ts
import { match } from '@formatjs/intl-localematcher'
import Negotiator from 'negotiator'
import { NextRequest, NextResponse } from 'next/server'
let defaultLocale = 'en'
let locales = ['cn', 'de', 'en']
// Get the preferred locale, similar to above or using a library
function getLocale(request: NextRequest) {
const acceptedLanguage = request.headers.get('accept-language') ?? undefined
let headers = { 'accept-language': acceptedLanguage }
let languages = new Negotiator({ headers }).languages()
return match(languages, locales, defaultLocale) // -> 'en-US'
}
export function middleware(request: NextRequest) {
// Check if there is any supported locale in the pathname
const pathname = request.nextUrl.pathname
const pathnameIsMissingLocale = locales.every(
(locale) => !pathname.startsWith(`/${locale}/`) && pathname !== `/${locale}`
)
// Redirect if there is no locale
if (pathnameIsMissingLocale) {
const locale = getLocale(request)
// e.g. incoming request is /products
// The new URL is now /en-US/products
return NextResponse.redirect(new URL(`/${locale}/${pathname}`, request.url))
}
}
export const config = {
matcher: [
// Skip all internal paths (_next, assets, api)
'/((?!api|assets|.*\\..*|_next).*)',
// Optional: only run on root (/) URL
// '/'
],
}
Organizing Files for Locale-Based Handling
To enable Next.js to dynamically manage different locales in the route, nest all special files within
app/[lang]
app/[lang]
.For example:
app/[lang]/page.tsx
app/[lang]/page.tsx
export default async function Page({ params: { lang } }) {
return ...
}
export default async function Page({ params: { lang } }) {
return ...
}
Implementing Localization
Localization involves adapting the displayed content based on the user's preferred locale. This can be achieved by maintaining separate dictionaries for each supported language.
Creating Dictionaries
For example, let's consider English, Dutch, and Chinese translations for Next js main page:
app/[lang]/dictionaries
app/[lang]/dictionaries
dictionaries/en.json
dictionaries/en.json
{
"get_started": "Get started by editing",
"doc": "Find in-depth information about Next.js features and API.",
"learn": "Learn about Next.js in an interactive course with quizzes!",
"template": "Explore the Next.js 13 playground.",
"deploy": "Instantly deploy your Next.js site to a shareable URL with Vercel."
}
{
"get_started": "Get started by editing",
"doc": "Find in-depth information about Next.js features and API.",
"learn": "Learn about Next.js in an interactive course with quizzes!",
"template": "Explore the Next.js 13 playground.",
"deploy": "Instantly deploy your Next.js site to a shareable URL with Vercel."
}
dictionaries/de.json
dictionaries/de.json
{
"get_started": "Beginnen Sie, indem Sie bearbeiten",
"doc": "Finden Sie ausführliche Informationen zu den Funktionen und der API von Next.js.",
"learn": "Erfahren Sie mehr über Next.js in einem interaktiven Kurs mit Quizfragen!",
"template": "Erkunden Sie den Next.js 13-Spielplatz.",
"deploy": "Bereiten Sie Ihre Next.js-Website sofort für die Bereitstellung auf einer gemeinsam nutzbaren URL mit Vercel vor."
}
{
"get_started": "Beginnen Sie, indem Sie bearbeiten",
"doc": "Finden Sie ausführliche Informationen zu den Funktionen und der API von Next.js.",
"learn": "Erfahren Sie mehr über Next.js in einem interaktiven Kurs mit Quizfragen!",
"template": "Erkunden Sie den Next.js 13-Spielplatz.",
"deploy": "Bereiten Sie Ihre Next.js-Website sofort für die Bereitstellung auf einer gemeinsam nutzbaren URL mit Vercel vor."
}
dictionaries/cn.json
dictionaries/cn.json
{
"get_started": "通过编辑开始",
"doc": "查找关于 Next.js 功能和 API 的深入信息。",
"learn": "通过互动课程学习 Next.js,包括测验!",
"template": "探索 Next.js 13 的示范环境。",
"deploy": "使用 Vercel 立即部署您的 Next.js 网站到可共享的 URL。"
}
{
"get_started": "通过编辑开始",
"doc": "查找关于 Next.js 功能和 API 的深入信息。",
"learn": "通过互动课程学习 Next.js,包括测验!",
"template": "探索 Next.js 13 的示范环境。",
"deploy": "使用 Vercel 立即部署您的 Next.js 网站到可共享的 URL。"
}
Loading translations
Create a function
getDictionary
getDictionary
to load the translations for the requested locale.import 'server-only'
export type Locale = keyof typeof dictionaries
const dictionaries = {
en: () => import('./dictionaries/en.json').then((module) => module.default),
de: () => import('./dictionaries/de.json').then((module) => module.default),
cn: () => import('./dictionaries/cn.json').then((module) => module.default),
}
export const getDictionary = async (locale: Locale) => dictionaries[locale]()
import 'server-only'
export type Locale = keyof typeof dictionaries
const dictionaries = {
en: () => import('./dictionaries/en.json').then((module) => module.default),
de: () => import('./dictionaries/de.json').then((module) => module.default),
cn: () => import('./dictionaries/cn.json').then((module) => module.default),
}
export const getDictionary = async (locale: Locale) => dictionaries[locale]()
Using Translations in Components
You can now fetch the dictionary inside a layout or page to display translated content.
import SwitchLang from '@/components/SwitchLang/SwitchLang'
import Image from 'next/image'
import { Locale, getDictionary } from './dictionaries'
import styles from './page.module.css'
type Props = {
params: {
lang: Locale
}
}
^^^^^^^^^^^^^^^
export default async function Home({ params: { lang } }: Props) {
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
const intl = await getDictionary(lang)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
return (
<main className={styles.main}>
<div className={styles.description}>
<p>
{intl.get_started}
^^^^^^^^^^^^^^^^^^
<code className={styles.code}>src/app/page.tsx</code>
</p>
<SwitchLang />
<div>
<a
href="https://vercel.com?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
target="_blank"
rel="noopener noreferrer"
>
By{' '}
<Image
src="/vercel.svg"
alt="Vercel Logo"
className={styles.vercelLogo}
width={100}
height={24}
priority
/>
</a>
</div>
</div>
<div className={styles.center}>
<Image
className={styles.logo}
src="/next.svg"
alt="Next.js Logo"
width={180}
height={37}
priority
/>
</div>
<div className={styles.grid}>
<a
href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Docs <span>-></span>
</h2>
<p>{intl.doc}</p>
^^^^^^^^^^
</a>
<a
href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Learn <span>-></span>
</h2>
<p>{intl.learn}</p>
^^^^^^^^^^^^
</a>
<a
href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Templates <span>-></span>
</h2>
<p>{intl.template}</p>
^^^^^^^^^^^^^^^
</a>
<a
href="https://vercel.com/new?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Deploy <span>-></span>
</h2>
<p>
{intl.deploy}
^^^^^^^^^^^^^
</p>
</a>
</div>
</main>
)
}
import SwitchLang from '@/components/SwitchLang/SwitchLang'
import Image from 'next/image'
import { Locale, getDictionary } from './dictionaries'
import styles from './page.module.css'
type Props = {
params: {
lang: Locale
}
}
^^^^^^^^^^^^^^^
export default async function Home({ params: { lang } }: Props) {
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
const intl = await getDictionary(lang)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
return (
<main className={styles.main}>
<div className={styles.description}>
<p>
{intl.get_started}
^^^^^^^^^^^^^^^^^^
<code className={styles.code}>src/app/page.tsx</code>
</p>
<SwitchLang />
<div>
<a
href="https://vercel.com?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
target="_blank"
rel="noopener noreferrer"
>
By{' '}
<Image
src="/vercel.svg"
alt="Vercel Logo"
className={styles.vercelLogo}
width={100}
height={24}
priority
/>
</a>
</div>
</div>
<div className={styles.center}>
<Image
className={styles.logo}
src="/next.svg"
alt="Next.js Logo"
width={180}
height={37}
priority
/>
</div>
<div className={styles.grid}>
<a
href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Docs <span>-></span>
</h2>
<p>{intl.doc}</p>
^^^^^^^^^^
</a>
<a
href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Learn <span>-></span>
</h2>
<p>{intl.learn}</p>
^^^^^^^^^^^^
</a>
<a
href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Templates <span>-></span>
</h2>
<p>{intl.template}</p>
^^^^^^^^^^^^^^^
</a>
<a
href="https://vercel.com/new?utm_source=create-next-app&utm_medium=appdir-template&utm_campaign=create-next-app"
className={styles.card}
target="_blank"
rel="noopener noreferrer"
>
<h2>
Deploy <span>-></span>
</h2>
<p>
{intl.deploy}
^^^^^^^^^^^^^
</p>
</a>
</div>
</main>
)
}
Static Generation for Multiple locales
To generate static routes for a set of locales, use
generateStaticParams
generateStaticParams
in any page or layout. This can be set globally, such as in the root layout:// app/[lang]/layout.ts
export async function generateStaticParams() {
return [{ lang: 'en' }, { lang: 'de' }, { lang: 'cn' }]
}
export default function Root({ children, params }) {
return (
<html lang={params.lang}>
<body>{children}</body>
</html>
)
}
// app/[lang]/layout.ts
export async function generateStaticParams() {
return [{ lang: 'en' }, { lang: 'de' }, { lang: 'cn' }]
}
export default function Root({ children, params }) {
return (
<html lang={params.lang}>
<body>{children}</body>
</html>
)
}
Conclusion
Implementing internationalization in Next.js provides a seamless experience for users from different linguistic backgrounds. By adapting routing.
Full Source Code
For the complete source code, including all files and dependencies, you can visit the GitHub Repository. Feel free to explore, fork, or clone the repository to delve deeper into the project.
Reference
Thanks for taking the time to read my post, I hope you enjoyed reading it! If you did I would greatly appreciate it if you shared it with your friends and colleagues.
Whether you did or you didn't I would love to hear your feedback; what works, what doesn't, did I leave anything out? Unfortunately I haven't implemented comments yet, but my socials are linked in the footer of this page if you wish to contact me.