Docs
Next.js 13
Client Components

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 (opens in a new tab). The goal is to prefix all routes with the locale, so we can retrieve it as a dynamic segment (opens in a new tab) 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.ts (4)

Now, set up these files as follows:

  1. Set up messages for a language, e.g. in messages/en.json.
messages/en.json
{
  "Index": {
    "title": "Hello world!"
  }
}
  1. Provide the document layout and set up NextIntlClientProvider in app/[locale]/layout.tsx.
app/[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>
  );
}
  1. Turn your component in app/[locale]/page.tsx into a Client Component to be able to use translations.
app/[locale]/page.tsx
'use client';
 
import {useTranslations} from 'next-intl';
 
export default function Index() {
  const t = useTranslations('Index');
  return <h1>{t('title')}</h1>;
}
  1. Create a middleware (opens in a new tab) in middleware.ts that handles redirects.
middleware.ts
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 (opens in a new tab) (demo (opens in a new tab)).

💡

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.

Link.tsx
import {useLocale} from 'next-intl';
import NextLink from 'next/link';
import {ComponentProps, forwardRef} from 'react';
 
type Props = ComponentProps<typeof NextLink>;
 
function Link({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 <NextLink ref={ref} href={localizedHref} {...rest} />;
}
 
export default forwardRef(Link);
// Usage while being on `/en`
<Link href="/">Goes to `/en`</Link>
<Link href="/nested">Goes to `/en/nested`</Link>
<Link href={{pathname: "/nested"}}>Goes to `/en/nested`</Link>
💡

next-intl is considering to provide built-in support for this component (see #149 (opens in a new tab)).

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>
    </>
  );
}