2019-12-07 13:28:52 +02:00
|
|
|
import { IntlShape, createIntl, createIntlCache } from 'react-intl';
|
2019-12-07 21:02:00 +02:00
|
|
|
import captcha from 'app/services/captcha';
|
|
|
|
import locales from 'app/i18n';
|
2019-11-11 10:40:05 +02:00
|
|
|
|
|
|
|
import intlPolyfill from './intlPolyfill';
|
|
|
|
|
2020-01-20 09:51:37 +02:00
|
|
|
export const SUPPORTED_LANGUAGES = Object.keys(locales);
|
|
|
|
export const DEFAULT_LANGUAGE = 'en';
|
2019-11-11 10:40:05 +02:00
|
|
|
|
|
|
|
function getBrowserPreferredLanguages(): string[] {
|
2020-05-24 02:08:24 +03:00
|
|
|
return ([] as string[]).concat(navigator['languages'] || []).concat(navigator['language'] || []);
|
2019-11-11 10:40:05 +02:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
function detectLanguage(userLanguages: string[], availableLanguages: string[], defaultLanguage: string): string {
|
|
|
|
return (
|
|
|
|
userLanguages
|
|
|
|
.map((lang) => (lang.split('-').shift() || '').toLowerCase())
|
|
|
|
.find((lang) => availableLanguages.indexOf(lang) !== -1) || defaultLanguage
|
|
|
|
);
|
2019-11-11 10:40:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const cache = createIntlCache();
|
|
|
|
|
|
|
|
let intl: IntlShape;
|
|
|
|
|
|
|
|
class I18N {
|
2020-05-24 02:08:24 +03:00
|
|
|
detectLanguage(lang: string = ''): string {
|
|
|
|
return detectLanguage(
|
|
|
|
[lang].concat(getBrowserPreferredLanguages()).filter((item) => !!item),
|
|
|
|
SUPPORTED_LANGUAGES,
|
|
|
|
DEFAULT_LANGUAGE,
|
|
|
|
);
|
2019-11-11 10:40:05 +02:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
getIntl(): IntlShape {
|
|
|
|
if (!intl) {
|
|
|
|
intl = createIntl(
|
|
|
|
{
|
|
|
|
locale: 'en',
|
|
|
|
messages: {},
|
|
|
|
},
|
|
|
|
cache,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return intl;
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
async changeLocale(locale: string = DEFAULT_LANGUAGE): Promise<IntlShape> {
|
|
|
|
const { messages } = await this.require(locale);
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
captcha.setLang(locale);
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
intl = createIntl(
|
|
|
|
{
|
|
|
|
locale,
|
|
|
|
messages,
|
|
|
|
},
|
|
|
|
cache,
|
|
|
|
);
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
return intl;
|
|
|
|
}
|
|
|
|
|
|
|
|
async ensureIntl() {
|
|
|
|
await intlPolyfill('en');
|
|
|
|
}
|
|
|
|
|
2021-07-13 22:40:31 +02:00
|
|
|
async require(locale: string): Promise<{
|
2020-05-24 02:08:24 +03:00
|
|
|
locale: string;
|
2020-06-05 18:24:41 +03:00
|
|
|
messages: Record<string, string>;
|
2020-05-24 02:08:24 +03:00
|
|
|
}> {
|
2020-06-09 21:02:03 +03:00
|
|
|
try {
|
|
|
|
const [{ default: messages }] = await Promise.all([
|
|
|
|
import(/* webpackChunkName: "locale-[request]" */ `app/i18n/${locale}.json`),
|
|
|
|
intlPolyfill(locale),
|
|
|
|
]);
|
|
|
|
|
|
|
|
return {
|
|
|
|
locale,
|
|
|
|
messages,
|
|
|
|
};
|
|
|
|
} catch (err) {
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
if (err.message === "Cannot find module './en.json'") {
|
|
|
|
console.warn(
|
|
|
|
[
|
|
|
|
"Locales module for the source language isn't exists.",
|
|
|
|
'You may generate this file by running yarn i18n:extract command.',
|
|
|
|
'Until then, defaultMessages will be used for displaying on the site.',
|
|
|
|
].join(' '),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return { locale, messages: {} };
|
|
|
|
}
|
|
|
|
|
|
|
|
throw err;
|
|
|
|
}
|
2020-05-24 02:08:24 +03:00
|
|
|
}
|
2019-11-11 10:40:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export default new I18N();
|