Rense Bakker

App translation with React-i18next

Published: August 31st 2021
ReactTypescriptReact i18next

In this article I will show you how to translate your React app with React-i18next and give a few examples that you may find useful!

I am using joy ui in this example, but I won't show any configuration for it because it is out of scope. You can see the full code in this code sandbox.

I am also assuming that you already have a react project up and running.

Setup

First of we need to add the react-i18next package and its dependencies:

1npm install react-i18next i18next i18next-http-backend i18next-browser-languagedetector 2

You do not strictly need i18next-http-backend and i18next-browser-languagedetector, however in my experience, almost every React app with translations will want to use these tools sooner or later.

Create config and add i18next provider

Now we need to create a i18n.ts file where we will configure the i18n instance:

1// /src/i18n.ts 2import i18n from 'i18next' 3import { initReactI18next } from 'react-i18next' 4import Backend from 'i18next-http-backend' 5import LanguageDetector from 'i18next-browser-languagedetector' 6 7i18n 8 .use(Backend) // default import from /public/locales/[lang]/[namespace].json 9 .use(LanguageDetector) // Detect browser language 10 .use(initReactI18next) 11 .init({ 12 fallbackLng: 'en', // Our default language 13 debug: true, // Only use this in dev mode 14 interpolation: { 15 escapeValue: false // We don't need this for React 16 } 17 }) 18 19export default i18n 20

Next we need to provide this config to our entire React app using the I18nextProvider:

1// /src/main.tsx 2import React from 'react' 3import ReactDOM from 'react-dom/client' 4import App from './App' 5import { I18nextProvider } from 'react-i18next' 6import i18n from './i18n' // import i18n config 7 8ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render( 9 <React.StrictMode> 10 <I18nextProvider i18n={i18n} defaultNS={'translation'}> 11 <App /> 12 </I18nextProvider> 13 </React.StrictMode> 14) 15

Add the translations files

English translation:

1// /public/locales/en/translation.json 2{ 3 "hello": "Hello World!" 4} 5

Dutch translation:

1// /public/locales/nl/translation.json 2{ 3 "hello": "Hallo wereld!" 4} 5

Typescript

For extra convenience, we can add a react-i18next.d.ts file so the compiler will know which translation keys are available in our app. This way we get autocomplete for translation keys and it will let us know when we are using a translation key that doesn't exist:

1// /src/react-i18next.d.ts 2import 'react-i18next' 3import translation from 'locales/en/translation.json' 4 5declare module 'react-i18next' { 6 interface Resources { 7 translation: typeof translation 8 } 9} 10 11declare module 'react-i18next' { 12 interface CustomTypeOptions { 13 defaultNS: 'translation' 14 resources: { 15 translation: typeof translation 16 } 17 } 18} 19

Using translations

To translate a part of our app, we just import that useTranslation hook and let it know which translation key to use:

1// /src/App.tsx 2import { useTranslation } from 'react-i18next' 3import { Typography } from '@mui/joy' 4 5function App(){ 6 const { t } = useTranslation() 7 8 return <Typography>{t('hello')}</Typography> 9} 10

More complex scenario's

Variables and pluralization

You might be thinking, what if some part of my text is a variable? Do I have to split the text out into chunks before and after the variable and translate them seperately? The answer is: nope, I18next has you covered!

1// translation.json 2{ 3 "welcome_one": "Welcome {{name}}, you have {{count}} message.", 4 "welcome_other": "Welcome {{name}}, you have {{count}} messages.", 5} 6
1// /src/App.tsx 2import { useTranslation } from 'react-i18next' 3import { Typography } from '@mui/joy' 4 5function App(){ 6 const { t } = useTranslation() 7 const [name, setName] = useState('') 8 const [count, setCount] = useState(1) 9 10 // ... 11 12 return <Typography>{t('welcome', { count, name })}</Typography> 13} 14

You can do a lot of other fancy stuff with i18next as well like context aware translations.

Custom React components

Sometimes you have some text with a link for example, or you may even want to use a custom React component inside a piece of translated text. For this you have to use the Trans component.

1// translations.json 2{ 3 "custom_link": "Translate some text with a custom <1>link</1>." 4} 5

Note the <1> placeholder that tells i18next where to expect a custom element.

1// /src/App.tsx 2import { Link } from '@mui/joy' 3import { Trans } from 'react-i18next' 4 5function App(){ 6 return <Trans i18nKey="custom_link"> 7 Translate some text with a custom <Link href="#link">link</Link>. 8 </Trans> 9} 10

Adding a language switcher

Building a language switcher with i18next is very simple, all we have to do is make some buttons and call the i18n.changeLanguage method with the language that we want to switch to:

1// /src/App.tsx 2import { Typography, IconButton } from '@mui/joy' 3import { useTranslation, Trans } from 'react-i18next' 4 5function App(){ 6 const { t, i18n } = useTranslation() 7 8 return <> 9 <Typography>{t('hello')}</Typography> 10 <IconButton onClick={() => i18n.changeLanguage('en')} size="sm" variant={i18n.language ==='en' ? 'solid' : 'soft'}>EN</IconButton> 11 <IconButton onClick={() => i18n.changeLanguage('nl')} size="sm" variant={i18n.language ==='nl' ? 'solid' : 'soft'}>NL</IconButton> 12 </> 13} 14
  • Code Sandbox
  • React-i18next Step-by-step guide

Table of contents

Setup Create config and add i18next provider Add the translations files Typescript Using translations More complex scenario's Variables and pluralization Custom React components Adding a language switcher Links