Use ParaglideJS with the NextJS Pages Router

This guide covers how to integrate ParaglideJS into a NextJS project using the pages router. It will cover i18n routing and SEO Pitfalls and more!

Loris Sigrist

Using ParaglideJS with Next.js' Pages Router

In this guide you will learn how to internationalise your NextJS app that's using the Pages router with ParaglideJS!

By the end you will have i18n routing set up in an SEO friendly manner, be able to use translations in your code, all using ParaglideJS which ships some very compact JS thanks to it's compiler based approach.

This guide assumes that you already have a Next.js project set up with the pages router. If you don't, you can follow the official Next.js Getting Started Guide.

Install ParaglideJS

We will need to install two things. ParaglideJS itself, and Paraglide-Next.

ParaglideJS is the i18n library we will use for string translations. Paraglide-Next makes it easier to integrate Paraglide with NextJS's i18n routing.

In your project root, run the following commands and follow the instructions in your terminal.

npx @inlang/paraglide-js init
npm install @inlang/paraglide-next

This will have done a few things:

  • Created an inlang project in your project root
  • Added the required devDependencies to your package.json
  • Added the paraglide compiler to your package.json build scripts
  • Installed the Paraglide-Next which will handle setting the locale

Setting up

In your next.config.js, import the Paraglide Plugin and apply it to your config.

You will need to pass it some config. The location of the Inlang Project & the output directory that messages should be compiled to. You should stick with the defaults unless you have a reason not to.

// next.config.js
const { paraglide } = require("@inlang/paraglide-next/plugin");

module.exports = paraglide({
  paraglide: {
    project: "./project.inlang",
    outdir: "./src/paraglide",
  },
});

This Plugin will make sure to automatically recompile messages whenever they change.

Next, let's set up NextJS' built in i18n routing.

Still in your next.config.js, add the i18n object to your config, pass it the languages you want to support and set the default language.

// next.config.js
const { paraglide } = require("@inlang/paraglide-next/plugin");

module.exports = paraglide({
  paraglide: {
    project: "./project.inlang",
    outdir: "./src/paraglide",
  },
  i18n: {
    locales: ["en", "de"], //list the languages you intend to support
    defaultLocale: "en",
  },
});

Last, let's add the <ParaglideJS> component to your _app.js file. This will manage the language state for you.

// _app.js
import type { AppProps } from "next/app"
import { ParaglideJS } from "@inlang/paraglide-next/pages"

export default function App({ Component, pageProps }: AppProps) {
	return (
		<ParaglideJS>
			<Component {...pageProps} />
		</ParaglideJS>
	)
}

Our first Message

Translations live in the messages/{lang}.json files that were generated by the init CLI. Add messages there

// messages/en.jsons
{
  "$schema": "https://inlang.com/schema/inlang-message-format",
  "hello_world": "Hello World!"
}

If you now start your app, you should see a new folder appear in your src folder called paraglide. This folder contains the compiled messages and any runtime code required by paraglide. Your messages live at ./src/paraglide/messages.js.

Let's use the hello_world message on our homepage. Open pages/index.js, import from paraglide/messages.js and use the hello_world message in the h1 tag.

// @ is next's default alias for the src/ folder
// By convention paraglide uses import * as m
import * as m from "@/paraglide/messages";

export default function Home() {
  return (
    <div>
      <h1>{m.hello_world()}</h1>
    </div>
  );
}

You should now see the message "Hello World!" on your homepage!

Note: If you are using Visual Studio Code, you should install the accompanying VS Code extension Sherlock. It will give you inline message previews & allow you to edit them right from your code.

Adding more languages

You need to add more languages in two places. First, in your next.config.js file, and then in your Inlang project settings.

Let's add german to both:

//next.config.js
module.exports = paraglide({
	i18n: {
		locales: ["en", "de"], // <-- Added german
		defaultLocale: "en",
	},
	...
})
// project.inlang/settings.json
{
  "$schema": "https://inlang.com/schema/project-settings",
  "sourceLanguageTag": "en",
  "languageTags": ["en", "de"], // <-- Added german
  "...": "..."
}

You can now add a de.json file to your messages folder and add a german translation for the hello_world message.

// messages/de.json
{
  "$schema": "https://inlang.com/schema/inlang-message-format",
  "hello_world": "Hallo Welt!"
}

With this config in place, you should get the following routing behaviour:

  • /some-page - Default language (English)
  • /en/some-page - English
  • /de/some-page - German

When you visit /de you should now see the german translation of "Hello World!".

Switching languages the right way

We will use a <Link> to switch languages. This will automatically rerender the page and load the correct messages. You can specify the locale prop to set which language to switch to.

// pages/index.js
import * as m from "@/paraglide/messages";
import Link from "next/link";
import { useRouter } from "next/router";

export default function Home() {
  const router = useRouter();

  return (
    <div>
      <h1>{m.hello_world()}</h1>
      <Link href={router.asPath} locale="en" hrefLang="en">
        English
      </Link>
      <Link href={router.asPath} locale="de" hrefLang="de">
        German
      </Link>
    </div>
  );
}

Alternatively, you could use Next's router.push function to switch languages programmatically.

// pages/index.js

import * as m from "@/paraglide/messages";
import { useRouter } from "next/router";

export default function Home() {
  const router = useRouter();

  return (
    <div>
      <h1>{m.hello_world()}</h1>
      <button
        onClick={() =>
          router.push(router.asPath, router.asPath, { locale: "en" })
        }
      >
        English
      </button>
      <button
        onClick={() =>
          router.push(router.asPath, router.asPath, { locale: "de" })
        }
      >
        German
      </button>
    </div>
  );
}

You should always switch languages by doing a navigation. This will ensure that the correct messages are loaded and that the correct language is set in the url.

Next Steps

That's it! You should now have a fully functional multilingual NextJS app using ParaglideJS. Wasn't that hard was it?

You can check out the full source code of this example here.

If you want to learn more about ParaglideJS, check out the ParaglideJS Documentation. If you need help or have some ideas, feel free to reach out to us on Discord or open a Discussion on GitHub.

Edit on GitHub

Keywords

Recommended