/* * Lightweight clientside i18n implementation. * Caution: Only available after DOM loaded because we need to detect the language * * For non-i18n stub implementation, see framework/javascript/src/i18nx.js * * Based on jQuery i18n plugin: 1.0.0 Feb-10-2008 * * Dual licensed under the MIT and GPL licenses: * http://www.opensource.org/licenses/mit-license.php * http://www.gnu.org/licenses/gpl.html * * Based on 'javascript i18n that almost doesn't suck' by markos * http://markos.gaivo.net/blog/?p=100 */ class i18n { constructor() { this.currentLocale = null; this.defaultLocale = 'en_US'; this.lang = {}; } /** * Set locale in long format, e.g. "de_AT" for Austrian German. * * @param string locale */ setLocale(locale) { this.currentLocale = locale; } /** * Get locale in long format. Falls back to i18n.defaut_locale. * * @return string */ getLocale() { return this.currentLocale !== null ? this.currentLocale : this.defaultLocale; } /** * The actual translation function. Looks the given string up in the * dictionary and returns the translation if one exists. If a translation * is not found, returns the original word. * * @param string entity - A "long" locale format, e.g. "de_DE" (Required) * @param string fallbackString - (Required) * @param int priority - (not used) * @param string context - Give translators context for the string * @return string : Translated word */ _t(entity, fallbackString, priority, context) { const langName = this.getLocale().replace(/_[\w]+/i, ''); const defaultlangName = this.defaultLocale.replace(/_[\w]+/i, ''); if (this.lang && this.lang[this.getLocale()] && this.lang[this.getLocale()][entity]) { return this.lang[this.getLocale()][entity]; } else if (this.lang && this.lang[langName] && this.lang[langName][entity]) { return this.lang[langName][entity]; } else if (this.lang && this.lang[this.defaultLocale] && this.lang[this.defaultLocale][entity]) { return this.lang[this.defaultLocale][entity]; } else if (this.lang && this.lang[defaultlangName] && this.lang[defaultlangName][entity]) { return this.lang[defaultlangName][entity]; } else if(fallbackString) { return fallbackString; } else { return ''; } } /** * Add entities to a dictionary. If a dictionary doesn't * exist for this locale, its automatically created. * Existing entities are overwritten. * * @param string locale * @param Object dict */ addDictionary(locale, dict) { if (typeof this.lang[locale] === 'undefined') { this.lang[locale] = {}; } for (let entity in dict) { this.lang[locale][entity] = dict[entity]; } } /** * Get dictionary for a specific locale. * * @param string locale */ getDictionary(locale) { return this.lang[locale]; } /** * @param string str - The string to strip. * @return string result - Stripped string. * */ stripStr(str) { return str.replace(/^\s*/, '').replace(/\s*$/, ''); } /** * @param string str - The multi-line string to strip. * @return string result - Stripped string. * */ stripStrML(str) { // Split because m flag doesn't exist before JS1.5 and we need to // strip newlines anyway var parts = str.split('\n'); for (let i = 0; i < parts.length; i += 1) { parts[i] = stripStr(parts[i]); } // Don't join with empty strings, because it "concats" words // And strip again return stripStr(parts.join(' ')); } /** * Substitutes %s with parameters * given in list. %%s is used to escape %s. * * @param string s - The string to perform the substitutions on. * @return string - The new string with substitutions made. */ sprintf(s, ...params) { if (params.length === 0) { return s; } const regx = new RegExp('(.?)(%s)', 'g'); let i = 0; return s.replace(regx, function (match, subMatch1, subMatch2, offset, string) { // skip %%s if (subMatch1 === '%') { return match; } return subMatch1 + params[i++]; }); } /** * Substitutes variables with a list of injections. * * @param string s - The string to perform the substitutions on. * @param object map - An object with the substitions map e.g. {var: value}. * @return string - The new string with substitutions made. */ inject(s, map) { const regx = new RegExp('\{([A-Za-z0-9_]*)\}', 'g'); return s.replace(regx, function (match, key, offset, string) { return (map[key]) ? map[key] : match; }); } /** * Detect document language settings by looking at tags. * If no match is found, returns this.defaultLocale. * * @todo get by - needs modification of SSViewer * * @return string - Locale in mixed lowercase/uppercase format suitable * for usage in i18n.lang arrays (e.g. 'en_US'). */ detectLocale() { var rawLocale; var detectedLocale; // Get by container tag rawLocale = jQuery('body').attr('lang'); // Get by meta if (!rawLocale) { var metas = document.getElementsByTagName('meta'); for (var i=0; i