Common i18n patterns in React
Following page describes the most common i18n patterns in React. It's a follow-up to tutorial with practical examples. See the API reference for detailed information about all components.
Macros
Using jsx macros is the most straightforward way how to translate your React components.
Trans
handles translations of messages including variables and other React components:
import { Trans } from "@lingui/macro"
function render() {
return (
<>
<h1><Trans>LinguiJS example</Trans></h1>
<p><Trans>Hello <a href="/profile">{name}</a>.</Trans></p>
</>
);
}
You don't need anything special to use Trans
inside your app (except of wrapping the root component in I18nProvider
).
Choosing between generated and explicit ID
Using generated IDs provides more scalability and gives a better developer experience. On the other hand, explicit IDs for phrases make it easier to identify phrases out of context and to track where they're used. IDs usually follow a naming scheme that includes where the phrase is used.
Using ID generated from message
With Trans
In the example code above, the content of Trans
is transformed into a message in MessageFormat syntax. By default, this message is used for generating the ID. Considering the example above, the catalog would be fulfilled by these entries:
const catalog = [
{
id: "uxV9Xq",
message: "LinguiJS example",
},
{
id: "9/omjw",
message: "Hello <0>{name}</0>.",
},
]
With t
In the following example, the message Image caption
will be extracted and used to create an ID.
import { t } from "@lingui/macro"
export default function ImageWithCaption() {
return <img src="..." alt={t`Image caption`} />
}
Providing context for a message
Use context
to add more contextual information for translators. It also influences the ID generation: the same text elements with different contexts are extracted with different IDs.
See Context for more details.
import { Trans } from "@lingui/macro";
<Trans context="direction">right</Trans>;
<Trans context="correctness">right</Trans>;
// ↓ ↓ ↓ ↓ ↓ ↓
import { Trans } from "@lingui/react";
<Trans id={"d1wX4r"} message="right" />;
<Trans id={"16eaSK"} message="right" />;
Using custom ID
With Trans
If you're using custom IDs in your project, add id
prop to i18n components:
import { Trans } from "@lingui/macro"
function render() {
return (
<>
<h1><Trans id="msg.header">LinguiJS example</Trans></h1>
<p><Trans id="msg.hello">Hello <a href="/profile">{name}</a>.</Trans></p>
</>
);
}
Messages msg.header
and msg.hello
will be extracted with default values LinguiJS example
and Hello <0>{name}</0>.
.
With t
If you're using custom IDs in your project, call t
with a message descriptor object and pass ID as id
prop:
import { t } from "@lingui/macro"
export default function ImageWithCaption() {
return <img src="..." alt={t({id: 'msg.caption', message: `Image caption`})} />
}
Message msg.caption
will be extracted with default value Image caption
.
For all other js macros (plural
, select
, selectOrdinal
, use them inside t
macro to pass ID (in this case, 'msg.caption'
).
import { t, plural } from "@lingui/macro"
export default function ImageWithCaption({ count }) {
return (
<img src="..." alt={t({id: 'msg.caption', message: plural(count, {
one: "# image caption",
other: "# image captions",
})})} />
)
}
Element attributes and string-only translations
Sometimes you can't use Trans
component, for example when translating element attributes:
<img src="..." alt="Image caption" />
In such case you need to use t
macro to wrap message. t
is equivalent for Trans
, plural
is equivalent to Plural
.
You also need to use useLingui
hook to subscribe your component for locale updates.
import { t } from "@lingui/macro"
import { useLingui } from "@lingui/react"
export default function ImageWithCaption() {
useLingui()
return <img src="..." alt={t`Image caption`} />
}
Translations outside React components
Another common pattern is when you need to access translations outside React components, for example inside redux-saga
. You can use t
macro outside React context as usual:
import { t } from "@lingui/macro"
export function alert() {
// use t as if you were inside a React component
alert(t`...`)
}
The t
macro can only be used in a reactive or re-executable context.
import { t } from "@lingui/macro"
// ❌ Bad! This won't work because the `t` macro is used at the module level.
// The `t` macro returns a string, and once this string is assigned, it won't react to changes.
const alertProps = {
header: t`Alert`,
subHeader: t`Important message`,
message: t`This is an alert!`,
buttons: [t`OK`]
};
// ✅ Good! Every time the function is executed, the `t` macro will be re-executed as well,
// and the actual result will be returned.
function getAlertProps() {
return {
header: t`Alert`,
subHeader: t`Important message`,
message: t`This is an alert!`,
buttons: [t`OK`]
}
}
Another option would be to use the Lazy Translations pattern described in the following paragraph.
Lazy Translations
Messages don't have to be declared at the same code location where they're displayed. Tag a string with the defineMessage
macro, and you've created a "message descriptor", which can then be passed around as a variable, and can be displayed as a translated string by passing its id
to Trans
as its id
prop:
import { msg } from "@lingui/macro"
import { Trans } from "@lingui/react"
const favoriteColors = [
msg`Red`,
msg`Orange`,
msg`Yellow`,
msg`Green`,
]
export default function ColorList() {
return (
<ul>
{favoriteColors.map(color => (
<li><Trans id={color.id}/></li>
))}
</ul>
)
}
Or to render the message descriptor as a string-only translation, pass it to the i18n._()
method:
import { i18n } from "@lingui/core"
import { msg } from "@lingui/macro"
const favoriteColors = [
msg`Red`,
msg`Orange`,
msg`Yellow`,
msg`Green`,
]
export function getTranslatedColorNames() {
return favoriteColors.map(
color => i18n._(color)
)
}
Passing messages as props
It's often convenient to pass messages around as component props, for example as a "label" prop on a button. The easiest way to do this is to pass a Trans
element as the prop:
import { Trans } from "@lingui/macro"
export default function FancyButton(props) {
return <button>{props.label}</button>
}
export function LoginLogoutButtons(props) {
return <div>
<FancyButton label={<Trans>Log in</Trans>} />
<FancyButton label={<Trans>Log out</Trans>} />
</div>
}
If you need the prop to be displayed as a string-only translation, you can pass a message tagged with the t
macro:
import { t } from "@lingui/macro"
import { useLingui } from "@lingui/react"
export default function ImageWithCaption(props) {
return <img src="..." alt={props.caption} />
}
export function HappySad(props) {
useLingui()
return <div>
<ImageWithCaption caption={t`I'm so happy!`} />
<ImageWithCaption caption={t`I'm so sad.`} />
</div>
}
Picking a message based on a variable
Sometimes you need to pick between different messages to display, depending on the value of a variable. For example, imagine you have a numeric "status" code that comes from an API, and you need to display a message representing the current status.
A simple way to do this, is to make an object that maps the possible values of "status" to message descriptors (tagged with the defineMessage
macro), and render them as needed with deferred translation:
import { msg } from "@lingui/macro";
import { useLingui } from "@lingui/react";
const statusMessages = {
['STATUS_OPEN']: msg`Open`,
['STATUS_CLOSED']: msg`Closed`,
['STATUS_CANCELLED']: msg`Cancelled`,
['STATUS_COMPLETED']: msg`Completed`,
}
export default function StatusDisplay({ statusCode }) {
const { i18n } = useLingui();
return <div>{i18n._(statusMessages[statusCode])}</div>
}
Memoization pitfall
In the following contrived example, we document how welcome message will or will not be updated when locale changes.
The documented behavior is expected, because of how useMemo
dependencies work. In order for translations to update, the useMemo
needs to depend on the i18n context.
We acknowledge that this is not intuitive, and we're open to accepting a solution to make this easier.
Please also note that useMemo
is meant as a performance optimization in React and you probably don't need to memoize your translations. Additionally, this issue is not present when using the Trans
component which we recommend to use when possible.
import { msg } from "@lingui/macro";
import { i18n } from "@lingui/core"
const welcomeMessage = msg`Welcome!`;
// ❌ Bad! This code won't work
export function Welcome() {
const buggyWelcome = useMemo(() => {
return i18n._(welcomeMessage);
}, []);
return <div>{buggyWelcome}</div>;
}
// ❌ Bad! This code won't work either because the reference to i18n does not change
export function Welcome() {
const { i18n } = useLingui();
const buggyWelcome = useMemo(() => {
return i18n._(welcomeMessage);
}, [i18n]);
return <div>{buggyWelcome}</div>;
}
// ✅ Good! `useMemo` has i18n context in the dependency
export function Welcome() {
const linguiCtx = useLingui();
const welcome = useMemo(() => {
return linguiCtx.i18n._(welcomeMessage);
}, [linguiCtx]);
return <div>{welcome}</div>;
}