2024-08-15 17:28:33 +02:00
|
|
|
class WebAppLocalizer {
|
|
|
|
constructor(translations, localStoragePrefix = 'webAppLocalizer_', languageSelector = null) {
|
|
|
|
this.translations = translations;
|
|
|
|
this.localStoragePrefix = localStoragePrefix;
|
|
|
|
this.currentLang = this.loadCurrentLanguage() || Object.keys(translations)[0];
|
2024-08-25 04:21:43 +02:00
|
|
|
|
|
|
|
if (typeof languageSelector === 'string') {
|
|
|
|
this.languageSelector = document.getElementById(languageSelector);
|
|
|
|
} else {
|
|
|
|
this.languageSelector = languageSelector;
|
|
|
|
}
|
2024-08-15 17:28:33 +02:00
|
|
|
|
|
|
|
if (this.languageSelector) {
|
|
|
|
this.initializeLanguageSelector();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
loadCurrentLanguage() {
|
|
|
|
return localStorage.getItem(this.localStoragePrefix + 'currentLang');
|
|
|
|
}
|
|
|
|
|
|
|
|
saveCurrentLanguage(lang) {
|
|
|
|
localStorage.setItem(this.localStoragePrefix + 'currentLang', lang);
|
|
|
|
}
|
|
|
|
|
|
|
|
setLanguage(lang) {
|
|
|
|
if (this.translations.hasOwnProperty(lang)) {
|
|
|
|
this.currentLang = lang;
|
|
|
|
this.saveCurrentLanguage(lang);
|
|
|
|
this.apply();
|
|
|
|
if (this.languageSelector) {
|
|
|
|
this.languageSelector.value = lang;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
console.warn(`Language '${lang}' not found in translations.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getCurrentLanguage() {
|
|
|
|
return this.currentLang;
|
|
|
|
}
|
|
|
|
|
|
|
|
getAvailableLanguages() {
|
|
|
|
return Object.keys(this.translations).map(lang => ({
|
|
|
|
code: lang,
|
|
|
|
name: this.translations[lang].name
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
translate(key) {
|
|
|
|
const translations = this.translations[this.currentLang].translations;
|
|
|
|
return translations[key] || key;
|
|
|
|
}
|
|
|
|
|
|
|
|
apply() {
|
|
|
|
const elements = document.querySelectorAll('[data-translate]');
|
|
|
|
elements.forEach(element => {
|
|
|
|
const key = element.getAttribute('data-translate');
|
2024-09-17 21:18:18 +02:00
|
|
|
const useHTML = element.hasAttribute('data-translate-html');
|
|
|
|
|
|
|
|
if (useHTML) {
|
|
|
|
element.innerHTML = this.translate(key);
|
|
|
|
} else {
|
|
|
|
element.textContent = this.translate(key);
|
|
|
|
}
|
2024-08-15 17:28:33 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
initializeLanguageSelector() {
|
|
|
|
if (!(this.languageSelector instanceof HTMLElement)) {
|
|
|
|
console.warn('Language selector is not a valid HTML element.');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.languageSelector.tagName.toLowerCase() !== 'select') {
|
|
|
|
// Create a select element if the provided element is not a select
|
|
|
|
const selectElement = document.createElement('select');
|
|
|
|
this.languageSelector.appendChild(selectElement);
|
|
|
|
this.languageSelector = selectElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear existing options
|
|
|
|
this.languageSelector.innerHTML = '';
|
|
|
|
|
|
|
|
// Add options for each available language
|
|
|
|
this.getAvailableLanguages().forEach(lang => {
|
|
|
|
const option = document.createElement('option');
|
|
|
|
option.value = lang.code;
|
|
|
|
option.textContent = lang.name;
|
|
|
|
this.languageSelector.appendChild(option);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Set the current language
|
|
|
|
this.languageSelector.value = this.currentLang;
|
|
|
|
|
|
|
|
// Add event listener for language change
|
|
|
|
this.languageSelector.addEventListener('change', (event) => {
|
|
|
|
this.setLanguage(event.target.value);
|
|
|
|
});
|
|
|
|
}
|
2024-08-27 01:59:25 +02:00
|
|
|
// helper functions
|
|
|
|
// A format prompt function
|
|
|
|
formatPrompt(prompt_name, values) {
|
2024-09-01 03:51:42 +02:00
|
|
|
return this.translations[this.currentLang].promptTranslations[prompt_name].replace(/\{(\w+)\}/g, (match, key) => values[key] || match);
|
2024-08-27 01:59:25 +02:00
|
|
|
}
|
2024-08-15 17:28:33 +02:00
|
|
|
}
|
2024-08-27 01:59:25 +02:00
|
|
|
|