mirror of
https://github.com/schlagmichdoch/PairDrop.git
synced 2025-04-21 07:16:18 -04:00
Refactor localization.js
This commit is contained in:
parent
d3a623d352
commit
c08b324d6a
2 changed files with 91 additions and 51 deletions
|
@ -1,40 +1,49 @@
|
||||||
class Localization {
|
class Localization {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
Localization.$htmlRoot = document.querySelector('html');
|
||||||
|
|
||||||
Localization.defaultLocale = "en";
|
Localization.defaultLocale = "en";
|
||||||
Localization.supportedLocales = ["ar", "ca", "de", "en", "es", "fr", "id", "it", "ja", "nb", "nl", "pt-BR", "ro", "ru", "tr", "zh-CN"];
|
Localization.supportedLocales = ["ar", "ca", "de", "en", "es", "fr", "id", "it", "ja", "nb", "nl", "pt-BR", "ro", "ru", "tr", "zh-CN"];
|
||||||
Localization.supportedLocalesRtl = ["ar"];
|
Localization.supportedLocalesRtl = ["ar"];
|
||||||
|
|
||||||
Localization.translations = {};
|
Localization.translations = {};
|
||||||
Localization.defaultTranslations = {};
|
Localization.translationsDefaultLocale = {};
|
||||||
|
|
||||||
Localization.systemLocale = Localization.getSupportedOrDefault(navigator.languages);
|
Localization.systemLocale = Localization.getSupportedOrDefaultLocales(navigator.languages);
|
||||||
|
|
||||||
let storedLanguageCode = localStorage.getItem('language_code');
|
let storedLanguageCode = localStorage.getItem('language_code');
|
||||||
|
|
||||||
Localization.initialLocale = storedLanguageCode && Localization.isSupported(storedLanguageCode)
|
Localization.initialLocale = storedLanguageCode && Localization.localeIsSupported(storedLanguageCode)
|
||||||
? storedLanguageCode
|
? storedLanguageCode
|
||||||
: Localization.systemLocale;
|
: Localization.systemLocale;
|
||||||
}
|
}
|
||||||
|
|
||||||
static isSupported(locale) {
|
static localeIsSupported(locale) {
|
||||||
return Localization.supportedLocales.indexOf(locale) > -1;
|
return Localization.supportedLocales.indexOf(locale) > -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static isRtlLanguage(locale) {
|
static localeIsRtl(locale) {
|
||||||
return Localization.supportedLocalesRtl.indexOf(locale) > -1;
|
return Localization.supportedLocalesRtl.indexOf(locale) > -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static isCurrentLocaleRtl() {
|
static currentLocaleIsRtl() {
|
||||||
return Localization.isRtlLanguage(Localization.locale);
|
return Localization.localeIsRtl(Localization.locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
static getSupportedOrDefault(locales) {
|
static currentLocaleIsDefault() {
|
||||||
|
return Localization.locale === Localization.defaultLocale
|
||||||
|
}
|
||||||
|
|
||||||
|
static getSupportedOrDefaultLocales(locales) {
|
||||||
|
// get generic locales not included in locales
|
||||||
|
// ["en-us", "de-CH", "fr"] --> ["en", "de"]
|
||||||
let localesGeneric = locales
|
let localesGeneric = locales
|
||||||
.map(locale => locale.split("-")[0])
|
.map(locale => locale.split("-")[0])
|
||||||
.filter(locale => locales.indexOf(locale) === -1);
|
.filter(locale => locales.indexOf(locale) === -1);
|
||||||
|
|
||||||
return locales.find(Localization.isSupported)
|
// If there is no perfect match for browser locales, try generic locales first before resorting to the default locale
|
||||||
|| localesGeneric.find(Localization.isSupported)
|
return locales.find(Localization.localeIsSupported)
|
||||||
|
|| localesGeneric.find(Localization.localeIsSupported)
|
||||||
|| Localization.defaultLocale;
|
|| Localization.defaultLocale;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,16 +57,14 @@ class Localization {
|
||||||
await Localization.setLocale(locale)
|
await Localization.setLocale(locale)
|
||||||
await Localization.translatePage();
|
await Localization.translatePage();
|
||||||
|
|
||||||
const htmlRootNode = document.querySelector('html');
|
if (Localization.localeIsRtl(locale)) {
|
||||||
|
Localization.$htmlRoot.setAttribute('dir', 'rtl');
|
||||||
if (Localization.isRtlLanguage(locale)) {
|
|
||||||
htmlRootNode.setAttribute('dir', 'rtl');
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
htmlRootNode.removeAttribute('dir');
|
Localization.$htmlRoot.removeAttribute('dir');
|
||||||
}
|
}
|
||||||
|
|
||||||
htmlRootNode.setAttribute('lang', locale);
|
Localization.$htmlRoot.setAttribute('lang', locale);
|
||||||
|
|
||||||
|
|
||||||
console.log("Page successfully translated",
|
console.log("Page successfully translated",
|
||||||
|
@ -111,75 +118,108 @@ class Localization {
|
||||||
const key = element.getAttribute("data-i18n-key");
|
const key = element.getAttribute("data-i18n-key");
|
||||||
const attrs = element.getAttribute("data-i18n-attrs").split(" ");
|
const attrs = element.getAttribute("data-i18n-attrs").split(" ");
|
||||||
|
|
||||||
for (let i in attrs) {
|
attrs.forEach(attr => {
|
||||||
let attr = attrs[i];
|
|
||||||
if (attr === "text") {
|
if (attr === "text") {
|
||||||
element.innerText = Localization.getTranslation(key);
|
element.innerText = Localization.getTranslation(key);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
element.setAttribute(attr, Localization.getTranslation(key, attr));
|
element.setAttribute(attr, Localization.getTranslation(key, attr));
|
||||||
}
|
}
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
static getTranslation(key, attr = null, data = {}, useDefault = false) {
|
static getTranslationFromTranslationsObj(translationObj, key, attr) {
|
||||||
const keys = key.split(".");
|
|
||||||
|
|
||||||
let translationCandidates = useDefault
|
|
||||||
? Localization.defaultTranslations
|
|
||||||
: Localization.translations;
|
|
||||||
|
|
||||||
let translation;
|
let translation;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
const keys = key.split(".");
|
||||||
|
|
||||||
for (let i = 0; i < keys.length - 1; i++) {
|
for (let i = 0; i < keys.length - 1; i++) {
|
||||||
translationCandidates = translationCandidates[keys[i]]
|
// iterate into translation object until last layer
|
||||||
|
translationObj = translationObj[keys[i]]
|
||||||
}
|
}
|
||||||
|
|
||||||
let lastKey = keys[keys.length - 1];
|
let lastKey = keys[keys.length - 1];
|
||||||
|
|
||||||
if (attr) lastKey += "_" + attr;
|
if (attr) lastKey += "_" + attr;
|
||||||
|
|
||||||
translation = translationCandidates[lastKey];
|
translation = translationObj[lastKey];
|
||||||
|
|
||||||
for (let j in data) {
|
|
||||||
if (translation.includes(`{{${j}}}`)) {
|
|
||||||
translation = translation.replace(`{{${j}}}`, data[j]);
|
|
||||||
} else {
|
|
||||||
console.warn(`Translation for your language ${Localization.locale.toUpperCase()} misses at least one data placeholder:`, key, attr, data);
|
|
||||||
Localization.logHelpCallKey(key);
|
|
||||||
Localization.logHelpCall();
|
|
||||||
translation = "";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
translation = "";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!translation) {
|
if (!translation) {
|
||||||
if (!useDefault) {
|
throw new Error(`Translation misses entry. Key: ${key} Attribute: ${attr}`);
|
||||||
console.warn(`Missing translation entry for your language ${Localization.locale.toUpperCase()}. Using ${Localization.defaultLocale.toUpperCase()} instead.`, key, attr);
|
}
|
||||||
Localization.logHelpCallKey(key);
|
|
||||||
Localization.logHelpCall();
|
return translation;
|
||||||
translation = this.getTranslation(key, attr, data, true);
|
}
|
||||||
|
|
||||||
|
static addDataToTranslation(translation, data) {
|
||||||
|
for (let j in data) {
|
||||||
|
if (!translation.includes(`{{${j}}}`)) {
|
||||||
|
throw new Error(`Translation misses data placeholder: ${j}`);
|
||||||
|
}
|
||||||
|
// Add data to translation
|
||||||
|
translation = translation.replace(`{{${j}}}`, data[j]);
|
||||||
|
}
|
||||||
|
return translation;
|
||||||
|
}
|
||||||
|
|
||||||
|
static getTranslation(key, attr = null, data = {}, useDefault = false) {
|
||||||
|
let translationObj = useDefault
|
||||||
|
? Localization.translationsDefaultLocale
|
||||||
|
: Localization.translations;
|
||||||
|
|
||||||
|
let translation;
|
||||||
|
|
||||||
|
try {
|
||||||
|
translation = Localization.getTranslationFromTranslationsObj(translationObj, key, attr);
|
||||||
|
translation = Localization.addDataToTranslation(translation, data);
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
// Log warnings and help calls
|
||||||
|
console.warn(e);
|
||||||
|
Localization.logTranslationMissingOrBroken(key, attr, data, useDefault);
|
||||||
|
Localization.logHelpCallKey(key, attr);
|
||||||
|
Localization.logHelpCall();
|
||||||
|
|
||||||
|
if (useDefault || Localization.currentLocaleIsDefault()) {
|
||||||
|
// Is default locale already
|
||||||
|
// Use empty string as translation
|
||||||
|
translation = ""
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
console.warn("Missing translation in default language:", key, attr);
|
// Is not default locale yet
|
||||||
Localization.logHelpCall();
|
// Get translation for default language with same arguments
|
||||||
|
console.log(`Using default language ${Localization.defaultLocale.toUpperCase()} instead.`);
|
||||||
|
translation = this.getTranslation(key, attr, data, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return Localization.escapeHTML(translation);
|
return Localization.escapeHTML(translation);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static logTranslationMissingOrBroken(key, attr, data, useDefault) {
|
||||||
|
let usedLocale = useDefault
|
||||||
|
? Localization.defaultLocale.toUpperCase()
|
||||||
|
: Localization.locale.toUpperCase();
|
||||||
|
|
||||||
|
console.warn(`Missing or broken translation for language ${usedLocale}.\n`, 'key:', key, 'attr:', attr, 'data:', data);
|
||||||
|
}
|
||||||
|
|
||||||
static logHelpCall() {
|
static logHelpCall() {
|
||||||
console.log("Help translating PairDrop: https://hosted.weblate.org/engage/pairdrop/");
|
console.log("Help translating PairDrop: https://hosted.weblate.org/engage/pairdrop/");
|
||||||
}
|
}
|
||||||
|
|
||||||
static logHelpCallKey(key) {
|
static logHelpCallKey(key, attr) {
|
||||||
console.warn(`Translate this string here: https://hosted.weblate.org/browse/pairdrop/pairdrop-spa/${Localization.locale.toLowerCase()}/?q=${key}`);
|
let locale = Localization.locale.toLowerCase();
|
||||||
|
|
||||||
|
let keyComplete = !attr || attr === "text"
|
||||||
|
? key
|
||||||
|
: `${key}_${attr}`;
|
||||||
|
|
||||||
|
console.warn(`Translate this string here: https://hosted.weblate.org/browse/pairdrop/pairdrop-spa/${locale}/?q=${keyComplete}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
static escapeHTML(unsafeText) {
|
static escapeHTML(unsafeText) {
|
||||||
|
|
|
@ -132,7 +132,7 @@ class HeaderUI {
|
||||||
this.$header.classList.remove('overflow-expanded');
|
this.$header.classList.remove('overflow-expanded');
|
||||||
|
|
||||||
|
|
||||||
const rtlLocale = Localization.isCurrentLocaleRtl();
|
const rtlLocale = Localization.currentLocaleIsRtl();
|
||||||
let icon;
|
let icon;
|
||||||
const $headerIconsShown = document.querySelectorAll('body > header:first-of-type > *:not([hidden])');
|
const $headerIconsShown = document.querySelectorAll('body > header:first-of-type > *:not([hidden])');
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue