Learn more about implementing multiple language support in your customizations.
en
, de
, and es
. The default language is en
.The Merchant Center locale setting can be changed within the user profile page.
When you develop a customization, you have the option to translate it in one or more languages.
Translations are usually included as JSON files that are dynamically imported into the customization based on the user's locale.
To facilitate the management of source messages and translations, we provide some built-in functionalities and tooling.
Define source messages
messages.js
file next to your React component.import { defineMessages } from 'react-intl';
export default defineMessages({
title: {
id: 'Channels.title',
defaultMessage: 'Channels list',
},
});
react-intl
, a library with React abstractions on top of the ECMAScript Internationalization API.The source messages can then be used in your React components.
Consume messages as React components
react-intl
library provides several React components to render the message, like <FormattedMessage>
.import { FormattedMessage } from 'react-intl';
import messages from './messages';
const Channels = () => {
// ...
return (
<Spacings.Stack scale="l">
<Spacings.Stack scale="s">
<Text.Headline as="h2">
<FormattedMessage message={messages.title}>
</Text.Headline>
</Spacings.Stack>
{/* ... */}
</Spacings.Stack>
)
};
intlMessage
. This lets you pass your raw message and the component takes care of rendering the message.import { FormattedMessage } from 'react-intl';
import messages from './messages';
const Channels = () => {
// ...
return (
<Spacings.Stack scale="l">
<Spacings.Stack scale="s">
<Text.Headline as="h2" intlMessage={messages.title} />
</Spacings.Stack>
{/* ... */}
</Spacings.Stack>
);
};
Consume messages as React Hooks
react-intl
library also provides an imperative API approach via useIntl
, which exposes an intl
object with several formatting methods.intl
object is usually necessary for props that only accept strings, for example aria-label
, title
, etc.import { useIntl } from 'react-intl';
const Button = () => {
const intl = useIntl();
return (
<FlatButton
label={intl.formatMessage(messages.save)}
// ...
/>
);
};
Extract messages for translations
@formatjs/cli
package to extract source messages into a core.json
source file.For example:
formatjs extract \
--out-file=./src/i18n/data/core.json \
'src/**/messages.js'
core.json
is the so-called source file, which should be used as the reference file for the translations in the other locales.i18n
folder:└── src
└── i18n
└── data
├── core.json
├── en.json
└── de.json
--format
option to the script.formatjs extract \
--format=./intl-transformer.js
--out-file=./src/i18n/data/core.json \
'src/**/messages.js'
core.json
file with the key being the message id and the value being the default message.{
"Channels.title": "Channels list"
}
Import translations
The JSON files that contain the translations need to be loaded within the customization.
- For Custom Applications: the
<ApplicationShell>
expects aapplicationMessages
prop that is used to load the messages in thereact-intl
provider. - For Custom Views: the
<CustomViewShell>
expects aapplicationMessages
prop that is used to load the messages in thereact-intl
provider.
applicationMessages
prop can either be a JSON object or a function returning a promise with the loaded messages.i18n/data
folder, you can define a function to dynamically load the messages.import {
parseChunkImport,
mapLocaleToIntlLocale,
} from '@commercetools-frontend/i18n';
const getChunkImport = (locale) => {
const intlLocale = mapLocaleToIntlLocale(locale);
switch (intlLocale) {
case 'de':
return import(
/* webpackChunkName: "app-i18n-de" */ './i18n/data/de.json'
);
default:
return import(
/* webpackChunkName: "app-i18n-en" */ './i18n/data/en.json'
);
}
};
const loadMessages = async (locale) => {
try {
const chunkImport = await getChunkImport(locale);
return parseChunkImport(chunkImport);
} catch (error) {
console.warn(
`Something went wrong while loading the app messages for ${locale}`,
error
);
return {};
}
};
export default loadMessages;
The starter templates already provide this function.
entry-point.jsx
component you can then pass the function to the applicationMessages
prop:For Custom Applications
For Custom Views
import loadMessages from '../../load-messages';
const EntryPoint = () => (
<ApplicationShell
applicationMessages={loadMessages}
// ...
/>
);