2020-01-21 01:14:10 +02:00
|
|
|
import LanguageDetector from "i18next-browser-languagedetector";
|
|
|
|
|
2020-06-29 19:22:27 +01:00
|
|
|
import fallbackLanguageData from "./locales/en.json";
|
2020-06-30 19:28:19 +01:00
|
|
|
import percentages from "./locales/percentages.json";
|
2020-06-29 19:22:27 +01:00
|
|
|
|
2020-06-30 19:28:19 +01:00
|
|
|
const COMPLETION_THRESHOLD_TO_EXCEED = 85;
|
|
|
|
|
|
|
|
interface Language {
|
|
|
|
lng: string;
|
|
|
|
label: string;
|
|
|
|
rtl?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const allLanguages: Language[] = [
|
2020-08-20 16:45:20 +02:00
|
|
|
{ lng: "bg-BG", label: "Български" },
|
|
|
|
{ lng: "de-DE", label: "Deutsch" },
|
|
|
|
{ lng: "es-ES", label: "Español" },
|
|
|
|
{ lng: "ca-ES", label: "Catalan" },
|
|
|
|
{ lng: "el-GR", label: "Ελληνικά" },
|
|
|
|
{ lng: "fr-FR", label: "Français" },
|
|
|
|
{ lng: "id-ID", label: "Bahasa Indonesia" },
|
|
|
|
{ lng: "it-IT", label: "Italiano" },
|
|
|
|
{ lng: "hu-HU", label: "Magyar" },
|
|
|
|
{ lng: "nl-NL", label: "Nederlands" },
|
|
|
|
{ lng: "nb-NO", label: "Norsk bokmål" },
|
|
|
|
{ lng: "nn-NO", label: "Norsk nynorsk" },
|
|
|
|
{ lng: "pl-PL", label: "Polski" },
|
|
|
|
{ lng: "pt-PT", label: "Português" },
|
|
|
|
{ lng: "ru-RU", label: "Русский" },
|
|
|
|
{ lng: "uk-UA", label: "Українська" },
|
|
|
|
{ lng: "fi-FI", label: "Suomi" },
|
|
|
|
{ lng: "tr-TR", label: "Türkçe" },
|
|
|
|
{ lng: "ja-JP", label: "日本語" },
|
|
|
|
{ lng: "ko-KR", label: "한국어" },
|
|
|
|
{ lng: "zh-TW", label: "繁體中文" },
|
|
|
|
{ lng: "zh-CN", label: "简体中文" },
|
|
|
|
{ lng: "ar-SA", label: "العربية", rtl: true },
|
|
|
|
{ lng: "he-IL", label: "עברית", rtl: true },
|
|
|
|
{ lng: "hi-IN", label: "हिन्दी" },
|
|
|
|
{ lng: "ta-IN", label: "தமிழ்" },
|
|
|
|
{ lng: "gl-ES", label: "Galego" },
|
2020-08-30 10:33:59 +03:00
|
|
|
{ lng: "ro-RO", label: "Română" },
|
2020-09-13 20:30:28 +03:00
|
|
|
{ lng: "sv-SE", label: "Svenska" },
|
2020-01-31 21:06:06 +00:00
|
|
|
];
|
|
|
|
|
2020-08-20 16:45:20 +02:00
|
|
|
export const languages: Language[] = [{ lng: "en", label: "English" }]
|
2020-06-30 19:28:19 +01:00
|
|
|
.concat(
|
|
|
|
allLanguages.sort((left, right) => (left.label > right.label ? 1 : -1)),
|
|
|
|
)
|
|
|
|
.filter(
|
|
|
|
(lang) =>
|
|
|
|
(percentages as Record<string, number>)[lang.lng] >
|
|
|
|
COMPLETION_THRESHOLD_TO_EXCEED,
|
|
|
|
);
|
|
|
|
|
2020-01-31 21:06:06 +00:00
|
|
|
let currentLanguage = languages[0];
|
2020-06-27 12:02:54 +01:00
|
|
|
let currentLanguageData = {};
|
2020-01-31 21:06:06 +00:00
|
|
|
const fallbackLanguage = languages[0];
|
|
|
|
|
2020-06-27 12:02:54 +01:00
|
|
|
export const setLanguage = async (newLng: string | undefined) => {
|
2020-01-31 21:06:06 +00:00
|
|
|
currentLanguage =
|
2020-03-23 13:05:07 +02:00
|
|
|
languages.find((language) => language.lng === newLng) || fallbackLanguage;
|
2020-01-31 21:06:06 +00:00
|
|
|
|
2020-04-02 12:21:19 -04:00
|
|
|
document.documentElement.dir = currentLanguage.rtl ? "rtl" : "ltr";
|
|
|
|
|
2020-10-18 23:06:25 +05:30
|
|
|
currentLanguageData = await import(
|
|
|
|
/* webpackChunkName: "i18n-[request]" */ `./locales/${currentLanguage.lng}.json`
|
|
|
|
);
|
2020-06-27 12:02:54 +01:00
|
|
|
|
|
|
|
languageDetector.cacheUserLanguage(currentLanguage.lng);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const setLanguageFirstTime = async () => {
|
|
|
|
const newLng: string | undefined = languageDetector.detect();
|
|
|
|
|
|
|
|
currentLanguage =
|
|
|
|
languages.find((language) => language.lng === newLng) || fallbackLanguage;
|
|
|
|
|
|
|
|
document.documentElement.dir = currentLanguage.rtl ? "rtl" : "ltr";
|
|
|
|
|
2020-10-18 23:06:25 +05:30
|
|
|
currentLanguageData = await import(
|
|
|
|
/* webpackChunkName: "i18n-[request]" */ `./locales/${currentLanguage.lng}.json`
|
|
|
|
);
|
2020-06-27 12:02:54 +01:00
|
|
|
|
2020-01-31 21:06:06 +00:00
|
|
|
languageDetector.cacheUserLanguage(currentLanguage.lng);
|
2020-05-20 16:21:37 +03:00
|
|
|
};
|
2020-01-31 21:06:06 +00:00
|
|
|
|
2020-05-20 16:21:37 +03:00
|
|
|
export const getLanguage = () => currentLanguage;
|
2020-01-22 16:25:04 +02:00
|
|
|
|
2020-05-20 16:21:37 +03:00
|
|
|
const findPartsForData = (data: any, parts: string[]) => {
|
2020-01-31 21:06:06 +00:00
|
|
|
for (var i = 0; i < parts.length; ++i) {
|
|
|
|
const part = parts[i];
|
|
|
|
if (data[part] === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
data = data[part];
|
2020-01-22 16:25:04 +02:00
|
|
|
}
|
2020-01-31 21:06:06 +00:00
|
|
|
if (typeof data !== "string") {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
return data;
|
2020-05-20 16:21:37 +03:00
|
|
|
};
|
2020-01-22 16:25:04 +02:00
|
|
|
|
2020-05-20 16:21:37 +03:00
|
|
|
export const t = (path: string, replacement?: { [key: string]: string }) => {
|
2020-01-31 21:06:06 +00:00
|
|
|
const parts = path.split(".");
|
|
|
|
let translation =
|
2020-06-27 12:02:54 +01:00
|
|
|
findPartsForData(currentLanguageData, parts) ||
|
|
|
|
findPartsForData(fallbackLanguageData, parts);
|
2020-01-31 21:06:06 +00:00
|
|
|
if (translation === undefined) {
|
2020-02-02 20:04:35 +02:00
|
|
|
throw new Error(`Can't find translation for ${path}`);
|
2020-01-31 21:06:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (replacement) {
|
|
|
|
for (var key in replacement) {
|
2020-02-02 20:04:35 +02:00
|
|
|
translation = translation.replace(`{{${key}}}`, replacement[key]);
|
2020-01-31 21:06:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return translation;
|
2020-05-20 16:21:37 +03:00
|
|
|
};
|
2020-01-31 21:06:06 +00:00
|
|
|
|
|
|
|
const languageDetector = new LanguageDetector();
|
|
|
|
languageDetector.init({
|
|
|
|
languageUtils: {
|
2020-05-20 16:21:37 +03:00
|
|
|
formatLanguageCode: (lng: string) => lng,
|
2020-01-31 21:06:06 +00:00
|
|
|
isWhitelisted: () => true,
|
|
|
|
},
|
|
|
|
checkWhitelist: false,
|
|
|
|
});
|