12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147 |
- /* */
- /**
- * constants
- */
- const numberFormatKeys = [
- 'compactDisplay',
- 'currency',
- 'currencyDisplay',
- 'currencySign',
- 'localeMatcher',
- 'notation',
- 'numberingSystem',
- 'signDisplay',
- 'style',
- 'unit',
- 'unitDisplay',
- 'useGrouping',
- 'minimumIntegerDigits',
- 'minimumFractionDigits',
- 'maximumFractionDigits',
- 'minimumSignificantDigits',
- 'maximumSignificantDigits'
- ];
- /**
- * utilities
- */
- function warn (msg, err) {
- if (typeof console !== 'undefined') {
- console.warn('[vue-i18n] ' + msg);
- /* istanbul ignore if */
- if (err) {
- console.warn(err.stack);
- }
- }
- }
- function error (msg, err) {
- if (typeof console !== 'undefined') {
- console.error('[vue-i18n] ' + msg);
- /* istanbul ignore if */
- if (err) {
- console.error(err.stack);
- }
- }
- }
- const isArray = Array.isArray;
- function isObject (obj) {
- return obj !== null && typeof obj === 'object'
- }
- function isBoolean (val) {
- return typeof val === 'boolean'
- }
- function isString (val) {
- return typeof val === 'string'
- }
- const toString = Object.prototype.toString;
- const OBJECT_STRING = '[object Object]';
- function isPlainObject (obj) {
- return toString.call(obj) === OBJECT_STRING
- }
- function isNull (val) {
- return val === null || val === undefined
- }
- function isFunction (val) {
- return typeof val === 'function'
- }
- function parseArgs (...args) {
- let locale = null;
- let params = null;
- if (args.length === 1) {
- if (isObject(args[0]) || isArray(args[0])) {
- params = args[0];
- } else if (typeof args[0] === 'string') {
- locale = args[0];
- }
- } else if (args.length === 2) {
- if (typeof args[0] === 'string') {
- locale = args[0];
- }
- /* istanbul ignore if */
- if (isObject(args[1]) || isArray(args[1])) {
- params = args[1];
- }
- }
- return { locale, params }
- }
- function looseClone (obj) {
- return JSON.parse(JSON.stringify(obj))
- }
- function remove (arr, item) {
- if (arr.length) {
- const index = arr.indexOf(item);
- if (index > -1) {
- return arr.splice(index, 1)
- }
- }
- }
- function includes (arr, item) {
- return !!~arr.indexOf(item)
- }
- const hasOwnProperty = Object.prototype.hasOwnProperty;
- function hasOwn (obj, key) {
- return hasOwnProperty.call(obj, key)
- }
- function merge (target) {
- const output = Object(target);
- for (let i = 1; i < arguments.length; i++) {
- const source = arguments[i];
- if (source !== undefined && source !== null) {
- let key;
- for (key in source) {
- if (hasOwn(source, key)) {
- if (isObject(source[key])) {
- output[key] = merge(output[key], source[key]);
- } else {
- output[key] = source[key];
- }
- }
- }
- }
- }
- return output
- }
- function looseEqual (a, b) {
- if (a === b) { return true }
- const isObjectA = isObject(a);
- const isObjectB = isObject(b);
- if (isObjectA && isObjectB) {
- try {
- const isArrayA = isArray(a);
- const isArrayB = isArray(b);
- if (isArrayA && isArrayB) {
- return a.length === b.length && a.every((e, i) => {
- return looseEqual(e, b[i])
- })
- } else if (!isArrayA && !isArrayB) {
- const keysA = Object.keys(a);
- const keysB = Object.keys(b);
- return keysA.length === keysB.length && keysA.every((key) => {
- return looseEqual(a[key], b[key])
- })
- } else {
- /* istanbul ignore next */
- return false
- }
- } catch (e) {
- /* istanbul ignore next */
- return false
- }
- } else if (!isObjectA && !isObjectB) {
- return String(a) === String(b)
- } else {
- return false
- }
- }
- /**
- * Sanitizes html special characters from input strings. For mitigating risk of XSS attacks.
- * @param rawText The raw input from the user that should be escaped.
- */
- function escapeHtml(rawText) {
- return rawText
- .replace(/</g, '<')
- .replace(/>/g, '>')
- .replace(/"/g, '"')
- .replace(/'/g, ''')
- }
- /**
- * Escapes html tags and special symbols from all provided params which were returned from parseArgs().params.
- * This method performs an in-place operation on the params object.
- *
- * @param {any} params Parameters as provided from `parseArgs().params`.
- * May be either an array of strings or a string->any map.
- *
- * @returns The manipulated `params` object.
- */
- function escapeParams(params) {
- if(params != null) {
- Object.keys(params).forEach(key => {
- if(typeof(params[key]) == 'string') {
- params[key] = escapeHtml(params[key]);
- }
- });
- }
- return params
- }
- /* */
- function extend (Vue) {
- if (!Vue.prototype.hasOwnProperty('$i18n')) {
- // $FlowFixMe
- Object.defineProperty(Vue.prototype, '$i18n', {
- get () { return this._i18n }
- });
- }
- Vue.prototype.$t = function (key, ...values) {
- const i18n = this.$i18n;
- return i18n._t(key, i18n.locale, i18n._getMessages(), this, ...values)
- };
- Vue.prototype.$tc = function (key, choice, ...values) {
- const i18n = this.$i18n;
- return i18n._tc(key, i18n.locale, i18n._getMessages(), this, choice, ...values)
- };
- Vue.prototype.$te = function (key, locale) {
- const i18n = this.$i18n;
- return i18n._te(key, i18n.locale, i18n._getMessages(), locale)
- };
- Vue.prototype.$d = function (value, ...args) {
- return this.$i18n.d(value, ...args)
- };
- Vue.prototype.$n = function (value, ...args) {
- return this.$i18n.n(value, ...args)
- };
- }
- /* */
- var mixin = {
- beforeCreate () {
- const options = this.$options;
- options.i18n = options.i18n || (options.__i18n ? {} : null);
- if (options.i18n) {
- if (options.i18n instanceof VueI18n) {
- // init locale messages via custom blocks
- if (options.__i18n) {
- try {
- let localeMessages = options.i18n && options.i18n.messages ? options.i18n.messages : {};
- options.__i18n.forEach(resource => {
- localeMessages = merge(localeMessages, JSON.parse(resource));
- });
- Object.keys(localeMessages).forEach((locale) => {
- options.i18n.mergeLocaleMessage(locale, localeMessages[locale]);
- });
- } catch (e) {
- {
- error(`Cannot parse locale messages via custom blocks.`, e);
- }
- }
- }
- this._i18n = options.i18n;
- this._i18nWatcher = this._i18n.watchI18nData();
- } else if (isPlainObject(options.i18n)) {
- const rootI18n = this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n
- ? this.$root.$i18n
- : null;
- // component local i18n
- if (rootI18n) {
- options.i18n.root = this.$root;
- options.i18n.formatter = rootI18n.formatter;
- options.i18n.fallbackLocale = rootI18n.fallbackLocale;
- options.i18n.formatFallbackMessages = rootI18n.formatFallbackMessages;
- options.i18n.silentTranslationWarn = rootI18n.silentTranslationWarn;
- options.i18n.silentFallbackWarn = rootI18n.silentFallbackWarn;
- options.i18n.pluralizationRules = rootI18n.pluralizationRules;
- options.i18n.preserveDirectiveContent = rootI18n.preserveDirectiveContent;
- }
- // init locale messages via custom blocks
- if (options.__i18n) {
- try {
- let localeMessages = options.i18n && options.i18n.messages ? options.i18n.messages : {};
- options.__i18n.forEach(resource => {
- localeMessages = merge(localeMessages, JSON.parse(resource));
- });
- options.i18n.messages = localeMessages;
- } catch (e) {
- {
- warn(`Cannot parse locale messages via custom blocks.`, e);
- }
- }
- }
- const { sharedMessages } = options.i18n;
- if (sharedMessages && isPlainObject(sharedMessages)) {
- options.i18n.messages = merge(options.i18n.messages, sharedMessages);
- }
- this._i18n = new VueI18n(options.i18n);
- this._i18nWatcher = this._i18n.watchI18nData();
- if (options.i18n.sync === undefined || !!options.i18n.sync) {
- this._localeWatcher = this.$i18n.watchLocale();
- }
- if (rootI18n) {
- rootI18n.onComponentInstanceCreated(this._i18n);
- }
- } else {
- {
- warn(`Cannot be interpreted 'i18n' option.`);
- }
- }
- } else if (this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n) {
- // root i18n
- this._i18n = this.$root.$i18n;
- } else if (options.parent && options.parent.$i18n && options.parent.$i18n instanceof VueI18n) {
- // parent i18n
- this._i18n = options.parent.$i18n;
- }
- },
- beforeMount () {
- const options = this.$options;
- options.i18n = options.i18n || (options.__i18n ? {} : null);
- if (options.i18n) {
- if (options.i18n instanceof VueI18n) {
- // init locale messages via custom blocks
- this._i18n.subscribeDataChanging(this);
- this._subscribing = true;
- } else if (isPlainObject(options.i18n)) {
- this._i18n.subscribeDataChanging(this);
- this._subscribing = true;
- } else {
- {
- warn(`Cannot be interpreted 'i18n' option.`);
- }
- }
- } else if (this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n) {
- this._i18n.subscribeDataChanging(this);
- this._subscribing = true;
- } else if (options.parent && options.parent.$i18n && options.parent.$i18n instanceof VueI18n) {
- this._i18n.subscribeDataChanging(this);
- this._subscribing = true;
- }
- },
- beforeDestroy () {
- if (!this._i18n) { return }
- const self = this;
- this.$nextTick(() => {
- if (self._subscribing) {
- self._i18n.unsubscribeDataChanging(self);
- delete self._subscribing;
- }
- if (self._i18nWatcher) {
- self._i18nWatcher();
- self._i18n.destroyVM();
- delete self._i18nWatcher;
- }
- if (self._localeWatcher) {
- self._localeWatcher();
- delete self._localeWatcher;
- }
- });
- }
- };
- /* */
- var interpolationComponent = {
- name: 'i18n',
- functional: true,
- props: {
- tag: {
- type: [String, Boolean, Object],
- default: 'span'
- },
- path: {
- type: String,
- required: true
- },
- locale: {
- type: String
- },
- places: {
- type: [Array, Object]
- }
- },
- render (h, { data, parent, props, slots }) {
- const { $i18n } = parent;
- if (!$i18n) {
- {
- warn('Cannot find VueI18n instance!');
- }
- return
- }
- const { path, locale, places } = props;
- const params = slots();
- const children = $i18n.i(
- path,
- locale,
- onlyHasDefaultPlace(params) || places
- ? useLegacyPlaces(params.default, places)
- : params
- );
- const tag = (!!props.tag && props.tag !== true) || props.tag === false ? props.tag : 'span';
- return tag ? h(tag, data, children) : children
- }
- };
- function onlyHasDefaultPlace (params) {
- let prop;
- for (prop in params) {
- if (prop !== 'default') { return false }
- }
- return Boolean(prop)
- }
- function useLegacyPlaces (children, places) {
- const params = places ? createParamsFromPlaces(places) : {};
- if (!children) { return params }
- // Filter empty text nodes
- children = children.filter(child => {
- return child.tag || child.text.trim() !== ''
- });
- const everyPlace = children.every(vnodeHasPlaceAttribute);
- if (everyPlace) {
- warn('`place` attribute is deprecated in next major version. Please switch to Vue slots.');
- }
- return children.reduce(
- everyPlace ? assignChildPlace : assignChildIndex,
- params
- )
- }
- function createParamsFromPlaces (places) {
- {
- warn('`places` prop is deprecated in next major version. Please switch to Vue slots.');
- }
- return Array.isArray(places)
- ? places.reduce(assignChildIndex, {})
- : Object.assign({}, places)
- }
- function assignChildPlace (params, child) {
- if (child.data && child.data.attrs && child.data.attrs.place) {
- params[child.data.attrs.place] = child;
- }
- return params
- }
- function assignChildIndex (params, child, index) {
- params[index] = child;
- return params
- }
- function vnodeHasPlaceAttribute (vnode) {
- return Boolean(vnode.data && vnode.data.attrs && vnode.data.attrs.place)
- }
- /* */
- var numberComponent = {
- name: 'i18n-n',
- functional: true,
- props: {
- tag: {
- type: [String, Boolean, Object],
- default: 'span'
- },
- value: {
- type: Number,
- required: true
- },
- format: {
- type: [String, Object]
- },
- locale: {
- type: String
- }
- },
- render (h, { props, parent, data }) {
- const i18n = parent.$i18n;
- if (!i18n) {
- {
- warn('Cannot find VueI18n instance!');
- }
- return null
- }
- let key = null;
- let options = null;
- if (isString(props.format)) {
- key = props.format;
- } else if (isObject(props.format)) {
- if (props.format.key) {
- key = props.format.key;
- }
- // Filter out number format options only
- options = Object.keys(props.format).reduce((acc, prop) => {
- if (includes(numberFormatKeys, prop)) {
- return Object.assign({}, acc, { [prop]: props.format[prop] })
- }
- return acc
- }, null);
- }
- const locale = props.locale || i18n.locale;
- const parts = i18n._ntp(props.value, locale, key, options);
- const values = parts.map((part, index) => {
- const slot = data.scopedSlots && data.scopedSlots[part.type];
- return slot ? slot({ [part.type]: part.value, index, parts }) : part.value
- });
- const tag = (!!props.tag && props.tag !== true) || props.tag === false ? props.tag : 'span';
- return tag
- ? h(tag, {
- attrs: data.attrs,
- 'class': data['class'],
- staticClass: data.staticClass
- }, values)
- : values
- }
- };
- /* */
- function bind (el, binding, vnode) {
- if (!assert(el, vnode)) { return }
- t(el, binding, vnode);
- }
- function update (el, binding, vnode, oldVNode) {
- if (!assert(el, vnode)) { return }
- const i18n = vnode.context.$i18n;
- if (localeEqual(el, vnode) &&
- (looseEqual(binding.value, binding.oldValue) &&
- looseEqual(el._localeMessage, i18n.getLocaleMessage(i18n.locale)))) { return }
- t(el, binding, vnode);
- }
- function unbind (el, binding, vnode, oldVNode) {
- const vm = vnode.context;
- if (!vm) {
- warn('Vue instance does not exists in VNode context');
- return
- }
- const i18n = vnode.context.$i18n || {};
- if (!binding.modifiers.preserve && !i18n.preserveDirectiveContent) {
- el.textContent = '';
- }
- el._vt = undefined;
- delete el['_vt'];
- el._locale = undefined;
- delete el['_locale'];
- el._localeMessage = undefined;
- delete el['_localeMessage'];
- }
- function assert (el, vnode) {
- const vm = vnode.context;
- if (!vm) {
- warn('Vue instance does not exists in VNode context');
- return false
- }
- if (!vm.$i18n) {
- warn('VueI18n instance does not exists in Vue instance');
- return false
- }
- return true
- }
- function localeEqual (el, vnode) {
- const vm = vnode.context;
- return el._locale === vm.$i18n.locale
- }
- function t (el, binding, vnode) {
- const value = binding.value;
- const { path, locale, args, choice } = parseValue(value);
- if (!path && !locale && !args) {
- warn('value type not supported');
- return
- }
- if (!path) {
- warn('`path` is required in v-t directive');
- return
- }
- const vm = vnode.context;
- if (choice != null) {
- el._vt = el.textContent = vm.$i18n.tc(path, choice, ...makeParams(locale, args));
- } else {
- el._vt = el.textContent = vm.$i18n.t(path, ...makeParams(locale, args));
- }
- el._locale = vm.$i18n.locale;
- el._localeMessage = vm.$i18n.getLocaleMessage(vm.$i18n.locale);
- }
- function parseValue (value) {
- let path;
- let locale;
- let args;
- let choice;
- if (isString(value)) {
- path = value;
- } else if (isPlainObject(value)) {
- path = value.path;
- locale = value.locale;
- args = value.args;
- choice = value.choice;
- }
- return { path, locale, args, choice }
- }
- function makeParams (locale, args) {
- const params = [];
- locale && params.push(locale);
- if (args && (Array.isArray(args) || isPlainObject(args))) {
- params.push(args);
- }
- return params
- }
- let Vue;
- function install (_Vue) {
- /* istanbul ignore if */
- if (install.installed && _Vue === Vue) {
- warn('already installed.');
- return
- }
- install.installed = true;
- Vue = _Vue;
- const version = (Vue.version && Number(Vue.version.split('.')[0])) || -1;
- /* istanbul ignore if */
- if (version < 2) {
- warn(`vue-i18n (${install.version}) need to use Vue 2.0 or later (Vue: ${Vue.version}).`);
- return
- }
- extend(Vue);
- Vue.mixin(mixin);
- Vue.directive('t', { bind, update, unbind });
- Vue.component(interpolationComponent.name, interpolationComponent);
- Vue.component(numberComponent.name, numberComponent);
- // use simple mergeStrategies to prevent i18n instance lose '__proto__'
- const strats = Vue.config.optionMergeStrategies;
- strats.i18n = function (parentVal, childVal) {
- return childVal === undefined
- ? parentVal
- : childVal
- };
- }
- /* */
- class BaseFormatter {
-
- constructor () {
- this._caches = Object.create(null);
- }
- interpolate (message, values) {
- if (!values) {
- return [message]
- }
- let tokens = this._caches[message];
- if (!tokens) {
- tokens = parse(message);
- this._caches[message] = tokens;
- }
- return compile(tokens, values)
- }
- }
- const RE_TOKEN_LIST_VALUE = /^(?:\d)+/;
- const RE_TOKEN_NAMED_VALUE = /^(?:\w)+/;
- function parse (format) {
- const tokens = [];
- let position = 0;
- let text = '';
- while (position < format.length) {
- let char = format[position++];
- if (char === '{') {
- if (text) {
- tokens.push({ type: 'text', value: text });
- }
- text = '';
- let sub = '';
- char = format[position++];
- while (char !== undefined && char !== '}') {
- sub += char;
- char = format[position++];
- }
- const isClosed = char === '}';
- const type = RE_TOKEN_LIST_VALUE.test(sub)
- ? 'list'
- : isClosed && RE_TOKEN_NAMED_VALUE.test(sub)
- ? 'named'
- : 'unknown';
- tokens.push({ value: sub, type });
- } else if (char === '%') {
- // when found rails i18n syntax, skip text capture
- if (format[(position)] !== '{') {
- text += char;
- }
- } else {
- text += char;
- }
- }
- text && tokens.push({ type: 'text', value: text });
- return tokens
- }
- function compile (tokens, values) {
- const compiled = [];
- let index = 0;
- const mode = Array.isArray(values)
- ? 'list'
- : isObject(values)
- ? 'named'
- : 'unknown';
- if (mode === 'unknown') { return compiled }
- while (index < tokens.length) {
- const token = tokens[index];
- switch (token.type) {
- case 'text':
- compiled.push(token.value);
- break
- case 'list':
- compiled.push(values[parseInt(token.value, 10)]);
- break
- case 'named':
- if (mode === 'named') {
- compiled.push((values)[token.value]);
- } else {
- {
- warn(`Type of token '${token.type}' and format of value '${mode}' don't match!`);
- }
- }
- break
- case 'unknown':
- {
- warn(`Detect 'unknown' type of token!`);
- }
- break
- }
- index++;
- }
- return compiled
- }
- /* */
- /**
- * Path parser
- * - Inspired:
- * Vue.js Path parser
- */
- // actions
- const APPEND = 0;
- const PUSH = 1;
- const INC_SUB_PATH_DEPTH = 2;
- const PUSH_SUB_PATH = 3;
- // states
- const BEFORE_PATH = 0;
- const IN_PATH = 1;
- const BEFORE_IDENT = 2;
- const IN_IDENT = 3;
- const IN_SUB_PATH = 4;
- const IN_SINGLE_QUOTE = 5;
- const IN_DOUBLE_QUOTE = 6;
- const AFTER_PATH = 7;
- const ERROR = 8;
- const pathStateMachine = [];
- pathStateMachine[BEFORE_PATH] = {
- 'ws': [BEFORE_PATH],
- 'ident': [IN_IDENT, APPEND],
- '[': [IN_SUB_PATH],
- 'eof': [AFTER_PATH]
- };
- pathStateMachine[IN_PATH] = {
- 'ws': [IN_PATH],
- '.': [BEFORE_IDENT],
- '[': [IN_SUB_PATH],
- 'eof': [AFTER_PATH]
- };
- pathStateMachine[BEFORE_IDENT] = {
- 'ws': [BEFORE_IDENT],
- 'ident': [IN_IDENT, APPEND],
- '0': [IN_IDENT, APPEND],
- 'number': [IN_IDENT, APPEND]
- };
- pathStateMachine[IN_IDENT] = {
- 'ident': [IN_IDENT, APPEND],
- '0': [IN_IDENT, APPEND],
- 'number': [IN_IDENT, APPEND],
- 'ws': [IN_PATH, PUSH],
- '.': [BEFORE_IDENT, PUSH],
- '[': [IN_SUB_PATH, PUSH],
- 'eof': [AFTER_PATH, PUSH]
- };
- pathStateMachine[IN_SUB_PATH] = {
- "'": [IN_SINGLE_QUOTE, APPEND],
- '"': [IN_DOUBLE_QUOTE, APPEND],
- '[': [IN_SUB_PATH, INC_SUB_PATH_DEPTH],
- ']': [IN_PATH, PUSH_SUB_PATH],
- 'eof': ERROR,
- 'else': [IN_SUB_PATH, APPEND]
- };
- pathStateMachine[IN_SINGLE_QUOTE] = {
- "'": [IN_SUB_PATH, APPEND],
- 'eof': ERROR,
- 'else': [IN_SINGLE_QUOTE, APPEND]
- };
- pathStateMachine[IN_DOUBLE_QUOTE] = {
- '"': [IN_SUB_PATH, APPEND],
- 'eof': ERROR,
- 'else': [IN_DOUBLE_QUOTE, APPEND]
- };
- /**
- * Check if an expression is a literal value.
- */
- const literalValueRE = /^\s?(?:true|false|-?[\d.]+|'[^']*'|"[^"]*")\s?$/;
- function isLiteral (exp) {
- return literalValueRE.test(exp)
- }
- /**
- * Strip quotes from a string
- */
- function stripQuotes (str) {
- const a = str.charCodeAt(0);
- const b = str.charCodeAt(str.length - 1);
- return a === b && (a === 0x22 || a === 0x27)
- ? str.slice(1, -1)
- : str
- }
- /**
- * Determine the type of a character in a keypath.
- */
- function getPathCharType (ch) {
- if (ch === undefined || ch === null) { return 'eof' }
- const code = ch.charCodeAt(0);
- switch (code) {
- case 0x5B: // [
- case 0x5D: // ]
- case 0x2E: // .
- case 0x22: // "
- case 0x27: // '
- return ch
- case 0x5F: // _
- case 0x24: // $
- case 0x2D: // -
- return 'ident'
- case 0x09: // Tab
- case 0x0A: // Newline
- case 0x0D: // Return
- case 0xA0: // No-break space
- case 0xFEFF: // Byte Order Mark
- case 0x2028: // Line Separator
- case 0x2029: // Paragraph Separator
- return 'ws'
- }
- return 'ident'
- }
- /**
- * Format a subPath, return its plain form if it is
- * a literal string or number. Otherwise prepend the
- * dynamic indicator (*).
- */
- function formatSubPath (path) {
- const trimmed = path.trim();
- // invalid leading 0
- if (path.charAt(0) === '0' && isNaN(path)) { return false }
- return isLiteral(trimmed) ? stripQuotes(trimmed) : '*' + trimmed
- }
- /**
- * Parse a string path into an array of segments
- */
- function parse$1 (path) {
- const keys = [];
- let index = -1;
- let mode = BEFORE_PATH;
- let subPathDepth = 0;
- let c;
- let key;
- let newChar;
- let type;
- let transition;
- let action;
- let typeMap;
- const actions = [];
- actions[PUSH] = function () {
- if (key !== undefined) {
- keys.push(key);
- key = undefined;
- }
- };
- actions[APPEND] = function () {
- if (key === undefined) {
- key = newChar;
- } else {
- key += newChar;
- }
- };
- actions[INC_SUB_PATH_DEPTH] = function () {
- actions[APPEND]();
- subPathDepth++;
- };
- actions[PUSH_SUB_PATH] = function () {
- if (subPathDepth > 0) {
- subPathDepth--;
- mode = IN_SUB_PATH;
- actions[APPEND]();
- } else {
- subPathDepth = 0;
- if (key === undefined) { return false }
- key = formatSubPath(key);
- if (key === false) {
- return false
- } else {
- actions[PUSH]();
- }
- }
- };
- function maybeUnescapeQuote () {
- const nextChar = path[index + 1];
- if ((mode === IN_SINGLE_QUOTE && nextChar === "'") ||
- (mode === IN_DOUBLE_QUOTE && nextChar === '"')) {
- index++;
- newChar = '\\' + nextChar;
- actions[APPEND]();
- return true
- }
- }
- while (mode !== null) {
- index++;
- c = path[index];
- if (c === '\\' && maybeUnescapeQuote()) {
- continue
- }
- type = getPathCharType(c);
- typeMap = pathStateMachine[mode];
- transition = typeMap[type] || typeMap['else'] || ERROR;
- if (transition === ERROR) {
- return // parse error
- }
- mode = transition[0];
- action = actions[transition[1]];
- if (action) {
- newChar = transition[2];
- newChar = newChar === undefined
- ? c
- : newChar;
- if (action() === false) {
- return
- }
- }
- if (mode === AFTER_PATH) {
- return keys
- }
- }
- }
- class I18nPath {
-
- constructor () {
- this._cache = Object.create(null);
- }
- /**
- * External parse that check for a cache hit first
- */
- parsePath (path) {
- let hit = this._cache[path];
- if (!hit) {
- hit = parse$1(path);
- if (hit) {
- this._cache[path] = hit;
- }
- }
- return hit || []
- }
- /**
- * Get path value from path string
- */
- getPathValue (obj, path) {
- if (!isObject(obj)) { return null }
- const paths = this.parsePath(path);
- if (paths.length === 0) {
- return null
- } else {
- const length = paths.length;
- let last = obj;
- let i = 0;
- while (i < length) {
- const value = last[paths[i]];
- if (value === undefined) {
- return null
- }
- last = value;
- i++;
- }
- return last
- }
- }
- }
- /* */
- const htmlTagMatcher = /<\/?[\w\s="/.':;#-\/]+>/;
- const linkKeyMatcher = /(?:@(?:\.[a-z]+)?:(?:[\w\-_|.]+|\([\w\-_|.]+\)))/g;
- const linkKeyPrefixMatcher = /^@(?:\.([a-z]+))?:/;
- const bracketsMatcher = /[()]/g;
- const defaultModifiers = {
- 'upper': str => str.toLocaleUpperCase(),
- 'lower': str => str.toLocaleLowerCase(),
- 'capitalize': str => `${str.charAt(0).toLocaleUpperCase()}${str.substr(1)}`
- };
- const defaultFormatter = new BaseFormatter();
- class VueI18n {
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- constructor (options = {}) {
- // Auto install if it is not done yet and `window` has `Vue`.
- // To allow users to avoid auto-installation in some cases,
- // this code should be placed here. See #290
- /* istanbul ignore if */
- if (!Vue && typeof window !== 'undefined' && window.Vue) {
- install(window.Vue);
- }
- const locale = options.locale || 'en-US';
- const fallbackLocale = options.fallbackLocale === false
- ? false
- : options.fallbackLocale || 'en-US';
- const messages = options.messages || {};
- const dateTimeFormats = options.dateTimeFormats || {};
- const numberFormats = options.numberFormats || {};
- this._vm = null;
- this._formatter = options.formatter || defaultFormatter;
- this._modifiers = options.modifiers || {};
- this._missing = options.missing || null;
- this._root = options.root || null;
- this._sync = options.sync === undefined ? true : !!options.sync;
- this._fallbackRoot = options.fallbackRoot === undefined
- ? true
- : !!options.fallbackRoot;
- this._formatFallbackMessages = options.formatFallbackMessages === undefined
- ? false
- : !!options.formatFallbackMessages;
- this._silentTranslationWarn = options.silentTranslationWarn === undefined
- ? false
- : options.silentTranslationWarn;
- this._silentFallbackWarn = options.silentFallbackWarn === undefined
- ? false
- : !!options.silentFallbackWarn;
- this._dateTimeFormatters = {};
- this._numberFormatters = {};
- this._path = new I18nPath();
- this._dataListeners = [];
- this._componentInstanceCreatedListener = options.componentInstanceCreatedListener || null;
- this._preserveDirectiveContent = options.preserveDirectiveContent === undefined
- ? false
- : !!options.preserveDirectiveContent;
- this.pluralizationRules = options.pluralizationRules || {};
- this._warnHtmlInMessage = options.warnHtmlInMessage || 'off';
- this._postTranslation = options.postTranslation || null;
- this._escapeParameterHtml = options.escapeParameterHtml || false;
- /**
- * @param choice {number} a choice index given by the input to $tc: `$tc('path.to.rule', choiceIndex)`
- * @param choicesLength {number} an overall amount of available choices
- * @returns a final choice index
- */
- this.getChoiceIndex = (choice, choicesLength) => {
- const thisPrototype = Object.getPrototypeOf(this);
- if (thisPrototype && thisPrototype.getChoiceIndex) {
- const prototypeGetChoiceIndex = (thisPrototype.getChoiceIndex);
- return (prototypeGetChoiceIndex).call(this, choice, choicesLength)
- }
- // Default (old) getChoiceIndex implementation - english-compatible
- const defaultImpl = (_choice, _choicesLength) => {
- _choice = Math.abs(_choice);
- if (_choicesLength === 2) {
- return _choice
- ? _choice > 1
- ? 1
- : 0
- : 1
- }
- return _choice ? Math.min(_choice, 2) : 0
- };
- if (this.locale in this.pluralizationRules) {
- return this.pluralizationRules[this.locale].apply(this, [choice, choicesLength])
- } else {
- return defaultImpl(choice, choicesLength)
- }
- };
- this._exist = (message, key) => {
- if (!message || !key) { return false }
- if (!isNull(this._path.getPathValue(message, key))) { return true }
- // fallback for flat key
- if (message[key]) { return true }
- return false
- };
- if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {
- Object.keys(messages).forEach(locale => {
- this._checkLocaleMessage(locale, this._warnHtmlInMessage, messages[locale]);
- });
- }
- this._initVM({
- locale,
- fallbackLocale,
- messages,
- dateTimeFormats,
- numberFormats
- });
- }
- _checkLocaleMessage (locale, level, message) {
- const paths = [];
- const fn = (level, locale, message, paths) => {
- if (isPlainObject(message)) {
- Object.keys(message).forEach(key => {
- const val = message[key];
- if (isPlainObject(val)) {
- paths.push(key);
- paths.push('.');
- fn(level, locale, val, paths);
- paths.pop();
- paths.pop();
- } else {
- paths.push(key);
- fn(level, locale, val, paths);
- paths.pop();
- }
- });
- } else if (isArray(message)) {
- message.forEach((item, index) => {
- if (isPlainObject(item)) {
- paths.push(`[${index}]`);
- paths.push('.');
- fn(level, locale, item, paths);
- paths.pop();
- paths.pop();
- } else {
- paths.push(`[${index}]`);
- fn(level, locale, item, paths);
- paths.pop();
- }
- });
- } else if (isString(message)) {
- const ret = htmlTagMatcher.test(message);
- if (ret) {
- const msg = `Detected HTML in message '${message}' of keypath '${paths.join('')}' at '${locale}'. Consider component interpolation with '<i18n>' to avoid XSS. See https://bit.ly/2ZqJzkp`;
- if (level === 'warn') {
- warn(msg);
- } else if (level === 'error') {
- error(msg);
- }
- }
- }
- };
- fn(level, locale, message, paths);
- }
- _initVM (data) {
- const silent = Vue.config.silent;
- Vue.config.silent = true;
- this._vm = new Vue({ data });
- Vue.config.silent = silent;
- }
- destroyVM () {
- this._vm.$destroy();
- }
- subscribeDataChanging (vm) {
- this._dataListeners.push(vm);
- }
- unsubscribeDataChanging (vm) {
- remove(this._dataListeners, vm);
- }
- watchI18nData () {
- const self = this;
- return this._vm.$watch('$data', () => {
- let i = self._dataListeners.length;
- while (i--) {
- Vue.nextTick(() => {
- self._dataListeners[i] && self._dataListeners[i].$forceUpdate();
- });
- }
- }, { deep: true })
- }
- watchLocale () {
- /* istanbul ignore if */
- if (!this._sync || !this._root) { return null }
- const target = this._vm;
- return this._root.$i18n.vm.$watch('locale', (val) => {
- target.$set(target, 'locale', val);
- target.$forceUpdate();
- }, { immediate: true })
- }
- onComponentInstanceCreated (newI18n) {
- if (this._componentInstanceCreatedListener) {
- this._componentInstanceCreatedListener(newI18n, this);
- }
- }
- get vm () { return this._vm }
- get messages () { return looseClone(this._getMessages()) }
- get dateTimeFormats () { return looseClone(this._getDateTimeFormats()) }
- get numberFormats () { return looseClone(this._getNumberFormats()) }
- get availableLocales () { return Object.keys(this.messages).sort() }
- get locale () { return this._vm.locale }
- set locale (locale) {
- this._vm.$set(this._vm, 'locale', locale);
- }
- get fallbackLocale () { return this._vm.fallbackLocale }
- set fallbackLocale (locale) {
- this._localeChainCache = {};
- this._vm.$set(this._vm, 'fallbackLocale', locale);
- }
- get formatFallbackMessages () { return this._formatFallbackMessages }
- set formatFallbackMessages (fallback) { this._formatFallbackMessages = fallback; }
- get missing () { return this._missing }
- set missing (handler) { this._missing = handler; }
- get formatter () { return this._formatter }
- set formatter (formatter) { this._formatter = formatter; }
- get silentTranslationWarn () { return this._silentTranslationWarn }
- set silentTranslationWarn (silent) { this._silentTranslationWarn = silent; }
- get silentFallbackWarn () { return this._silentFallbackWarn }
- set silentFallbackWarn (silent) { this._silentFallbackWarn = silent; }
- get preserveDirectiveContent () { return this._preserveDirectiveContent }
- set preserveDirectiveContent (preserve) { this._preserveDirectiveContent = preserve; }
- get warnHtmlInMessage () { return this._warnHtmlInMessage }
- set warnHtmlInMessage (level) {
- const orgLevel = this._warnHtmlInMessage;
- this._warnHtmlInMessage = level;
- if (orgLevel !== level && (level === 'warn' || level === 'error')) {
- const messages = this._getMessages();
- Object.keys(messages).forEach(locale => {
- this._checkLocaleMessage(locale, this._warnHtmlInMessage, messages[locale]);
- });
- }
- }
- get postTranslation () { return this._postTranslation }
- set postTranslation (handler) { this._postTranslation = handler; }
- _getMessages () { return this._vm.messages }
- _getDateTimeFormats () { return this._vm.dateTimeFormats }
- _getNumberFormats () { return this._vm.numberFormats }
- _warnDefault (locale, key, result, vm, values, interpolateMode) {
- if (!isNull(result)) { return result }
- if (this._missing) {
- const missingRet = this._missing.apply(null, [locale, key, vm, values]);
- if (isString(missingRet)) {
- return missingRet
- }
- } else {
- if (!this._isSilentTranslationWarn(key)) {
- warn(
- `Cannot translate the value of keypath '${key}'. ` +
- 'Use the value of keypath as default.'
- );
- }
- }
- if (this._formatFallbackMessages) {
- const parsedArgs = parseArgs(...values);
- return this._render(key, interpolateMode, parsedArgs.params, key)
- } else {
- return key
- }
- }
- _isFallbackRoot (val) {
- return !val && !isNull(this._root) && this._fallbackRoot
- }
- _isSilentFallbackWarn (key) {
- return this._silentFallbackWarn instanceof RegExp
- ? this._silentFallbackWarn.test(key)
- : this._silentFallbackWarn
- }
- _isSilentFallback (locale, key) {
- return this._isSilentFallbackWarn(key) && (this._isFallbackRoot() || locale !== this.fallbackLocale)
- }
- _isSilentTranslationWarn (key) {
- return this._silentTranslationWarn instanceof RegExp
- ? this._silentTranslationWarn.test(key)
- : this._silentTranslationWarn
- }
- _interpolate (
- locale,
- message,
- key,
- host,
- interpolateMode,
- values,
- visitedLinkStack
- ) {
- if (!message) { return null }
- const pathRet = this._path.getPathValue(message, key);
- if (isArray(pathRet) || isPlainObject(pathRet)) { return pathRet }
- let ret;
- if (isNull(pathRet)) {
- /* istanbul ignore else */
- if (isPlainObject(message)) {
- ret = message[key];
- if (!(isString(ret) || isFunction(ret))) {
- if (!this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {
- warn(`Value of key '${key}' is not a string or function !`);
- }
- return null
- }
- } else {
- return null
- }
- } else {
- /* istanbul ignore else */
- if (isString(pathRet) || isFunction(pathRet)) {
- ret = pathRet;
- } else {
- if (!this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {
- warn(`Value of key '${key}' is not a string or function!`);
- }
- return null
- }
- }
- // Check for the existence of links within the translated string
- if (isString(ret) && (ret.indexOf('@:') >= 0 || ret.indexOf('@.') >= 0)) {
- ret = this._link(locale, message, ret, host, 'raw', values, visitedLinkStack);
- }
- return this._render(ret, interpolateMode, values, key)
- }
- _link (
- locale,
- message,
- str,
- host,
- interpolateMode,
- values,
- visitedLinkStack
- ) {
- let ret = str;
- // Match all the links within the local
- // We are going to replace each of
- // them with its translation
- const matches = ret.match(linkKeyMatcher);
- for (let idx in matches) {
- // ie compatible: filter custom array
- // prototype method
- if (!matches.hasOwnProperty(idx)) {
- continue
- }
- const link = matches[idx];
- const linkKeyPrefixMatches = link.match(linkKeyPrefixMatcher);
- const [linkPrefix, formatterName] = linkKeyPrefixMatches;
- // Remove the leading @:, @.case: and the brackets
- const linkPlaceholder = link.replace(linkPrefix, '').replace(bracketsMatcher, '');
- if (includes(visitedLinkStack, linkPlaceholder)) {
- {
- warn(`Circular reference found. "${link}" is already visited in the chain of ${visitedLinkStack.reverse().join(' <- ')}`);
- }
- return ret
- }
- visitedLinkStack.push(linkPlaceholder);
- // Translate the link
- let translated = this._interpolate(
- locale, message, linkPlaceholder, host,
- interpolateMode === 'raw' ? 'string' : interpolateMode,
- interpolateMode === 'raw' ? undefined : values,
- visitedLinkStack
- );
- if (this._isFallbackRoot(translated)) {
- if (!this._isSilentTranslationWarn(linkPlaceholder)) {
- warn(`Fall back to translate the link placeholder '${linkPlaceholder}' with root locale.`);
- }
- /* istanbul ignore if */
- if (!this._root) { throw Error('unexpected error') }
- const root = this._root.$i18n;
- translated = root._translate(
- root._getMessages(), root.locale, root.fallbackLocale,
- linkPlaceholder, host, interpolateMode, values
- );
- }
- translated = this._warnDefault(
- locale, linkPlaceholder, translated, host,
- isArray(values) ? values : [values],
- interpolateMode
- );
- if (this._modifiers.hasOwnProperty(formatterName)) {
- translated = this._modifiers[formatterName](translated);
- } else if (defaultModifiers.hasOwnProperty(formatterName)) {
- translated = defaultModifiers[formatterName](translated);
- }
- visitedLinkStack.pop();
- // Replace the link with the translated
- ret = !translated ? ret : ret.replace(link, translated);
- }
- return ret
- }
- _createMessageContext (values) {
- const _list = isArray(values) ? values : [];
- const _named = isObject(values) ? values : {};
- const list = (index) => _list[index];
- const named = (key) => _named[key];
- return {
- list,
- named
- }
- }
- _render (message, interpolateMode, values, path) {
- if (isFunction(message)) {
- return message(this._createMessageContext(values))
- }
- let ret = this._formatter.interpolate(message, values, path);
- // If the custom formatter refuses to work - apply the default one
- if (!ret) {
- ret = defaultFormatter.interpolate(message, values, path);
- }
- // if interpolateMode is **not** 'string' ('row'),
- // return the compiled data (e.g. ['foo', VNode, 'bar']) with formatter
- return interpolateMode === 'string' && !isString(ret) ? ret.join('') : ret
- }
- _appendItemToChain (chain, item, blocks) {
- let follow = false;
- if (!includes(chain, item)) {
- follow = true;
- if (item) {
- follow = item[item.length - 1] !== '!';
- item = item.replace(/!/g, '');
- chain.push(item);
- if (blocks && blocks[item]) {
- follow = blocks[item];
- }
- }
- }
- return follow
- }
- _appendLocaleToChain (chain, locale, blocks) {
- let follow;
- const tokens = locale.split('-');
- do {
- const item = tokens.join('-');
- follow = this._appendItemToChain(chain, item, blocks);
- tokens.splice(-1, 1);
- } while (tokens.length && (follow === true))
- return follow
- }
- _appendBlockToChain (chain, block, blocks) {
- let follow = true;
- for (let i = 0; (i < block.length) && (isBoolean(follow)); i++) {
- const locale = block[i];
- if (isString(locale)) {
- follow = this._appendLocaleToChain(chain, locale, blocks);
- }
- }
- return follow
- }
- _getLocaleChain (start, fallbackLocale) {
- if (start === '') { return [] }
- if (!this._localeChainCache) {
- this._localeChainCache = {};
- }
- let chain = this._localeChainCache[start];
- if (!chain) {
- if (!fallbackLocale) {
- fallbackLocale = this.fallbackLocale;
- }
- chain = [];
- // first block defined by start
- let block = [start];
- // while any intervening block found
- while (isArray(block)) {
- block = this._appendBlockToChain(
- chain,
- block,
- fallbackLocale
- );
- }
- // last block defined by default
- let defaults;
- if (isArray(fallbackLocale)) {
- defaults = fallbackLocale;
- } else if (isObject(fallbackLocale)) {
- /* $FlowFixMe */
- if (fallbackLocale['default']) {
- defaults = fallbackLocale['default'];
- } else {
- defaults = null;
- }
- } else {
- defaults = fallbackLocale;
- }
- // convert defaults to array
- if (isString(defaults)) {
- block = [defaults];
- } else {
- block = defaults;
- }
- if (block) {
- this._appendBlockToChain(
- chain,
- block,
- null
- );
- }
- this._localeChainCache[start] = chain;
- }
- return chain
- }
- _translate (
- messages,
- locale,
- fallback,
- key,
- host,
- interpolateMode,
- args
- ) {
- const chain = this._getLocaleChain(locale, fallback);
- let res;
- for (let i = 0; i < chain.length; i++) {
- const step = chain[i];
- res =
- this._interpolate(step, messages[step], key, host, interpolateMode, args, [key]);
- if (!isNull(res)) {
- if (step !== locale && "development" !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(("Fall back to translate the keypath '" + key + "' with '" + step + "' locale."));
- }
- return res
- }
- }
- return null
- }
- _t (key, _locale, messages, host, ...values) {
- if (!key) { return '' }
- const parsedArgs = parseArgs(...values);
- if(this._escapeParameterHtml) {
- parsedArgs.params = escapeParams(parsedArgs.params);
- }
- const locale = parsedArgs.locale || _locale;
- let ret = this._translate(
- messages, locale, this.fallbackLocale, key,
- host, 'string', parsedArgs.params
- );
- if (this._isFallbackRoot(ret)) {
- if (!this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(`Fall back to translate the keypath '${key}' with root locale.`);
- }
- /* istanbul ignore if */
- if (!this._root) { throw Error('unexpected error') }
- return this._root.$t(key, ...values)
- } else {
- ret = this._warnDefault(locale, key, ret, host, values, 'string');
- if (this._postTranslation && ret !== null && ret !== undefined) {
- ret = this._postTranslation(ret, key);
- }
- return ret
- }
- }
- t (key, ...values) {
- return this._t(key, this.locale, this._getMessages(), null, ...values)
- }
- _i (key, locale, messages, host, values) {
- const ret =
- this._translate(messages, locale, this.fallbackLocale, key, host, 'raw', values);
- if (this._isFallbackRoot(ret)) {
- if (!this._isSilentTranslationWarn(key)) {
- warn(`Fall back to interpolate the keypath '${key}' with root locale.`);
- }
- if (!this._root) { throw Error('unexpected error') }
- return this._root.$i18n.i(key, locale, values)
- } else {
- return this._warnDefault(locale, key, ret, host, [values], 'raw')
- }
- }
- i (key, locale, values) {
- /* istanbul ignore if */
- if (!key) { return '' }
- if (!isString(locale)) {
- locale = this.locale;
- }
- return this._i(key, locale, this._getMessages(), null, values)
- }
- _tc (
- key,
- _locale,
- messages,
- host,
- choice,
- ...values
- ) {
- if (!key) { return '' }
- if (choice === undefined) {
- choice = 1;
- }
- const predefined = { 'count': choice, 'n': choice };
- const parsedArgs = parseArgs(...values);
- parsedArgs.params = Object.assign(predefined, parsedArgs.params);
- values = parsedArgs.locale === null ? [parsedArgs.params] : [parsedArgs.locale, parsedArgs.params];
- return this.fetchChoice(this._t(key, _locale, messages, host, ...values), choice)
- }
- fetchChoice (message, choice) {
- /* istanbul ignore if */
- if (!message || !isString(message)) { return null }
- const choices = message.split('|');
- choice = this.getChoiceIndex(choice, choices.length);
- if (!choices[choice]) { return message }
- return choices[choice].trim()
- }
- tc (key, choice, ...values) {
- return this._tc(key, this.locale, this._getMessages(), null, choice, ...values)
- }
- _te (key, locale, messages, ...args) {
- const _locale = parseArgs(...args).locale || locale;
- return this._exist(messages[_locale], key)
- }
- te (key, locale) {
- return this._te(key, this.locale, this._getMessages(), locale)
- }
- getLocaleMessage (locale) {
- return looseClone(this._vm.messages[locale] || {})
- }
- setLocaleMessage (locale, message) {
- if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {
- this._checkLocaleMessage(locale, this._warnHtmlInMessage, message);
- }
- this._vm.$set(this._vm.messages, locale, message);
- }
- mergeLocaleMessage (locale, message) {
- if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {
- this._checkLocaleMessage(locale, this._warnHtmlInMessage, message);
- }
- this._vm.$set(this._vm.messages, locale, merge({}, this._vm.messages[locale] || {}, message));
- }
- getDateTimeFormat (locale) {
- return looseClone(this._vm.dateTimeFormats[locale] || {})
- }
- setDateTimeFormat (locale, format) {
- this._vm.$set(this._vm.dateTimeFormats, locale, format);
- this._clearDateTimeFormat(locale, format);
- }
- mergeDateTimeFormat (locale, format) {
- this._vm.$set(this._vm.dateTimeFormats, locale, merge(this._vm.dateTimeFormats[locale] || {}, format));
- this._clearDateTimeFormat(locale, format);
- }
- _clearDateTimeFormat (locale, format) {
- for (let key in format) {
- const id = `${locale}__${key}`;
- if (!this._dateTimeFormatters.hasOwnProperty(id)) {
- continue
- }
- delete this._dateTimeFormatters[id];
- }
- }
- _localizeDateTime (
- value,
- locale,
- fallback,
- dateTimeFormats,
- key
- ) {
- let _locale = locale;
- let formats = dateTimeFormats[_locale];
- const chain = this._getLocaleChain(locale, fallback);
- for (let i = 0; i < chain.length; i++) {
- const current = _locale;
- const step = chain[i];
- formats = dateTimeFormats[step];
- _locale = step;
- // fallback locale
- if (isNull(formats) || isNull(formats[key])) {
- if (step !== locale && "development" !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(`Fall back to '${step}' datetime formats from '${current}' datetime formats.`);
- }
- } else {
- break
- }
- }
- if (isNull(formats) || isNull(formats[key])) {
- return null
- } else {
- const format = formats[key];
- const id = `${_locale}__${key}`;
- let formatter = this._dateTimeFormatters[id];
- if (!formatter) {
- formatter = this._dateTimeFormatters[id] = new Intl.DateTimeFormat(_locale, format);
- }
- return formatter.format(value)
- }
- }
- _d (value, locale, key) {
- /* istanbul ignore if */
- if (!VueI18n.availabilities.dateTimeFormat) {
- warn('Cannot format a Date value due to not supported Intl.DateTimeFormat.');
- return ''
- }
- if (!key) {
- return new Intl.DateTimeFormat(locale).format(value)
- }
- const ret =
- this._localizeDateTime(value, locale, this.fallbackLocale, this._getDateTimeFormats(), key);
- if (this._isFallbackRoot(ret)) {
- if (!this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(`Fall back to datetime localization of root: key '${key}'.`);
- }
- /* istanbul ignore if */
- if (!this._root) { throw Error('unexpected error') }
- return this._root.$i18n.d(value, key, locale)
- } else {
- return ret || ''
- }
- }
- d (value, ...args) {
- let locale = this.locale;
- let key = null;
- if (args.length === 1) {
- if (isString(args[0])) {
- key = args[0];
- } else if (isObject(args[0])) {
- if (args[0].locale) {
- locale = args[0].locale;
- }
- if (args[0].key) {
- key = args[0].key;
- }
- }
- } else if (args.length === 2) {
- if (isString(args[0])) {
- key = args[0];
- }
- if (isString(args[1])) {
- locale = args[1];
- }
- }
- return this._d(value, locale, key)
- }
- getNumberFormat (locale) {
- return looseClone(this._vm.numberFormats[locale] || {})
- }
- setNumberFormat (locale, format) {
- this._vm.$set(this._vm.numberFormats, locale, format);
- this._clearNumberFormat(locale, format);
- }
- mergeNumberFormat (locale, format) {
- this._vm.$set(this._vm.numberFormats, locale, merge(this._vm.numberFormats[locale] || {}, format));
- this._clearNumberFormat(locale, format);
- }
- _clearNumberFormat (locale, format) {
- for (let key in format) {
- const id = `${locale}__${key}`;
- if (!this._numberFormatters.hasOwnProperty(id)) {
- continue
- }
- delete this._numberFormatters[id];
- }
- }
- _getNumberFormatter (
- value,
- locale,
- fallback,
- numberFormats,
- key,
- options
- ) {
- let _locale = locale;
- let formats = numberFormats[_locale];
- const chain = this._getLocaleChain(locale, fallback);
- for (let i = 0; i < chain.length; i++) {
- const current = _locale;
- const step = chain[i];
- formats = numberFormats[step];
- _locale = step;
- // fallback locale
- if (isNull(formats) || isNull(formats[key])) {
- if (step !== locale && "development" !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(`Fall back to '${step}' number formats from '${current}' number formats.`);
- }
- } else {
- break
- }
- }
- if (isNull(formats) || isNull(formats[key])) {
- return null
- } else {
- const format = formats[key];
- let formatter;
- if (options) {
- // If options specified - create one time number formatter
- formatter = new Intl.NumberFormat(_locale, Object.assign({}, format, options));
- } else {
- const id = `${_locale}__${key}`;
- formatter = this._numberFormatters[id];
- if (!formatter) {
- formatter = this._numberFormatters[id] = new Intl.NumberFormat(_locale, format);
- }
- }
- return formatter
- }
- }
- _n (value, locale, key, options) {
- /* istanbul ignore if */
- if (!VueI18n.availabilities.numberFormat) {
- {
- warn('Cannot format a Number value due to not supported Intl.NumberFormat.');
- }
- return ''
- }
- if (!key) {
- const nf = !options ? new Intl.NumberFormat(locale) : new Intl.NumberFormat(locale, options);
- return nf.format(value)
- }
- const formatter = this._getNumberFormatter(value, locale, this.fallbackLocale, this._getNumberFormats(), key, options);
- const ret = formatter && formatter.format(value);
- if (this._isFallbackRoot(ret)) {
- if (!this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {
- warn(`Fall back to number localization of root: key '${key}'.`);
- }
- /* istanbul ignore if */
- if (!this._root) { throw Error('unexpected error') }
- return this._root.$i18n.n(value, Object.assign({}, { key, locale }, options))
- } else {
- return ret || ''
- }
- }
- n (value, ...args) {
- let locale = this.locale;
- let key = null;
- let options = null;
- if (args.length === 1) {
- if (isString(args[0])) {
- key = args[0];
- } else if (isObject(args[0])) {
- if (args[0].locale) {
- locale = args[0].locale;
- }
- if (args[0].key) {
- key = args[0].key;
- }
- // Filter out number format options only
- options = Object.keys(args[0]).reduce((acc, key) => {
- if (includes(numberFormatKeys, key)) {
- return Object.assign({}, acc, { [key]: args[0][key] })
- }
- return acc
- }, null);
- }
- } else if (args.length === 2) {
- if (isString(args[0])) {
- key = args[0];
- }
- if (isString(args[1])) {
- locale = args[1];
- }
- }
- return this._n(value, locale, key, options)
- }
- _ntp (value, locale, key, options) {
- /* istanbul ignore if */
- if (!VueI18n.availabilities.numberFormat) {
- {
- warn('Cannot format to parts a Number value due to not supported Intl.NumberFormat.');
- }
- return []
- }
- if (!key) {
- const nf = !options ? new Intl.NumberFormat(locale) : new Intl.NumberFormat(locale, options);
- return nf.formatToParts(value)
- }
- const formatter = this._getNumberFormatter(value, locale, this.fallbackLocale, this._getNumberFormats(), key, options);
- const ret = formatter && formatter.formatToParts(value);
- if (this._isFallbackRoot(ret)) {
- if (!this._isSilentTranslationWarn(key)) {
- warn(`Fall back to format number to parts of root: key '${key}' .`);
- }
- /* istanbul ignore if */
- if (!this._root) { throw Error('unexpected error') }
- return this._root.$i18n._ntp(value, locale, key, options)
- } else {
- return ret || []
- }
- }
- }
- let availabilities;
- // $FlowFixMe
- Object.defineProperty(VueI18n, 'availabilities', {
- get () {
- if (!availabilities) {
- const intlDefined = typeof Intl !== 'undefined';
- availabilities = {
- dateTimeFormat: intlDefined && typeof Intl.DateTimeFormat !== 'undefined',
- numberFormat: intlDefined && typeof Intl.NumberFormat !== 'undefined'
- };
- }
- return availabilities
- }
- });
- VueI18n.install = install;
- VueI18n.version = '8.22.2';
- export default VueI18n;
|