2086 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			2086 lines
		
	
	
		
			53 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*  */
 | |
| 
 | |
| /**
 | |
|  * constants
 | |
|  */
 | |
| 
 | |
| const numberFormatKeys = [
 | |
|   'style',
 | |
|   'currency',
 | |
|   'currencyDisplay',
 | |
|   'useGrouping',
 | |
|   'minimumIntegerDigits',
 | |
|   'minimumFractionDigits',
 | |
|   'maximumFractionDigits',
 | |
|   'minimumSignificantDigits',
 | |
|   'maximumSignificantDigits',
 | |
|   'localeMatcher',
 | |
|   'formatMatcher',
 | |
|   'unit'
 | |
| ];
 | |
| 
 | |
| /**
 | |
|  * 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 parseArgs (...args) {
 | |
|   let locale = null;
 | |
|   let params = null;
 | |
|   if (args.length === 1) {
 | |
|     if (isObject(args[0]) || Array.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]) || Array.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 = Array.isArray(a);
 | |
|       const isArrayB = Array.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
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*  */
 | |
| 
 | |
| 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.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.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;
 | |
| 
 | |
|     /**
 | |
|      * @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 (Array.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 (Array.isArray(pathRet) || isPlainObject(pathRet)) { return pathRet }
 | |
| 
 | |
|     let ret;
 | |
|     if (isNull(pathRet)) {
 | |
|       /* istanbul ignore else */
 | |
|       if (isPlainObject(message)) {
 | |
|         ret = message[key];
 | |
|         if (!isString(ret)) {
 | |
|           if (!this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {
 | |
|             warn(`Value of key '${key}' is not a string!`);
 | |
|           }
 | |
|           return null
 | |
|         }
 | |
|       } else {
 | |
|         return null
 | |
|       }
 | |
|     } else {
 | |
|       /* istanbul ignore else */
 | |
|       if (isString(pathRet)) {
 | |
|         ret = pathRet;
 | |
|       } else {
 | |
|         if (!this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {
 | |
|           warn(`Value of key '${key}' is not a string!`);
 | |
|         }
 | |
|         return null
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Check for the existence of links within the translated string
 | |
|     if (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,
 | |
|         Array.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
 | |
|   }
 | |
| 
 | |
|   _render (message, interpolateMode, values, path) {
 | |
|     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);
 | |
|     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.20.0';
 | |
| 
 | |
| export default VueI18n;
 |