Usage in Client Components
next-intl
is available in the app
directory from version 2.10.0
onwards.
Getting started
If you haven't done so already, create a Next.js 13 app that uses the app
directory. The goal is to prefix all routes with the locale
, so we can retrieve it as a dynamic segment and use it to configure next-intl
.
Start by creating the following file structure:
├── messages (1)
│ ├── en.json
│ └── ...
├── app
│ └── [locale]
│ ├── layout.tsx (2)
│ └── page.tsx (3)
└── middleware.tsx (4)
Now, set up these files as follows:
- Set up messages for a language, e.g. in
messages/en.json
:
{
"Index": {
"title": "Hello world!"
}
}
- Provide the document layout and set up
NextIntlClientProvider
inapp/[locale]/layout.tsx
:
import {NextIntlClientProvider} from 'next-intl/client';
import {notFound} from 'next/navigation';
export function generateStaticParams() {
return [{locale: 'en'}, {locale: 'de'}];
}
export default async function LocaleLayout({children, params: {locale}}) {
let messages;
try {
messages = (await import(`../../messages/${locale}.json`)).default;
} catch (error) {
notFound();
}
return (
<html lang={locale}>
<body>
<NextIntlClientProvider locale={locale} messages={messages}>
{children}
</NextIntlClientProvider>
</body>
</html>
);
}
- Turn your component in
app/[locale]/page.tsx
into a Client Component to be able to use translations:
'use client';
import {useTranslations} from 'next-intl';
export default function Index() {
const t = useTranslations('Index');
return <h1>{t('title')}</h1>;
}
- Create a middleware in
middleware.tsx
that handles redirects:
import {createIntlMiddleware} from 'next-intl/server';
// This middleware intercepts requests to `/` and will redirect
// to the best matching locale instead (e.g. `/en`). A cookie
// is set in the background, so if the user switches to a new
// language, this will take precedence from now on.
export default createIntlMiddleware({
locales: ['en', 'de'],
defaultLocale: 'en'
});
That's all you need to do to start using translations in the app
directory!
If you've encountered an issue, you can explore the code for a working example (demo).
Note that you have to make use of 'use client';
in all components that use
features from next-intl
if you use this approach. Support for next-intl
APIs in Server Components is available in a beta
version.
Linking between pages
In the pages
folder, Next.js automatically considers the current locale for next/link
. Since this is no longer the case for the app
directory, you can add a component that wraps next/link
accordingly as a drop-in replacement.
// LocalizedLink.tsx
import {useLocale} from 'next-intl';
import Link from 'next/link';
import {ComponentProps, forwardRef} from 'react';
type Props = ComponentProps<typeof Link>;
function LocalizedLink({href, ...rest}: Props, ref: Props['ref']) {
const locale = useLocale();
function getLocalizedHref(originalHref: string) {
return originalHref.replace(/^\//, '/' + locale + '/');
}
const localizedHref =
typeof href === 'string'
? getLocalizedHref(href)
: href.pathname != null
? {...href, pathname: getLocalizedHref(href.pathname)}
: href;
return <Link ref={ref} href={localizedHref} {...rest} />;
}
export default forwardRef(LocalizedLink);
// Usage while being on `/en`
<LocalizedLink href="/">Goes to `/en`</LocalizedLink>
<LocalizedLink href="/nested">Goes to `/en/nested`</LocalizedLink>
<LocalizedLink href={{pathname: "/nested"}}>Goes to `/en/nested`</LocalizedLink>
next-intl
is considering to provide built-in support for this component (see
#149).
Usage in head.js
This is the one component that Next.js doesn't allow to be marked with 'use client;'
.
Support for Server Components in next-intl
will allow you to use the same APIs that you already use in other components to handle i18n in head.js
. In the meantime, as a stopgap solution, you can use the core library from next-intl
for this use case.
// app/[locale]/head.tsx
import {createTranslator} from 'next-intl';
import {notFound} from 'next/navigation';
type Props = {
params: {
locale: string;
};
};
export default async function Head({params: {locale}}: Props) {
let messages;
try {
messages = (await import(`../../messages/${locale}.json`)).default;
} catch (error) {
notFound();
}
// Currently you can use the core (non-React) APIs when you
// have to use next-intl in a Server Component like `Head`.
// In the future you'll be able to use the React APIs here as
// well (see https://next-intl-docs.vercel.app/docs/next-13).
const t = createTranslator({locale, messages});
return (
<>
<title>{t('Head.title')}</title>
</>
);
}