mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			248 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			248 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { registerDebugFunction } from './power-user.js';
 | |
| import { updateSecretDisplay } from './secrets.js';
 | |
| 
 | |
| const storageKey = 'language';
 | |
| const overrideLanguage = localStorage.getItem(storageKey);
 | |
| const localeFile = String(overrideLanguage || navigator.language || navigator.userLanguage || 'en').toLowerCase();
 | |
| var langs;
 | |
| // Don't change to let/const! It will break module loading.
 | |
| // eslint-disable-next-line prefer-const
 | |
| var localeData;
 | |
| 
 | |
| export const getCurrentLocale = () => localeFile;
 | |
| 
 | |
| /**
 | |
|  * An observer that will check if any new i18n elements are added to the document
 | |
|  * @type {MutationObserver}
 | |
|  */
 | |
| const observer = new MutationObserver(mutations => {
 | |
|     mutations.forEach(mutation => {
 | |
|         mutation.addedNodes.forEach(node => {
 | |
|             if (node.nodeType === Node.ELEMENT_NODE && node instanceof Element) {
 | |
|                 if (node.hasAttribute('data-i18n')) {
 | |
|                     translateElement(node);
 | |
|                 }
 | |
|                 node.querySelectorAll('[data-i18n]').forEach(element => {
 | |
|                     translateElement(element);
 | |
|                 });
 | |
|             }
 | |
|         });
 | |
|         if (mutation.attributeName === 'data-i18n' && mutation.target instanceof Element) {
 | |
|             translateElement(mutation.target);
 | |
|         }
 | |
|     });
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * Translates a template string with named arguments
 | |
|  *
 | |
|  * Uses the template literal with all values replaced by index placeholder for translation key.
 | |
|  *
 | |
|  * @example
 | |
|  * ```js
 | |
|  * toastr.warn(t`Tag ${tagName} not found.`);
 | |
|  * ```
 | |
|  * Should be translated in the translation files as:
 | |
|  * ```
 | |
|  * Tag ${0} not found. -> Tag ${0} nicht gefunden.
 | |
|  * ```
 | |
|  *
 | |
|  * @param {TemplateStringsArray} strings - Template strings array
 | |
|  * @param  {...any} values - Values for placeholders in the template string
 | |
|  * @returns {string} Translated and formatted string
 | |
|  */
 | |
| export function t(strings, ...values) {
 | |
|     let str = strings.reduce((result, string, i) => result + string + (values[i] !== undefined ? `\${${i}}` : ''), '');
 | |
|     let translatedStr = translate(str);
 | |
| 
 | |
|     // Replace indexed placeholders with actual values
 | |
|     return translatedStr.replace(/\$\{(\d+)\}/g, (match, index) => values[index]);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Translates a given key or text
 | |
|  *
 | |
|  * If the translation is based on a key, that one is used to find a possible translation in the translation file.
 | |
|  * The original text still has to be provided, as that is the default value being returned if no translation is found.
 | |
|  *
 | |
|  * For in-code text translation on a format string, using the template literal `t` is preferred.
 | |
|  *
 | |
|  * @param {string} text - The text to translate
 | |
|  * @param {string?} key - The key to use for translation. If not provided, text is used as the key.
 | |
|  * @returns {string} - The translated text
 | |
|  */
 | |
| export function translate(text, key = null) {
 | |
|     const translationKey = key || text;
 | |
|     return localeData?.[translationKey] || text;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Fetches the locale data for the given language.
 | |
|  * @param {string} language Language code
 | |
|  * @returns {Promise<Record<string, string>>} Locale data
 | |
|  */
 | |
| async function getLocaleData(language) {
 | |
|     let supportedLang = findLang(language);
 | |
|     if (!supportedLang) {
 | |
|         return {};
 | |
|     }
 | |
| 
 | |
|     const data = await fetch(`./locales/${language}.json`).then(response => {
 | |
|         console.log(`Loading locale data from ./locales/${language}.json`);
 | |
|         if (!response.ok) {
 | |
|             return {};
 | |
|         }
 | |
|         return response.json();
 | |
|     });
 | |
| 
 | |
|     return data;
 | |
| }
 | |
| 
 | |
| function findLang(language) {
 | |
|     var supportedLang = langs.find(x => x.lang === language);
 | |
| 
 | |
|     if (!supportedLang) {
 | |
|         console.warn(`Unsupported language: ${language}`);
 | |
|     }
 | |
|     return supportedLang;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Translates a given element based on its data-i18n attribute.
 | |
|  * @param {Element} element The element to translate
 | |
|  */
 | |
| function translateElement(element) {
 | |
|     const keys = element.getAttribute('data-i18n').split(';'); // Multi-key entries are ; delimited
 | |
|     for (const key of keys) {
 | |
|         const attributeMatch = key.match(/\[(\S+)\](.+)/); // [attribute]key
 | |
|         if (attributeMatch) { // attribute-tagged key
 | |
|             const localizedValue = localeData?.[attributeMatch[2]];
 | |
|             if (localizedValue || localizedValue === '') {
 | |
|                 element.setAttribute(attributeMatch[1], localizedValue);
 | |
|             }
 | |
|         } else { // No attribute tag, treat as 'text'
 | |
|             const localizedValue = localeData?.[key];
 | |
|             if (localizedValue || localizedValue === '') {
 | |
|                 element.textContent = localizedValue;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| async function getMissingTranslations() {
 | |
|     const missingData = [];
 | |
| 
 | |
|     // Determine locales to search for untranslated strings
 | |
|     const isNotSupported = !findLang(localeFile);
 | |
|     const langsToProcess = (isNotSupported || localeFile == 'en') ? langs : [findLang(localeFile)];
 | |
| 
 | |
|     for (const language of langsToProcess) {
 | |
|         const localeData = await getLocaleData(language.lang);
 | |
|         $(document).find('[data-i18n]').each(function () {
 | |
|             const keys = $(this).data('i18n').split(';'); // Multi-key entries are ; delimited
 | |
|             for (const key of keys) {
 | |
|                 const attributeMatch = key.match(/\[(\S+)\](.+)/); // [attribute]key
 | |
|                 if (attributeMatch) { // attribute-tagged key
 | |
|                     const localizedValue = localeData?.[attributeMatch[2]];
 | |
|                     if (!localizedValue) {
 | |
|                         missingData.push({ key, language: language.lang, value: $(this).attr(attributeMatch[1]) });
 | |
|                     }
 | |
|                 } else { // No attribute tag, treat as 'text'
 | |
|                     const localizedValue = localeData?.[key];
 | |
|                     if (!localizedValue) {
 | |
|                         missingData.push({ key, language: language.lang, value: $(this).text().trim() });
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     // Remove duplicates
 | |
|     const uniqueMissingData = [];
 | |
|     for (const { key, language, value } of missingData) {
 | |
|         if (!uniqueMissingData.some(x => x.key === key && x.language === language && x.value === value)) {
 | |
|             uniqueMissingData.push({ key, language, value });
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Sort by language, then key
 | |
|     uniqueMissingData.sort((a, b) => a.language.localeCompare(b.language) || a.key.localeCompare(b.key));
 | |
| 
 | |
|     // Map to { language: { key: value } }
 | |
|     let missingDataMap = {};
 | |
|     for (const { key, value } of uniqueMissingData) {
 | |
|         if (!missingDataMap) {
 | |
|             missingDataMap = {};
 | |
|         }
 | |
|         missingDataMap[key] = value;
 | |
|     }
 | |
| 
 | |
|     console.table(uniqueMissingData);
 | |
|     console.log(missingDataMap);
 | |
| 
 | |
|     toastr.success(`Found ${uniqueMissingData.length} missing translations. See browser console for details.`);
 | |
| }
 | |
| 
 | |
| export function applyLocale(root = document) {
 | |
|     if (!localeData || Object.keys(localeData).length === 0) {
 | |
|         return root;
 | |
|     }
 | |
| 
 | |
|     const $root = root instanceof Document ? $(root) : $(new DOMParser().parseFromString(root, 'text/html'));
 | |
| 
 | |
|     //find all the elements with `data-i18n` attribute
 | |
|     $root.find('[data-i18n]').each(function () {
 | |
|         translateElement(this);
 | |
|     });
 | |
| 
 | |
|     if (root !== document) {
 | |
|         return $root.get(0).body.innerHTML;
 | |
|     }
 | |
| }
 | |
| 
 | |
| function addLanguagesToDropdown() {
 | |
|     const uiLanguageSelects = $('#ui_language_select, #onboarding_ui_language_select');
 | |
|     for (const langObj of langs) { // Set the value to the language code
 | |
|         const option = document.createElement('option');
 | |
|         option.value = langObj['lang']; // Set the value to the language code
 | |
|         option.innerText = langObj['display']; // Set the display text to the language name
 | |
|         uiLanguageSelects.append(option);
 | |
|     }
 | |
| 
 | |
|     const selectedLanguage = localStorage.getItem(storageKey);
 | |
|     if (selectedLanguage) {
 | |
|         uiLanguageSelects.val(selectedLanguage);
 | |
|     }
 | |
| }
 | |
| 
 | |
| export async function initLocales() {
 | |
|     langs = await fetch('/locales/lang.json').then(response => response.json());
 | |
|     localeData = await getLocaleData(localeFile);
 | |
|     applyLocale();
 | |
|     addLanguagesToDropdown();
 | |
|     updateSecretDisplay();
 | |
| 
 | |
|     $('#ui_language_select, #onboarding_ui_language_select').on('change', async function () {
 | |
|         const language = String($(this).val());
 | |
| 
 | |
|         if (language) {
 | |
|             localStorage.setItem(storageKey, language);
 | |
|         } else {
 | |
|             localStorage.removeItem(storageKey);
 | |
|         }
 | |
| 
 | |
|         location.reload();
 | |
|     });
 | |
| 
 | |
|     observer.observe(document, {
 | |
|         childList: true,
 | |
|         subtree: true,
 | |
|         attributes: true,
 | |
|         attributeFilter: ['data-i18n'],
 | |
|     });
 | |
| 
 | |
|     registerDebugFunction('getMissingTranslations', 'Get missing translations', 'Detects missing localization data in the current locale and dumps the data into the browser console. If the current locale is English, searches all other locales.', getMissingTranslations);
 | |
|     registerDebugFunction('applyLocale', 'Apply locale', 'Reapplies the currently selected locale to the page.', applyLocale);
 | |
| }
 |