You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
9188 lines
327 KiB
9188 lines
327 KiB
(function () {
|
|
'use strict';
|
|
|
|
/**
|
|
* Make a map and return a function for checking if a key
|
|
* is in that map.
|
|
* IMPORTANT: all calls of this function must be prefixed with
|
|
* \/\*#\_\_PURE\_\_\*\/
|
|
* So that rollup can tree-shake them if necessary.
|
|
*/
|
|
function makeMap(str, expectsLowerCase) {
|
|
const map = Object.create(null);
|
|
const list = str.split(',');
|
|
for (let i = 0; i < list.length; i++) {
|
|
map[list[i]] = true;
|
|
}
|
|
return expectsLowerCase ? val => !!map[val.toLowerCase()] : val => !!map[val];
|
|
}
|
|
|
|
const GLOBALS_WHITE_LISTED = 'Infinity,undefined,NaN,isFinite,isNaN,parseFloat,parseInt,decodeURI,' +
|
|
'decodeURIComponent,encodeURI,encodeURIComponent,Math,Number,Date,Array,' +
|
|
'Object,Boolean,String,RegExp,Map,Set,JSON,Intl';
|
|
const isGloballyWhitelisted = /*#__PURE__*/ makeMap(GLOBALS_WHITE_LISTED);
|
|
|
|
/**
|
|
* On the client we only need to offer special cases for boolean attributes that
|
|
* have different names from their corresponding dom properties:
|
|
* - itemscope -> N/A
|
|
* - allowfullscreen -> allowFullscreen
|
|
* - formnovalidate -> formNoValidate
|
|
* - ismap -> isMap
|
|
* - nomodule -> noModule
|
|
* - novalidate -> noValidate
|
|
* - readonly -> readOnly
|
|
*/
|
|
const specialBooleanAttrs = `itemscope,allowfullscreen,formnovalidate,ismap,nomodule,novalidate,readonly`;
|
|
const isSpecialBooleanAttr = /*#__PURE__*/ makeMap(specialBooleanAttrs);
|
|
|
|
function normalizeStyle(value) {
|
|
if (isArray(value)) {
|
|
const res = {};
|
|
for (let i = 0; i < value.length; i++) {
|
|
const item = value[i];
|
|
const normalized = normalizeStyle(isString(item) ? parseStringStyle(item) : item);
|
|
if (normalized) {
|
|
for (const key in normalized) {
|
|
res[key] = normalized[key];
|
|
}
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
else if (isObject(value)) {
|
|
return value;
|
|
}
|
|
}
|
|
const listDelimiterRE = /;(?![^(]*\))/g;
|
|
const propertyDelimiterRE = /:(.+)/;
|
|
function parseStringStyle(cssText) {
|
|
const ret = {};
|
|
cssText.split(listDelimiterRE).forEach(item => {
|
|
if (item) {
|
|
const tmp = item.split(propertyDelimiterRE);
|
|
tmp.length > 1 && (ret[tmp[0].trim()] = tmp[1].trim());
|
|
}
|
|
});
|
|
return ret;
|
|
}
|
|
function normalizeClass(value) {
|
|
let res = '';
|
|
if (isString(value)) {
|
|
res = value;
|
|
}
|
|
else if (isArray(value)) {
|
|
for (let i = 0; i < value.length; i++) {
|
|
res += normalizeClass(value[i]) + ' ';
|
|
}
|
|
}
|
|
else if (isObject(value)) {
|
|
for (const name in value) {
|
|
if (value[name]) {
|
|
res += name + ' ';
|
|
}
|
|
}
|
|
}
|
|
return res.trim();
|
|
}
|
|
|
|
function looseCompareArrays(a, b) {
|
|
if (a.length !== b.length)
|
|
return false;
|
|
let equal = true;
|
|
for (let i = 0; equal && i < a.length; i++) {
|
|
equal = looseEqual(a[i], b[i]);
|
|
}
|
|
return equal;
|
|
}
|
|
function looseEqual(a, b) {
|
|
if (a === b)
|
|
return true;
|
|
let aValidType = isDate(a);
|
|
let bValidType = isDate(b);
|
|
if (aValidType || bValidType) {
|
|
return aValidType && bValidType ? a.getTime() === b.getTime() : false;
|
|
}
|
|
aValidType = isArray(a);
|
|
bValidType = isArray(b);
|
|
if (aValidType || bValidType) {
|
|
return aValidType && bValidType ? looseCompareArrays(a, b) : false;
|
|
}
|
|
aValidType = isObject(a);
|
|
bValidType = isObject(b);
|
|
if (aValidType || bValidType) {
|
|
/* istanbul ignore if: this if will probably never be called */
|
|
if (!aValidType || !bValidType) {
|
|
return false;
|
|
}
|
|
const aKeysCount = Object.keys(a).length;
|
|
const bKeysCount = Object.keys(b).length;
|
|
if (aKeysCount !== bKeysCount) {
|
|
return false;
|
|
}
|
|
for (const key in a) {
|
|
const aHasKey = a.hasOwnProperty(key);
|
|
const bHasKey = b.hasOwnProperty(key);
|
|
if ((aHasKey && !bHasKey) ||
|
|
(!aHasKey && bHasKey) ||
|
|
!looseEqual(a[key], b[key])) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return String(a) === String(b);
|
|
}
|
|
function looseIndexOf(arr, val) {
|
|
return arr.findIndex(item => looseEqual(item, val));
|
|
}
|
|
|
|
/**
|
|
* For converting {{ interpolation }} values to displayed strings.
|
|
* @private
|
|
*/
|
|
const toDisplayString = (val) => {
|
|
return val == null
|
|
? ''
|
|
: isObject(val)
|
|
? JSON.stringify(val, replacer, 2)
|
|
: String(val);
|
|
};
|
|
const replacer = (_key, val) => {
|
|
if (isMap(val)) {
|
|
return {
|
|
[`Map(${val.size})`]: [...val.entries()].reduce((entries, [key, val]) => {
|
|
entries[`${key} =>`] = val;
|
|
return entries;
|
|
}, {})
|
|
};
|
|
}
|
|
else if (isSet(val)) {
|
|
return {
|
|
[`Set(${val.size})`]: [...val.values()]
|
|
};
|
|
}
|
|
else if (isObject(val) && !isArray(val) && !isPlainObject(val)) {
|
|
return String(val);
|
|
}
|
|
return val;
|
|
};
|
|
const EMPTY_OBJ = {};
|
|
const EMPTY_ARR = [];
|
|
const NOOP = () => { };
|
|
/**
|
|
* Always return false.
|
|
*/
|
|
const NO = () => false;
|
|
const onRE = /^on[^a-z]/;
|
|
const isOn = (key) => onRE.test(key);
|
|
const isModelListener = (key) => key.startsWith('onUpdate:');
|
|
const extend = Object.assign;
|
|
const remove = (arr, el) => {
|
|
const i = arr.indexOf(el);
|
|
if (i > -1) {
|
|
arr.splice(i, 1);
|
|
}
|
|
};
|
|
const hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
const hasOwn = (val, key) => hasOwnProperty.call(val, key);
|
|
const isArray = Array.isArray;
|
|
const isMap = (val) => toTypeString(val) === '[object Map]';
|
|
const isSet = (val) => toTypeString(val) === '[object Set]';
|
|
const isDate = (val) => val instanceof Date;
|
|
const isFunction = (val) => typeof val === 'function';
|
|
const isString = (val) => typeof val === 'string';
|
|
const isSymbol = (val) => typeof val === 'symbol';
|
|
const isObject = (val) => val !== null && typeof val === 'object';
|
|
const isPromise = (val) => {
|
|
return isObject(val) && isFunction(val.then) && isFunction(val.catch);
|
|
};
|
|
const objectToString = Object.prototype.toString;
|
|
const toTypeString = (value) => objectToString.call(value);
|
|
const toRawType = (value) => {
|
|
// extract "RawType" from strings like "[object RawType]"
|
|
return toTypeString(value).slice(8, -1);
|
|
};
|
|
const isPlainObject = (val) => toTypeString(val) === '[object Object]';
|
|
const isIntegerKey = (key) => isString(key) &&
|
|
key !== 'NaN' &&
|
|
key[0] !== '-' &&
|
|
'' + parseInt(key, 10) === key;
|
|
const isReservedProp = /*#__PURE__*/ makeMap(
|
|
// the leading comma is intentional so empty string "" is also included
|
|
',key,ref,' +
|
|
'onVnodeBeforeMount,onVnodeMounted,' +
|
|
'onVnodeBeforeUpdate,onVnodeUpdated,' +
|
|
'onVnodeBeforeUnmount,onVnodeUnmounted');
|
|
const cacheStringFunction = (fn) => {
|
|
const cache = Object.create(null);
|
|
return ((str) => {
|
|
const hit = cache[str];
|
|
return hit || (cache[str] = fn(str));
|
|
});
|
|
};
|
|
const camelizeRE = /-(\w)/g;
|
|
/**
|
|
* @private
|
|
*/
|
|
const camelize = cacheStringFunction((str) => {
|
|
return str.replace(camelizeRE, (_, c) => (c ? c.toUpperCase() : ''));
|
|
});
|
|
const hyphenateRE = /\B([A-Z])/g;
|
|
/**
|
|
* @private
|
|
*/
|
|
const hyphenate = cacheStringFunction((str) => str.replace(hyphenateRE, '-$1').toLowerCase());
|
|
/**
|
|
* @private
|
|
*/
|
|
const capitalize = cacheStringFunction((str) => str.charAt(0).toUpperCase() + str.slice(1));
|
|
/**
|
|
* @private
|
|
*/
|
|
const toHandlerKey = cacheStringFunction((str) => (str ? `on${capitalize(str)}` : ``));
|
|
// compare whether a value has changed, accounting for NaN.
|
|
const hasChanged = (value, oldValue) => value !== oldValue && (value === value || oldValue === oldValue);
|
|
const invokeArrayFns = (fns, arg) => {
|
|
for (let i = 0; i < fns.length; i++) {
|
|
fns[i](arg);
|
|
}
|
|
};
|
|
const def = (obj, key, value) => {
|
|
Object.defineProperty(obj, key, {
|
|
configurable: true,
|
|
enumerable: false,
|
|
value
|
|
});
|
|
};
|
|
const toNumber = (val) => {
|
|
const n = parseFloat(val);
|
|
return isNaN(n) ? val : n;
|
|
};
|
|
|
|
const targetMap = new WeakMap();
|
|
const effectStack = [];
|
|
let activeEffect;
|
|
const ITERATE_KEY = Symbol('');
|
|
const MAP_KEY_ITERATE_KEY = Symbol('');
|
|
function isEffect(fn) {
|
|
return fn && fn._isEffect === true;
|
|
}
|
|
function effect(fn, options = EMPTY_OBJ) {
|
|
if (isEffect(fn)) {
|
|
fn = fn.raw;
|
|
}
|
|
const effect = createReactiveEffect(fn, options);
|
|
if (!options.lazy) {
|
|
effect();
|
|
}
|
|
return effect;
|
|
}
|
|
function stop(effect) {
|
|
if (effect.active) {
|
|
cleanup(effect);
|
|
if (effect.options.onStop) {
|
|
effect.options.onStop();
|
|
}
|
|
effect.active = false;
|
|
}
|
|
}
|
|
let uid = 0;
|
|
function createReactiveEffect(fn, options) {
|
|
const effect = function reactiveEffect() {
|
|
if (!effect.active) {
|
|
return options.scheduler ? undefined : fn();
|
|
}
|
|
if (!effectStack.includes(effect)) {
|
|
cleanup(effect);
|
|
try {
|
|
enableTracking();
|
|
effectStack.push(effect);
|
|
activeEffect = effect;
|
|
return fn();
|
|
}
|
|
finally {
|
|
effectStack.pop();
|
|
resetTracking();
|
|
activeEffect = effectStack[effectStack.length - 1];
|
|
}
|
|
}
|
|
};
|
|
effect.id = uid++;
|
|
effect.allowRecurse = !!options.allowRecurse;
|
|
effect._isEffect = true;
|
|
effect.active = true;
|
|
effect.raw = fn;
|
|
effect.deps = [];
|
|
effect.options = options;
|
|
return effect;
|
|
}
|
|
function cleanup(effect) {
|
|
const { deps } = effect;
|
|
if (deps.length) {
|
|
for (let i = 0; i < deps.length; i++) {
|
|
deps[i].delete(effect);
|
|
}
|
|
deps.length = 0;
|
|
}
|
|
}
|
|
let shouldTrack = true;
|
|
const trackStack = [];
|
|
function pauseTracking() {
|
|
trackStack.push(shouldTrack);
|
|
shouldTrack = false;
|
|
}
|
|
function enableTracking() {
|
|
trackStack.push(shouldTrack);
|
|
shouldTrack = true;
|
|
}
|
|
function resetTracking() {
|
|
const last = trackStack.pop();
|
|
shouldTrack = last === undefined ? true : last;
|
|
}
|
|
function track(target, type, key) {
|
|
if (!shouldTrack || activeEffect === undefined) {
|
|
return;
|
|
}
|
|
let depsMap = targetMap.get(target);
|
|
if (!depsMap) {
|
|
targetMap.set(target, (depsMap = new Map()));
|
|
}
|
|
let dep = depsMap.get(key);
|
|
if (!dep) {
|
|
depsMap.set(key, (dep = new Set()));
|
|
}
|
|
if (!dep.has(activeEffect)) {
|
|
dep.add(activeEffect);
|
|
activeEffect.deps.push(dep);
|
|
}
|
|
}
|
|
function trigger(target, type, key, newValue, oldValue, oldTarget) {
|
|
const depsMap = targetMap.get(target);
|
|
if (!depsMap) {
|
|
// never been tracked
|
|
return;
|
|
}
|
|
const effects = new Set();
|
|
const add = (effectsToAdd) => {
|
|
if (effectsToAdd) {
|
|
effectsToAdd.forEach(effect => {
|
|
if (effect !== activeEffect || effect.allowRecurse) {
|
|
effects.add(effect);
|
|
}
|
|
});
|
|
}
|
|
};
|
|
if (type === "clear" /* CLEAR */) {
|
|
// collection being cleared
|
|
// trigger all effects for target
|
|
depsMap.forEach(add);
|
|
}
|
|
else if (key === 'length' && isArray(target)) {
|
|
depsMap.forEach((dep, key) => {
|
|
if (key === 'length' || key >= newValue) {
|
|
add(dep);
|
|
}
|
|
});
|
|
}
|
|
else {
|
|
// schedule runs for SET | ADD | DELETE
|
|
if (key !== void 0) {
|
|
add(depsMap.get(key));
|
|
}
|
|
// also run for iteration key on ADD | DELETE | Map.SET
|
|
switch (type) {
|
|
case "add" /* ADD */:
|
|
if (!isArray(target)) {
|
|
add(depsMap.get(ITERATE_KEY));
|
|
if (isMap(target)) {
|
|
add(depsMap.get(MAP_KEY_ITERATE_KEY));
|
|
}
|
|
}
|
|
else if (isIntegerKey(key)) {
|
|
// new index added to array -> length changes
|
|
add(depsMap.get('length'));
|
|
}
|
|
break;
|
|
case "delete" /* DELETE */:
|
|
if (!isArray(target)) {
|
|
add(depsMap.get(ITERATE_KEY));
|
|
if (isMap(target)) {
|
|
add(depsMap.get(MAP_KEY_ITERATE_KEY));
|
|
}
|
|
}
|
|
break;
|
|
case "set" /* SET */:
|
|
if (isMap(target)) {
|
|
add(depsMap.get(ITERATE_KEY));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
const run = (effect) => {
|
|
if (effect.options.scheduler) {
|
|
effect.options.scheduler(effect);
|
|
}
|
|
else {
|
|
effect();
|
|
}
|
|
};
|
|
effects.forEach(run);
|
|
}
|
|
|
|
const builtInSymbols = new Set(Object.getOwnPropertyNames(Symbol)
|
|
.map(key => Symbol[key])
|
|
.filter(isSymbol));
|
|
const get = /*#__PURE__*/ createGetter();
|
|
const shallowGet = /*#__PURE__*/ createGetter(false, true);
|
|
const readonlyGet = /*#__PURE__*/ createGetter(true);
|
|
const shallowReadonlyGet = /*#__PURE__*/ createGetter(true, true);
|
|
const arrayInstrumentations = {};
|
|
['includes', 'indexOf', 'lastIndexOf'].forEach(key => {
|
|
const method = Array.prototype[key];
|
|
arrayInstrumentations[key] = function (...args) {
|
|
const arr = toRaw(this);
|
|
for (let i = 0, l = this.length; i < l; i++) {
|
|
track(arr, "get" /* GET */, i + '');
|
|
}
|
|
// we run the method using the original args first (which may be reactive)
|
|
const res = method.apply(arr, args);
|
|
if (res === -1 || res === false) {
|
|
// if that didn't work, run it again using raw values.
|
|
return method.apply(arr, args.map(toRaw));
|
|
}
|
|
else {
|
|
return res;
|
|
}
|
|
};
|
|
});
|
|
['push', 'pop', 'shift', 'unshift', 'splice'].forEach(key => {
|
|
const method = Array.prototype[key];
|
|
arrayInstrumentations[key] = function (...args) {
|
|
pauseTracking();
|
|
const res = method.apply(this, args);
|
|
resetTracking();
|
|
return res;
|
|
};
|
|
});
|
|
function createGetter(isReadonly = false, shallow = false) {
|
|
return function get(target, key, receiver) {
|
|
if (key === "__v_isReactive" /* IS_REACTIVE */) {
|
|
return !isReadonly;
|
|
}
|
|
else if (key === "__v_isReadonly" /* IS_READONLY */) {
|
|
return isReadonly;
|
|
}
|
|
else if (key === "__v_raw" /* RAW */ &&
|
|
receiver === (isReadonly ? readonlyMap : reactiveMap).get(target)) {
|
|
return target;
|
|
}
|
|
const targetIsArray = isArray(target);
|
|
if (!isReadonly && targetIsArray && hasOwn(arrayInstrumentations, key)) {
|
|
return Reflect.get(arrayInstrumentations, key, receiver);
|
|
}
|
|
const res = Reflect.get(target, key, receiver);
|
|
if (isSymbol(key)
|
|
? builtInSymbols.has(key)
|
|
: key === `__proto__` || key === `__v_isRef`) {
|
|
return res;
|
|
}
|
|
if (!isReadonly) {
|
|
track(target, "get" /* GET */, key);
|
|
}
|
|
if (shallow) {
|
|
return res;
|
|
}
|
|
if (isRef(res)) {
|
|
// ref unwrapping - does not apply for Array + integer key.
|
|
const shouldUnwrap = !targetIsArray || !isIntegerKey(key);
|
|
return shouldUnwrap ? res.value : res;
|
|
}
|
|
if (isObject(res)) {
|
|
// Convert returned value into a proxy as well. we do the isObject check
|
|
// here to avoid invalid value warning. Also need to lazy access readonly
|
|
// and reactive here to avoid circular dependency.
|
|
return isReadonly ? readonly(res) : reactive(res);
|
|
}
|
|
return res;
|
|
};
|
|
}
|
|
const set = /*#__PURE__*/ createSetter();
|
|
const shallowSet = /*#__PURE__*/ createSetter(true);
|
|
function createSetter(shallow = false) {
|
|
return function set(target, key, value, receiver) {
|
|
const oldValue = target[key];
|
|
if (!shallow) {
|
|
value = toRaw(value);
|
|
if (!isArray(target) && isRef(oldValue) && !isRef(value)) {
|
|
oldValue.value = value;
|
|
return true;
|
|
}
|
|
}
|
|
const hadKey = isArray(target) && isIntegerKey(key)
|
|
? Number(key) < target.length
|
|
: hasOwn(target, key);
|
|
const result = Reflect.set(target, key, value, receiver);
|
|
// don't trigger if target is something up in the prototype chain of original
|
|
if (target === toRaw(receiver)) {
|
|
if (!hadKey) {
|
|
trigger(target, "add" /* ADD */, key, value);
|
|
}
|
|
else if (hasChanged(value, oldValue)) {
|
|
trigger(target, "set" /* SET */, key, value);
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
function deleteProperty(target, key) {
|
|
const hadKey = hasOwn(target, key);
|
|
target[key];
|
|
const result = Reflect.deleteProperty(target, key);
|
|
if (result && hadKey) {
|
|
trigger(target, "delete" /* DELETE */, key, undefined);
|
|
}
|
|
return result;
|
|
}
|
|
function has(target, key) {
|
|
const result = Reflect.has(target, key);
|
|
if (!isSymbol(key) || !builtInSymbols.has(key)) {
|
|
track(target, "has" /* HAS */, key);
|
|
}
|
|
return result;
|
|
}
|
|
function ownKeys(target) {
|
|
track(target, "iterate" /* ITERATE */, isArray(target) ? 'length' : ITERATE_KEY);
|
|
return Reflect.ownKeys(target);
|
|
}
|
|
const mutableHandlers = {
|
|
get,
|
|
set,
|
|
deleteProperty,
|
|
has,
|
|
ownKeys
|
|
};
|
|
const readonlyHandlers = {
|
|
get: readonlyGet,
|
|
set(target, key) {
|
|
return true;
|
|
},
|
|
deleteProperty(target, key) {
|
|
return true;
|
|
}
|
|
};
|
|
const shallowReactiveHandlers = extend({}, mutableHandlers, {
|
|
get: shallowGet,
|
|
set: shallowSet
|
|
});
|
|
// Props handlers are special in the sense that it should not unwrap top-level
|
|
// refs (in order to allow refs to be explicitly passed down), but should
|
|
// retain the reactivity of the normal readonly object.
|
|
extend({}, readonlyHandlers, {
|
|
get: shallowReadonlyGet
|
|
});
|
|
|
|
const toReactive = (value) => isObject(value) ? reactive(value) : value;
|
|
const toReadonly = (value) => isObject(value) ? readonly(value) : value;
|
|
const toShallow = (value) => value;
|
|
const getProto = (v) => Reflect.getPrototypeOf(v);
|
|
function get$1(target, key, isReadonly = false, isShallow = false) {
|
|
// #1772: readonly(reactive(Map)) should return readonly + reactive version
|
|
// of the value
|
|
target = target["__v_raw" /* RAW */];
|
|
const rawTarget = toRaw(target);
|
|
const rawKey = toRaw(key);
|
|
if (key !== rawKey) {
|
|
!isReadonly && track(rawTarget, "get" /* GET */, key);
|
|
}
|
|
!isReadonly && track(rawTarget, "get" /* GET */, rawKey);
|
|
const { has } = getProto(rawTarget);
|
|
const wrap = isReadonly ? toReadonly : isShallow ? toShallow : toReactive;
|
|
if (has.call(rawTarget, key)) {
|
|
return wrap(target.get(key));
|
|
}
|
|
else if (has.call(rawTarget, rawKey)) {
|
|
return wrap(target.get(rawKey));
|
|
}
|
|
}
|
|
function has$1(key, isReadonly = false) {
|
|
const target = this["__v_raw" /* RAW */];
|
|
const rawTarget = toRaw(target);
|
|
const rawKey = toRaw(key);
|
|
if (key !== rawKey) {
|
|
!isReadonly && track(rawTarget, "has" /* HAS */, key);
|
|
}
|
|
!isReadonly && track(rawTarget, "has" /* HAS */, rawKey);
|
|
return key === rawKey
|
|
? target.has(key)
|
|
: target.has(key) || target.has(rawKey);
|
|
}
|
|
function size(target, isReadonly = false) {
|
|
target = target["__v_raw" /* RAW */];
|
|
!isReadonly && track(toRaw(target), "iterate" /* ITERATE */, ITERATE_KEY);
|
|
return Reflect.get(target, 'size', target);
|
|
}
|
|
function add(value) {
|
|
value = toRaw(value);
|
|
const target = toRaw(this);
|
|
const proto = getProto(target);
|
|
const hadKey = proto.has.call(target, value);
|
|
target.add(value);
|
|
if (!hadKey) {
|
|
trigger(target, "add" /* ADD */, value, value);
|
|
}
|
|
return this;
|
|
}
|
|
function set$1(key, value) {
|
|
value = toRaw(value);
|
|
const target = toRaw(this);
|
|
const { has, get } = getProto(target);
|
|
let hadKey = has.call(target, key);
|
|
if (!hadKey) {
|
|
key = toRaw(key);
|
|
hadKey = has.call(target, key);
|
|
}
|
|
const oldValue = get.call(target, key);
|
|
target.set(key, value);
|
|
if (!hadKey) {
|
|
trigger(target, "add" /* ADD */, key, value);
|
|
}
|
|
else if (hasChanged(value, oldValue)) {
|
|
trigger(target, "set" /* SET */, key, value);
|
|
}
|
|
return this;
|
|
}
|
|
function deleteEntry(key) {
|
|
const target = toRaw(this);
|
|
const { has, get } = getProto(target);
|
|
let hadKey = has.call(target, key);
|
|
if (!hadKey) {
|
|
key = toRaw(key);
|
|
hadKey = has.call(target, key);
|
|
}
|
|
get ? get.call(target, key) : undefined;
|
|
// forward the operation before queueing reactions
|
|
const result = target.delete(key);
|
|
if (hadKey) {
|
|
trigger(target, "delete" /* DELETE */, key, undefined);
|
|
}
|
|
return result;
|
|
}
|
|
function clear() {
|
|
const target = toRaw(this);
|
|
const hadItems = target.size !== 0;
|
|
// forward the operation before queueing reactions
|
|
const result = target.clear();
|
|
if (hadItems) {
|
|
trigger(target, "clear" /* CLEAR */, undefined, undefined);
|
|
}
|
|
return result;
|
|
}
|
|
function createForEach(isReadonly, isShallow) {
|
|
return function forEach(callback, thisArg) {
|
|
const observed = this;
|
|
const target = observed["__v_raw" /* RAW */];
|
|
const rawTarget = toRaw(target);
|
|
const wrap = isReadonly ? toReadonly : isShallow ? toShallow : toReactive;
|
|
!isReadonly && track(rawTarget, "iterate" /* ITERATE */, ITERATE_KEY);
|
|
return target.forEach((value, key) => {
|
|
// important: make sure the callback is
|
|
// 1. invoked with the reactive map as `this` and 3rd arg
|
|
// 2. the value received should be a corresponding reactive/readonly.
|
|
return callback.call(thisArg, wrap(value), wrap(key), observed);
|
|
});
|
|
};
|
|
}
|
|
function createIterableMethod(method, isReadonly, isShallow) {
|
|
return function (...args) {
|
|
const target = this["__v_raw" /* RAW */];
|
|
const rawTarget = toRaw(target);
|
|
const targetIsMap = isMap(rawTarget);
|
|
const isPair = method === 'entries' || (method === Symbol.iterator && targetIsMap);
|
|
const isKeyOnly = method === 'keys' && targetIsMap;
|
|
const innerIterator = target[method](...args);
|
|
const wrap = isReadonly ? toReadonly : isShallow ? toShallow : toReactive;
|
|
!isReadonly &&
|
|
track(rawTarget, "iterate" /* ITERATE */, isKeyOnly ? MAP_KEY_ITERATE_KEY : ITERATE_KEY);
|
|
// return a wrapped iterator which returns observed versions of the
|
|
// values emitted from the real iterator
|
|
return {
|
|
// iterator protocol
|
|
next() {
|
|
const { value, done } = innerIterator.next();
|
|
return done
|
|
? { value, done }
|
|
: {
|
|
value: isPair ? [wrap(value[0]), wrap(value[1])] : wrap(value),
|
|
done
|
|
};
|
|
},
|
|
// iterable protocol
|
|
[Symbol.iterator]() {
|
|
return this;
|
|
}
|
|
};
|
|
};
|
|
}
|
|
function createReadonlyMethod(type) {
|
|
return function (...args) {
|
|
return type === "delete" /* DELETE */ ? false : this;
|
|
};
|
|
}
|
|
const mutableInstrumentations = {
|
|
get(key) {
|
|
return get$1(this, key);
|
|
},
|
|
get size() {
|
|
return size(this);
|
|
},
|
|
has: has$1,
|
|
add,
|
|
set: set$1,
|
|
delete: deleteEntry,
|
|
clear,
|
|
forEach: createForEach(false, false)
|
|
};
|
|
const shallowInstrumentations = {
|
|
get(key) {
|
|
return get$1(this, key, false, true);
|
|
},
|
|
get size() {
|
|
return size(this);
|
|
},
|
|
has: has$1,
|
|
add,
|
|
set: set$1,
|
|
delete: deleteEntry,
|
|
clear,
|
|
forEach: createForEach(false, true)
|
|
};
|
|
const readonlyInstrumentations = {
|
|
get(key) {
|
|
return get$1(this, key, true);
|
|
},
|
|
get size() {
|
|
return size(this, true);
|
|
},
|
|
has(key) {
|
|
return has$1.call(this, key, true);
|
|
},
|
|
add: createReadonlyMethod("add" /* ADD */),
|
|
set: createReadonlyMethod("set" /* SET */),
|
|
delete: createReadonlyMethod("delete" /* DELETE */),
|
|
clear: createReadonlyMethod("clear" /* CLEAR */),
|
|
forEach: createForEach(true, false)
|
|
};
|
|
const iteratorMethods = ['keys', 'values', 'entries', Symbol.iterator];
|
|
iteratorMethods.forEach(method => {
|
|
mutableInstrumentations[method] = createIterableMethod(method, false, false);
|
|
readonlyInstrumentations[method] = createIterableMethod(method, true, false);
|
|
shallowInstrumentations[method] = createIterableMethod(method, false, true);
|
|
});
|
|
function createInstrumentationGetter(isReadonly, shallow) {
|
|
const instrumentations = shallow
|
|
? shallowInstrumentations
|
|
: isReadonly
|
|
? readonlyInstrumentations
|
|
: mutableInstrumentations;
|
|
return (target, key, receiver) => {
|
|
if (key === "__v_isReactive" /* IS_REACTIVE */) {
|
|
return !isReadonly;
|
|
}
|
|
else if (key === "__v_isReadonly" /* IS_READONLY */) {
|
|
return isReadonly;
|
|
}
|
|
else if (key === "__v_raw" /* RAW */) {
|
|
return target;
|
|
}
|
|
return Reflect.get(hasOwn(instrumentations, key) && key in target
|
|
? instrumentations
|
|
: target, key, receiver);
|
|
};
|
|
}
|
|
const mutableCollectionHandlers = {
|
|
get: createInstrumentationGetter(false, false)
|
|
};
|
|
const shallowCollectionHandlers = {
|
|
get: createInstrumentationGetter(false, true)
|
|
};
|
|
const readonlyCollectionHandlers = {
|
|
get: createInstrumentationGetter(true, false)
|
|
};
|
|
|
|
const reactiveMap = new WeakMap();
|
|
const readonlyMap = new WeakMap();
|
|
function targetTypeMap(rawType) {
|
|
switch (rawType) {
|
|
case 'Object':
|
|
case 'Array':
|
|
return 1 /* COMMON */;
|
|
case 'Map':
|
|
case 'Set':
|
|
case 'WeakMap':
|
|
case 'WeakSet':
|
|
return 2 /* COLLECTION */;
|
|
default:
|
|
return 0 /* INVALID */;
|
|
}
|
|
}
|
|
function getTargetType(value) {
|
|
return value["__v_skip" /* SKIP */] || !Object.isExtensible(value)
|
|
? 0 /* INVALID */
|
|
: targetTypeMap(toRawType(value));
|
|
}
|
|
function reactive(target) {
|
|
// if trying to observe a readonly proxy, return the readonly version.
|
|
if (target && target["__v_isReadonly" /* IS_READONLY */]) {
|
|
return target;
|
|
}
|
|
return createReactiveObject(target, false, mutableHandlers, mutableCollectionHandlers);
|
|
}
|
|
/**
|
|
* Return a shallowly-reactive copy of the original object, where only the root
|
|
* level properties are reactive. It also does not auto-unwrap refs (even at the
|
|
* root level).
|
|
*/
|
|
function shallowReactive(target) {
|
|
return createReactiveObject(target, false, shallowReactiveHandlers, shallowCollectionHandlers);
|
|
}
|
|
/**
|
|
* Creates a readonly copy of the original object. Note the returned copy is not
|
|
* made reactive, but `readonly` can be called on an already reactive object.
|
|
*/
|
|
function readonly(target) {
|
|
return createReactiveObject(target, true, readonlyHandlers, readonlyCollectionHandlers);
|
|
}
|
|
function createReactiveObject(target, isReadonly, baseHandlers, collectionHandlers) {
|
|
if (!isObject(target)) {
|
|
return target;
|
|
}
|
|
// target is already a Proxy, return it.
|
|
// exception: calling readonly() on a reactive object
|
|
if (target["__v_raw" /* RAW */] &&
|
|
!(isReadonly && target["__v_isReactive" /* IS_REACTIVE */])) {
|
|
return target;
|
|
}
|
|
// target already has corresponding Proxy
|
|
const proxyMap = isReadonly ? readonlyMap : reactiveMap;
|
|
const existingProxy = proxyMap.get(target);
|
|
if (existingProxy) {
|
|
return existingProxy;
|
|
}
|
|
// only a whitelist of value types can be observed.
|
|
const targetType = getTargetType(target);
|
|
if (targetType === 0 /* INVALID */) {
|
|
return target;
|
|
}
|
|
const proxy = new Proxy(target, targetType === 2 /* COLLECTION */ ? collectionHandlers : baseHandlers);
|
|
proxyMap.set(target, proxy);
|
|
return proxy;
|
|
}
|
|
function isReactive(value) {
|
|
if (isReadonly(value)) {
|
|
return isReactive(value["__v_raw" /* RAW */]);
|
|
}
|
|
return !!(value && value["__v_isReactive" /* IS_REACTIVE */]);
|
|
}
|
|
function isReadonly(value) {
|
|
return !!(value && value["__v_isReadonly" /* IS_READONLY */]);
|
|
}
|
|
function isProxy(value) {
|
|
return isReactive(value) || isReadonly(value);
|
|
}
|
|
function toRaw(observed) {
|
|
return ((observed && toRaw(observed["__v_raw" /* RAW */])) || observed);
|
|
}
|
|
function isRef(r) {
|
|
return Boolean(r && r.__v_isRef === true);
|
|
}
|
|
function unref(ref) {
|
|
return isRef(ref) ? ref.value : ref;
|
|
}
|
|
const shallowUnwrapHandlers = {
|
|
get: (target, key, receiver) => unref(Reflect.get(target, key, receiver)),
|
|
set: (target, key, value, receiver) => {
|
|
const oldValue = target[key];
|
|
if (isRef(oldValue) && !isRef(value)) {
|
|
oldValue.value = value;
|
|
return true;
|
|
}
|
|
else {
|
|
return Reflect.set(target, key, value, receiver);
|
|
}
|
|
}
|
|
};
|
|
function proxyRefs(objectWithRefs) {
|
|
return isReactive(objectWithRefs)
|
|
? objectWithRefs
|
|
: new Proxy(objectWithRefs, shallowUnwrapHandlers);
|
|
}
|
|
class ObjectRefImpl {
|
|
constructor(_object, _key) {
|
|
this._object = _object;
|
|
this._key = _key;
|
|
this.__v_isRef = true;
|
|
}
|
|
get value() {
|
|
return this._object[this._key];
|
|
}
|
|
set value(newVal) {
|
|
this._object[this._key] = newVal;
|
|
}
|
|
}
|
|
function toRef(object, key) {
|
|
return isRef(object[key])
|
|
? object[key]
|
|
: new ObjectRefImpl(object, key);
|
|
}
|
|
|
|
class ComputedRefImpl {
|
|
constructor(getter, _setter, isReadonly) {
|
|
this._setter = _setter;
|
|
this._dirty = true;
|
|
this.__v_isRef = true;
|
|
this.effect = effect(getter, {
|
|
lazy: true,
|
|
scheduler: () => {
|
|
if (!this._dirty) {
|
|
this._dirty = true;
|
|
trigger(toRaw(this), "set" /* SET */, 'value');
|
|
}
|
|
}
|
|
});
|
|
this["__v_isReadonly" /* IS_READONLY */] = isReadonly;
|
|
}
|
|
get value() {
|
|
if (this._dirty) {
|
|
this._value = this.effect();
|
|
this._dirty = false;
|
|
}
|
|
track(toRaw(this), "get" /* GET */, 'value');
|
|
return this._value;
|
|
}
|
|
set value(newValue) {
|
|
this._setter(newValue);
|
|
}
|
|
}
|
|
function computed(getterOrOptions) {
|
|
let getter;
|
|
let setter;
|
|
if (isFunction(getterOrOptions)) {
|
|
getter = getterOrOptions;
|
|
setter = NOOP;
|
|
}
|
|
else {
|
|
getter = getterOrOptions.get;
|
|
setter = getterOrOptions.set;
|
|
}
|
|
return new ComputedRefImpl(getter, setter, isFunction(getterOrOptions) || !getterOrOptions.set);
|
|
}
|
|
|
|
const stack = [];
|
|
function warn(msg, ...args) {
|
|
// avoid props formatting or warn handler tracking deps that might be mutated
|
|
// during patch, leading to infinite recursion.
|
|
pauseTracking();
|
|
const instance = stack.length ? stack[stack.length - 1].component : null;
|
|
const appWarnHandler = instance && instance.appContext.config.warnHandler;
|
|
const trace = getComponentTrace();
|
|
if (appWarnHandler) {
|
|
callWithErrorHandling(appWarnHandler, instance, 11 /* APP_WARN_HANDLER */, [
|
|
msg + args.join(''),
|
|
instance && instance.proxy,
|
|
trace
|
|
.map(({ vnode }) => `at <${formatComponentName(instance, vnode.type)}>`)
|
|
.join('\n'),
|
|
trace
|
|
]);
|
|
}
|
|
else {
|
|
const warnArgs = [`[Vue warn]: ${msg}`, ...args];
|
|
/* istanbul ignore if */
|
|
if (trace.length &&
|
|
// avoid spamming console during tests
|
|
!false) {
|
|
warnArgs.push(`\n`, ...formatTrace(trace));
|
|
}
|
|
console.warn(...warnArgs);
|
|
}
|
|
resetTracking();
|
|
}
|
|
function getComponentTrace() {
|
|
let currentVNode = stack[stack.length - 1];
|
|
if (!currentVNode) {
|
|
return [];
|
|
}
|
|
// we can't just use the stack because it will be incomplete during updates
|
|
// that did not start from the root. Re-construct the parent chain using
|
|
// instance parent pointers.
|
|
const normalizedStack = [];
|
|
while (currentVNode) {
|
|
const last = normalizedStack[0];
|
|
if (last && last.vnode === currentVNode) {
|
|
last.recurseCount++;
|
|
}
|
|
else {
|
|
normalizedStack.push({
|
|
vnode: currentVNode,
|
|
recurseCount: 0
|
|
});
|
|
}
|
|
const parentInstance = currentVNode.component && currentVNode.component.parent;
|
|
currentVNode = parentInstance && parentInstance.vnode;
|
|
}
|
|
return normalizedStack;
|
|
}
|
|
/* istanbul ignore next */
|
|
function formatTrace(trace) {
|
|
const logs = [];
|
|
trace.forEach((entry, i) => {
|
|
logs.push(...(i === 0 ? [] : [`\n`]), ...formatTraceEntry(entry));
|
|
});
|
|
return logs;
|
|
}
|
|
function formatTraceEntry({ vnode, recurseCount }) {
|
|
const postfix = recurseCount > 0 ? `... (${recurseCount} recursive calls)` : ``;
|
|
const isRoot = vnode.component ? vnode.component.parent == null : false;
|
|
const open = ` at <${formatComponentName(vnode.component, vnode.type, isRoot)}`;
|
|
const close = `>` + postfix;
|
|
return vnode.props
|
|
? [open, ...formatProps(vnode.props), close]
|
|
: [open + close];
|
|
}
|
|
/* istanbul ignore next */
|
|
function formatProps(props) {
|
|
const res = [];
|
|
const keys = Object.keys(props);
|
|
keys.slice(0, 3).forEach(key => {
|
|
res.push(...formatProp(key, props[key]));
|
|
});
|
|
if (keys.length > 3) {
|
|
res.push(` ...`);
|
|
}
|
|
return res;
|
|
}
|
|
/* istanbul ignore next */
|
|
function formatProp(key, value, raw) {
|
|
if (isString(value)) {
|
|
value = JSON.stringify(value);
|
|
return raw ? value : [`${key}=${value}`];
|
|
}
|
|
else if (typeof value === 'number' ||
|
|
typeof value === 'boolean' ||
|
|
value == null) {
|
|
return raw ? value : [`${key}=${value}`];
|
|
}
|
|
else if (isRef(value)) {
|
|
value = formatProp(key, toRaw(value.value), true);
|
|
return raw ? value : [`${key}=Ref<`, value, `>`];
|
|
}
|
|
else if (isFunction(value)) {
|
|
return [`${key}=fn${value.name ? `<${value.name}>` : ``}`];
|
|
}
|
|
else {
|
|
value = toRaw(value);
|
|
return raw ? value : [`${key}=`, value];
|
|
}
|
|
}
|
|
function callWithErrorHandling(fn, instance, type, args) {
|
|
let res;
|
|
try {
|
|
res = args ? fn(...args) : fn();
|
|
}
|
|
catch (err) {
|
|
handleError(err, instance, type);
|
|
}
|
|
return res;
|
|
}
|
|
function callWithAsyncErrorHandling(fn, instance, type, args) {
|
|
if (isFunction(fn)) {
|
|
const res = callWithErrorHandling(fn, instance, type, args);
|
|
if (res && isPromise(res)) {
|
|
res.catch(err => {
|
|
handleError(err, instance, type);
|
|
});
|
|
}
|
|
return res;
|
|
}
|
|
const values = [];
|
|
for (let i = 0; i < fn.length; i++) {
|
|
values.push(callWithAsyncErrorHandling(fn[i], instance, type, args));
|
|
}
|
|
return values;
|
|
}
|
|
function handleError(err, instance, type, throwInDev = true) {
|
|
const contextVNode = instance ? instance.vnode : null;
|
|
if (instance) {
|
|
let cur = instance.parent;
|
|
// the exposed instance is the render proxy to keep it consistent with 2.x
|
|
const exposedInstance = instance.proxy;
|
|
// in production the hook receives only the error code
|
|
const errorInfo = type;
|
|
while (cur) {
|
|
const errorCapturedHooks = cur.ec;
|
|
if (errorCapturedHooks) {
|
|
for (let i = 0; i < errorCapturedHooks.length; i++) {
|
|
if (errorCapturedHooks[i](err, exposedInstance, errorInfo) === false) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
cur = cur.parent;
|
|
}
|
|
// app-level handling
|
|
const appErrorHandler = instance.appContext.config.errorHandler;
|
|
if (appErrorHandler) {
|
|
callWithErrorHandling(appErrorHandler, null, 10 /* APP_ERROR_HANDLER */, [err, exposedInstance, errorInfo]);
|
|
return;
|
|
}
|
|
}
|
|
logError(err, type, contextVNode, throwInDev);
|
|
}
|
|
function logError(err, type, contextVNode, throwInDev = true) {
|
|
{
|
|
// recover in prod to reduce the impact on end-user
|
|
console.error(err);
|
|
}
|
|
}
|
|
|
|
let isFlushing = false;
|
|
let isFlushPending = false;
|
|
const queue = [];
|
|
let flushIndex = 0;
|
|
const pendingPreFlushCbs = [];
|
|
let activePreFlushCbs = null;
|
|
let preFlushIndex = 0;
|
|
const pendingPostFlushCbs = [];
|
|
let activePostFlushCbs = null;
|
|
let postFlushIndex = 0;
|
|
const resolvedPromise = Promise.resolve();
|
|
let currentFlushPromise = null;
|
|
let currentPreFlushParentJob = null;
|
|
const RECURSION_LIMIT = 100;
|
|
function nextTick(fn) {
|
|
const p = currentFlushPromise || resolvedPromise;
|
|
return fn ? p.then(this ? fn.bind(this) : fn) : p;
|
|
}
|
|
function queueJob(job) {
|
|
// the dedupe search uses the startIndex argument of Array.includes()
|
|
// by default the search index includes the current job that is being run
|
|
// so it cannot recursively trigger itself again.
|
|
// if the job is a watch() callback, the search will start with a +1 index to
|
|
// allow it recursively trigger itself - it is the user's responsibility to
|
|
// ensure it doesn't end up in an infinite loop.
|
|
if ((!queue.length ||
|
|
!queue.includes(job, isFlushing && job.allowRecurse ? flushIndex + 1 : flushIndex)) &&
|
|
job !== currentPreFlushParentJob) {
|
|
queue.push(job);
|
|
queueFlush();
|
|
}
|
|
}
|
|
function queueFlush() {
|
|
if (!isFlushing && !isFlushPending) {
|
|
isFlushPending = true;
|
|
currentFlushPromise = resolvedPromise.then(flushJobs);
|
|
}
|
|
}
|
|
function invalidateJob(job) {
|
|
const i = queue.indexOf(job);
|
|
if (i > -1) {
|
|
queue.splice(i, 1);
|
|
}
|
|
}
|
|
function queueCb(cb, activeQueue, pendingQueue, index) {
|
|
if (!isArray(cb)) {
|
|
if (!activeQueue ||
|
|
!activeQueue.includes(cb, cb.allowRecurse ? index + 1 : index)) {
|
|
pendingQueue.push(cb);
|
|
}
|
|
}
|
|
else {
|
|
// if cb is an array, it is a component lifecycle hook which can only be
|
|
// triggered by a job, which is already deduped in the main queue, so
|
|
// we can skip duplicate check here to improve perf
|
|
pendingQueue.push(...cb);
|
|
}
|
|
queueFlush();
|
|
}
|
|
function queuePreFlushCb(cb) {
|
|
queueCb(cb, activePreFlushCbs, pendingPreFlushCbs, preFlushIndex);
|
|
}
|
|
function queuePostFlushCb(cb) {
|
|
queueCb(cb, activePostFlushCbs, pendingPostFlushCbs, postFlushIndex);
|
|
}
|
|
function flushPreFlushCbs(seen, parentJob = null) {
|
|
if (pendingPreFlushCbs.length) {
|
|
currentPreFlushParentJob = parentJob;
|
|
activePreFlushCbs = [...new Set(pendingPreFlushCbs)];
|
|
pendingPreFlushCbs.length = 0;
|
|
for (preFlushIndex = 0; preFlushIndex < activePreFlushCbs.length; preFlushIndex++) {
|
|
activePreFlushCbs[preFlushIndex]();
|
|
}
|
|
activePreFlushCbs = null;
|
|
preFlushIndex = 0;
|
|
currentPreFlushParentJob = null;
|
|
// recursively flush until it drains
|
|
flushPreFlushCbs(seen, parentJob);
|
|
}
|
|
}
|
|
function flushPostFlushCbs(seen) {
|
|
if (pendingPostFlushCbs.length) {
|
|
const deduped = [...new Set(pendingPostFlushCbs)];
|
|
pendingPostFlushCbs.length = 0;
|
|
// #1947 already has active queue, nested flushPostFlushCbs call
|
|
if (activePostFlushCbs) {
|
|
activePostFlushCbs.push(...deduped);
|
|
return;
|
|
}
|
|
activePostFlushCbs = deduped;
|
|
activePostFlushCbs.sort((a, b) => getId(a) - getId(b));
|
|
for (postFlushIndex = 0; postFlushIndex < activePostFlushCbs.length; postFlushIndex++) {
|
|
activePostFlushCbs[postFlushIndex]();
|
|
}
|
|
activePostFlushCbs = null;
|
|
postFlushIndex = 0;
|
|
}
|
|
}
|
|
const getId = (job) => job.id == null ? Infinity : job.id;
|
|
function flushJobs(seen) {
|
|
isFlushPending = false;
|
|
isFlushing = true;
|
|
flushPreFlushCbs(seen);
|
|
// Sort queue before flush.
|
|
// This ensures that:
|
|
// 1. Components are updated from parent to child. (because parent is always
|
|
// created before the child so its render effect will have smaller
|
|
// priority number)
|
|
// 2. If a component is unmounted during a parent component's update,
|
|
// its update can be skipped.
|
|
queue.sort((a, b) => getId(a) - getId(b));
|
|
try {
|
|
for (flushIndex = 0; flushIndex < queue.length; flushIndex++) {
|
|
const job = queue[flushIndex];
|
|
if (job) {
|
|
if (("production" !== 'production')) ;
|
|
callWithErrorHandling(job, null, 14 /* SCHEDULER */);
|
|
}
|
|
}
|
|
}
|
|
finally {
|
|
flushIndex = 0;
|
|
queue.length = 0;
|
|
flushPostFlushCbs();
|
|
isFlushing = false;
|
|
currentFlushPromise = null;
|
|
// some postFlushCb queued jobs!
|
|
// keep flushing until it drains.
|
|
if (queue.length || pendingPostFlushCbs.length) {
|
|
flushJobs(seen);
|
|
}
|
|
}
|
|
}
|
|
function checkRecursiveUpdates(seen, fn) {
|
|
if (!seen.has(fn)) {
|
|
seen.set(fn, 1);
|
|
}
|
|
else {
|
|
const count = seen.get(fn);
|
|
if (count > RECURSION_LIMIT) {
|
|
throw new Error(`Maximum recursive updates exceeded. ` +
|
|
`This means you have a reactive effect that is mutating its own ` +
|
|
`dependencies and thus recursively triggering itself. Possible sources ` +
|
|
`include component template, render function, updated hook or ` +
|
|
`watcher source function.`);
|
|
}
|
|
else {
|
|
seen.set(fn, count + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
function emit(instance, event, ...rawArgs) {
|
|
const props = instance.vnode.props || EMPTY_OBJ;
|
|
let args = rawArgs;
|
|
const isModelListener = event.startsWith('update:');
|
|
// for v-model update:xxx events, apply modifiers on args
|
|
const modelArg = isModelListener && event.slice(7);
|
|
if (modelArg && modelArg in props) {
|
|
const modifiersKey = `${modelArg === 'modelValue' ? 'model' : modelArg}Modifiers`;
|
|
const { number, trim } = props[modifiersKey] || EMPTY_OBJ;
|
|
if (trim) {
|
|
args = rawArgs.map(a => a.trim());
|
|
}
|
|
else if (number) {
|
|
args = rawArgs.map(toNumber);
|
|
}
|
|
}
|
|
// convert handler name to camelCase. See issue #2249
|
|
let handlerName = toHandlerKey(camelize(event));
|
|
let handler = props[handlerName];
|
|
// for v-model update:xxx events, also trigger kebab-case equivalent
|
|
// for props passed via kebab-case
|
|
if (!handler && isModelListener) {
|
|
handlerName = toHandlerKey(hyphenate(event));
|
|
handler = props[handlerName];
|
|
}
|
|
if (handler) {
|
|
callWithAsyncErrorHandling(handler, instance, 6 /* COMPONENT_EVENT_HANDLER */, args);
|
|
}
|
|
const onceHandler = props[handlerName + `Once`];
|
|
if (onceHandler) {
|
|
if (!instance.emitted) {
|
|
(instance.emitted = {})[handlerName] = true;
|
|
}
|
|
else if (instance.emitted[handlerName]) {
|
|
return;
|
|
}
|
|
callWithAsyncErrorHandling(onceHandler, instance, 6 /* COMPONENT_EVENT_HANDLER */, args);
|
|
}
|
|
}
|
|
function normalizeEmitsOptions(comp, appContext, asMixin = false) {
|
|
if (!appContext.deopt && comp.__emits !== undefined) {
|
|
return comp.__emits;
|
|
}
|
|
const raw = comp.emits;
|
|
let normalized = {};
|
|
// apply mixin/extends props
|
|
let hasExtends = false;
|
|
if (!isFunction(comp)) {
|
|
const extendEmits = (raw) => {
|
|
hasExtends = true;
|
|
extend(normalized, normalizeEmitsOptions(raw, appContext, true));
|
|
};
|
|
if (!asMixin && appContext.mixins.length) {
|
|
appContext.mixins.forEach(extendEmits);
|
|
}
|
|
if (comp.extends) {
|
|
extendEmits(comp.extends);
|
|
}
|
|
if (comp.mixins) {
|
|
comp.mixins.forEach(extendEmits);
|
|
}
|
|
}
|
|
if (!raw && !hasExtends) {
|
|
return (comp.__emits = null);
|
|
}
|
|
if (isArray(raw)) {
|
|
raw.forEach(key => (normalized[key] = null));
|
|
}
|
|
else {
|
|
extend(normalized, raw);
|
|
}
|
|
return (comp.__emits = normalized);
|
|
}
|
|
// Check if an incoming prop key is a declared emit event listener.
|
|
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
|
// both considered matched listeners.
|
|
function isEmitListener(options, key) {
|
|
if (!options || !isOn(key)) {
|
|
return false;
|
|
}
|
|
key = key.slice(2).replace(/Once$/, '');
|
|
return (hasOwn(options, key[0].toLowerCase() + key.slice(1)) ||
|
|
hasOwn(options, hyphenate(key)) ||
|
|
hasOwn(options, key));
|
|
}
|
|
|
|
/**
|
|
* mark the current rendering instance for asset resolution (e.g.
|
|
* resolveComponent, resolveDirective) during render
|
|
*/
|
|
let currentRenderingInstance = null;
|
|
function setCurrentRenderingInstance(instance) {
|
|
currentRenderingInstance = instance;
|
|
}
|
|
/**
|
|
* dev only flag to track whether $attrs was used during render.
|
|
* If $attrs was used during render then the warning for failed attrs
|
|
* fallthrough can be suppressed.
|
|
*/
|
|
let accessedAttrs = false;
|
|
function markAttrsAccessed() {
|
|
accessedAttrs = true;
|
|
}
|
|
function renderComponentRoot(instance) {
|
|
const { type: Component, vnode, proxy, withProxy, props, propsOptions: [propsOptions], slots, attrs, emit, render, renderCache, data, setupState, ctx } = instance;
|
|
let result;
|
|
currentRenderingInstance = instance;
|
|
try {
|
|
let fallthroughAttrs;
|
|
if (vnode.shapeFlag & 4 /* STATEFUL_COMPONENT */) {
|
|
// withProxy is a proxy with a different `has` trap only for
|
|
// runtime-compiled render functions using `with` block.
|
|
const proxyToUse = withProxy || proxy;
|
|
result = normalizeVNode(render.call(proxyToUse, proxyToUse, renderCache, props, setupState, data, ctx));
|
|
fallthroughAttrs = attrs;
|
|
}
|
|
else {
|
|
// functional
|
|
const render = Component;
|
|
// in dev, mark attrs accessed if optional props (attrs === props)
|
|
if (("production" !== 'production') && attrs === props) ;
|
|
result = normalizeVNode(render.length > 1
|
|
? render(props, ("production" !== 'production')
|
|
? {
|
|
get attrs() {
|
|
markAttrsAccessed();
|
|
return attrs;
|
|
},
|
|
slots,
|
|
emit
|
|
}
|
|
: { attrs, slots, emit })
|
|
: render(props, null /* we know it doesn't need it */));
|
|
fallthroughAttrs = Component.props
|
|
? attrs
|
|
: getFunctionalFallthrough(attrs);
|
|
}
|
|
// attr merging
|
|
// in dev mode, comments are preserved, and it's possible for a template
|
|
// to have comments along side the root element which makes it a fragment
|
|
let root = result;
|
|
let setRoot = undefined;
|
|
if (("production" !== 'production') && result.patchFlag & 2048 /* DEV_ROOT_FRAGMENT */) ;
|
|
if (Component.inheritAttrs !== false && fallthroughAttrs) {
|
|
const keys = Object.keys(fallthroughAttrs);
|
|
const { shapeFlag } = root;
|
|
if (keys.length) {
|
|
if (shapeFlag & 1 /* ELEMENT */ ||
|
|
shapeFlag & 6 /* COMPONENT */) {
|
|
if (propsOptions && keys.some(isModelListener)) {
|
|
// If a v-model listener (onUpdate:xxx) has a corresponding declared
|
|
// prop, it indicates this component expects to handle v-model and
|
|
// it should not fallthrough.
|
|
// related: #1543, #1643, #1989
|
|
fallthroughAttrs = filterModelListeners(fallthroughAttrs, propsOptions);
|
|
}
|
|
root = cloneVNode(root, fallthroughAttrs);
|
|
}
|
|
else if (("production" !== 'production') && !accessedAttrs && root.type !== Comment) ;
|
|
}
|
|
}
|
|
// inherit directives
|
|
if (vnode.dirs) {
|
|
if (("production" !== 'production') && !isElementRoot(root)) ;
|
|
root.dirs = root.dirs ? root.dirs.concat(vnode.dirs) : vnode.dirs;
|
|
}
|
|
// inherit transition data
|
|
if (vnode.transition) {
|
|
if (("production" !== 'production') && !isElementRoot(root)) ;
|
|
root.transition = vnode.transition;
|
|
}
|
|
if (("production" !== 'production') && setRoot) ;
|
|
else {
|
|
result = root;
|
|
}
|
|
}
|
|
catch (err) {
|
|
handleError(err, instance, 1 /* RENDER_FUNCTION */);
|
|
result = createVNode(Comment);
|
|
}
|
|
currentRenderingInstance = null;
|
|
return result;
|
|
}
|
|
/**
|
|
* dev only
|
|
* In dev mode, template root level comments are rendered, which turns the
|
|
* template into a fragment root, but we need to locate the single element
|
|
* root for attrs and scope id processing.
|
|
*/
|
|
const getChildRoot = (vnode) => {
|
|
const rawChildren = vnode.children;
|
|
const dynamicChildren = vnode.dynamicChildren;
|
|
const childRoot = filterSingleRoot(rawChildren);
|
|
if (!childRoot) {
|
|
return [vnode, undefined];
|
|
}
|
|
const index = rawChildren.indexOf(childRoot);
|
|
const dynamicIndex = dynamicChildren ? dynamicChildren.indexOf(childRoot) : -1;
|
|
const setRoot = (updatedRoot) => {
|
|
rawChildren[index] = updatedRoot;
|
|
if (dynamicChildren) {
|
|
if (dynamicIndex > -1) {
|
|
dynamicChildren[dynamicIndex] = updatedRoot;
|
|
}
|
|
else if (updatedRoot.patchFlag > 0) {
|
|
vnode.dynamicChildren = [...dynamicChildren, updatedRoot];
|
|
}
|
|
}
|
|
};
|
|
return [normalizeVNode(childRoot), setRoot];
|
|
};
|
|
function filterSingleRoot(children) {
|
|
let singleRoot;
|
|
for (let i = 0; i < children.length; i++) {
|
|
const child = children[i];
|
|
if (isVNode(child)) {
|
|
// ignore user comment
|
|
if (child.type !== Comment || child.children === 'v-if') {
|
|
if (singleRoot) {
|
|
// has more than 1 non-comment child, return now
|
|
return;
|
|
}
|
|
else {
|
|
singleRoot = child;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
}
|
|
return singleRoot;
|
|
}
|
|
const getFunctionalFallthrough = (attrs) => {
|
|
let res;
|
|
for (const key in attrs) {
|
|
if (key === 'class' || key === 'style' || isOn(key)) {
|
|
(res || (res = {}))[key] = attrs[key];
|
|
}
|
|
}
|
|
return res;
|
|
};
|
|
const filterModelListeners = (attrs, props) => {
|
|
const res = {};
|
|
for (const key in attrs) {
|
|
if (!isModelListener(key) || !(key.slice(9) in props)) {
|
|
res[key] = attrs[key];
|
|
}
|
|
}
|
|
return res;
|
|
};
|
|
const isElementRoot = (vnode) => {
|
|
return (vnode.shapeFlag & 6 /* COMPONENT */ ||
|
|
vnode.shapeFlag & 1 /* ELEMENT */ ||
|
|
vnode.type === Comment // potential v-if branch switch
|
|
);
|
|
};
|
|
function shouldUpdateComponent(prevVNode, nextVNode, optimized) {
|
|
const { props: prevProps, children: prevChildren, component } = prevVNode;
|
|
const { props: nextProps, children: nextChildren, patchFlag } = nextVNode;
|
|
const emits = component.emitsOptions;
|
|
// force child update for runtime directive or transition on component vnode.
|
|
if (nextVNode.dirs || nextVNode.transition) {
|
|
return true;
|
|
}
|
|
if (optimized && patchFlag >= 0) {
|
|
if (patchFlag & 1024 /* DYNAMIC_SLOTS */) {
|
|
// slot content that references values that might have changed,
|
|
// e.g. in a v-for
|
|
return true;
|
|
}
|
|
if (patchFlag & 16 /* FULL_PROPS */) {
|
|
if (!prevProps) {
|
|
return !!nextProps;
|
|
}
|
|
// presence of this flag indicates props are always non-null
|
|
return hasPropsChanged(prevProps, nextProps, emits);
|
|
}
|
|
else if (patchFlag & 8 /* PROPS */) {
|
|
const dynamicProps = nextVNode.dynamicProps;
|
|
for (let i = 0; i < dynamicProps.length; i++) {
|
|
const key = dynamicProps[i];
|
|
if (nextProps[key] !== prevProps[key] &&
|
|
!isEmitListener(emits, key)) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// this path is only taken by manually written render functions
|
|
// so presence of any children leads to a forced update
|
|
if (prevChildren || nextChildren) {
|
|
if (!nextChildren || !nextChildren.$stable) {
|
|
return true;
|
|
}
|
|
}
|
|
if (prevProps === nextProps) {
|
|
return false;
|
|
}
|
|
if (!prevProps) {
|
|
return !!nextProps;
|
|
}
|
|
if (!nextProps) {
|
|
return true;
|
|
}
|
|
return hasPropsChanged(prevProps, nextProps, emits);
|
|
}
|
|
return false;
|
|
}
|
|
function hasPropsChanged(prevProps, nextProps, emitsOptions) {
|
|
const nextKeys = Object.keys(nextProps);
|
|
if (nextKeys.length !== Object.keys(prevProps).length) {
|
|
return true;
|
|
}
|
|
for (let i = 0; i < nextKeys.length; i++) {
|
|
const key = nextKeys[i];
|
|
if (nextProps[key] !== prevProps[key] &&
|
|
!isEmitListener(emitsOptions, key)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
function updateHOCHostEl({ vnode, parent }, el // HostNode
|
|
) {
|
|
while (parent && parent.subTree === vnode) {
|
|
(vnode = parent.vnode).el = el;
|
|
parent = parent.parent;
|
|
}
|
|
}
|
|
|
|
const isSuspense = (type) => type.__isSuspense;
|
|
function normalizeSuspenseChildren(vnode) {
|
|
const { shapeFlag, children } = vnode;
|
|
let content;
|
|
let fallback;
|
|
if (shapeFlag & 32 /* SLOTS_CHILDREN */) {
|
|
content = normalizeSuspenseSlot(children.default);
|
|
fallback = normalizeSuspenseSlot(children.fallback);
|
|
}
|
|
else {
|
|
content = normalizeSuspenseSlot(children);
|
|
fallback = normalizeVNode(null);
|
|
}
|
|
return {
|
|
content,
|
|
fallback
|
|
};
|
|
}
|
|
function normalizeSuspenseSlot(s) {
|
|
if (isFunction(s)) {
|
|
s = s();
|
|
}
|
|
if (isArray(s)) {
|
|
const singleChild = filterSingleRoot(s);
|
|
s = singleChild;
|
|
}
|
|
return normalizeVNode(s);
|
|
}
|
|
function queueEffectWithSuspense(fn, suspense) {
|
|
if (suspense && suspense.pendingBranch) {
|
|
if (isArray(fn)) {
|
|
suspense.effects.push(...fn);
|
|
}
|
|
else {
|
|
suspense.effects.push(fn);
|
|
}
|
|
}
|
|
else {
|
|
queuePostFlushCb(fn);
|
|
}
|
|
}
|
|
|
|
let isRenderingCompiledSlot = 0;
|
|
const setCompiledSlotRendering = (n) => (isRenderingCompiledSlot += n);
|
|
|
|
/**
|
|
* Wrap a slot function to memoize current rendering instance
|
|
* @private
|
|
*/
|
|
function withCtx(fn, ctx = currentRenderingInstance) {
|
|
if (!ctx)
|
|
return fn;
|
|
const renderFnWithContext = (...args) => {
|
|
// If a user calls a compiled slot inside a template expression (#1745), it
|
|
// can mess up block tracking, so by default we need to push a null block to
|
|
// avoid that. This isn't necessary if rendering a compiled `<slot>`.
|
|
if (!isRenderingCompiledSlot) {
|
|
openBlock(true /* null block that disables tracking */);
|
|
}
|
|
const owner = currentRenderingInstance;
|
|
setCurrentRenderingInstance(ctx);
|
|
const res = fn(...args);
|
|
setCurrentRenderingInstance(owner);
|
|
if (!isRenderingCompiledSlot) {
|
|
closeBlock();
|
|
}
|
|
return res;
|
|
};
|
|
renderFnWithContext._c = true;
|
|
return renderFnWithContext;
|
|
}
|
|
|
|
// SFC scoped style ID management.
|
|
let currentScopeId = null;
|
|
const scopeIdStack = [];
|
|
/**
|
|
* @private
|
|
*/
|
|
function pushScopeId(id) {
|
|
scopeIdStack.push((currentScopeId = id));
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
function popScopeId() {
|
|
scopeIdStack.pop();
|
|
currentScopeId = scopeIdStack[scopeIdStack.length - 1] || null;
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
function withScopeId(id) {
|
|
return ((fn) => withCtx(function () {
|
|
pushScopeId(id);
|
|
const res = fn.apply(this, arguments);
|
|
popScopeId();
|
|
return res;
|
|
}));
|
|
}
|
|
|
|
function initProps(instance, rawProps, isStateful, // result of bitwise flag comparison
|
|
isSSR = false) {
|
|
const props = {};
|
|
const attrs = {};
|
|
def(attrs, InternalObjectKey, 1);
|
|
setFullProps(instance, rawProps, props, attrs);
|
|
if (isStateful) {
|
|
// stateful
|
|
instance.props = isSSR ? props : shallowReactive(props);
|
|
}
|
|
else {
|
|
if (!instance.type.props) {
|
|
// functional w/ optional props, props === attrs
|
|
instance.props = attrs;
|
|
}
|
|
else {
|
|
// functional w/ declared props
|
|
instance.props = props;
|
|
}
|
|
}
|
|
instance.attrs = attrs;
|
|
}
|
|
function updateProps(instance, rawProps, rawPrevProps, optimized) {
|
|
const { props, attrs, vnode: { patchFlag } } = instance;
|
|
const rawCurrentProps = toRaw(props);
|
|
const [options] = instance.propsOptions;
|
|
if (
|
|
// always force full diff in dev
|
|
// - #1942 if hmr is enabled with sfc component
|
|
// - vite#872 non-sfc component used by sfc component
|
|
(optimized || patchFlag > 0) &&
|
|
!(patchFlag & 16 /* FULL_PROPS */)) {
|
|
if (patchFlag & 8 /* PROPS */) {
|
|
// Compiler-generated props & no keys change, just set the updated
|
|
// the props.
|
|
const propsToUpdate = instance.vnode.dynamicProps;
|
|
for (let i = 0; i < propsToUpdate.length; i++) {
|
|
const key = propsToUpdate[i];
|
|
// PROPS flag guarantees rawProps to be non-null
|
|
const value = rawProps[key];
|
|
if (options) {
|
|
// attr / props separation was done on init and will be consistent
|
|
// in this code path, so just check if attrs have it.
|
|
if (hasOwn(attrs, key)) {
|
|
attrs[key] = value;
|
|
}
|
|
else {
|
|
const camelizedKey = camelize(key);
|
|
props[camelizedKey] = resolvePropValue(options, rawCurrentProps, camelizedKey, value, instance);
|
|
}
|
|
}
|
|
else {
|
|
attrs[key] = value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// full props update.
|
|
setFullProps(instance, rawProps, props, attrs);
|
|
// in case of dynamic props, check if we need to delete keys from
|
|
// the props object
|
|
let kebabKey;
|
|
for (const key in rawCurrentProps) {
|
|
if (!rawProps ||
|
|
// for camelCase
|
|
(!hasOwn(rawProps, key) &&
|
|
// it's possible the original props was passed in as kebab-case
|
|
// and converted to camelCase (#955)
|
|
((kebabKey = hyphenate(key)) === key || !hasOwn(rawProps, kebabKey)))) {
|
|
if (options) {
|
|
if (rawPrevProps &&
|
|
// for camelCase
|
|
(rawPrevProps[key] !== undefined ||
|
|
// for kebab-case
|
|
rawPrevProps[kebabKey] !== undefined)) {
|
|
props[key] = resolvePropValue(options, rawProps || EMPTY_OBJ, key, undefined, instance);
|
|
}
|
|
}
|
|
else {
|
|
delete props[key];
|
|
}
|
|
}
|
|
}
|
|
// in the case of functional component w/o props declaration, props and
|
|
// attrs point to the same object so it should already have been updated.
|
|
if (attrs !== rawCurrentProps) {
|
|
for (const key in attrs) {
|
|
if (!rawProps || !hasOwn(rawProps, key)) {
|
|
delete attrs[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// trigger updates for $attrs in case it's used in component slots
|
|
trigger(instance, "set" /* SET */, '$attrs');
|
|
}
|
|
function setFullProps(instance, rawProps, props, attrs) {
|
|
const [options, needCastKeys] = instance.propsOptions;
|
|
if (rawProps) {
|
|
for (const key in rawProps) {
|
|
const value = rawProps[key];
|
|
// key, ref are reserved and never passed down
|
|
if (isReservedProp(key)) {
|
|
continue;
|
|
}
|
|
// prop option names are camelized during normalization, so to support
|
|
// kebab -> camel conversion here we need to camelize the key.
|
|
let camelKey;
|
|
if (options && hasOwn(options, (camelKey = camelize(key)))) {
|
|
props[camelKey] = value;
|
|
}
|
|
else if (!isEmitListener(instance.emitsOptions, key)) {
|
|
// Any non-declared (either as a prop or an emitted event) props are put
|
|
// into a separate `attrs` object for spreading. Make sure to preserve
|
|
// original key casing
|
|
attrs[key] = value;
|
|
}
|
|
}
|
|
}
|
|
if (needCastKeys) {
|
|
const rawCurrentProps = toRaw(props);
|
|
for (let i = 0; i < needCastKeys.length; i++) {
|
|
const key = needCastKeys[i];
|
|
props[key] = resolvePropValue(options, rawCurrentProps, key, rawCurrentProps[key], instance);
|
|
}
|
|
}
|
|
}
|
|
function resolvePropValue(options, props, key, value, instance) {
|
|
const opt = options[key];
|
|
if (opt != null) {
|
|
const hasDefault = hasOwn(opt, 'default');
|
|
// default values
|
|
if (hasDefault && value === undefined) {
|
|
const defaultValue = opt.default;
|
|
if (opt.type !== Function && isFunction(defaultValue)) {
|
|
setCurrentInstance(instance);
|
|
value = defaultValue(props);
|
|
setCurrentInstance(null);
|
|
}
|
|
else {
|
|
value = defaultValue;
|
|
}
|
|
}
|
|
// boolean casting
|
|
if (opt[0 /* shouldCast */]) {
|
|
if (!hasOwn(props, key) && !hasDefault) {
|
|
value = false;
|
|
}
|
|
else if (opt[1 /* shouldCastTrue */] &&
|
|
(value === '' || value === hyphenate(key))) {
|
|
value = true;
|
|
}
|
|
}
|
|
}
|
|
return value;
|
|
}
|
|
function normalizePropsOptions(comp, appContext, asMixin = false) {
|
|
if (!appContext.deopt && comp.__props) {
|
|
return comp.__props;
|
|
}
|
|
const raw = comp.props;
|
|
const normalized = {};
|
|
const needCastKeys = [];
|
|
// apply mixin/extends props
|
|
let hasExtends = false;
|
|
if (!isFunction(comp)) {
|
|
const extendProps = (raw) => {
|
|
hasExtends = true;
|
|
const [props, keys] = normalizePropsOptions(raw, appContext, true);
|
|
extend(normalized, props);
|
|
if (keys)
|
|
needCastKeys.push(...keys);
|
|
};
|
|
if (!asMixin && appContext.mixins.length) {
|
|
appContext.mixins.forEach(extendProps);
|
|
}
|
|
if (comp.extends) {
|
|
extendProps(comp.extends);
|
|
}
|
|
if (comp.mixins) {
|
|
comp.mixins.forEach(extendProps);
|
|
}
|
|
}
|
|
if (!raw && !hasExtends) {
|
|
return (comp.__props = EMPTY_ARR);
|
|
}
|
|
if (isArray(raw)) {
|
|
for (let i = 0; i < raw.length; i++) {
|
|
const normalizedKey = camelize(raw[i]);
|
|
if (validatePropName(normalizedKey)) {
|
|
normalized[normalizedKey] = EMPTY_OBJ;
|
|
}
|
|
}
|
|
}
|
|
else if (raw) {
|
|
for (const key in raw) {
|
|
const normalizedKey = camelize(key);
|
|
if (validatePropName(normalizedKey)) {
|
|
const opt = raw[key];
|
|
const prop = (normalized[normalizedKey] =
|
|
isArray(opt) || isFunction(opt) ? { type: opt } : opt);
|
|
if (prop) {
|
|
const booleanIndex = getTypeIndex(Boolean, prop.type);
|
|
const stringIndex = getTypeIndex(String, prop.type);
|
|
prop[0 /* shouldCast */] = booleanIndex > -1;
|
|
prop[1 /* shouldCastTrue */] =
|
|
stringIndex < 0 || booleanIndex < stringIndex;
|
|
// if the prop needs boolean casting or default value
|
|
if (booleanIndex > -1 || hasOwn(prop, 'default')) {
|
|
needCastKeys.push(normalizedKey);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return (comp.__props = [normalized, needCastKeys]);
|
|
}
|
|
function validatePropName(key) {
|
|
if (key[0] !== '$') {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
// use function string name to check type constructors
|
|
// so that it works across vms / iframes.
|
|
function getType(ctor) {
|
|
const match = ctor && ctor.toString().match(/^\s*function (\w+)/);
|
|
return match ? match[1] : '';
|
|
}
|
|
function isSameType(a, b) {
|
|
return getType(a) === getType(b);
|
|
}
|
|
function getTypeIndex(type, expectedTypes) {
|
|
if (isArray(expectedTypes)) {
|
|
for (let i = 0, len = expectedTypes.length; i < len; i++) {
|
|
if (isSameType(expectedTypes[i], type)) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
else if (isFunction(expectedTypes)) {
|
|
return isSameType(expectedTypes, type) ? 0 : -1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
function injectHook(type, hook, target = currentInstance, prepend = false) {
|
|
if (target) {
|
|
const hooks = target[type] || (target[type] = []);
|
|
// cache the error handling wrapper for injected hooks so the same hook
|
|
// can be properly deduped by the scheduler. "__weh" stands for "with error
|
|
// handling".
|
|
const wrappedHook = hook.__weh ||
|
|
(hook.__weh = (...args) => {
|
|
if (target.isUnmounted) {
|
|
return;
|
|
}
|
|
// disable tracking inside all lifecycle hooks
|
|
// since they can potentially be called inside effects.
|
|
pauseTracking();
|
|
// Set currentInstance during hook invocation.
|
|
// This assumes the hook does not synchronously trigger other hooks, which
|
|
// can only be false when the user does something really funky.
|
|
setCurrentInstance(target);
|
|
const res = callWithAsyncErrorHandling(hook, target, type, args);
|
|
setCurrentInstance(null);
|
|
resetTracking();
|
|
return res;
|
|
});
|
|
if (prepend) {
|
|
hooks.unshift(wrappedHook);
|
|
}
|
|
else {
|
|
hooks.push(wrappedHook);
|
|
}
|
|
return wrappedHook;
|
|
}
|
|
}
|
|
const createHook = (lifecycle) => (hook, target = currentInstance) =>
|
|
// post-create lifecycle registrations are noops during SSR
|
|
!isInSSRComponentSetup && injectHook(lifecycle, hook, target);
|
|
const onBeforeMount = createHook("bm" /* BEFORE_MOUNT */);
|
|
const onMounted = createHook("m" /* MOUNTED */);
|
|
const onBeforeUpdate = createHook("bu" /* BEFORE_UPDATE */);
|
|
const onUpdated = createHook("u" /* UPDATED */);
|
|
const onBeforeUnmount = createHook("bum" /* BEFORE_UNMOUNT */);
|
|
const onUnmounted = createHook("um" /* UNMOUNTED */);
|
|
const onRenderTriggered = createHook("rtg" /* RENDER_TRIGGERED */);
|
|
const onRenderTracked = createHook("rtc" /* RENDER_TRACKED */);
|
|
const onErrorCaptured = (hook, target = currentInstance) => {
|
|
injectHook("ec" /* ERROR_CAPTURED */, hook, target);
|
|
};
|
|
// initial value for watchers to trigger on undefined initial values
|
|
const INITIAL_WATCHER_VALUE = {};
|
|
// implementation
|
|
function watch(source, cb, options) {
|
|
return doWatch(source, cb, options);
|
|
}
|
|
function doWatch(source, cb, { immediate, deep, flush, onTrack, onTrigger } = EMPTY_OBJ, instance = currentInstance) {
|
|
let getter;
|
|
let forceTrigger = false;
|
|
if (isRef(source)) {
|
|
getter = () => source.value;
|
|
forceTrigger = !!source._shallow;
|
|
}
|
|
else if (isReactive(source)) {
|
|
getter = () => source;
|
|
deep = true;
|
|
}
|
|
else if (isArray(source)) {
|
|
getter = () => source.map(s => {
|
|
if (isRef(s)) {
|
|
return s.value;
|
|
}
|
|
else if (isReactive(s)) {
|
|
return traverse(s);
|
|
}
|
|
else if (isFunction(s)) {
|
|
return callWithErrorHandling(s, instance, 2 /* WATCH_GETTER */);
|
|
}
|
|
else ;
|
|
});
|
|
}
|
|
else if (isFunction(source)) {
|
|
if (cb) {
|
|
// getter with cb
|
|
getter = () => callWithErrorHandling(source, instance, 2 /* WATCH_GETTER */);
|
|
}
|
|
else {
|
|
// no cb -> simple effect
|
|
getter = () => {
|
|
if (instance && instance.isUnmounted) {
|
|
return;
|
|
}
|
|
if (cleanup) {
|
|
cleanup();
|
|
}
|
|
return callWithErrorHandling(source, instance, 3 /* WATCH_CALLBACK */, [onInvalidate]);
|
|
};
|
|
}
|
|
}
|
|
else {
|
|
getter = NOOP;
|
|
}
|
|
if (cb && deep) {
|
|
const baseGetter = getter;
|
|
getter = () => traverse(baseGetter());
|
|
}
|
|
let cleanup;
|
|
const onInvalidate = (fn) => {
|
|
cleanup = runner.options.onStop = () => {
|
|
callWithErrorHandling(fn, instance, 4 /* WATCH_CLEANUP */);
|
|
};
|
|
};
|
|
let oldValue = isArray(source) ? [] : INITIAL_WATCHER_VALUE;
|
|
const job = () => {
|
|
if (!runner.active) {
|
|
return;
|
|
}
|
|
if (cb) {
|
|
// watch(source, cb)
|
|
const newValue = runner();
|
|
if (deep || forceTrigger || hasChanged(newValue, oldValue)) {
|
|
// cleanup before running cb again
|
|
if (cleanup) {
|
|
cleanup();
|
|
}
|
|
callWithAsyncErrorHandling(cb, instance, 3 /* WATCH_CALLBACK */, [
|
|
newValue,
|
|
// pass undefined as the old value when it's changed for the first time
|
|
oldValue === INITIAL_WATCHER_VALUE ? undefined : oldValue,
|
|
onInvalidate
|
|
]);
|
|
oldValue = newValue;
|
|
}
|
|
}
|
|
else {
|
|
// watchEffect
|
|
runner();
|
|
}
|
|
};
|
|
// important: mark the job as a watcher callback so that scheduler knows
|
|
// it is allowed to self-trigger (#1727)
|
|
job.allowRecurse = !!cb;
|
|
let scheduler;
|
|
if (flush === 'sync') {
|
|
scheduler = job;
|
|
}
|
|
else if (flush === 'post') {
|
|
scheduler = () => queuePostRenderEffect(job, instance && instance.suspense);
|
|
}
|
|
else {
|
|
// default: 'pre'
|
|
scheduler = () => {
|
|
if (!instance || instance.isMounted) {
|
|
queuePreFlushCb(job);
|
|
}
|
|
else {
|
|
// with 'pre' option, the first call must happen before
|
|
// the component is mounted so it is called synchronously.
|
|
job();
|
|
}
|
|
};
|
|
}
|
|
const runner = effect(getter, {
|
|
lazy: true,
|
|
onTrack,
|
|
onTrigger,
|
|
scheduler
|
|
});
|
|
recordInstanceBoundEffect(runner, instance);
|
|
// initial run
|
|
if (cb) {
|
|
if (immediate) {
|
|
job();
|
|
}
|
|
else {
|
|
oldValue = runner();
|
|
}
|
|
}
|
|
else if (flush === 'post') {
|
|
queuePostRenderEffect(runner, instance && instance.suspense);
|
|
}
|
|
else {
|
|
runner();
|
|
}
|
|
return () => {
|
|
stop(runner);
|
|
if (instance) {
|
|
remove(instance.effects, runner);
|
|
}
|
|
};
|
|
}
|
|
// this.$watch
|
|
function instanceWatch(source, cb, options) {
|
|
const publicThis = this.proxy;
|
|
const getter = isString(source)
|
|
? () => publicThis[source]
|
|
: source.bind(publicThis);
|
|
return doWatch(getter, cb.bind(publicThis), options, this);
|
|
}
|
|
function traverse(value, seen = new Set()) {
|
|
if (!isObject(value) || seen.has(value)) {
|
|
return value;
|
|
}
|
|
seen.add(value);
|
|
if (isRef(value)) {
|
|
traverse(value.value, seen);
|
|
}
|
|
else if (isArray(value)) {
|
|
for (let i = 0; i < value.length; i++) {
|
|
traverse(value[i], seen);
|
|
}
|
|
}
|
|
else if (isSet(value) || isMap(value)) {
|
|
value.forEach((v) => {
|
|
traverse(v, seen);
|
|
});
|
|
}
|
|
else {
|
|
for (const key in value) {
|
|
traverse(value[key], seen);
|
|
}
|
|
}
|
|
return value;
|
|
}
|
|
|
|
const isKeepAlive = (vnode) => vnode.type.__isKeepAlive;
|
|
function onActivated(hook, target) {
|
|
registerKeepAliveHook(hook, "a" /* ACTIVATED */, target);
|
|
}
|
|
function onDeactivated(hook, target) {
|
|
registerKeepAliveHook(hook, "da" /* DEACTIVATED */, target);
|
|
}
|
|
function registerKeepAliveHook(hook, type, target = currentInstance) {
|
|
// cache the deactivate branch check wrapper for injected hooks so the same
|
|
// hook can be properly deduped by the scheduler. "__wdc" stands for "with
|
|
// deactivation check".
|
|
const wrappedHook = hook.__wdc ||
|
|
(hook.__wdc = () => {
|
|
// only fire the hook if the target instance is NOT in a deactivated branch.
|
|
let current = target;
|
|
while (current) {
|
|
if (current.isDeactivated) {
|
|
return;
|
|
}
|
|
current = current.parent;
|
|
}
|
|
hook();
|
|
});
|
|
injectHook(type, wrappedHook, target);
|
|
// In addition to registering it on the target instance, we walk up the parent
|
|
// chain and register it on all ancestor instances that are keep-alive roots.
|
|
// This avoids the need to walk the entire component tree when invoking these
|
|
// hooks, and more importantly, avoids the need to track child components in
|
|
// arrays.
|
|
if (target) {
|
|
let current = target.parent;
|
|
while (current && current.parent) {
|
|
if (isKeepAlive(current.parent.vnode)) {
|
|
injectToKeepAliveRoot(wrappedHook, type, target, current);
|
|
}
|
|
current = current.parent;
|
|
}
|
|
}
|
|
}
|
|
function injectToKeepAliveRoot(hook, type, target, keepAliveRoot) {
|
|
// injectHook wraps the original for error handling, so make sure to remove
|
|
// the wrapped version.
|
|
const injected = injectHook(type, hook, keepAliveRoot, true /* prepend */);
|
|
onUnmounted(() => {
|
|
remove(keepAliveRoot[type], injected);
|
|
}, target);
|
|
}
|
|
|
|
const isInternalKey = (key) => key[0] === '_' || key === '$stable';
|
|
const normalizeSlotValue = (value) => isArray(value)
|
|
? value.map(normalizeVNode)
|
|
: [normalizeVNode(value)];
|
|
const normalizeSlot = (key, rawSlot, ctx) => withCtx((props) => {
|
|
return normalizeSlotValue(rawSlot(props));
|
|
}, ctx);
|
|
const normalizeObjectSlots = (rawSlots, slots) => {
|
|
const ctx = rawSlots._ctx;
|
|
for (const key in rawSlots) {
|
|
if (isInternalKey(key))
|
|
continue;
|
|
const value = rawSlots[key];
|
|
if (isFunction(value)) {
|
|
slots[key] = normalizeSlot(key, value, ctx);
|
|
}
|
|
else if (value != null) {
|
|
const normalized = normalizeSlotValue(value);
|
|
slots[key] = () => normalized;
|
|
}
|
|
}
|
|
};
|
|
const normalizeVNodeSlots = (instance, children) => {
|
|
const normalized = normalizeSlotValue(children);
|
|
instance.slots.default = () => normalized;
|
|
};
|
|
const initSlots = (instance, children) => {
|
|
if (instance.vnode.shapeFlag & 32 /* SLOTS_CHILDREN */) {
|
|
const type = children._;
|
|
if (type) {
|
|
instance.slots = children;
|
|
// make compiler marker non-enumerable
|
|
def(children, '_', type);
|
|
}
|
|
else {
|
|
normalizeObjectSlots(children, (instance.slots = {}));
|
|
}
|
|
}
|
|
else {
|
|
instance.slots = {};
|
|
if (children) {
|
|
normalizeVNodeSlots(instance, children);
|
|
}
|
|
}
|
|
def(instance.slots, InternalObjectKey, 1);
|
|
};
|
|
const updateSlots = (instance, children) => {
|
|
const { vnode, slots } = instance;
|
|
let needDeletionCheck = true;
|
|
let deletionComparisonTarget = EMPTY_OBJ;
|
|
if (vnode.shapeFlag & 32 /* SLOTS_CHILDREN */) {
|
|
const type = children._;
|
|
if (type) {
|
|
// compiled slots.
|
|
if (type === 1 /* STABLE */) {
|
|
// compiled AND stable.
|
|
// no need to update, and skip stale slots removal.
|
|
needDeletionCheck = false;
|
|
}
|
|
else {
|
|
// compiled but dynamic (v-if/v-for on slots) - update slots, but skip
|
|
// normalization.
|
|
extend(slots, children);
|
|
}
|
|
}
|
|
else {
|
|
needDeletionCheck = !children.$stable;
|
|
normalizeObjectSlots(children, slots);
|
|
}
|
|
deletionComparisonTarget = children;
|
|
}
|
|
else if (children) {
|
|
// non slot object children (direct value) passed to a component
|
|
normalizeVNodeSlots(instance, children);
|
|
deletionComparisonTarget = { default: 1 };
|
|
}
|
|
// delete stale slots
|
|
if (needDeletionCheck) {
|
|
for (const key in slots) {
|
|
if (!isInternalKey(key) && !(key in deletionComparisonTarget)) {
|
|
delete slots[key];
|
|
}
|
|
}
|
|
}
|
|
};
|
|
/**
|
|
* Adds directives to a VNode.
|
|
*/
|
|
function withDirectives(vnode, directives) {
|
|
const internalInstance = currentRenderingInstance;
|
|
if (internalInstance === null) {
|
|
return vnode;
|
|
}
|
|
const instance = internalInstance.proxy;
|
|
const bindings = vnode.dirs || (vnode.dirs = []);
|
|
for (let i = 0; i < directives.length; i++) {
|
|
let [dir, value, arg, modifiers = EMPTY_OBJ] = directives[i];
|
|
if (isFunction(dir)) {
|
|
dir = {
|
|
mounted: dir,
|
|
updated: dir
|
|
};
|
|
}
|
|
bindings.push({
|
|
dir,
|
|
instance,
|
|
value,
|
|
oldValue: void 0,
|
|
arg,
|
|
modifiers
|
|
});
|
|
}
|
|
return vnode;
|
|
}
|
|
function invokeDirectiveHook(vnode, prevVNode, instance, name) {
|
|
const bindings = vnode.dirs;
|
|
const oldBindings = prevVNode && prevVNode.dirs;
|
|
for (let i = 0; i < bindings.length; i++) {
|
|
const binding = bindings[i];
|
|
if (oldBindings) {
|
|
binding.oldValue = oldBindings[i].value;
|
|
}
|
|
const hook = binding.dir[name];
|
|
if (hook) {
|
|
callWithAsyncErrorHandling(hook, instance, 8 /* DIRECTIVE_HOOK */, [
|
|
vnode.el,
|
|
binding,
|
|
vnode,
|
|
prevVNode
|
|
]);
|
|
}
|
|
}
|
|
}
|
|
|
|
function createAppContext() {
|
|
return {
|
|
app: null,
|
|
config: {
|
|
isNativeTag: NO,
|
|
performance: false,
|
|
globalProperties: {},
|
|
optionMergeStrategies: {},
|
|
isCustomElement: NO,
|
|
errorHandler: undefined,
|
|
warnHandler: undefined
|
|
},
|
|
mixins: [],
|
|
components: {},
|
|
directives: {},
|
|
provides: Object.create(null)
|
|
};
|
|
}
|
|
let uid$1 = 0;
|
|
function createAppAPI(render, hydrate) {
|
|
return function createApp(rootComponent, rootProps = null) {
|
|
if (rootProps != null && !isObject(rootProps)) {
|
|
rootProps = null;
|
|
}
|
|
const context = createAppContext();
|
|
const installedPlugins = new Set();
|
|
let isMounted = false;
|
|
const app = (context.app = {
|
|
_uid: uid$1++,
|
|
_component: rootComponent,
|
|
_props: rootProps,
|
|
_container: null,
|
|
_context: context,
|
|
version,
|
|
get config() {
|
|
return context.config;
|
|
},
|
|
set config(v) {
|
|
},
|
|
use(plugin, ...options) {
|
|
if (installedPlugins.has(plugin)) ;
|
|
else if (plugin && isFunction(plugin.install)) {
|
|
installedPlugins.add(plugin);
|
|
plugin.install(app, ...options);
|
|
}
|
|
else if (isFunction(plugin)) {
|
|
installedPlugins.add(plugin);
|
|
plugin(app, ...options);
|
|
}
|
|
else ;
|
|
return app;
|
|
},
|
|
mixin(mixin) {
|
|
{
|
|
if (!context.mixins.includes(mixin)) {
|
|
context.mixins.push(mixin);
|
|
// global mixin with props/emits de-optimizes props/emits
|
|
// normalization caching.
|
|
if (mixin.props || mixin.emits) {
|
|
context.deopt = true;
|
|
}
|
|
}
|
|
}
|
|
return app;
|
|
},
|
|
component(name, component) {
|
|
if (!component) {
|
|
return context.components[name];
|
|
}
|
|
context.components[name] = component;
|
|
return app;
|
|
},
|
|
directive(name, directive) {
|
|
if (!directive) {
|
|
return context.directives[name];
|
|
}
|
|
context.directives[name] = directive;
|
|
return app;
|
|
},
|
|
mount(rootContainer, isHydrate) {
|
|
if (!isMounted) {
|
|
const vnode = createVNode(rootComponent, rootProps);
|
|
// store app context on the root VNode.
|
|
// this will be set on the root instance on initial mount.
|
|
vnode.appContext = context;
|
|
if (isHydrate && hydrate) {
|
|
hydrate(vnode, rootContainer);
|
|
}
|
|
else {
|
|
render(vnode, rootContainer);
|
|
}
|
|
isMounted = true;
|
|
app._container = rootContainer;
|
|
rootContainer.__vue_app__ = app;
|
|
return vnode.component.proxy;
|
|
}
|
|
},
|
|
unmount() {
|
|
if (isMounted) {
|
|
render(null, app._container);
|
|
}
|
|
},
|
|
provide(key, value) {
|
|
// TypeScript doesn't allow symbols as index type
|
|
// https://github.com/Microsoft/TypeScript/issues/24587
|
|
context.provides[key] = value;
|
|
return app;
|
|
}
|
|
});
|
|
return app;
|
|
};
|
|
}
|
|
|
|
const isAsyncWrapper = (i) => !!i.type.__asyncLoader;
|
|
|
|
const prodEffectOptions = {
|
|
scheduler: queueJob,
|
|
// #1801, #2043 component render effects should allow recursive updates
|
|
allowRecurse: true
|
|
};
|
|
const queuePostRenderEffect = queueEffectWithSuspense
|
|
;
|
|
const setRef = (rawRef, oldRawRef, parentSuspense, vnode) => {
|
|
if (isArray(rawRef)) {
|
|
rawRef.forEach((r, i) => setRef(r, oldRawRef && (isArray(oldRawRef) ? oldRawRef[i] : oldRawRef), parentSuspense, vnode));
|
|
return;
|
|
}
|
|
let value;
|
|
if (!vnode || isAsyncWrapper(vnode)) {
|
|
value = null;
|
|
}
|
|
else {
|
|
if (vnode.shapeFlag & 4 /* STATEFUL_COMPONENT */) {
|
|
value = vnode.component.exposed || vnode.component.proxy;
|
|
}
|
|
else {
|
|
value = vnode.el;
|
|
}
|
|
}
|
|
const { i: owner, r: ref } = rawRef;
|
|
const oldRef = oldRawRef && oldRawRef.r;
|
|
const refs = owner.refs === EMPTY_OBJ ? (owner.refs = {}) : owner.refs;
|
|
const setupState = owner.setupState;
|
|
// unset old ref
|
|
if (oldRef != null && oldRef !== ref) {
|
|
if (isString(oldRef)) {
|
|
refs[oldRef] = null;
|
|
if (hasOwn(setupState, oldRef)) {
|
|
setupState[oldRef] = null;
|
|
}
|
|
}
|
|
else if (isRef(oldRef)) {
|
|
oldRef.value = null;
|
|
}
|
|
}
|
|
if (isString(ref)) {
|
|
const doSet = () => {
|
|
refs[ref] = value;
|
|
if (hasOwn(setupState, ref)) {
|
|
setupState[ref] = value;
|
|
}
|
|
};
|
|
// #1789: for non-null values, set them after render
|
|
// null values means this is unmount and it should not overwrite another
|
|
// ref with the same key
|
|
if (value) {
|
|
doSet.id = -1;
|
|
queuePostRenderEffect(doSet, parentSuspense);
|
|
}
|
|
else {
|
|
doSet();
|
|
}
|
|
}
|
|
else if (isRef(ref)) {
|
|
const doSet = () => {
|
|
ref.value = value;
|
|
};
|
|
if (value) {
|
|
doSet.id = -1;
|
|
queuePostRenderEffect(doSet, parentSuspense);
|
|
}
|
|
else {
|
|
doSet();
|
|
}
|
|
}
|
|
else if (isFunction(ref)) {
|
|
callWithErrorHandling(ref, owner, 12 /* FUNCTION_REF */, [value, refs]);
|
|
}
|
|
else ;
|
|
};
|
|
/**
|
|
* The createRenderer function accepts two generic arguments:
|
|
* HostNode and HostElement, corresponding to Node and Element types in the
|
|
* host environment. For example, for runtime-dom, HostNode would be the DOM
|
|
* `Node` interface and HostElement would be the DOM `Element` interface.
|
|
*
|
|
* Custom renderers can pass in the platform specific types like this:
|
|
*
|
|
* ``` js
|
|
* const { render, createApp } = createRenderer<Node, Element>({
|
|
* patchProp,
|
|
* ...nodeOps
|
|
* })
|
|
* ```
|
|
*/
|
|
function createRenderer(options) {
|
|
return baseCreateRenderer(options);
|
|
}
|
|
// implementation
|
|
function baseCreateRenderer(options, createHydrationFns) {
|
|
const { insert: hostInsert, remove: hostRemove, patchProp: hostPatchProp, forcePatchProp: hostForcePatchProp, createElement: hostCreateElement, createText: hostCreateText, createComment: hostCreateComment, setText: hostSetText, setElementText: hostSetElementText, parentNode: hostParentNode, nextSibling: hostNextSibling, setScopeId: hostSetScopeId = NOOP, cloneNode: hostCloneNode, insertStaticContent: hostInsertStaticContent } = options;
|
|
// Note: functions inside this closure should use `const xxx = () => {}`
|
|
// style in order to prevent being inlined by minifiers.
|
|
const patch = (n1, n2, container, anchor = null, parentComponent = null, parentSuspense = null, isSVG = false, optimized = false) => {
|
|
// patching & not same type, unmount old tree
|
|
if (n1 && !isSameVNodeType(n1, n2)) {
|
|
anchor = getNextHostNode(n1);
|
|
unmount(n1, parentComponent, parentSuspense, true);
|
|
n1 = null;
|
|
}
|
|
if (n2.patchFlag === -2 /* BAIL */) {
|
|
optimized = false;
|
|
n2.dynamicChildren = null;
|
|
}
|
|
const { type, ref, shapeFlag } = n2;
|
|
switch (type) {
|
|
case Text:
|
|
processText(n1, n2, container, anchor);
|
|
break;
|
|
case Comment:
|
|
processCommentNode(n1, n2, container, anchor);
|
|
break;
|
|
case Static:
|
|
if (n1 == null) {
|
|
mountStaticNode(n2, container, anchor, isSVG);
|
|
}
|
|
break;
|
|
case Fragment:
|
|
processFragment(n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
break;
|
|
default:
|
|
if (shapeFlag & 1 /* ELEMENT */) {
|
|
processElement(n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else if (shapeFlag & 6 /* COMPONENT */) {
|
|
processComponent(n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else if (shapeFlag & 64 /* TELEPORT */) {
|
|
type.process(n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized, internals);
|
|
}
|
|
else if ( shapeFlag & 128 /* SUSPENSE */) {
|
|
type.process(n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized, internals);
|
|
}
|
|
else ;
|
|
}
|
|
// set ref
|
|
if (ref != null && parentComponent) {
|
|
setRef(ref, n1 && n1.ref, parentSuspense, n2);
|
|
}
|
|
};
|
|
const processText = (n1, n2, container, anchor) => {
|
|
if (n1 == null) {
|
|
hostInsert((n2.el = hostCreateText(n2.children)), container, anchor);
|
|
}
|
|
else {
|
|
const el = (n2.el = n1.el);
|
|
if (n2.children !== n1.children) {
|
|
hostSetText(el, n2.children);
|
|
}
|
|
}
|
|
};
|
|
const processCommentNode = (n1, n2, container, anchor) => {
|
|
if (n1 == null) {
|
|
hostInsert((n2.el = hostCreateComment(n2.children || '')), container, anchor);
|
|
}
|
|
else {
|
|
// there's no support for dynamic comments
|
|
n2.el = n1.el;
|
|
}
|
|
};
|
|
const mountStaticNode = (n2, container, anchor, isSVG) => {
|
|
[n2.el, n2.anchor] = hostInsertStaticContent(n2.children, container, anchor, isSVG);
|
|
};
|
|
const moveStaticNode = ({ el, anchor }, container, nextSibling) => {
|
|
let next;
|
|
while (el && el !== anchor) {
|
|
next = hostNextSibling(el);
|
|
hostInsert(el, container, nextSibling);
|
|
el = next;
|
|
}
|
|
hostInsert(anchor, container, nextSibling);
|
|
};
|
|
const removeStaticNode = ({ el, anchor }) => {
|
|
let next;
|
|
while (el && el !== anchor) {
|
|
next = hostNextSibling(el);
|
|
hostRemove(el);
|
|
el = next;
|
|
}
|
|
hostRemove(anchor);
|
|
};
|
|
const processElement = (n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
isSVG = isSVG || n2.type === 'svg';
|
|
if (n1 == null) {
|
|
mountElement(n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else {
|
|
patchElement(n1, n2, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
};
|
|
const mountElement = (vnode, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
let el;
|
|
let vnodeHook;
|
|
const { type, props, shapeFlag, transition, scopeId, patchFlag, dirs } = vnode;
|
|
if (vnode.el &&
|
|
hostCloneNode !== undefined &&
|
|
patchFlag === -1 /* HOISTED */) {
|
|
// If a vnode has non-null el, it means it's being reused.
|
|
// Only static vnodes can be reused, so its mounted DOM nodes should be
|
|
// exactly the same, and we can simply do a clone here.
|
|
// only do this in production since cloned trees cannot be HMR updated.
|
|
el = vnode.el = hostCloneNode(vnode.el);
|
|
}
|
|
else {
|
|
el = vnode.el = hostCreateElement(vnode.type, isSVG, props && props.is);
|
|
// mount children first, since some props may rely on child content
|
|
// being already rendered, e.g. `<select value>`
|
|
if (shapeFlag & 8 /* TEXT_CHILDREN */) {
|
|
hostSetElementText(el, vnode.children);
|
|
}
|
|
else if (shapeFlag & 16 /* ARRAY_CHILDREN */) {
|
|
mountChildren(vnode.children, el, null, parentComponent, parentSuspense, isSVG && type !== 'foreignObject', optimized || !!vnode.dynamicChildren);
|
|
}
|
|
if (dirs) {
|
|
invokeDirectiveHook(vnode, null, parentComponent, 'created');
|
|
}
|
|
// props
|
|
if (props) {
|
|
for (const key in props) {
|
|
if (!isReservedProp(key)) {
|
|
hostPatchProp(el, key, null, props[key], isSVG, vnode.children, parentComponent, parentSuspense, unmountChildren);
|
|
}
|
|
}
|
|
if ((vnodeHook = props.onVnodeBeforeMount)) {
|
|
invokeVNodeHook(vnodeHook, parentComponent, vnode);
|
|
}
|
|
}
|
|
// scopeId
|
|
setScopeId(el, scopeId, vnode, parentComponent);
|
|
}
|
|
if (dirs) {
|
|
invokeDirectiveHook(vnode, null, parentComponent, 'beforeMount');
|
|
}
|
|
// #1583 For inside suspense + suspense not resolved case, enter hook should call when suspense resolved
|
|
// #1689 For inside suspense + suspense resolved case, just call it
|
|
const needCallTransitionHooks = (!parentSuspense || (parentSuspense && !parentSuspense.pendingBranch)) &&
|
|
transition &&
|
|
!transition.persisted;
|
|
if (needCallTransitionHooks) {
|
|
transition.beforeEnter(el);
|
|
}
|
|
hostInsert(el, container, anchor);
|
|
if ((vnodeHook = props && props.onVnodeMounted) ||
|
|
needCallTransitionHooks ||
|
|
dirs) {
|
|
queuePostRenderEffect(() => {
|
|
vnodeHook && invokeVNodeHook(vnodeHook, parentComponent, vnode);
|
|
needCallTransitionHooks && transition.enter(el);
|
|
dirs && invokeDirectiveHook(vnode, null, parentComponent, 'mounted');
|
|
}, parentSuspense);
|
|
}
|
|
};
|
|
const setScopeId = (el, scopeId, vnode, parentComponent) => {
|
|
if (scopeId) {
|
|
hostSetScopeId(el, scopeId);
|
|
}
|
|
if (parentComponent) {
|
|
const treeOwnerId = parentComponent.type.__scopeId;
|
|
// vnode's own scopeId and the current patched component's scopeId is
|
|
// different - this is a slot content node.
|
|
if (treeOwnerId && treeOwnerId !== scopeId) {
|
|
hostSetScopeId(el, treeOwnerId + '-s');
|
|
}
|
|
let subTree = parentComponent.subTree;
|
|
if (vnode === subTree) {
|
|
setScopeId(el, parentComponent.vnode.scopeId, parentComponent.vnode, parentComponent.parent);
|
|
}
|
|
}
|
|
};
|
|
const mountChildren = (children, container, anchor, parentComponent, parentSuspense, isSVG, optimized, start = 0) => {
|
|
for (let i = start; i < children.length; i++) {
|
|
const child = (children[i] = optimized
|
|
? cloneIfMounted(children[i])
|
|
: normalizeVNode(children[i]));
|
|
patch(null, child, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
};
|
|
const patchElement = (n1, n2, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
const el = (n2.el = n1.el);
|
|
let { patchFlag, dynamicChildren, dirs } = n2;
|
|
// #1426 take the old vnode's patch flag into account since user may clone a
|
|
// compiler-generated vnode, which de-opts to FULL_PROPS
|
|
patchFlag |= n1.patchFlag & 16 /* FULL_PROPS */;
|
|
const oldProps = n1.props || EMPTY_OBJ;
|
|
const newProps = n2.props || EMPTY_OBJ;
|
|
let vnodeHook;
|
|
if ((vnodeHook = newProps.onVnodeBeforeUpdate)) {
|
|
invokeVNodeHook(vnodeHook, parentComponent, n2, n1);
|
|
}
|
|
if (dirs) {
|
|
invokeDirectiveHook(n2, n1, parentComponent, 'beforeUpdate');
|
|
}
|
|
if (patchFlag > 0) {
|
|
// the presence of a patchFlag means this element's render code was
|
|
// generated by the compiler and can take the fast path.
|
|
// in this path old node and new node are guaranteed to have the same shape
|
|
// (i.e. at the exact same position in the source template)
|
|
if (patchFlag & 16 /* FULL_PROPS */) {
|
|
// element props contain dynamic keys, full diff needed
|
|
patchProps(el, n2, oldProps, newProps, parentComponent, parentSuspense, isSVG);
|
|
}
|
|
else {
|
|
// class
|
|
// this flag is matched when the element has dynamic class bindings.
|
|
if (patchFlag & 2 /* CLASS */) {
|
|
if (oldProps.class !== newProps.class) {
|
|
hostPatchProp(el, 'class', null, newProps.class, isSVG);
|
|
}
|
|
}
|
|
// style
|
|
// this flag is matched when the element has dynamic style bindings
|
|
if (patchFlag & 4 /* STYLE */) {
|
|
hostPatchProp(el, 'style', oldProps.style, newProps.style, isSVG);
|
|
}
|
|
// props
|
|
// This flag is matched when the element has dynamic prop/attr bindings
|
|
// other than class and style. The keys of dynamic prop/attrs are saved for
|
|
// faster iteration.
|
|
// Note dynamic keys like :[foo]="bar" will cause this optimization to
|
|
// bail out and go through a full diff because we need to unset the old key
|
|
if (patchFlag & 8 /* PROPS */) {
|
|
// if the flag is present then dynamicProps must be non-null
|
|
const propsToUpdate = n2.dynamicProps;
|
|
for (let i = 0; i < propsToUpdate.length; i++) {
|
|
const key = propsToUpdate[i];
|
|
const prev = oldProps[key];
|
|
const next = newProps[key];
|
|
if (next !== prev ||
|
|
(hostForcePatchProp && hostForcePatchProp(el, key))) {
|
|
hostPatchProp(el, key, prev, next, isSVG, n1.children, parentComponent, parentSuspense, unmountChildren);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// text
|
|
// This flag is matched when the element has only dynamic text children.
|
|
if (patchFlag & 1 /* TEXT */) {
|
|
if (n1.children !== n2.children) {
|
|
hostSetElementText(el, n2.children);
|
|
}
|
|
}
|
|
}
|
|
else if (!optimized && dynamicChildren == null) {
|
|
// unoptimized, full diff
|
|
patchProps(el, n2, oldProps, newProps, parentComponent, parentSuspense, isSVG);
|
|
}
|
|
const areChildrenSVG = isSVG && n2.type !== 'foreignObject';
|
|
if (dynamicChildren) {
|
|
patchBlockChildren(n1.dynamicChildren, dynamicChildren, el, parentComponent, parentSuspense, areChildrenSVG);
|
|
}
|
|
else if (!optimized) {
|
|
// full diff
|
|
patchChildren(n1, n2, el, null, parentComponent, parentSuspense, areChildrenSVG);
|
|
}
|
|
if ((vnodeHook = newProps.onVnodeUpdated) || dirs) {
|
|
queuePostRenderEffect(() => {
|
|
vnodeHook && invokeVNodeHook(vnodeHook, parentComponent, n2, n1);
|
|
dirs && invokeDirectiveHook(n2, n1, parentComponent, 'updated');
|
|
}, parentSuspense);
|
|
}
|
|
};
|
|
// The fast path for blocks.
|
|
const patchBlockChildren = (oldChildren, newChildren, fallbackContainer, parentComponent, parentSuspense, isSVG) => {
|
|
for (let i = 0; i < newChildren.length; i++) {
|
|
const oldVNode = oldChildren[i];
|
|
const newVNode = newChildren[i];
|
|
// Determine the container (parent element) for the patch.
|
|
const container =
|
|
// - In the case of a Fragment, we need to provide the actual parent
|
|
// of the Fragment itself so it can move its children.
|
|
oldVNode.type === Fragment ||
|
|
// - In the case of different nodes, there is going to be a replacement
|
|
// which also requires the correct parent container
|
|
!isSameVNodeType(oldVNode, newVNode) ||
|
|
// - In the case of a component, it could contain anything.
|
|
oldVNode.shapeFlag & 6 /* COMPONENT */ ||
|
|
oldVNode.shapeFlag & 64 /* TELEPORT */
|
|
? hostParentNode(oldVNode.el)
|
|
: // In other cases, the parent container is not actually used so we
|
|
// just pass the block element here to avoid a DOM parentNode call.
|
|
fallbackContainer;
|
|
patch(oldVNode, newVNode, container, null, parentComponent, parentSuspense, isSVG, true);
|
|
}
|
|
};
|
|
const patchProps = (el, vnode, oldProps, newProps, parentComponent, parentSuspense, isSVG) => {
|
|
if (oldProps !== newProps) {
|
|
for (const key in newProps) {
|
|
// empty string is not valid prop
|
|
if (isReservedProp(key))
|
|
continue;
|
|
const next = newProps[key];
|
|
const prev = oldProps[key];
|
|
if (next !== prev ||
|
|
(hostForcePatchProp && hostForcePatchProp(el, key))) {
|
|
hostPatchProp(el, key, prev, next, isSVG, vnode.children, parentComponent, parentSuspense, unmountChildren);
|
|
}
|
|
}
|
|
if (oldProps !== EMPTY_OBJ) {
|
|
for (const key in oldProps) {
|
|
if (!isReservedProp(key) && !(key in newProps)) {
|
|
hostPatchProp(el, key, oldProps[key], null, isSVG, vnode.children, parentComponent, parentSuspense, unmountChildren);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
const processFragment = (n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
const fragmentStartAnchor = (n2.el = n1 ? n1.el : hostCreateText(''));
|
|
const fragmentEndAnchor = (n2.anchor = n1 ? n1.anchor : hostCreateText(''));
|
|
let { patchFlag, dynamicChildren } = n2;
|
|
if (patchFlag > 0) {
|
|
optimized = true;
|
|
}
|
|
if (n1 == null) {
|
|
hostInsert(fragmentStartAnchor, container, anchor);
|
|
hostInsert(fragmentEndAnchor, container, anchor);
|
|
// a fragment can only have array children
|
|
// since they are either generated by the compiler, or implicitly created
|
|
// from arrays.
|
|
mountChildren(n2.children, container, fragmentEndAnchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else {
|
|
if (patchFlag > 0 &&
|
|
patchFlag & 64 /* STABLE_FRAGMENT */ &&
|
|
dynamicChildren &&
|
|
// #2715 the previous fragment could've been a BAILed one as a result
|
|
// of renderSlot() with no valid children
|
|
n1.dynamicChildren) {
|
|
// a stable fragment (template root or <template v-for>) doesn't need to
|
|
// patch children order, but it may contain dynamicChildren.
|
|
patchBlockChildren(n1.dynamicChildren, dynamicChildren, container, parentComponent, parentSuspense, isSVG);
|
|
if (
|
|
// #2080 if the stable fragment has a key, it's a <template v-for> that may
|
|
// get moved around. Make sure all root level vnodes inherit el.
|
|
// #2134 or if it's a component root, it may also get moved around
|
|
// as the component is being moved.
|
|
n2.key != null ||
|
|
(parentComponent && n2 === parentComponent.subTree)) {
|
|
traverseStaticChildren(n1, n2, true /* shallow */);
|
|
}
|
|
}
|
|
else {
|
|
// keyed / unkeyed, or manual fragments.
|
|
// for keyed & unkeyed, since they are compiler generated from v-for,
|
|
// each child is guaranteed to be a block so the fragment will never
|
|
// have dynamicChildren.
|
|
patchChildren(n1, n2, container, fragmentEndAnchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
}
|
|
};
|
|
const processComponent = (n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
if (n1 == null) {
|
|
if (n2.shapeFlag & 512 /* COMPONENT_KEPT_ALIVE */) {
|
|
parentComponent.ctx.activate(n2, container, anchor, isSVG, optimized);
|
|
}
|
|
else {
|
|
mountComponent(n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
}
|
|
else {
|
|
updateComponent(n1, n2, optimized);
|
|
}
|
|
};
|
|
const mountComponent = (initialVNode, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
const instance = (initialVNode.component = createComponentInstance(initialVNode, parentComponent, parentSuspense));
|
|
// inject renderer internals for keepAlive
|
|
if (isKeepAlive(initialVNode)) {
|
|
instance.ctx.renderer = internals;
|
|
}
|
|
setupComponent(instance);
|
|
// setup() is async. This component relies on async logic to be resolved
|
|
// before proceeding
|
|
if ( instance.asyncDep) {
|
|
parentSuspense && parentSuspense.registerDep(instance, setupRenderEffect);
|
|
// Give it a placeholder if this is not hydration
|
|
// TODO handle self-defined fallback
|
|
if (!initialVNode.el) {
|
|
const placeholder = (instance.subTree = createVNode(Comment));
|
|
processCommentNode(null, placeholder, container, anchor);
|
|
}
|
|
return;
|
|
}
|
|
setupRenderEffect(instance, initialVNode, container, anchor, parentSuspense, isSVG, optimized);
|
|
};
|
|
const updateComponent = (n1, n2, optimized) => {
|
|
const instance = (n2.component = n1.component);
|
|
if (shouldUpdateComponent(n1, n2, optimized)) {
|
|
if (
|
|
instance.asyncDep &&
|
|
!instance.asyncResolved) {
|
|
updateComponentPreRender(instance, n2, optimized);
|
|
return;
|
|
}
|
|
else {
|
|
// normal update
|
|
instance.next = n2;
|
|
// in case the child component is also queued, remove it to avoid
|
|
// double updating the same child component in the same flush.
|
|
invalidateJob(instance.update);
|
|
// instance.update is the reactive effect runner.
|
|
instance.update();
|
|
}
|
|
}
|
|
else {
|
|
// no update needed. just copy over properties
|
|
n2.component = n1.component;
|
|
n2.el = n1.el;
|
|
instance.vnode = n2;
|
|
}
|
|
};
|
|
const setupRenderEffect = (instance, initialVNode, container, anchor, parentSuspense, isSVG, optimized) => {
|
|
// create reactive effect for rendering
|
|
instance.update = effect(function componentEffect() {
|
|
if (!instance.isMounted) {
|
|
let vnodeHook;
|
|
const { el, props } = initialVNode;
|
|
const { bm, m, parent } = instance;
|
|
// beforeMount hook
|
|
if (bm) {
|
|
invokeArrayFns(bm);
|
|
}
|
|
// onVnodeBeforeMount
|
|
if ((vnodeHook = props && props.onVnodeBeforeMount)) {
|
|
invokeVNodeHook(vnodeHook, parent, initialVNode);
|
|
}
|
|
const subTree = (instance.subTree = renderComponentRoot(instance));
|
|
if (el && hydrateNode) {
|
|
// vnode has adopted host node - perform hydration instead of mount.
|
|
hydrateNode(initialVNode.el, subTree, instance, parentSuspense);
|
|
}
|
|
else {
|
|
patch(null, subTree, container, anchor, instance, parentSuspense, isSVG);
|
|
initialVNode.el = subTree.el;
|
|
}
|
|
// mounted hook
|
|
if (m) {
|
|
queuePostRenderEffect(m, parentSuspense);
|
|
}
|
|
// onVnodeMounted
|
|
if ((vnodeHook = props && props.onVnodeMounted)) {
|
|
const scopedInitialVNode = initialVNode;
|
|
queuePostRenderEffect(() => {
|
|
invokeVNodeHook(vnodeHook, parent, scopedInitialVNode);
|
|
}, parentSuspense);
|
|
}
|
|
// activated hook for keep-alive roots.
|
|
// #1742 activated hook must be accessed after first render
|
|
// since the hook may be injected by a child keep-alive
|
|
const { a } = instance;
|
|
if (a &&
|
|
initialVNode.shapeFlag & 256 /* COMPONENT_SHOULD_KEEP_ALIVE */) {
|
|
queuePostRenderEffect(a, parentSuspense);
|
|
}
|
|
instance.isMounted = true;
|
|
// #2458: deference mount-only object parameters to prevent memleaks
|
|
initialVNode = container = anchor = null;
|
|
}
|
|
else {
|
|
// updateComponent
|
|
// This is triggered by mutation of component's own state (next: null)
|
|
// OR parent calling processComponent (next: VNode)
|
|
let { next, bu, u, parent, vnode } = instance;
|
|
let originNext = next;
|
|
let vnodeHook;
|
|
if (next) {
|
|
next.el = vnode.el;
|
|
updateComponentPreRender(instance, next, optimized);
|
|
}
|
|
else {
|
|
next = vnode;
|
|
}
|
|
// beforeUpdate hook
|
|
if (bu) {
|
|
invokeArrayFns(bu);
|
|
}
|
|
// onVnodeBeforeUpdate
|
|
if ((vnodeHook = next.props && next.props.onVnodeBeforeUpdate)) {
|
|
invokeVNodeHook(vnodeHook, parent, next, vnode);
|
|
}
|
|
const nextTree = renderComponentRoot(instance);
|
|
const prevTree = instance.subTree;
|
|
instance.subTree = nextTree;
|
|
patch(prevTree, nextTree,
|
|
// parent may have changed if it's in a teleport
|
|
hostParentNode(prevTree.el),
|
|
// anchor may have changed if it's in a fragment
|
|
getNextHostNode(prevTree), instance, parentSuspense, isSVG);
|
|
next.el = nextTree.el;
|
|
if (originNext === null) {
|
|
// self-triggered update. In case of HOC, update parent component
|
|
// vnode el. HOC is indicated by parent instance's subTree pointing
|
|
// to child component's vnode
|
|
updateHOCHostEl(instance, nextTree.el);
|
|
}
|
|
// updated hook
|
|
if (u) {
|
|
queuePostRenderEffect(u, parentSuspense);
|
|
}
|
|
// onVnodeUpdated
|
|
if ((vnodeHook = next.props && next.props.onVnodeUpdated)) {
|
|
queuePostRenderEffect(() => {
|
|
invokeVNodeHook(vnodeHook, parent, next, vnode);
|
|
}, parentSuspense);
|
|
}
|
|
}
|
|
}, prodEffectOptions);
|
|
};
|
|
const updateComponentPreRender = (instance, nextVNode, optimized) => {
|
|
nextVNode.component = instance;
|
|
const prevProps = instance.vnode.props;
|
|
instance.vnode = nextVNode;
|
|
instance.next = null;
|
|
updateProps(instance, nextVNode.props, prevProps, optimized);
|
|
updateSlots(instance, nextVNode.children);
|
|
// props update may have triggered pre-flush watchers.
|
|
// flush them before the render update.
|
|
flushPreFlushCbs(undefined, instance.update);
|
|
};
|
|
const patchChildren = (n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized = false) => {
|
|
const c1 = n1 && n1.children;
|
|
const prevShapeFlag = n1 ? n1.shapeFlag : 0;
|
|
const c2 = n2.children;
|
|
const { patchFlag, shapeFlag } = n2;
|
|
// fast path
|
|
if (patchFlag > 0) {
|
|
if (patchFlag & 128 /* KEYED_FRAGMENT */) {
|
|
// this could be either fully-keyed or mixed (some keyed some not)
|
|
// presence of patchFlag means children are guaranteed to be arrays
|
|
patchKeyedChildren(c1, c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
return;
|
|
}
|
|
else if (patchFlag & 256 /* UNKEYED_FRAGMENT */) {
|
|
// unkeyed
|
|
patchUnkeyedChildren(c1, c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
return;
|
|
}
|
|
}
|
|
// children has 3 possibilities: text, array or no children.
|
|
if (shapeFlag & 8 /* TEXT_CHILDREN */) {
|
|
// text children fast path
|
|
if (prevShapeFlag & 16 /* ARRAY_CHILDREN */) {
|
|
unmountChildren(c1, parentComponent, parentSuspense);
|
|
}
|
|
if (c2 !== c1) {
|
|
hostSetElementText(container, c2);
|
|
}
|
|
}
|
|
else {
|
|
if (prevShapeFlag & 16 /* ARRAY_CHILDREN */) {
|
|
// prev children was array
|
|
if (shapeFlag & 16 /* ARRAY_CHILDREN */) {
|
|
// two arrays, cannot assume anything, do full diff
|
|
patchKeyedChildren(c1, c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else {
|
|
// no new children, just unmount old
|
|
unmountChildren(c1, parentComponent, parentSuspense, true);
|
|
}
|
|
}
|
|
else {
|
|
// prev children was text OR null
|
|
// new children is array OR null
|
|
if (prevShapeFlag & 8 /* TEXT_CHILDREN */) {
|
|
hostSetElementText(container, '');
|
|
}
|
|
// mount new if array
|
|
if (shapeFlag & 16 /* ARRAY_CHILDREN */) {
|
|
mountChildren(c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
const patchUnkeyedChildren = (c1, c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
c1 = c1 || EMPTY_ARR;
|
|
c2 = c2 || EMPTY_ARR;
|
|
const oldLength = c1.length;
|
|
const newLength = c2.length;
|
|
const commonLength = Math.min(oldLength, newLength);
|
|
let i;
|
|
for (i = 0; i < commonLength; i++) {
|
|
const nextChild = (c2[i] = optimized
|
|
? cloneIfMounted(c2[i])
|
|
: normalizeVNode(c2[i]));
|
|
patch(c1[i], nextChild, container, null, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
if (oldLength > newLength) {
|
|
// remove old
|
|
unmountChildren(c1, parentComponent, parentSuspense, true, false, commonLength);
|
|
}
|
|
else {
|
|
// mount new
|
|
mountChildren(c2, container, anchor, parentComponent, parentSuspense, isSVG, optimized, commonLength);
|
|
}
|
|
};
|
|
// can be all-keyed or mixed
|
|
const patchKeyedChildren = (c1, c2, container, parentAnchor, parentComponent, parentSuspense, isSVG, optimized) => {
|
|
let i = 0;
|
|
const l2 = c2.length;
|
|
let e1 = c1.length - 1; // prev ending index
|
|
let e2 = l2 - 1; // next ending index
|
|
// 1. sync from start
|
|
// (a b) c
|
|
// (a b) d e
|
|
while (i <= e1 && i <= e2) {
|
|
const n1 = c1[i];
|
|
const n2 = (c2[i] = optimized
|
|
? cloneIfMounted(c2[i])
|
|
: normalizeVNode(c2[i]));
|
|
if (isSameVNodeType(n1, n2)) {
|
|
patch(n1, n2, container, null, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else {
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
// 2. sync from end
|
|
// a (b c)
|
|
// d e (b c)
|
|
while (i <= e1 && i <= e2) {
|
|
const n1 = c1[e1];
|
|
const n2 = (c2[e2] = optimized
|
|
? cloneIfMounted(c2[e2])
|
|
: normalizeVNode(c2[e2]));
|
|
if (isSameVNodeType(n1, n2)) {
|
|
patch(n1, n2, container, null, parentComponent, parentSuspense, isSVG, optimized);
|
|
}
|
|
else {
|
|
break;
|
|
}
|
|
e1--;
|
|
e2--;
|
|
}
|
|
// 3. common sequence + mount
|
|
// (a b)
|
|
// (a b) c
|
|
// i = 2, e1 = 1, e2 = 2
|
|
// (a b)
|
|
// c (a b)
|
|
// i = 0, e1 = -1, e2 = 0
|
|
if (i > e1) {
|
|
if (i <= e2) {
|
|
const nextPos = e2 + 1;
|
|
const anchor = nextPos < l2 ? c2[nextPos].el : parentAnchor;
|
|
while (i <= e2) {
|
|
patch(null, (c2[i] = optimized
|
|
? cloneIfMounted(c2[i])
|
|
: normalizeVNode(c2[i])), container, anchor, parentComponent, parentSuspense, isSVG);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
// 4. common sequence + unmount
|
|
// (a b) c
|
|
// (a b)
|
|
// i = 2, e1 = 2, e2 = 1
|
|
// a (b c)
|
|
// (b c)
|
|
// i = 0, e1 = 0, e2 = -1
|
|
else if (i > e2) {
|
|
while (i <= e1) {
|
|
unmount(c1[i], parentComponent, parentSuspense, true);
|
|
i++;
|
|
}
|
|
}
|
|
// 5. unknown sequence
|
|
// [i ... e1 + 1]: a b [c d e] f g
|
|
// [i ... e2 + 1]: a b [e d c h] f g
|
|
// i = 2, e1 = 4, e2 = 5
|
|
else {
|
|
const s1 = i; // prev starting index
|
|
const s2 = i; // next starting index
|
|
// 5.1 build key:index map for newChildren
|
|
const keyToNewIndexMap = new Map();
|
|
for (i = s2; i <= e2; i++) {
|
|
const nextChild = (c2[i] = optimized
|
|
? cloneIfMounted(c2[i])
|
|
: normalizeVNode(c2[i]));
|
|
if (nextChild.key != null) {
|
|
keyToNewIndexMap.set(nextChild.key, i);
|
|
}
|
|
}
|
|
// 5.2 loop through old children left to be patched and try to patch
|
|
// matching nodes & remove nodes that are no longer present
|
|
let j;
|
|
let patched = 0;
|
|
const toBePatched = e2 - s2 + 1;
|
|
let moved = false;
|
|
// used to track whether any node has moved
|
|
let maxNewIndexSoFar = 0;
|
|
// works as Map<newIndex, oldIndex>
|
|
// Note that oldIndex is offset by +1
|
|
// and oldIndex = 0 is a special value indicating the new node has
|
|
// no corresponding old node.
|
|
// used for determining longest stable subsequence
|
|
const newIndexToOldIndexMap = new Array(toBePatched);
|
|
for (i = 0; i < toBePatched; i++)
|
|
newIndexToOldIndexMap[i] = 0;
|
|
for (i = s1; i <= e1; i++) {
|
|
const prevChild = c1[i];
|
|
if (patched >= toBePatched) {
|
|
// all new children have been patched so this can only be a removal
|
|
unmount(prevChild, parentComponent, parentSuspense, true);
|
|
continue;
|
|
}
|
|
let newIndex;
|
|
if (prevChild.key != null) {
|
|
newIndex = keyToNewIndexMap.get(prevChild.key);
|
|
}
|
|
else {
|
|
// key-less node, try to locate a key-less node of the same type
|
|
for (j = s2; j <= e2; j++) {
|
|
if (newIndexToOldIndexMap[j - s2] === 0 &&
|
|
isSameVNodeType(prevChild, c2[j])) {
|
|
newIndex = j;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (newIndex === undefined) {
|
|
unmount(prevChild, parentComponent, parentSuspense, true);
|
|
}
|
|
else {
|
|
newIndexToOldIndexMap[newIndex - s2] = i + 1;
|
|
if (newIndex >= maxNewIndexSoFar) {
|
|
maxNewIndexSoFar = newIndex;
|
|
}
|
|
else {
|
|
moved = true;
|
|
}
|
|
patch(prevChild, c2[newIndex], container, null, parentComponent, parentSuspense, isSVG, optimized);
|
|
patched++;
|
|
}
|
|
}
|
|
// 5.3 move and mount
|
|
// generate longest stable subsequence only when nodes have moved
|
|
const increasingNewIndexSequence = moved
|
|
? getSequence(newIndexToOldIndexMap)
|
|
: EMPTY_ARR;
|
|
j = increasingNewIndexSequence.length - 1;
|
|
// looping backwards so that we can use last patched node as anchor
|
|
for (i = toBePatched - 1; i >= 0; i--) {
|
|
const nextIndex = s2 + i;
|
|
const nextChild = c2[nextIndex];
|
|
const anchor = nextIndex + 1 < l2 ? c2[nextIndex + 1].el : parentAnchor;
|
|
if (newIndexToOldIndexMap[i] === 0) {
|
|
// mount new
|
|
patch(null, nextChild, container, anchor, parentComponent, parentSuspense, isSVG);
|
|
}
|
|
else if (moved) {
|
|
// move if:
|
|
// There is no stable subsequence (e.g. a reverse)
|
|
// OR current node is not among the stable sequence
|
|
if (j < 0 || i !== increasingNewIndexSequence[j]) {
|
|
move(nextChild, container, anchor, 2 /* REORDER */);
|
|
}
|
|
else {
|
|
j--;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
const move = (vnode, container, anchor, moveType, parentSuspense = null) => {
|
|
const { el, type, transition, children, shapeFlag } = vnode;
|
|
if (shapeFlag & 6 /* COMPONENT */) {
|
|
move(vnode.component.subTree, container, anchor, moveType);
|
|
return;
|
|
}
|
|
if ( shapeFlag & 128 /* SUSPENSE */) {
|
|
vnode.suspense.move(container, anchor, moveType);
|
|
return;
|
|
}
|
|
if (shapeFlag & 64 /* TELEPORT */) {
|
|
type.move(vnode, container, anchor, internals);
|
|
return;
|
|
}
|
|
if (type === Fragment) {
|
|
hostInsert(el, container, anchor);
|
|
for (let i = 0; i < children.length; i++) {
|
|
move(children[i], container, anchor, moveType);
|
|
}
|
|
hostInsert(vnode.anchor, container, anchor);
|
|
return;
|
|
}
|
|
if (type === Static) {
|
|
moveStaticNode(vnode, container, anchor);
|
|
return;
|
|
}
|
|
// single nodes
|
|
const needTransition = moveType !== 2 /* REORDER */ &&
|
|
shapeFlag & 1 /* ELEMENT */ &&
|
|
transition;
|
|
if (needTransition) {
|
|
if (moveType === 0 /* ENTER */) {
|
|
transition.beforeEnter(el);
|
|
hostInsert(el, container, anchor);
|
|
queuePostRenderEffect(() => transition.enter(el), parentSuspense);
|
|
}
|
|
else {
|
|
const { leave, delayLeave, afterLeave } = transition;
|
|
const remove = () => hostInsert(el, container, anchor);
|
|
const performLeave = () => {
|
|
leave(el, () => {
|
|
remove();
|
|
afterLeave && afterLeave();
|
|
});
|
|
};
|
|
if (delayLeave) {
|
|
delayLeave(el, remove, performLeave);
|
|
}
|
|
else {
|
|
performLeave();
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
hostInsert(el, container, anchor);
|
|
}
|
|
};
|
|
const unmount = (vnode, parentComponent, parentSuspense, doRemove = false, optimized = false) => {
|
|
const { type, props, ref, children, dynamicChildren, shapeFlag, patchFlag, dirs } = vnode;
|
|
// unset ref
|
|
if (ref != null) {
|
|
setRef(ref, null, parentSuspense, null);
|
|
}
|
|
if (shapeFlag & 256 /* COMPONENT_SHOULD_KEEP_ALIVE */) {
|
|
parentComponent.ctx.deactivate(vnode);
|
|
return;
|
|
}
|
|
const shouldInvokeDirs = shapeFlag & 1 /* ELEMENT */ && dirs;
|
|
let vnodeHook;
|
|
if ((vnodeHook = props && props.onVnodeBeforeUnmount)) {
|
|
invokeVNodeHook(vnodeHook, parentComponent, vnode);
|
|
}
|
|
if (shapeFlag & 6 /* COMPONENT */) {
|
|
unmountComponent(vnode.component, parentSuspense, doRemove);
|
|
}
|
|
else {
|
|
if ( shapeFlag & 128 /* SUSPENSE */) {
|
|
vnode.suspense.unmount(parentSuspense, doRemove);
|
|
return;
|
|
}
|
|
if (shouldInvokeDirs) {
|
|
invokeDirectiveHook(vnode, null, parentComponent, 'beforeUnmount');
|
|
}
|
|
if (dynamicChildren &&
|
|
// #1153: fast path should not be taken for non-stable (v-for) fragments
|
|
(type !== Fragment ||
|
|
(patchFlag > 0 && patchFlag & 64 /* STABLE_FRAGMENT */))) {
|
|
// fast path for block nodes: only need to unmount dynamic children.
|
|
unmountChildren(dynamicChildren, parentComponent, parentSuspense, false, true);
|
|
}
|
|
else if ((type === Fragment &&
|
|
(patchFlag & 128 /* KEYED_FRAGMENT */ ||
|
|
patchFlag & 256 /* UNKEYED_FRAGMENT */)) ||
|
|
(!optimized && shapeFlag & 16 /* ARRAY_CHILDREN */)) {
|
|
unmountChildren(children, parentComponent, parentSuspense);
|
|
}
|
|
// an unmounted teleport should always remove its children if not disabled
|
|
if (shapeFlag & 64 /* TELEPORT */ &&
|
|
(doRemove || !isTeleportDisabled(vnode.props))) {
|
|
vnode.type.remove(vnode, internals);
|
|
}
|
|
if (doRemove) {
|
|
remove(vnode);
|
|
}
|
|
}
|
|
if ((vnodeHook = props && props.onVnodeUnmounted) || shouldInvokeDirs) {
|
|
queuePostRenderEffect(() => {
|
|
vnodeHook && invokeVNodeHook(vnodeHook, parentComponent, vnode);
|
|
shouldInvokeDirs &&
|
|
invokeDirectiveHook(vnode, null, parentComponent, 'unmounted');
|
|
}, parentSuspense);
|
|
}
|
|
};
|
|
const remove = vnode => {
|
|
const { type, el, anchor, transition } = vnode;
|
|
if (type === Fragment) {
|
|
removeFragment(el, anchor);
|
|
return;
|
|
}
|
|
if (type === Static) {
|
|
removeStaticNode(vnode);
|
|
return;
|
|
}
|
|
const performRemove = () => {
|
|
hostRemove(el);
|
|
if (transition && !transition.persisted && transition.afterLeave) {
|
|
transition.afterLeave();
|
|
}
|
|
};
|
|
if (vnode.shapeFlag & 1 /* ELEMENT */ &&
|
|
transition &&
|
|
!transition.persisted) {
|
|
const { leave, delayLeave } = transition;
|
|
const performLeave = () => leave(el, performRemove);
|
|
if (delayLeave) {
|
|
delayLeave(vnode.el, performRemove, performLeave);
|
|
}
|
|
else {
|
|
performLeave();
|
|
}
|
|
}
|
|
else {
|
|
performRemove();
|
|
}
|
|
};
|
|
const removeFragment = (cur, end) => {
|
|
// For fragments, directly remove all contained DOM nodes.
|
|
// (fragment child nodes cannot have transition)
|
|
let next;
|
|
while (cur !== end) {
|
|
next = hostNextSibling(cur);
|
|
hostRemove(cur);
|
|
cur = next;
|
|
}
|
|
hostRemove(end);
|
|
};
|
|
const unmountComponent = (instance, parentSuspense, doRemove) => {
|
|
const { bum, effects, update, subTree, um } = instance;
|
|
// beforeUnmount hook
|
|
if (bum) {
|
|
invokeArrayFns(bum);
|
|
}
|
|
if (effects) {
|
|
for (let i = 0; i < effects.length; i++) {
|
|
stop(effects[i]);
|
|
}
|
|
}
|
|
// update may be null if a component is unmounted before its async
|
|
// setup has resolved.
|
|
if (update) {
|
|
stop(update);
|
|
unmount(subTree, instance, parentSuspense, doRemove);
|
|
}
|
|
// unmounted hook
|
|
if (um) {
|
|
queuePostRenderEffect(um, parentSuspense);
|
|
}
|
|
queuePostRenderEffect(() => {
|
|
instance.isUnmounted = true;
|
|
}, parentSuspense);
|
|
// A component with async dep inside a pending suspense is unmounted before
|
|
// its async dep resolves. This should remove the dep from the suspense, and
|
|
// cause the suspense to resolve immediately if that was the last dep.
|
|
if (
|
|
parentSuspense &&
|
|
parentSuspense.pendingBranch &&
|
|
!parentSuspense.isUnmounted &&
|
|
instance.asyncDep &&
|
|
!instance.asyncResolved &&
|
|
instance.suspenseId === parentSuspense.pendingId) {
|
|
parentSuspense.deps--;
|
|
if (parentSuspense.deps === 0) {
|
|
parentSuspense.resolve();
|
|
}
|
|
}
|
|
};
|
|
const unmountChildren = (children, parentComponent, parentSuspense, doRemove = false, optimized = false, start = 0) => {
|
|
for (let i = start; i < children.length; i++) {
|
|
unmount(children[i], parentComponent, parentSuspense, doRemove, optimized);
|
|
}
|
|
};
|
|
const getNextHostNode = vnode => {
|
|
if (vnode.shapeFlag & 6 /* COMPONENT */) {
|
|
return getNextHostNode(vnode.component.subTree);
|
|
}
|
|
if ( vnode.shapeFlag & 128 /* SUSPENSE */) {
|
|
return vnode.suspense.next();
|
|
}
|
|
return hostNextSibling((vnode.anchor || vnode.el));
|
|
};
|
|
const render = (vnode, container) => {
|
|
if (vnode == null) {
|
|
if (container._vnode) {
|
|
unmount(container._vnode, null, null, true);
|
|
}
|
|
}
|
|
else {
|
|
patch(container._vnode || null, vnode, container);
|
|
}
|
|
flushPostFlushCbs();
|
|
container._vnode = vnode;
|
|
};
|
|
const internals = {
|
|
p: patch,
|
|
um: unmount,
|
|
m: move,
|
|
r: remove,
|
|
mt: mountComponent,
|
|
mc: mountChildren,
|
|
pc: patchChildren,
|
|
pbc: patchBlockChildren,
|
|
n: getNextHostNode,
|
|
o: options
|
|
};
|
|
let hydrate;
|
|
let hydrateNode;
|
|
if (createHydrationFns) {
|
|
[hydrate, hydrateNode] = createHydrationFns(internals);
|
|
}
|
|
return {
|
|
render,
|
|
hydrate,
|
|
createApp: createAppAPI(render, hydrate)
|
|
};
|
|
}
|
|
function invokeVNodeHook(hook, instance, vnode, prevVNode = null) {
|
|
callWithAsyncErrorHandling(hook, instance, 7 /* VNODE_HOOK */, [
|
|
vnode,
|
|
prevVNode
|
|
]);
|
|
}
|
|
/**
|
|
* #1156
|
|
* When a component is HMR-enabled, we need to make sure that all static nodes
|
|
* inside a block also inherit the DOM element from the previous tree so that
|
|
* HMR updates (which are full updates) can retrieve the element for patching.
|
|
*
|
|
* #2080
|
|
* Inside keyed `template` fragment static children, if a fragment is moved,
|
|
* the children will always moved so that need inherit el form previous nodes
|
|
* to ensure correct moved position.
|
|
*/
|
|
function traverseStaticChildren(n1, n2, shallow = false) {
|
|
const ch1 = n1.children;
|
|
const ch2 = n2.children;
|
|
if (isArray(ch1) && isArray(ch2)) {
|
|
for (let i = 0; i < ch1.length; i++) {
|
|
// this is only called in the optimized path so array children are
|
|
// guaranteed to be vnodes
|
|
const c1 = ch1[i];
|
|
let c2 = ch2[i];
|
|
if (c2.shapeFlag & 1 /* ELEMENT */ && !c2.dynamicChildren) {
|
|
if (c2.patchFlag <= 0 || c2.patchFlag === 32 /* HYDRATE_EVENTS */) {
|
|
c2 = ch2[i] = cloneIfMounted(ch2[i]);
|
|
c2.el = c1.el;
|
|
}
|
|
if (!shallow)
|
|
traverseStaticChildren(c1, c2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// https://en.wikipedia.org/wiki/Longest_increasing_subsequence
|
|
function getSequence(arr) {
|
|
const p = arr.slice();
|
|
const result = [0];
|
|
let i, j, u, v, c;
|
|
const len = arr.length;
|
|
for (i = 0; i < len; i++) {
|
|
const arrI = arr[i];
|
|
if (arrI !== 0) {
|
|
j = result[result.length - 1];
|
|
if (arr[j] < arrI) {
|
|
p[i] = j;
|
|
result.push(i);
|
|
continue;
|
|
}
|
|
u = 0;
|
|
v = result.length - 1;
|
|
while (u < v) {
|
|
c = ((u + v) / 2) | 0;
|
|
if (arr[result[c]] < arrI) {
|
|
u = c + 1;
|
|
}
|
|
else {
|
|
v = c;
|
|
}
|
|
}
|
|
if (arrI < arr[result[u]]) {
|
|
if (u > 0) {
|
|
p[i] = result[u - 1];
|
|
}
|
|
result[u] = i;
|
|
}
|
|
}
|
|
}
|
|
u = result.length;
|
|
v = result[u - 1];
|
|
while (u-- > 0) {
|
|
result[u] = v;
|
|
v = p[v];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
const isTeleport = (type) => type.__isTeleport;
|
|
const isTeleportDisabled = (props) => props && (props.disabled || props.disabled === '');
|
|
|
|
const COMPONENTS = 'components';
|
|
/**
|
|
* @private
|
|
*/
|
|
function resolveComponent(name) {
|
|
return resolveAsset(COMPONENTS, name) || name;
|
|
}
|
|
const NULL_DYNAMIC_COMPONENT = Symbol();
|
|
// implementation
|
|
function resolveAsset(type, name, warnMissing = true) {
|
|
const instance = currentRenderingInstance || currentInstance;
|
|
if (instance) {
|
|
const Component = instance.type;
|
|
// self name has highest priority
|
|
if (type === COMPONENTS) {
|
|
// special self referencing call generated by compiler
|
|
// inferred from SFC filename
|
|
if (name === `_self`) {
|
|
return Component;
|
|
}
|
|
const selfName = getComponentName(Component);
|
|
if (selfName &&
|
|
(selfName === name ||
|
|
selfName === camelize(name) ||
|
|
selfName === capitalize(camelize(name)))) {
|
|
return Component;
|
|
}
|
|
}
|
|
const res =
|
|
// local registration
|
|
// check instance[type] first for components with mixin or extends.
|
|
resolve(instance[type] || Component[type], name) ||
|
|
// global registration
|
|
resolve(instance.appContext[type], name);
|
|
return res;
|
|
}
|
|
}
|
|
function resolve(registry, name) {
|
|
return (registry &&
|
|
(registry[name] ||
|
|
registry[camelize(name)] ||
|
|
registry[capitalize(camelize(name))]));
|
|
}
|
|
|
|
const Fragment = Symbol(undefined);
|
|
const Text = Symbol(undefined);
|
|
const Comment = Symbol(undefined);
|
|
const Static = Symbol(undefined);
|
|
// Since v-if and v-for are the two possible ways node structure can dynamically
|
|
// change, once we consider v-if branches and each v-for fragment a block, we
|
|
// can divide a template into nested blocks, and within each block the node
|
|
// structure would be stable. This allows us to skip most children diffing
|
|
// and only worry about the dynamic nodes (indicated by patch flags).
|
|
const blockStack = [];
|
|
let currentBlock = null;
|
|
/**
|
|
* Open a block.
|
|
* This must be called before `createBlock`. It cannot be part of `createBlock`
|
|
* because the children of the block are evaluated before `createBlock` itself
|
|
* is called. The generated code typically looks like this:
|
|
*
|
|
* ```js
|
|
* function render() {
|
|
* return (openBlock(),createBlock('div', null, [...]))
|
|
* }
|
|
* ```
|
|
* disableTracking is true when creating a v-for fragment block, since a v-for
|
|
* fragment always diffs its children.
|
|
*
|
|
* @private
|
|
*/
|
|
function openBlock(disableTracking = false) {
|
|
blockStack.push((currentBlock = disableTracking ? null : []));
|
|
}
|
|
function closeBlock() {
|
|
blockStack.pop();
|
|
currentBlock = blockStack[blockStack.length - 1] || null;
|
|
}
|
|
/**
|
|
* Create a block root vnode. Takes the same exact arguments as `createVNode`.
|
|
* A block root keeps track of dynamic nodes within the block in the
|
|
* `dynamicChildren` array.
|
|
*
|
|
* @private
|
|
*/
|
|
function createBlock(type, props, children, patchFlag, dynamicProps) {
|
|
const vnode = createVNode(type, props, children, patchFlag, dynamicProps, true /* isBlock: prevent a block from tracking itself */);
|
|
// save current block children on the block vnode
|
|
vnode.dynamicChildren = currentBlock || EMPTY_ARR;
|
|
// close block
|
|
closeBlock();
|
|
// a block is always going to be patched, so track it as a child of its
|
|
// parent block
|
|
if (currentBlock) {
|
|
currentBlock.push(vnode);
|
|
}
|
|
return vnode;
|
|
}
|
|
function isVNode(value) {
|
|
return value ? value.__v_isVNode === true : false;
|
|
}
|
|
function isSameVNodeType(n1, n2) {
|
|
return n1.type === n2.type && n1.key === n2.key;
|
|
}
|
|
const InternalObjectKey = `__vInternal`;
|
|
const normalizeKey = ({ key }) => key != null ? key : null;
|
|
const normalizeRef = ({ ref }) => {
|
|
return (ref != null
|
|
? isString(ref) || isRef(ref) || isFunction(ref)
|
|
? { i: currentRenderingInstance, r: ref }
|
|
: ref
|
|
: null);
|
|
};
|
|
const createVNode = (_createVNode);
|
|
function _createVNode(type, props = null, children = null, patchFlag = 0, dynamicProps = null, isBlockNode = false) {
|
|
if (!type || type === NULL_DYNAMIC_COMPONENT) {
|
|
type = Comment;
|
|
}
|
|
if (isVNode(type)) {
|
|
// createVNode receiving an existing vnode. This happens in cases like
|
|
// <component :is="vnode"/>
|
|
// #2078 make sure to merge refs during the clone instead of overwriting it
|
|
const cloned = cloneVNode(type, props, true /* mergeRef: true */);
|
|
if (children) {
|
|
normalizeChildren(cloned, children);
|
|
}
|
|
return cloned;
|
|
}
|
|
// class component normalization.
|
|
if (isClassComponent(type)) {
|
|
type = type.__vccOpts;
|
|
}
|
|
// class & style normalization.
|
|
if (props) {
|
|
// for reactive or proxy objects, we need to clone it to enable mutation.
|
|
if (isProxy(props) || InternalObjectKey in props) {
|
|
props = extend({}, props);
|
|
}
|
|
let { class: klass, style } = props;
|
|
if (klass && !isString(klass)) {
|
|
props.class = normalizeClass(klass);
|
|
}
|
|
if (isObject(style)) {
|
|
// reactive state objects need to be cloned since they are likely to be
|
|
// mutated
|
|
if (isProxy(style) && !isArray(style)) {
|
|
style = extend({}, style);
|
|
}
|
|
props.style = normalizeStyle(style);
|
|
}
|
|
}
|
|
// encode the vnode type information into a bitmap
|
|
const shapeFlag = isString(type)
|
|
? 1 /* ELEMENT */
|
|
: isSuspense(type)
|
|
? 128 /* SUSPENSE */
|
|
: isTeleport(type)
|
|
? 64 /* TELEPORT */
|
|
: isObject(type)
|
|
? 4 /* STATEFUL_COMPONENT */
|
|
: isFunction(type)
|
|
? 2 /* FUNCTIONAL_COMPONENT */
|
|
: 0;
|
|
const vnode = {
|
|
__v_isVNode: true,
|
|
["__v_skip" /* SKIP */]: true,
|
|
type,
|
|
props,
|
|
key: props && normalizeKey(props),
|
|
ref: props && normalizeRef(props),
|
|
scopeId: currentScopeId,
|
|
children: null,
|
|
component: null,
|
|
suspense: null,
|
|
ssContent: null,
|
|
ssFallback: null,
|
|
dirs: null,
|
|
transition: null,
|
|
el: null,
|
|
anchor: null,
|
|
target: null,
|
|
targetAnchor: null,
|
|
staticCount: 0,
|
|
shapeFlag,
|
|
patchFlag,
|
|
dynamicProps,
|
|
dynamicChildren: null,
|
|
appContext: null
|
|
};
|
|
normalizeChildren(vnode, children);
|
|
// normalize suspense children
|
|
if ( shapeFlag & 128 /* SUSPENSE */) {
|
|
const { content, fallback } = normalizeSuspenseChildren(vnode);
|
|
vnode.ssContent = content;
|
|
vnode.ssFallback = fallback;
|
|
}
|
|
if (// avoid a block node from tracking itself
|
|
!isBlockNode &&
|
|
// has current parent block
|
|
currentBlock &&
|
|
// presence of a patch flag indicates this node needs patching on updates.
|
|
// component nodes also should always be patched, because even if the
|
|
// component doesn't need to update, it needs to persist the instance on to
|
|
// the next vnode so that it can be properly unmounted later.
|
|
(patchFlag > 0 || shapeFlag & 6 /* COMPONENT */) &&
|
|
// the EVENTS flag is only for hydration and if it is the only flag, the
|
|
// vnode should not be considered dynamic due to handler caching.
|
|
patchFlag !== 32 /* HYDRATE_EVENTS */) {
|
|
currentBlock.push(vnode);
|
|
}
|
|
return vnode;
|
|
}
|
|
function cloneVNode(vnode, extraProps, mergeRef = false) {
|
|
// This is intentionally NOT using spread or extend to avoid the runtime
|
|
// key enumeration cost.
|
|
const { props, ref, patchFlag } = vnode;
|
|
const mergedProps = extraProps ? mergeProps(props || {}, extraProps) : props;
|
|
return {
|
|
__v_isVNode: true,
|
|
["__v_skip" /* SKIP */]: true,
|
|
type: vnode.type,
|
|
props: mergedProps,
|
|
key: mergedProps && normalizeKey(mergedProps),
|
|
ref: extraProps && extraProps.ref
|
|
? // #2078 in the case of <component :is="vnode" ref="extra"/>
|
|
// if the vnode itself already has a ref, cloneVNode will need to merge
|
|
// the refs so the single vnode can be set on multiple refs
|
|
mergeRef && ref
|
|
? isArray(ref)
|
|
? ref.concat(normalizeRef(extraProps))
|
|
: [ref, normalizeRef(extraProps)]
|
|
: normalizeRef(extraProps)
|
|
: ref,
|
|
scopeId: vnode.scopeId,
|
|
children: vnode.children,
|
|
target: vnode.target,
|
|
targetAnchor: vnode.targetAnchor,
|
|
staticCount: vnode.staticCount,
|
|
shapeFlag: vnode.shapeFlag,
|
|
// if the vnode is cloned with extra props, we can no longer assume its
|
|
// existing patch flag to be reliable and need to add the FULL_PROPS flag.
|
|
// note: perserve flag for fragments since they use the flag for children
|
|
// fast paths only.
|
|
patchFlag: extraProps && vnode.type !== Fragment
|
|
? patchFlag === -1 // hoisted node
|
|
? 16 /* FULL_PROPS */
|
|
: patchFlag | 16 /* FULL_PROPS */
|
|
: patchFlag,
|
|
dynamicProps: vnode.dynamicProps,
|
|
dynamicChildren: vnode.dynamicChildren,
|
|
appContext: vnode.appContext,
|
|
dirs: vnode.dirs,
|
|
transition: vnode.transition,
|
|
// These should technically only be non-null on mounted VNodes. However,
|
|
// they *should* be copied for kept-alive vnodes. So we just always copy
|
|
// them since them being non-null during a mount doesn't affect the logic as
|
|
// they will simply be overwritten.
|
|
component: vnode.component,
|
|
suspense: vnode.suspense,
|
|
ssContent: vnode.ssContent && cloneVNode(vnode.ssContent),
|
|
ssFallback: vnode.ssFallback && cloneVNode(vnode.ssFallback),
|
|
el: vnode.el,
|
|
anchor: vnode.anchor
|
|
};
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
function createTextVNode(text = ' ', flag = 0) {
|
|
return createVNode(Text, null, text, flag);
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
function createCommentVNode(text = '',
|
|
// when used as the v-else branch, the comment node must be created as a
|
|
// block to ensure correct updates.
|
|
asBlock = false) {
|
|
return asBlock
|
|
? (openBlock(), createBlock(Comment, null, text))
|
|
: createVNode(Comment, null, text);
|
|
}
|
|
function normalizeVNode(child) {
|
|
if (child == null || typeof child === 'boolean') {
|
|
// empty placeholder
|
|
return createVNode(Comment);
|
|
}
|
|
else if (isArray(child)) {
|
|
// fragment
|
|
return createVNode(Fragment, null, child);
|
|
}
|
|
else if (typeof child === 'object') {
|
|
// already vnode, this should be the most common since compiled templates
|
|
// always produce all-vnode children arrays
|
|
return child.el === null ? child : cloneVNode(child);
|
|
}
|
|
else {
|
|
// strings and numbers
|
|
return createVNode(Text, null, String(child));
|
|
}
|
|
}
|
|
// optimized normalization for template-compiled render fns
|
|
function cloneIfMounted(child) {
|
|
return child.el === null ? child : cloneVNode(child);
|
|
}
|
|
function normalizeChildren(vnode, children) {
|
|
let type = 0;
|
|
const { shapeFlag } = vnode;
|
|
if (children == null) {
|
|
children = null;
|
|
}
|
|
else if (isArray(children)) {
|
|
type = 16 /* ARRAY_CHILDREN */;
|
|
}
|
|
else if (typeof children === 'object') {
|
|
if (shapeFlag & 1 /* ELEMENT */ || shapeFlag & 64 /* TELEPORT */) {
|
|
// Normalize slot to plain children for plain element and Teleport
|
|
const slot = children.default;
|
|
if (slot) {
|
|
// _c marker is added by withCtx() indicating this is a compiled slot
|
|
slot._c && setCompiledSlotRendering(1);
|
|
normalizeChildren(vnode, slot());
|
|
slot._c && setCompiledSlotRendering(-1);
|
|
}
|
|
return;
|
|
}
|
|
else {
|
|
type = 32 /* SLOTS_CHILDREN */;
|
|
const slotFlag = children._;
|
|
if (!slotFlag && !(InternalObjectKey in children)) {
|
|
children._ctx = currentRenderingInstance;
|
|
}
|
|
else if (slotFlag === 3 /* FORWARDED */ && currentRenderingInstance) {
|
|
// a child component receives forwarded slots from the parent.
|
|
// its slot type is determined by its parent's slot type.
|
|
if (currentRenderingInstance.vnode.patchFlag & 1024 /* DYNAMIC_SLOTS */) {
|
|
children._ = 2 /* DYNAMIC */;
|
|
vnode.patchFlag |= 1024 /* DYNAMIC_SLOTS */;
|
|
}
|
|
else {
|
|
children._ = 1 /* STABLE */;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (isFunction(children)) {
|
|
children = { default: children, _ctx: currentRenderingInstance };
|
|
type = 32 /* SLOTS_CHILDREN */;
|
|
}
|
|
else {
|
|
children = String(children);
|
|
// force teleport children to array so it can be moved around
|
|
if (shapeFlag & 64 /* TELEPORT */) {
|
|
type = 16 /* ARRAY_CHILDREN */;
|
|
children = [createTextVNode(children)];
|
|
}
|
|
else {
|
|
type = 8 /* TEXT_CHILDREN */;
|
|
}
|
|
}
|
|
vnode.children = children;
|
|
vnode.shapeFlag |= type;
|
|
}
|
|
function mergeProps(...args) {
|
|
const ret = extend({}, args[0]);
|
|
for (let i = 1; i < args.length; i++) {
|
|
const toMerge = args[i];
|
|
for (const key in toMerge) {
|
|
if (key === 'class') {
|
|
if (ret.class !== toMerge.class) {
|
|
ret.class = normalizeClass([ret.class, toMerge.class]);
|
|
}
|
|
}
|
|
else if (key === 'style') {
|
|
ret.style = normalizeStyle([ret.style, toMerge.style]);
|
|
}
|
|
else if (isOn(key)) {
|
|
const existing = ret[key];
|
|
const incoming = toMerge[key];
|
|
if (existing !== incoming) {
|
|
ret[key] = existing
|
|
? [].concat(existing, toMerge[key])
|
|
: incoming;
|
|
}
|
|
}
|
|
else if (key !== '') {
|
|
ret[key] = toMerge[key];
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
function provide(key, value) {
|
|
if (!currentInstance) ;
|
|
else {
|
|
let provides = currentInstance.provides;
|
|
// by default an instance inherits its parent's provides object
|
|
// but when it needs to provide values of its own, it creates its
|
|
// own provides object using parent provides object as prototype.
|
|
// this way in `inject` we can simply look up injections from direct
|
|
// parent and let the prototype chain do the work.
|
|
const parentProvides = currentInstance.parent && currentInstance.parent.provides;
|
|
if (parentProvides === provides) {
|
|
provides = currentInstance.provides = Object.create(parentProvides);
|
|
}
|
|
// TS doesn't allow symbol as index type
|
|
provides[key] = value;
|
|
}
|
|
}
|
|
function inject(key, defaultValue, treatDefaultAsFactory = false) {
|
|
// fallback to `currentRenderingInstance` so that this can be called in
|
|
// a functional component
|
|
const instance = currentInstance || currentRenderingInstance;
|
|
if (instance) {
|
|
// #2400
|
|
// to support `app.use` plugins,
|
|
// fallback to appContext's `provides` if the intance is at root
|
|
const provides = instance.parent == null
|
|
? instance.vnode.appContext && instance.vnode.appContext.provides
|
|
: instance.parent.provides;
|
|
if (provides && key in provides) {
|
|
// TS doesn't allow symbol as index type
|
|
return provides[key];
|
|
}
|
|
else if (arguments.length > 1) {
|
|
return treatDefaultAsFactory && isFunction(defaultValue)
|
|
? defaultValue()
|
|
: defaultValue;
|
|
}
|
|
else ;
|
|
}
|
|
}
|
|
let isInBeforeCreate = false;
|
|
function applyOptions(instance, options, deferredData = [], deferredWatch = [], deferredProvide = [], asMixin = false) {
|
|
const {
|
|
// composition
|
|
mixins, extends: extendsOptions,
|
|
// state
|
|
data: dataOptions, computed: computedOptions, methods, watch: watchOptions, provide: provideOptions, inject: injectOptions,
|
|
// assets
|
|
components, directives,
|
|
// lifecycle
|
|
beforeMount, mounted, beforeUpdate, updated, activated, deactivated, beforeDestroy, beforeUnmount, destroyed, unmounted, render, renderTracked, renderTriggered, errorCaptured,
|
|
// public API
|
|
expose } = options;
|
|
const publicThis = instance.proxy;
|
|
const ctx = instance.ctx;
|
|
const globalMixins = instance.appContext.mixins;
|
|
if (asMixin && render && instance.render === NOOP) {
|
|
instance.render = render;
|
|
}
|
|
// applyOptions is called non-as-mixin once per instance
|
|
if (!asMixin) {
|
|
isInBeforeCreate = true;
|
|
callSyncHook('beforeCreate', "bc" /* BEFORE_CREATE */, options, instance, globalMixins);
|
|
isInBeforeCreate = false;
|
|
// global mixins are applied first
|
|
applyMixins(instance, globalMixins, deferredData, deferredWatch, deferredProvide);
|
|
}
|
|
// extending a base component...
|
|
if (extendsOptions) {
|
|
applyOptions(instance, extendsOptions, deferredData, deferredWatch, deferredProvide, true);
|
|
}
|
|
// local mixins
|
|
if (mixins) {
|
|
applyMixins(instance, mixins, deferredData, deferredWatch, deferredProvide);
|
|
}
|
|
// options initialization order (to be consistent with Vue 2):
|
|
// - props (already done outside of this function)
|
|
// - inject
|
|
// - methods
|
|
// - data (deferred since it relies on `this` access)
|
|
// - computed
|
|
// - watch (deferred since it relies on `this` access)
|
|
if (injectOptions) {
|
|
if (isArray(injectOptions)) {
|
|
for (let i = 0; i < injectOptions.length; i++) {
|
|
const key = injectOptions[i];
|
|
ctx[key] = inject(key);
|
|
}
|
|
}
|
|
else {
|
|
for (const key in injectOptions) {
|
|
const opt = injectOptions[key];
|
|
if (isObject(opt)) {
|
|
ctx[key] = inject(opt.from || key, opt.default, true /* treat default function as factory */);
|
|
}
|
|
else {
|
|
ctx[key] = inject(opt);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (methods) {
|
|
for (const key in methods) {
|
|
const methodHandler = methods[key];
|
|
if (isFunction(methodHandler)) {
|
|
ctx[key] = methodHandler.bind(publicThis);
|
|
}
|
|
}
|
|
}
|
|
if (!asMixin) {
|
|
if (deferredData.length) {
|
|
deferredData.forEach(dataFn => resolveData(instance, dataFn, publicThis));
|
|
}
|
|
if (dataOptions) {
|
|
// @ts-ignore dataOptions is not fully type safe
|
|
resolveData(instance, dataOptions, publicThis);
|
|
}
|
|
}
|
|
else if (dataOptions) {
|
|
deferredData.push(dataOptions);
|
|
}
|
|
if (computedOptions) {
|
|
for (const key in computedOptions) {
|
|
const opt = computedOptions[key];
|
|
const get = isFunction(opt)
|
|
? opt.bind(publicThis, publicThis)
|
|
: isFunction(opt.get)
|
|
? opt.get.bind(publicThis, publicThis)
|
|
: NOOP;
|
|
const set = !isFunction(opt) && isFunction(opt.set)
|
|
? opt.set.bind(publicThis)
|
|
: NOOP;
|
|
const c = computed$1({
|
|
get,
|
|
set
|
|
});
|
|
Object.defineProperty(ctx, key, {
|
|
enumerable: true,
|
|
configurable: true,
|
|
get: () => c.value,
|
|
set: v => (c.value = v)
|
|
});
|
|
}
|
|
}
|
|
if (watchOptions) {
|
|
deferredWatch.push(watchOptions);
|
|
}
|
|
if (!asMixin && deferredWatch.length) {
|
|
deferredWatch.forEach(watchOptions => {
|
|
for (const key in watchOptions) {
|
|
createWatcher(watchOptions[key], ctx, publicThis, key);
|
|
}
|
|
});
|
|
}
|
|
if (provideOptions) {
|
|
deferredProvide.push(provideOptions);
|
|
}
|
|
if (!asMixin && deferredProvide.length) {
|
|
deferredProvide.forEach(provideOptions => {
|
|
const provides = isFunction(provideOptions)
|
|
? provideOptions.call(publicThis)
|
|
: provideOptions;
|
|
Reflect.ownKeys(provides).forEach(key => {
|
|
provide(key, provides[key]);
|
|
});
|
|
});
|
|
}
|
|
// asset options.
|
|
// To reduce memory usage, only components with mixins or extends will have
|
|
// resolved asset registry attached to instance.
|
|
if (asMixin) {
|
|
if (components) {
|
|
extend(instance.components ||
|
|
(instance.components = extend({}, instance.type.components)), components);
|
|
}
|
|
if (directives) {
|
|
extend(instance.directives ||
|
|
(instance.directives = extend({}, instance.type.directives)), directives);
|
|
}
|
|
}
|
|
// lifecycle options
|
|
if (!asMixin) {
|
|
callSyncHook('created', "c" /* CREATED */, options, instance, globalMixins);
|
|
}
|
|
if (beforeMount) {
|
|
onBeforeMount(beforeMount.bind(publicThis));
|
|
}
|
|
if (mounted) {
|
|
onMounted(mounted.bind(publicThis));
|
|
}
|
|
if (beforeUpdate) {
|
|
onBeforeUpdate(beforeUpdate.bind(publicThis));
|
|
}
|
|
if (updated) {
|
|
onUpdated(updated.bind(publicThis));
|
|
}
|
|
if (activated) {
|
|
onActivated(activated.bind(publicThis));
|
|
}
|
|
if (deactivated) {
|
|
onDeactivated(deactivated.bind(publicThis));
|
|
}
|
|
if (errorCaptured) {
|
|
onErrorCaptured(errorCaptured.bind(publicThis));
|
|
}
|
|
if (renderTracked) {
|
|
onRenderTracked(renderTracked.bind(publicThis));
|
|
}
|
|
if (renderTriggered) {
|
|
onRenderTriggered(renderTriggered.bind(publicThis));
|
|
}
|
|
if (beforeUnmount) {
|
|
onBeforeUnmount(beforeUnmount.bind(publicThis));
|
|
}
|
|
if (unmounted) {
|
|
onUnmounted(unmounted.bind(publicThis));
|
|
}
|
|
if (isArray(expose)) {
|
|
if (!asMixin) {
|
|
if (expose.length) {
|
|
const exposed = instance.exposed || (instance.exposed = proxyRefs({}));
|
|
expose.forEach(key => {
|
|
exposed[key] = toRef(publicThis, key);
|
|
});
|
|
}
|
|
else if (!instance.exposed) {
|
|
instance.exposed = EMPTY_OBJ;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
function callSyncHook(name, type, options, instance, globalMixins) {
|
|
callHookFromMixins(name, type, globalMixins, instance);
|
|
const { extends: base, mixins } = options;
|
|
if (base) {
|
|
callHookFromExtends(name, type, base, instance);
|
|
}
|
|
if (mixins) {
|
|
callHookFromMixins(name, type, mixins, instance);
|
|
}
|
|
const selfHook = options[name];
|
|
if (selfHook) {
|
|
callWithAsyncErrorHandling(selfHook.bind(instance.proxy), instance, type);
|
|
}
|
|
}
|
|
function callHookFromExtends(name, type, base, instance) {
|
|
if (base.extends) {
|
|
callHookFromExtends(name, type, base.extends, instance);
|
|
}
|
|
const baseHook = base[name];
|
|
if (baseHook) {
|
|
callWithAsyncErrorHandling(baseHook.bind(instance.proxy), instance, type);
|
|
}
|
|
}
|
|
function callHookFromMixins(name, type, mixins, instance) {
|
|
for (let i = 0; i < mixins.length; i++) {
|
|
const chainedMixins = mixins[i].mixins;
|
|
if (chainedMixins) {
|
|
callHookFromMixins(name, type, chainedMixins, instance);
|
|
}
|
|
const fn = mixins[i][name];
|
|
if (fn) {
|
|
callWithAsyncErrorHandling(fn.bind(instance.proxy), instance, type);
|
|
}
|
|
}
|
|
}
|
|
function applyMixins(instance, mixins, deferredData, deferredWatch, deferredProvide) {
|
|
for (let i = 0; i < mixins.length; i++) {
|
|
applyOptions(instance, mixins[i], deferredData, deferredWatch, deferredProvide, true);
|
|
}
|
|
}
|
|
function resolveData(instance, dataFn, publicThis) {
|
|
const data = dataFn.call(publicThis, publicThis);
|
|
if (!isObject(data)) ;
|
|
else if (instance.data === EMPTY_OBJ) {
|
|
instance.data = reactive(data);
|
|
}
|
|
else {
|
|
// existing data: this is a mixin or extends.
|
|
extend(instance.data, data);
|
|
}
|
|
}
|
|
function createWatcher(raw, ctx, publicThis, key) {
|
|
const getter = key.includes('.')
|
|
? createPathGetter(publicThis, key)
|
|
: () => publicThis[key];
|
|
if (isString(raw)) {
|
|
const handler = ctx[raw];
|
|
if (isFunction(handler)) {
|
|
watch(getter, handler);
|
|
}
|
|
}
|
|
else if (isFunction(raw)) {
|
|
watch(getter, raw.bind(publicThis));
|
|
}
|
|
else if (isObject(raw)) {
|
|
if (isArray(raw)) {
|
|
raw.forEach(r => createWatcher(r, ctx, publicThis, key));
|
|
}
|
|
else {
|
|
const handler = isFunction(raw.handler)
|
|
? raw.handler.bind(publicThis)
|
|
: ctx[raw.handler];
|
|
if (isFunction(handler)) {
|
|
watch(getter, handler, raw);
|
|
}
|
|
}
|
|
}
|
|
else ;
|
|
}
|
|
function createPathGetter(ctx, path) {
|
|
const segments = path.split('.');
|
|
return () => {
|
|
let cur = ctx;
|
|
for (let i = 0; i < segments.length && cur; i++) {
|
|
cur = cur[segments[i]];
|
|
}
|
|
return cur;
|
|
};
|
|
}
|
|
function resolveMergedOptions(instance) {
|
|
const raw = instance.type;
|
|
const { __merged, mixins, extends: extendsOptions } = raw;
|
|
if (__merged)
|
|
return __merged;
|
|
const globalMixins = instance.appContext.mixins;
|
|
if (!globalMixins.length && !mixins && !extendsOptions)
|
|
return raw;
|
|
const options = {};
|
|
globalMixins.forEach(m => mergeOptions(options, m, instance));
|
|
mergeOptions(options, raw, instance);
|
|
return (raw.__merged = options);
|
|
}
|
|
function mergeOptions(to, from, instance) {
|
|
const strats = instance.appContext.config.optionMergeStrategies;
|
|
const { mixins, extends: extendsOptions } = from;
|
|
extendsOptions && mergeOptions(to, extendsOptions, instance);
|
|
mixins &&
|
|
mixins.forEach((m) => mergeOptions(to, m, instance));
|
|
for (const key in from) {
|
|
if (strats && hasOwn(strats, key)) {
|
|
to[key] = strats[key](to[key], from[key], instance.proxy, key);
|
|
}
|
|
else {
|
|
to[key] = from[key];
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* #2437 In Vue 3, functional components do not have a public instance proxy but
|
|
* they exist in the internal parent chain. For code that relies on traversing
|
|
* public $parent chains, skip functional ones and go to the parent instead.
|
|
*/
|
|
const getPublicInstance = (i) => i && (i.proxy ? i.proxy : getPublicInstance(i.parent));
|
|
const publicPropertiesMap = extend(Object.create(null), {
|
|
$: i => i,
|
|
$el: i => i.vnode.el,
|
|
$data: i => i.data,
|
|
$props: i => (i.props),
|
|
$attrs: i => (i.attrs),
|
|
$slots: i => (i.slots),
|
|
$refs: i => (i.refs),
|
|
$parent: i => getPublicInstance(i.parent),
|
|
$root: i => i.root && i.root.proxy,
|
|
$emit: i => i.emit,
|
|
$options: i => (resolveMergedOptions(i) ),
|
|
$forceUpdate: i => () => queueJob(i.update),
|
|
$nextTick: i => nextTick.bind(i.proxy),
|
|
$watch: i => (instanceWatch.bind(i) )
|
|
});
|
|
const PublicInstanceProxyHandlers = {
|
|
get({ _: instance }, key) {
|
|
const { ctx, setupState, data, props, accessCache, type, appContext } = instance;
|
|
// let @vue/reactivity know it should never observe Vue public instances.
|
|
if (key === "__v_skip" /* SKIP */) {
|
|
return true;
|
|
}
|
|
// data / props / ctx
|
|
// This getter gets called for every property access on the render context
|
|
// during render and is a major hotspot. The most expensive part of this
|
|
// is the multiple hasOwn() calls. It's much faster to do a simple property
|
|
// access on a plain object, so we use an accessCache object (with null
|
|
// prototype) to memoize what access type a key corresponds to.
|
|
let normalizedProps;
|
|
if (key[0] !== '$') {
|
|
const n = accessCache[key];
|
|
if (n !== undefined) {
|
|
switch (n) {
|
|
case 0 /* SETUP */:
|
|
return setupState[key];
|
|
case 1 /* DATA */:
|
|
return data[key];
|
|
case 3 /* CONTEXT */:
|
|
return ctx[key];
|
|
case 2 /* PROPS */:
|
|
return props[key];
|
|
// default: just fallthrough
|
|
}
|
|
}
|
|
else if (setupState !== EMPTY_OBJ && hasOwn(setupState, key)) {
|
|
accessCache[key] = 0 /* SETUP */;
|
|
return setupState[key];
|
|
}
|
|
else if (data !== EMPTY_OBJ && hasOwn(data, key)) {
|
|
accessCache[key] = 1 /* DATA */;
|
|
return data[key];
|
|
}
|
|
else if (
|
|
// only cache other properties when instance has declared (thus stable)
|
|
// props
|
|
(normalizedProps = instance.propsOptions[0]) &&
|
|
hasOwn(normalizedProps, key)) {
|
|
accessCache[key] = 2 /* PROPS */;
|
|
return props[key];
|
|
}
|
|
else if (ctx !== EMPTY_OBJ && hasOwn(ctx, key)) {
|
|
accessCache[key] = 3 /* CONTEXT */;
|
|
return ctx[key];
|
|
}
|
|
else if (!isInBeforeCreate) {
|
|
accessCache[key] = 4 /* OTHER */;
|
|
}
|
|
}
|
|
const publicGetter = publicPropertiesMap[key];
|
|
let cssModule, globalProperties;
|
|
// public $xxx properties
|
|
if (publicGetter) {
|
|
if (key === '$attrs') {
|
|
track(instance, "get" /* GET */, key);
|
|
}
|
|
return publicGetter(instance);
|
|
}
|
|
else if (
|
|
// css module (injected by vue-loader)
|
|
(cssModule = type.__cssModules) &&
|
|
(cssModule = cssModule[key])) {
|
|
return cssModule;
|
|
}
|
|
else if (ctx !== EMPTY_OBJ && hasOwn(ctx, key)) {
|
|
// user may set custom properties to `this` that start with `$`
|
|
accessCache[key] = 3 /* CONTEXT */;
|
|
return ctx[key];
|
|
}
|
|
else if (
|
|
// global properties
|
|
((globalProperties = appContext.config.globalProperties),
|
|
hasOwn(globalProperties, key))) {
|
|
return globalProperties[key];
|
|
}
|
|
else ;
|
|
},
|
|
set({ _: instance }, key, value) {
|
|
const { data, setupState, ctx } = instance;
|
|
if (setupState !== EMPTY_OBJ && hasOwn(setupState, key)) {
|
|
setupState[key] = value;
|
|
}
|
|
else if (data !== EMPTY_OBJ && hasOwn(data, key)) {
|
|
data[key] = value;
|
|
}
|
|
else if (key in instance.props) {
|
|
return false;
|
|
}
|
|
if (key[0] === '$' && key.slice(1) in instance) {
|
|
return false;
|
|
}
|
|
else {
|
|
{
|
|
ctx[key] = value;
|
|
}
|
|
}
|
|
return true;
|
|
},
|
|
has({ _: { data, setupState, accessCache, ctx, appContext, propsOptions } }, key) {
|
|
let normalizedProps;
|
|
return (accessCache[key] !== undefined ||
|
|
(data !== EMPTY_OBJ && hasOwn(data, key)) ||
|
|
(setupState !== EMPTY_OBJ && hasOwn(setupState, key)) ||
|
|
((normalizedProps = propsOptions[0]) && hasOwn(normalizedProps, key)) ||
|
|
hasOwn(ctx, key) ||
|
|
hasOwn(publicPropertiesMap, key) ||
|
|
hasOwn(appContext.config.globalProperties, key));
|
|
}
|
|
};
|
|
const RuntimeCompiledPublicInstanceProxyHandlers = extend({}, PublicInstanceProxyHandlers, {
|
|
get(target, key) {
|
|
// fast path for unscopables when using `with` block
|
|
if (key === Symbol.unscopables) {
|
|
return;
|
|
}
|
|
return PublicInstanceProxyHandlers.get(target, key, target);
|
|
},
|
|
has(_, key) {
|
|
const has = key[0] !== '_' && !isGloballyWhitelisted(key);
|
|
return has;
|
|
}
|
|
});
|
|
|
|
const emptyAppContext = createAppContext();
|
|
let uid$1$1 = 0;
|
|
function createComponentInstance(vnode, parent, suspense) {
|
|
const type = vnode.type;
|
|
// inherit parent app context - or - if root, adopt from root vnode
|
|
const appContext = (parent ? parent.appContext : vnode.appContext) || emptyAppContext;
|
|
const instance = {
|
|
uid: uid$1$1++,
|
|
vnode,
|
|
type,
|
|
parent,
|
|
appContext,
|
|
root: null,
|
|
next: null,
|
|
subTree: null,
|
|
update: null,
|
|
render: null,
|
|
proxy: null,
|
|
exposed: null,
|
|
withProxy: null,
|
|
effects: null,
|
|
provides: parent ? parent.provides : Object.create(appContext.provides),
|
|
accessCache: null,
|
|
renderCache: [],
|
|
// local resovled assets
|
|
components: null,
|
|
directives: null,
|
|
// resolved props and emits options
|
|
propsOptions: normalizePropsOptions(type, appContext),
|
|
emitsOptions: normalizeEmitsOptions(type, appContext),
|
|
// emit
|
|
emit: null,
|
|
emitted: null,
|
|
// state
|
|
ctx: EMPTY_OBJ,
|
|
data: EMPTY_OBJ,
|
|
props: EMPTY_OBJ,
|
|
attrs: EMPTY_OBJ,
|
|
slots: EMPTY_OBJ,
|
|
refs: EMPTY_OBJ,
|
|
setupState: EMPTY_OBJ,
|
|
setupContext: null,
|
|
// suspense related
|
|
suspense,
|
|
suspenseId: suspense ? suspense.pendingId : 0,
|
|
asyncDep: null,
|
|
asyncResolved: false,
|
|
// lifecycle hooks
|
|
// not using enums here because it results in computed properties
|
|
isMounted: false,
|
|
isUnmounted: false,
|
|
isDeactivated: false,
|
|
bc: null,
|
|
c: null,
|
|
bm: null,
|
|
m: null,
|
|
bu: null,
|
|
u: null,
|
|
um: null,
|
|
bum: null,
|
|
da: null,
|
|
a: null,
|
|
rtg: null,
|
|
rtc: null,
|
|
ec: null
|
|
};
|
|
{
|
|
instance.ctx = { _: instance };
|
|
}
|
|
instance.root = parent ? parent.root : instance;
|
|
instance.emit = emit.bind(null, instance);
|
|
return instance;
|
|
}
|
|
let currentInstance = null;
|
|
const setCurrentInstance = (instance) => {
|
|
currentInstance = instance;
|
|
};
|
|
let isInSSRComponentSetup = false;
|
|
function setupComponent(instance, isSSR = false) {
|
|
isInSSRComponentSetup = isSSR;
|
|
const { props, children, shapeFlag } = instance.vnode;
|
|
const isStateful = shapeFlag & 4 /* STATEFUL_COMPONENT */;
|
|
initProps(instance, props, isStateful, isSSR);
|
|
initSlots(instance, children);
|
|
const setupResult = isStateful
|
|
? setupStatefulComponent(instance, isSSR)
|
|
: undefined;
|
|
isInSSRComponentSetup = false;
|
|
return setupResult;
|
|
}
|
|
function setupStatefulComponent(instance, isSSR) {
|
|
const Component = instance.type;
|
|
// 0. create render proxy property access cache
|
|
instance.accessCache = Object.create(null);
|
|
// 1. create public instance / render proxy
|
|
// also mark it raw so it's never observed
|
|
instance.proxy = new Proxy(instance.ctx, PublicInstanceProxyHandlers);
|
|
// 2. call setup()
|
|
const { setup } = Component;
|
|
if (setup) {
|
|
const setupContext = (instance.setupContext =
|
|
setup.length > 1 ? createSetupContext(instance) : null);
|
|
currentInstance = instance;
|
|
pauseTracking();
|
|
const setupResult = callWithErrorHandling(setup, instance, 0 /* SETUP_FUNCTION */, [instance.props, setupContext]);
|
|
resetTracking();
|
|
currentInstance = null;
|
|
if (isPromise(setupResult)) {
|
|
if (isSSR) {
|
|
// return the promise so server-renderer can wait on it
|
|
return setupResult.then((resolvedResult) => {
|
|
handleSetupResult(instance, resolvedResult);
|
|
});
|
|
}
|
|
else {
|
|
// async setup returned Promise.
|
|
// bail here and wait for re-entry.
|
|
instance.asyncDep = setupResult;
|
|
}
|
|
}
|
|
else {
|
|
handleSetupResult(instance, setupResult);
|
|
}
|
|
}
|
|
else {
|
|
finishComponentSetup(instance);
|
|
}
|
|
}
|
|
function handleSetupResult(instance, setupResult, isSSR) {
|
|
if (isFunction(setupResult)) {
|
|
// setup returned an inline render function
|
|
{
|
|
instance.render = setupResult;
|
|
}
|
|
}
|
|
else if (isObject(setupResult)) {
|
|
instance.setupState = proxyRefs(setupResult);
|
|
}
|
|
else ;
|
|
finishComponentSetup(instance);
|
|
}
|
|
function finishComponentSetup(instance, isSSR) {
|
|
const Component = instance.type;
|
|
// template / render function normalization
|
|
if (!instance.render) {
|
|
instance.render = (Component.render || NOOP);
|
|
// for runtime-compiled render functions using `with` blocks, the render
|
|
// proxy used needs a different `has` handler which is more performant and
|
|
// also only allows a whitelist of globals to fallthrough.
|
|
if (instance.render._rc) {
|
|
instance.withProxy = new Proxy(instance.ctx, RuntimeCompiledPublicInstanceProxyHandlers);
|
|
}
|
|
}
|
|
// support for 2.x options
|
|
{
|
|
currentInstance = instance;
|
|
pauseTracking();
|
|
applyOptions(instance, Component);
|
|
resetTracking();
|
|
currentInstance = null;
|
|
}
|
|
}
|
|
function createSetupContext(instance) {
|
|
const expose = exposed => {
|
|
instance.exposed = proxyRefs(exposed);
|
|
};
|
|
{
|
|
return {
|
|
attrs: instance.attrs,
|
|
slots: instance.slots,
|
|
emit: instance.emit,
|
|
expose
|
|
};
|
|
}
|
|
}
|
|
// record effects created during a component's setup() so that they can be
|
|
// stopped when the component unmounts
|
|
function recordInstanceBoundEffect(effect, instance = currentInstance) {
|
|
if (instance) {
|
|
(instance.effects || (instance.effects = [])).push(effect);
|
|
}
|
|
}
|
|
const classifyRE = /(?:^|[-_])(\w)/g;
|
|
const classify = (str) => str.replace(classifyRE, c => c.toUpperCase()).replace(/[-_]/g, '');
|
|
function getComponentName(Component) {
|
|
return isFunction(Component)
|
|
? Component.displayName || Component.name
|
|
: Component.name;
|
|
}
|
|
/* istanbul ignore next */
|
|
function formatComponentName(instance, Component, isRoot = false) {
|
|
let name = getComponentName(Component);
|
|
if (!name && Component.__file) {
|
|
const match = Component.__file.match(/([^/\\]+)\.\w+$/);
|
|
if (match) {
|
|
name = match[1];
|
|
}
|
|
}
|
|
if (!name && instance && instance.parent) {
|
|
// try to infer the name based on reverse resolution
|
|
const inferFromRegistry = (registry) => {
|
|
for (const key in registry) {
|
|
if (registry[key] === Component) {
|
|
return key;
|
|
}
|
|
}
|
|
};
|
|
name =
|
|
inferFromRegistry(instance.components ||
|
|
instance.parent.type.components) || inferFromRegistry(instance.appContext.components);
|
|
}
|
|
return name ? classify(name) : isRoot ? `App` : `Anonymous`;
|
|
}
|
|
function isClassComponent(value) {
|
|
return isFunction(value) && '__vccOpts' in value;
|
|
}
|
|
|
|
function computed$1(getterOrOptions) {
|
|
const c = computed(getterOrOptions);
|
|
recordInstanceBoundEffect(c.effect);
|
|
return c;
|
|
}
|
|
|
|
/**
|
|
* Actual implementation
|
|
*/
|
|
function renderList(source, renderItem) {
|
|
let ret;
|
|
if (isArray(source) || isString(source)) {
|
|
ret = new Array(source.length);
|
|
for (let i = 0, l = source.length; i < l; i++) {
|
|
ret[i] = renderItem(source[i], i);
|
|
}
|
|
}
|
|
else if (typeof source === 'number') {
|
|
ret = new Array(source);
|
|
for (let i = 0; i < source; i++) {
|
|
ret[i] = renderItem(i + 1, i);
|
|
}
|
|
}
|
|
else if (isObject(source)) {
|
|
if (source[Symbol.iterator]) {
|
|
ret = Array.from(source, renderItem);
|
|
}
|
|
else {
|
|
const keys = Object.keys(source);
|
|
ret = new Array(keys.length);
|
|
for (let i = 0, l = keys.length; i < l; i++) {
|
|
const key = keys[i];
|
|
ret[i] = renderItem(source[key], key, i);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
ret = [];
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// Core API ------------------------------------------------------------------
|
|
const version = "3.0.5";
|
|
|
|
const svgNS = 'http://www.w3.org/2000/svg';
|
|
const doc = (typeof document !== 'undefined' ? document : null);
|
|
let tempContainer;
|
|
let tempSVGContainer;
|
|
const nodeOps = {
|
|
insert: (child, parent, anchor) => {
|
|
parent.insertBefore(child, anchor || null);
|
|
},
|
|
remove: child => {
|
|
const parent = child.parentNode;
|
|
if (parent) {
|
|
parent.removeChild(child);
|
|
}
|
|
},
|
|
createElement: (tag, isSVG, is) => isSVG
|
|
? doc.createElementNS(svgNS, tag)
|
|
: doc.createElement(tag, is ? { is } : undefined),
|
|
createText: text => doc.createTextNode(text),
|
|
createComment: text => doc.createComment(text),
|
|
setText: (node, text) => {
|
|
node.nodeValue = text;
|
|
},
|
|
setElementText: (el, text) => {
|
|
el.textContent = text;
|
|
},
|
|
parentNode: node => node.parentNode,
|
|
nextSibling: node => node.nextSibling,
|
|
querySelector: selector => doc.querySelector(selector),
|
|
setScopeId(el, id) {
|
|
el.setAttribute(id, '');
|
|
},
|
|
cloneNode(el) {
|
|
return el.cloneNode(true);
|
|
},
|
|
// __UNSAFE__
|
|
// Reason: innerHTML.
|
|
// Static content here can only come from compiled templates.
|
|
// As long as the user only uses trusted templates, this is safe.
|
|
insertStaticContent(content, parent, anchor, isSVG) {
|
|
const temp = isSVG
|
|
? tempSVGContainer ||
|
|
(tempSVGContainer = doc.createElementNS(svgNS, 'svg'))
|
|
: tempContainer || (tempContainer = doc.createElement('div'));
|
|
temp.innerHTML = content;
|
|
const first = temp.firstChild;
|
|
let node = first;
|
|
let last = node;
|
|
while (node) {
|
|
last = node;
|
|
nodeOps.insert(node, parent, anchor);
|
|
node = temp.firstChild;
|
|
}
|
|
return [first, last];
|
|
}
|
|
};
|
|
|
|
// compiler should normalize class + :class bindings on the same element
|
|
// into a single binding ['staticClass', dynamic]
|
|
function patchClass(el, value, isSVG) {
|
|
if (value == null) {
|
|
value = '';
|
|
}
|
|
if (isSVG) {
|
|
el.setAttribute('class', value);
|
|
}
|
|
else {
|
|
// directly setting className should be faster than setAttribute in theory
|
|
// if this is an element during a transition, take the temporary transition
|
|
// classes into account.
|
|
const transitionClasses = el._vtc;
|
|
if (transitionClasses) {
|
|
value = (value
|
|
? [value, ...transitionClasses]
|
|
: [...transitionClasses]).join(' ');
|
|
}
|
|
el.className = value;
|
|
}
|
|
}
|
|
|
|
function patchStyle(el, prev, next) {
|
|
const style = el.style;
|
|
if (!next) {
|
|
el.removeAttribute('style');
|
|
}
|
|
else if (isString(next)) {
|
|
if (prev !== next) {
|
|
style.cssText = next;
|
|
}
|
|
}
|
|
else {
|
|
for (const key in next) {
|
|
setStyle(style, key, next[key]);
|
|
}
|
|
if (prev && !isString(prev)) {
|
|
for (const key in prev) {
|
|
if (next[key] == null) {
|
|
setStyle(style, key, '');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
const importantRE = /\s*!important$/;
|
|
function setStyle(style, name, val) {
|
|
if (isArray(val)) {
|
|
val.forEach(v => setStyle(style, name, v));
|
|
}
|
|
else {
|
|
if (name.startsWith('--')) {
|
|
// custom property definition
|
|
style.setProperty(name, val);
|
|
}
|
|
else {
|
|
const prefixed = autoPrefix(style, name);
|
|
if (importantRE.test(val)) {
|
|
// !important
|
|
style.setProperty(hyphenate(prefixed), val.replace(importantRE, ''), 'important');
|
|
}
|
|
else {
|
|
style[prefixed] = val;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
const prefixes = ['Webkit', 'Moz', 'ms'];
|
|
const prefixCache = {};
|
|
function autoPrefix(style, rawName) {
|
|
const cached = prefixCache[rawName];
|
|
if (cached) {
|
|
return cached;
|
|
}
|
|
let name = camelize(rawName);
|
|
if (name !== 'filter' && name in style) {
|
|
return (prefixCache[rawName] = name);
|
|
}
|
|
name = capitalize(name);
|
|
for (let i = 0; i < prefixes.length; i++) {
|
|
const prefixed = prefixes[i] + name;
|
|
if (prefixed in style) {
|
|
return (prefixCache[rawName] = prefixed);
|
|
}
|
|
}
|
|
return rawName;
|
|
}
|
|
|
|
const xlinkNS = 'http://www.w3.org/1999/xlink';
|
|
function patchAttr(el, key, value, isSVG) {
|
|
if (isSVG && key.startsWith('xlink:')) {
|
|
if (value == null) {
|
|
el.removeAttributeNS(xlinkNS, key.slice(6, key.length));
|
|
}
|
|
else {
|
|
el.setAttributeNS(xlinkNS, key, value);
|
|
}
|
|
}
|
|
else {
|
|
// note we are only checking boolean attributes that don't have a
|
|
// corresponding dom prop of the same name here.
|
|
const isBoolean = isSpecialBooleanAttr(key);
|
|
if (value == null || (isBoolean && value === false)) {
|
|
el.removeAttribute(key);
|
|
}
|
|
else {
|
|
el.setAttribute(key, isBoolean ? '' : value);
|
|
}
|
|
}
|
|
}
|
|
|
|
// __UNSAFE__
|
|
// functions. The user is responsible for using them with only trusted content.
|
|
function patchDOMProp(el, key, value,
|
|
// the following args are passed only due to potential innerHTML/textContent
|
|
// overriding existing VNodes, in which case the old tree must be properly
|
|
// unmounted.
|
|
prevChildren, parentComponent, parentSuspense, unmountChildren) {
|
|
if (key === 'innerHTML' || key === 'textContent') {
|
|
if (prevChildren) {
|
|
unmountChildren(prevChildren, parentComponent, parentSuspense);
|
|
}
|
|
el[key] = value == null ? '' : value;
|
|
return;
|
|
}
|
|
if (key === 'value' && el.tagName !== 'PROGRESS') {
|
|
// store value as _value as well since
|
|
// non-string values will be stringified.
|
|
el._value = value;
|
|
const newValue = value == null ? '' : value;
|
|
if (el.value !== newValue) {
|
|
el.value = newValue;
|
|
}
|
|
return;
|
|
}
|
|
if (value === '' || value == null) {
|
|
const type = typeof el[key];
|
|
if (value === '' && type === 'boolean') {
|
|
// e.g. <select multiple> compiles to { multiple: '' }
|
|
el[key] = true;
|
|
return;
|
|
}
|
|
else if (value == null && type === 'string') {
|
|
// e.g. <div :id="null">
|
|
el[key] = '';
|
|
el.removeAttribute(key);
|
|
return;
|
|
}
|
|
else if (type === 'number') {
|
|
// e.g. <img :width="null">
|
|
el[key] = 0;
|
|
el.removeAttribute(key);
|
|
return;
|
|
}
|
|
}
|
|
// some properties perform value validation and throw
|
|
try {
|
|
el[key] = value;
|
|
}
|
|
catch (e) {
|
|
}
|
|
}
|
|
|
|
// Async edge case fix requires storing an event listener's attach timestamp.
|
|
let _getNow = Date.now;
|
|
// Determine what event timestamp the browser is using. Annoyingly, the
|
|
// timestamp can either be hi-res (relative to page load) or low-res
|
|
// (relative to UNIX epoch), so in order to compare time we have to use the
|
|
// same timestamp type when saving the flush timestamp.
|
|
if (typeof document !== 'undefined' &&
|
|
_getNow() > document.createEvent('Event').timeStamp) {
|
|
// if the low-res timestamp which is bigger than the event timestamp
|
|
// (which is evaluated AFTER) it means the event is using a hi-res timestamp,
|
|
// and we need to use the hi-res version for event listeners as well.
|
|
_getNow = () => performance.now();
|
|
}
|
|
// To avoid the overhead of repeatedly calling performance.now(), we cache
|
|
// and use the same timestamp for all event listeners attached in the same tick.
|
|
let cachedNow = 0;
|
|
const p = Promise.resolve();
|
|
const reset = () => {
|
|
cachedNow = 0;
|
|
};
|
|
const getNow = () => cachedNow || (p.then(reset), (cachedNow = _getNow()));
|
|
function addEventListener(el, event, handler, options) {
|
|
el.addEventListener(event, handler, options);
|
|
}
|
|
function removeEventListener(el, event, handler, options) {
|
|
el.removeEventListener(event, handler, options);
|
|
}
|
|
function patchEvent(el, rawName, prevValue, nextValue, instance = null) {
|
|
// vei = vue event invokers
|
|
const invokers = el._vei || (el._vei = {});
|
|
const existingInvoker = invokers[rawName];
|
|
if (nextValue && existingInvoker) {
|
|
// patch
|
|
existingInvoker.value = nextValue;
|
|
}
|
|
else {
|
|
const [name, options] = parseName(rawName);
|
|
if (nextValue) {
|
|
// add
|
|
const invoker = (invokers[rawName] = createInvoker(nextValue, instance));
|
|
addEventListener(el, name, invoker, options);
|
|
}
|
|
else if (existingInvoker) {
|
|
// remove
|
|
removeEventListener(el, name, existingInvoker, options);
|
|
invokers[rawName] = undefined;
|
|
}
|
|
}
|
|
}
|
|
const optionsModifierRE = /(?:Once|Passive|Capture)$/;
|
|
function parseName(name) {
|
|
let options;
|
|
if (optionsModifierRE.test(name)) {
|
|
options = {};
|
|
let m;
|
|
while ((m = name.match(optionsModifierRE))) {
|
|
name = name.slice(0, name.length - m[0].length);
|
|
options[m[0].toLowerCase()] = true;
|
|
}
|
|
}
|
|
return [name.slice(2).toLowerCase(), options];
|
|
}
|
|
function createInvoker(initialValue, instance) {
|
|
const invoker = (e) => {
|
|
// async edge case #6566: inner click event triggers patch, event handler
|
|
// attached to outer element during patch, and triggered again. This
|
|
// happens because browsers fire microtask ticks between event propagation.
|
|
// the solution is simple: we save the timestamp when a handler is attached,
|
|
// and the handler would only fire if the event passed to it was fired
|
|
// AFTER it was attached.
|
|
const timeStamp = e.timeStamp || _getNow();
|
|
if (timeStamp >= invoker.attached - 1) {
|
|
callWithAsyncErrorHandling(patchStopImmediatePropagation(e, invoker.value), instance, 5 /* NATIVE_EVENT_HANDLER */, [e]);
|
|
}
|
|
};
|
|
invoker.value = initialValue;
|
|
invoker.attached = getNow();
|
|
return invoker;
|
|
}
|
|
function patchStopImmediatePropagation(e, value) {
|
|
if (isArray(value)) {
|
|
const originalStop = e.stopImmediatePropagation;
|
|
e.stopImmediatePropagation = () => {
|
|
originalStop.call(e);
|
|
e._stopped = true;
|
|
};
|
|
return value.map(fn => (e) => !e._stopped && fn(e));
|
|
}
|
|
else {
|
|
return value;
|
|
}
|
|
}
|
|
|
|
const nativeOnRE = /^on[a-z]/;
|
|
const forcePatchProp = (_, key) => key === 'value';
|
|
const patchProp = (el, key, prevValue, nextValue, isSVG = false, prevChildren, parentComponent, parentSuspense, unmountChildren) => {
|
|
switch (key) {
|
|
// special
|
|
case 'class':
|
|
patchClass(el, nextValue, isSVG);
|
|
break;
|
|
case 'style':
|
|
patchStyle(el, prevValue, nextValue);
|
|
break;
|
|
default:
|
|
if (isOn(key)) {
|
|
// ignore v-model listeners
|
|
if (!isModelListener(key)) {
|
|
patchEvent(el, key, prevValue, nextValue, parentComponent);
|
|
}
|
|
}
|
|
else if (shouldSetAsProp(el, key, nextValue, isSVG)) {
|
|
patchDOMProp(el, key, nextValue, prevChildren, parentComponent, parentSuspense, unmountChildren);
|
|
}
|
|
else {
|
|
// special case for <input v-model type="checkbox"> with
|
|
// :true-value & :false-value
|
|
// store value as dom properties since non-string values will be
|
|
// stringified.
|
|
if (key === 'true-value') {
|
|
el._trueValue = nextValue;
|
|
}
|
|
else if (key === 'false-value') {
|
|
el._falseValue = nextValue;
|
|
}
|
|
patchAttr(el, key, nextValue, isSVG);
|
|
}
|
|
break;
|
|
}
|
|
};
|
|
function shouldSetAsProp(el, key, value, isSVG) {
|
|
if (isSVG) {
|
|
// most keys must be set as attribute on svg elements to work
|
|
// ...except innerHTML
|
|
if (key === 'innerHTML') {
|
|
return true;
|
|
}
|
|
// or native onclick with function values
|
|
if (key in el && nativeOnRE.test(key) && isFunction(value)) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
// spellcheck and draggable are numerated attrs, however their
|
|
// corresponding DOM properties are actually booleans - this leads to
|
|
// setting it with a string "false" value leading it to be coerced to
|
|
// `true`, so we need to always treat them as attributes.
|
|
// Note that `contentEditable` doesn't have this problem: its DOM
|
|
// property is also enumerated string values.
|
|
if (key === 'spellcheck' || key === 'draggable') {
|
|
return false;
|
|
}
|
|
// #1787 form as an attribute must be a string, while it accepts an Element as
|
|
// a prop
|
|
if (key === 'form' && typeof value === 'string') {
|
|
return false;
|
|
}
|
|
// #1526 <input list> must be set as attribute
|
|
if (key === 'list' && el.tagName === 'INPUT') {
|
|
return false;
|
|
}
|
|
// native onclick with string value, must be set as attribute
|
|
if (nativeOnRE.test(key) && isString(value)) {
|
|
return false;
|
|
}
|
|
return key in el;
|
|
}
|
|
|
|
const getModelAssigner = (vnode) => {
|
|
const fn = vnode.props['onUpdate:modelValue'];
|
|
return isArray(fn) ? value => invokeArrayFns(fn, value) : fn;
|
|
};
|
|
function onCompositionStart(e) {
|
|
e.target.composing = true;
|
|
}
|
|
function onCompositionEnd(e) {
|
|
const target = e.target;
|
|
if (target.composing) {
|
|
target.composing = false;
|
|
trigger$1(target, 'input');
|
|
}
|
|
}
|
|
function trigger$1(el, type) {
|
|
const e = document.createEvent('HTMLEvents');
|
|
e.initEvent(type, true, true);
|
|
el.dispatchEvent(e);
|
|
}
|
|
// We are exporting the v-model runtime directly as vnode hooks so that it can
|
|
// be tree-shaken in case v-model is never used.
|
|
const vModelText = {
|
|
created(el, { modifiers: { lazy, trim, number } }, vnode) {
|
|
el._assign = getModelAssigner(vnode);
|
|
const castToNumber = number || el.type === 'number';
|
|
addEventListener(el, lazy ? 'change' : 'input', e => {
|
|
if (e.target.composing)
|
|
return;
|
|
let domValue = el.value;
|
|
if (trim) {
|
|
domValue = domValue.trim();
|
|
}
|
|
else if (castToNumber) {
|
|
domValue = toNumber(domValue);
|
|
}
|
|
el._assign(domValue);
|
|
});
|
|
if (trim) {
|
|
addEventListener(el, 'change', () => {
|
|
el.value = el.value.trim();
|
|
});
|
|
}
|
|
if (!lazy) {
|
|
addEventListener(el, 'compositionstart', onCompositionStart);
|
|
addEventListener(el, 'compositionend', onCompositionEnd);
|
|
// Safari < 10.2 & UIWebView doesn't fire compositionend when
|
|
// switching focus before confirming composition choice
|
|
// this also fixes the issue where some browsers e.g. iOS Chrome
|
|
// fires "change" instead of "input" on autocomplete.
|
|
addEventListener(el, 'change', onCompositionEnd);
|
|
}
|
|
},
|
|
// set value on mounted so it's after min/max for type="range"
|
|
mounted(el, { value }) {
|
|
el.value = value == null ? '' : value;
|
|
},
|
|
beforeUpdate(el, { value, modifiers: { trim, number } }, vnode) {
|
|
el._assign = getModelAssigner(vnode);
|
|
// avoid clearing unresolved text. #2302
|
|
if (el.composing)
|
|
return;
|
|
if (document.activeElement === el) {
|
|
if (trim && el.value.trim() === value) {
|
|
return;
|
|
}
|
|
if ((number || el.type === 'number') && toNumber(el.value) === value) {
|
|
return;
|
|
}
|
|
}
|
|
const newValue = value == null ? '' : value;
|
|
if (el.value !== newValue) {
|
|
el.value = newValue;
|
|
}
|
|
}
|
|
};
|
|
const vModelCheckbox = {
|
|
created(el, _, vnode) {
|
|
el._assign = getModelAssigner(vnode);
|
|
addEventListener(el, 'change', () => {
|
|
const modelValue = el._modelValue;
|
|
const elementValue = getValue(el);
|
|
const checked = el.checked;
|
|
const assign = el._assign;
|
|
if (isArray(modelValue)) {
|
|
const index = looseIndexOf(modelValue, elementValue);
|
|
const found = index !== -1;
|
|
if (checked && !found) {
|
|
assign(modelValue.concat(elementValue));
|
|
}
|
|
else if (!checked && found) {
|
|
const filtered = [...modelValue];
|
|
filtered.splice(index, 1);
|
|
assign(filtered);
|
|
}
|
|
}
|
|
else if (isSet(modelValue)) {
|
|
const cloned = new Set(modelValue);
|
|
if (checked) {
|
|
cloned.add(elementValue);
|
|
}
|
|
else {
|
|
cloned.delete(elementValue);
|
|
}
|
|
assign(cloned);
|
|
}
|
|
else {
|
|
assign(getCheckboxValue(el, checked));
|
|
}
|
|
});
|
|
},
|
|
// set initial checked on mount to wait for true-value/false-value
|
|
mounted: setChecked,
|
|
beforeUpdate(el, binding, vnode) {
|
|
el._assign = getModelAssigner(vnode);
|
|
setChecked(el, binding, vnode);
|
|
}
|
|
};
|
|
function setChecked(el, { value, oldValue }, vnode) {
|
|
el._modelValue = value;
|
|
if (isArray(value)) {
|
|
el.checked = looseIndexOf(value, vnode.props.value) > -1;
|
|
}
|
|
else if (isSet(value)) {
|
|
el.checked = value.has(vnode.props.value);
|
|
}
|
|
else if (value !== oldValue) {
|
|
el.checked = looseEqual(value, getCheckboxValue(el, true));
|
|
}
|
|
}
|
|
const vModelSelect = {
|
|
created(el, { value, modifiers: { number } }, vnode) {
|
|
const isSetModel = isSet(value);
|
|
addEventListener(el, 'change', () => {
|
|
const selectedVal = Array.prototype.filter
|
|
.call(el.options, (o) => o.selected)
|
|
.map((o) => number ? toNumber(getValue(o)) : getValue(o));
|
|
el._assign(el.multiple
|
|
? isSetModel
|
|
? new Set(selectedVal)
|
|
: selectedVal
|
|
: selectedVal[0]);
|
|
});
|
|
el._assign = getModelAssigner(vnode);
|
|
},
|
|
// set value in mounted & updated because <select> relies on its children
|
|
// <option>s.
|
|
mounted(el, { value }) {
|
|
setSelected(el, value);
|
|
},
|
|
beforeUpdate(el, _binding, vnode) {
|
|
el._assign = getModelAssigner(vnode);
|
|
},
|
|
updated(el, { value }) {
|
|
setSelected(el, value);
|
|
}
|
|
};
|
|
function setSelected(el, value) {
|
|
const isMultiple = el.multiple;
|
|
if (isMultiple && !isArray(value) && !isSet(value)) {
|
|
return;
|
|
}
|
|
for (let i = 0, l = el.options.length; i < l; i++) {
|
|
const option = el.options[i];
|
|
const optionValue = getValue(option);
|
|
if (isMultiple) {
|
|
if (isArray(value)) {
|
|
option.selected = looseIndexOf(value, optionValue) > -1;
|
|
}
|
|
else {
|
|
option.selected = value.has(optionValue);
|
|
}
|
|
}
|
|
else {
|
|
if (looseEqual(getValue(option), value)) {
|
|
el.selectedIndex = i;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
if (!isMultiple) {
|
|
el.selectedIndex = -1;
|
|
}
|
|
}
|
|
// retrieve raw value set via :value bindings
|
|
function getValue(el) {
|
|
return '_value' in el ? el._value : el.value;
|
|
}
|
|
// retrieve raw value for true-value and false-value set via :true-value or :false-value bindings
|
|
function getCheckboxValue(el, checked) {
|
|
const key = checked ? '_trueValue' : '_falseValue';
|
|
return key in el ? el[key] : checked;
|
|
}
|
|
|
|
const rendererOptions = extend({ patchProp, forcePatchProp }, nodeOps);
|
|
// lazy create the renderer - this makes core renderer logic tree-shakable
|
|
// in case the user only imports reactivity utilities from Vue.
|
|
let renderer;
|
|
function ensureRenderer() {
|
|
return renderer || (renderer = createRenderer(rendererOptions));
|
|
}
|
|
const createApp = ((...args) => {
|
|
const app = ensureRenderer().createApp(...args);
|
|
const { mount } = app;
|
|
app.mount = (containerOrSelector) => {
|
|
const container = normalizeContainer(containerOrSelector);
|
|
if (!container)
|
|
return;
|
|
const component = app._component;
|
|
if (!isFunction(component) && !component.render && !component.template) {
|
|
component.template = container.innerHTML;
|
|
}
|
|
// clear content before mounting
|
|
container.innerHTML = '';
|
|
const proxy = mount(container);
|
|
if (container instanceof Element) {
|
|
container.removeAttribute('v-cloak');
|
|
container.setAttribute('data-v-app', '');
|
|
}
|
|
return proxy;
|
|
};
|
|
return app;
|
|
});
|
|
function normalizeContainer(container) {
|
|
if (isString(container)) {
|
|
const res = document.querySelector(container);
|
|
return res;
|
|
}
|
|
return container;
|
|
}
|
|
|
|
var script = {
|
|
name: "StringValue",
|
|
props: {
|
|
id: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
value: Object
|
|
},
|
|
methods: {
|
|
focus() {
|
|
this.$refs.input.focus();
|
|
}
|
|
},
|
|
computed: {
|
|
inputValue: {
|
|
set(value) {
|
|
this.value.String = value;
|
|
},
|
|
get() {
|
|
return this.value.String;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function render(_ctx, _cache, $props, $setup, $data, $options) {
|
|
return withDirectives((openBlock(), createBlock("input", {
|
|
ref: "input",
|
|
type: "text",
|
|
id: $props.id,
|
|
"onUpdate:modelValue": _cache[1] || (_cache[1] = $event => ($options.inputValue = $event))
|
|
}, null, 8 /* PROPS */, ["id"])), [
|
|
[vModelText, $options.inputValue]
|
|
])
|
|
}
|
|
|
|
script.render = render;
|
|
script.__file = "src/components/StringValue.vue";
|
|
|
|
var script$1 = {
|
|
name: "DecimalValue",
|
|
props: {
|
|
id: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
value: Object
|
|
},
|
|
methods: {
|
|
focus() {
|
|
this.$refs.input.focus();
|
|
}
|
|
},
|
|
computed: {
|
|
inputValue: {
|
|
set(value) {
|
|
this.value.Decimal = parseFloat(value);
|
|
},
|
|
get() {
|
|
return this.value.Decimal;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function render$1(_ctx, _cache, $props, $setup, $data, $options) {
|
|
return withDirectives((openBlock(), createBlock("input", {
|
|
ref: "input",
|
|
type: "number",
|
|
id: $props.id,
|
|
step: "any",
|
|
"onUpdate:modelValue": _cache[1] || (_cache[1] = $event => ($options.inputValue = $event))
|
|
}, null, 8 /* PROPS */, ["id"])), [
|
|
[vModelText, $options.inputValue]
|
|
])
|
|
}
|
|
|
|
script$1.render = render$1;
|
|
script$1.__file = "src/components/DecimalValue.vue";
|
|
|
|
var script$2 = {
|
|
name: "BooleanValue",
|
|
props: {
|
|
id: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
value: Object
|
|
},
|
|
methods: {
|
|
focus() {
|
|
this.$refs.input.focus();
|
|
}
|
|
},
|
|
computed: {
|
|
inputValue: {
|
|
set(value) {
|
|
this.value.Boolean = !!value;
|
|
},
|
|
get() {
|
|
return this.value.Boolean;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function render$2(_ctx, _cache, $props, $setup, $data, $options) {
|
|
return withDirectives((openBlock(), createBlock("input", {
|
|
ref: "input",
|
|
type: "checkbox",
|
|
id: $props.id,
|
|
value: "true",
|
|
"onUpdate:modelValue": _cache[1] || (_cache[1] = $event => ($options.inputValue = $event))
|
|
}, null, 8 /* PROPS */, ["id"])), [
|
|
[vModelCheckbox, $options.inputValue]
|
|
])
|
|
}
|
|
|
|
script$2.render = render$2;
|
|
script$2.__file = "src/components/BooleanValue.vue";
|
|
|
|
var script$3 = {
|
|
name: "IntegerValue",
|
|
props: {
|
|
id: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
value: Object
|
|
},
|
|
methods: {
|
|
focus() {
|
|
this.$refs.input.focus();
|
|
}
|
|
},
|
|
computed: {
|
|
inputValue: {
|
|
set(value) {
|
|
this.value.Integer = parseInt(value);
|
|
},
|
|
get() {
|
|
return this.value.Integer;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function render$3(_ctx, _cache, $props, $setup, $data, $options) {
|
|
return withDirectives((openBlock(), createBlock("input", {
|
|
ref: "input",
|
|
type: "number",
|
|
id: $props.id,
|
|
step: "1",
|
|
"onUpdate:modelValue": _cache[1] || (_cache[1] = $event => ($options.inputValue = $event))
|
|
}, null, 8 /* PROPS */, ["id"])), [
|
|
[vModelText, $options.inputValue]
|
|
])
|
|
}
|
|
|
|
script$3.render = render$3;
|
|
script$3.__file = "src/components/IntegerValue.vue";
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto = Object.prototype;
|
|
|
|
/**
|
|
* Checks if `value` is likely a prototype object.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a prototype, else `false`.
|
|
*/
|
|
function isPrototype(value) {
|
|
var Ctor = value && value.constructor,
|
|
proto = (typeof Ctor == 'function' && Ctor.prototype) || objectProto;
|
|
|
|
return value === proto;
|
|
}
|
|
|
|
/**
|
|
* Creates a unary function that invokes `func` with its argument transformed.
|
|
*
|
|
* @private
|
|
* @param {Function} func The function to wrap.
|
|
* @param {Function} transform The argument transform.
|
|
* @returns {Function} Returns the new function.
|
|
*/
|
|
function overArg(func, transform) {
|
|
return function(arg) {
|
|
return func(transform(arg));
|
|
};
|
|
}
|
|
|
|
/* Built-in method references for those with the same name as other `lodash` methods. */
|
|
var nativeKeys = overArg(Object.keys, Object);
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$1 = Object.prototype;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$1 = objectProto$1.hasOwnProperty;
|
|
|
|
/**
|
|
* The base implementation of `_.keys` which doesn't treat sparse arrays as dense.
|
|
*
|
|
* @private
|
|
* @param {Object} object The object to query.
|
|
* @returns {Array} Returns the array of property names.
|
|
*/
|
|
function baseKeys(object) {
|
|
if (!isPrototype(object)) {
|
|
return nativeKeys(object);
|
|
}
|
|
var result = [];
|
|
for (var key in Object(object)) {
|
|
if (hasOwnProperty$1.call(object, key) && key != 'constructor') {
|
|
result.push(key);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/** Detect free variable `global` from Node.js. */
|
|
var freeGlobal = typeof global == 'object' && global && global.Object === Object && global;
|
|
|
|
/** Detect free variable `self`. */
|
|
var freeSelf = typeof self == 'object' && self && self.Object === Object && self;
|
|
|
|
/** Used as a reference to the global object. */
|
|
var root = freeGlobal || freeSelf || Function('return this')();
|
|
|
|
/** Built-in value references. */
|
|
var Symbol$1 = root.Symbol;
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$2 = Object.prototype;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$2 = objectProto$2.hasOwnProperty;
|
|
|
|
/**
|
|
* Used to resolve the
|
|
* [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
|
|
* of values.
|
|
*/
|
|
var nativeObjectToString = objectProto$2.toString;
|
|
|
|
/** Built-in value references. */
|
|
var symToStringTag = Symbol$1 ? Symbol$1.toStringTag : undefined;
|
|
|
|
/**
|
|
* A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to query.
|
|
* @returns {string} Returns the raw `toStringTag`.
|
|
*/
|
|
function getRawTag(value) {
|
|
var isOwn = hasOwnProperty$2.call(value, symToStringTag),
|
|
tag = value[symToStringTag];
|
|
|
|
try {
|
|
value[symToStringTag] = undefined;
|
|
var unmasked = true;
|
|
} catch (e) {}
|
|
|
|
var result = nativeObjectToString.call(value);
|
|
if (unmasked) {
|
|
if (isOwn) {
|
|
value[symToStringTag] = tag;
|
|
} else {
|
|
delete value[symToStringTag];
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$3 = Object.prototype;
|
|
|
|
/**
|
|
* Used to resolve the
|
|
* [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
|
|
* of values.
|
|
*/
|
|
var nativeObjectToString$1 = objectProto$3.toString;
|
|
|
|
/**
|
|
* Converts `value` to a string using `Object.prototype.toString`.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to convert.
|
|
* @returns {string} Returns the converted string.
|
|
*/
|
|
function objectToString$1(value) {
|
|
return nativeObjectToString$1.call(value);
|
|
}
|
|
|
|
/** `Object#toString` result references. */
|
|
var nullTag = '[object Null]',
|
|
undefinedTag = '[object Undefined]';
|
|
|
|
/** Built-in value references. */
|
|
var symToStringTag$1 = Symbol$1 ? Symbol$1.toStringTag : undefined;
|
|
|
|
/**
|
|
* The base implementation of `getTag` without fallbacks for buggy environments.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to query.
|
|
* @returns {string} Returns the `toStringTag`.
|
|
*/
|
|
function baseGetTag(value) {
|
|
if (value == null) {
|
|
return value === undefined ? undefinedTag : nullTag;
|
|
}
|
|
return (symToStringTag$1 && symToStringTag$1 in Object(value))
|
|
? getRawTag(value)
|
|
: objectToString$1(value);
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is the
|
|
* [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
|
|
* of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is an object, else `false`.
|
|
* @example
|
|
*
|
|
* _.isObject({});
|
|
* // => true
|
|
*
|
|
* _.isObject([1, 2, 3]);
|
|
* // => true
|
|
*
|
|
* _.isObject(_.noop);
|
|
* // => true
|
|
*
|
|
* _.isObject(null);
|
|
* // => false
|
|
*/
|
|
function isObject$1(value) {
|
|
var type = typeof value;
|
|
return value != null && (type == 'object' || type == 'function');
|
|
}
|
|
|
|
/** `Object#toString` result references. */
|
|
var asyncTag = '[object AsyncFunction]',
|
|
funcTag = '[object Function]',
|
|
genTag = '[object GeneratorFunction]',
|
|
proxyTag = '[object Proxy]';
|
|
|
|
/**
|
|
* Checks if `value` is classified as a `Function` object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a function, else `false`.
|
|
* @example
|
|
*
|
|
* _.isFunction(_);
|
|
* // => true
|
|
*
|
|
* _.isFunction(/abc/);
|
|
* // => false
|
|
*/
|
|
function isFunction$1(value) {
|
|
if (!isObject$1(value)) {
|
|
return false;
|
|
}
|
|
// The use of `Object#toString` avoids issues with the `typeof` operator
|
|
// in Safari 9 which returns 'object' for typed arrays and other constructors.
|
|
var tag = baseGetTag(value);
|
|
return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag;
|
|
}
|
|
|
|
/** Used to detect overreaching core-js shims. */
|
|
var coreJsData = root['__core-js_shared__'];
|
|
|
|
/** Used to detect methods masquerading as native. */
|
|
var maskSrcKey = (function() {
|
|
var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || '');
|
|
return uid ? ('Symbol(src)_1.' + uid) : '';
|
|
}());
|
|
|
|
/**
|
|
* Checks if `func` has its source masked.
|
|
*
|
|
* @private
|
|
* @param {Function} func The function to check.
|
|
* @returns {boolean} Returns `true` if `func` is masked, else `false`.
|
|
*/
|
|
function isMasked(func) {
|
|
return !!maskSrcKey && (maskSrcKey in func);
|
|
}
|
|
|
|
/** Used for built-in method references. */
|
|
var funcProto = Function.prototype;
|
|
|
|
/** Used to resolve the decompiled source of functions. */
|
|
var funcToString = funcProto.toString;
|
|
|
|
/**
|
|
* Converts `func` to its source code.
|
|
*
|
|
* @private
|
|
* @param {Function} func The function to convert.
|
|
* @returns {string} Returns the source code.
|
|
*/
|
|
function toSource(func) {
|
|
if (func != null) {
|
|
try {
|
|
return funcToString.call(func);
|
|
} catch (e) {}
|
|
try {
|
|
return (func + '');
|
|
} catch (e) {}
|
|
}
|
|
return '';
|
|
}
|
|
|
|
/**
|
|
* Used to match `RegExp`
|
|
* [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns).
|
|
*/
|
|
var reRegExpChar = /[\\^$.*+?()[\]{}|]/g;
|
|
|
|
/** Used to detect host constructors (Safari). */
|
|
var reIsHostCtor = /^\[object .+?Constructor\]$/;
|
|
|
|
/** Used for built-in method references. */
|
|
var funcProto$1 = Function.prototype,
|
|
objectProto$4 = Object.prototype;
|
|
|
|
/** Used to resolve the decompiled source of functions. */
|
|
var funcToString$1 = funcProto$1.toString;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$3 = objectProto$4.hasOwnProperty;
|
|
|
|
/** Used to detect if a method is native. */
|
|
var reIsNative = RegExp('^' +
|
|
funcToString$1.call(hasOwnProperty$3).replace(reRegExpChar, '\\$&')
|
|
.replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$'
|
|
);
|
|
|
|
/**
|
|
* The base implementation of `_.isNative` without bad shim checks.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a native function,
|
|
* else `false`.
|
|
*/
|
|
function baseIsNative(value) {
|
|
if (!isObject$1(value) || isMasked(value)) {
|
|
return false;
|
|
}
|
|
var pattern = isFunction$1(value) ? reIsNative : reIsHostCtor;
|
|
return pattern.test(toSource(value));
|
|
}
|
|
|
|
/**
|
|
* Gets the value at `key` of `object`.
|
|
*
|
|
* @private
|
|
* @param {Object} [object] The object to query.
|
|
* @param {string} key The key of the property to get.
|
|
* @returns {*} Returns the property value.
|
|
*/
|
|
function getValue$1(object, key) {
|
|
return object == null ? undefined : object[key];
|
|
}
|
|
|
|
/**
|
|
* Gets the native function at `key` of `object`.
|
|
*
|
|
* @private
|
|
* @param {Object} object The object to query.
|
|
* @param {string} key The key of the method to get.
|
|
* @returns {*} Returns the function if it's native, else `undefined`.
|
|
*/
|
|
function getNative(object, key) {
|
|
var value = getValue$1(object, key);
|
|
return baseIsNative(value) ? value : undefined;
|
|
}
|
|
|
|
/* Built-in method references that are verified to be native. */
|
|
var DataView = getNative(root, 'DataView');
|
|
|
|
/* Built-in method references that are verified to be native. */
|
|
var Map$1 = getNative(root, 'Map');
|
|
|
|
/* Built-in method references that are verified to be native. */
|
|
var Promise$1 = getNative(root, 'Promise');
|
|
|
|
/* Built-in method references that are verified to be native. */
|
|
var Set$1 = getNative(root, 'Set');
|
|
|
|
/* Built-in method references that are verified to be native. */
|
|
var WeakMap$1 = getNative(root, 'WeakMap');
|
|
|
|
/** `Object#toString` result references. */
|
|
var mapTag = '[object Map]',
|
|
objectTag = '[object Object]',
|
|
promiseTag = '[object Promise]',
|
|
setTag = '[object Set]',
|
|
weakMapTag = '[object WeakMap]';
|
|
|
|
var dataViewTag = '[object DataView]';
|
|
|
|
/** Used to detect maps, sets, and weakmaps. */
|
|
var dataViewCtorString = toSource(DataView),
|
|
mapCtorString = toSource(Map$1),
|
|
promiseCtorString = toSource(Promise$1),
|
|
setCtorString = toSource(Set$1),
|
|
weakMapCtorString = toSource(WeakMap$1);
|
|
|
|
/**
|
|
* Gets the `toStringTag` of `value`.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to query.
|
|
* @returns {string} Returns the `toStringTag`.
|
|
*/
|
|
var getTag = baseGetTag;
|
|
|
|
// Fallback for data views, maps, sets, and weak maps in IE 11 and promises in Node.js < 6.
|
|
if ((DataView && getTag(new DataView(new ArrayBuffer(1))) != dataViewTag) ||
|
|
(Map$1 && getTag(new Map$1) != mapTag) ||
|
|
(Promise$1 && getTag(Promise$1.resolve()) != promiseTag) ||
|
|
(Set$1 && getTag(new Set$1) != setTag) ||
|
|
(WeakMap$1 && getTag(new WeakMap$1) != weakMapTag)) {
|
|
getTag = function(value) {
|
|
var result = baseGetTag(value),
|
|
Ctor = result == objectTag ? value.constructor : undefined,
|
|
ctorString = Ctor ? toSource(Ctor) : '';
|
|
|
|
if (ctorString) {
|
|
switch (ctorString) {
|
|
case dataViewCtorString: return dataViewTag;
|
|
case mapCtorString: return mapTag;
|
|
case promiseCtorString: return promiseTag;
|
|
case setCtorString: return setTag;
|
|
case weakMapCtorString: return weakMapTag;
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
|
|
var getTag$1 = getTag;
|
|
|
|
/**
|
|
* Checks if `value` is object-like. A value is object-like if it's not `null`
|
|
* and has a `typeof` result of "object".
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 4.0.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is object-like, else `false`.
|
|
* @example
|
|
*
|
|
* _.isObjectLike({});
|
|
* // => true
|
|
*
|
|
* _.isObjectLike([1, 2, 3]);
|
|
* // => true
|
|
*
|
|
* _.isObjectLike(_.noop);
|
|
* // => false
|
|
*
|
|
* _.isObjectLike(null);
|
|
* // => false
|
|
*/
|
|
function isObjectLike(value) {
|
|
return value != null && typeof value == 'object';
|
|
}
|
|
|
|
/** `Object#toString` result references. */
|
|
var argsTag = '[object Arguments]';
|
|
|
|
/**
|
|
* The base implementation of `_.isArguments`.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is an `arguments` object,
|
|
*/
|
|
function baseIsArguments(value) {
|
|
return isObjectLike(value) && baseGetTag(value) == argsTag;
|
|
}
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$5 = Object.prototype;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$4 = objectProto$5.hasOwnProperty;
|
|
|
|
/** Built-in value references. */
|
|
var propertyIsEnumerable = objectProto$5.propertyIsEnumerable;
|
|
|
|
/**
|
|
* Checks if `value` is likely an `arguments` object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is an `arguments` object,
|
|
* else `false`.
|
|
* @example
|
|
*
|
|
* _.isArguments(function() { return arguments; }());
|
|
* // => true
|
|
*
|
|
* _.isArguments([1, 2, 3]);
|
|
* // => false
|
|
*/
|
|
var isArguments = baseIsArguments(function() { return arguments; }()) ? baseIsArguments : function(value) {
|
|
return isObjectLike(value) && hasOwnProperty$4.call(value, 'callee') &&
|
|
!propertyIsEnumerable.call(value, 'callee');
|
|
};
|
|
|
|
/**
|
|
* Checks if `value` is classified as an `Array` object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is an array, else `false`.
|
|
* @example
|
|
*
|
|
* _.isArray([1, 2, 3]);
|
|
* // => true
|
|
*
|
|
* _.isArray(document.body.children);
|
|
* // => false
|
|
*
|
|
* _.isArray('abc');
|
|
* // => false
|
|
*
|
|
* _.isArray(_.noop);
|
|
* // => false
|
|
*/
|
|
var isArray$1 = Array.isArray;
|
|
|
|
/** Used as references for various `Number` constants. */
|
|
var MAX_SAFE_INTEGER = 9007199254740991;
|
|
|
|
/**
|
|
* Checks if `value` is a valid array-like length.
|
|
*
|
|
* **Note:** This method is loosely based on
|
|
* [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 4.0.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a valid length, else `false`.
|
|
* @example
|
|
*
|
|
* _.isLength(3);
|
|
* // => true
|
|
*
|
|
* _.isLength(Number.MIN_VALUE);
|
|
* // => false
|
|
*
|
|
* _.isLength(Infinity);
|
|
* // => false
|
|
*
|
|
* _.isLength('3');
|
|
* // => false
|
|
*/
|
|
function isLength(value) {
|
|
return typeof value == 'number' &&
|
|
value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is array-like. A value is considered array-like if it's
|
|
* not a function and has a `value.length` that's an integer greater than or
|
|
* equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 4.0.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is array-like, else `false`.
|
|
* @example
|
|
*
|
|
* _.isArrayLike([1, 2, 3]);
|
|
* // => true
|
|
*
|
|
* _.isArrayLike(document.body.children);
|
|
* // => true
|
|
*
|
|
* _.isArrayLike('abc');
|
|
* // => true
|
|
*
|
|
* _.isArrayLike(_.noop);
|
|
* // => false
|
|
*/
|
|
function isArrayLike(value) {
|
|
return value != null && isLength(value.length) && !isFunction$1(value);
|
|
}
|
|
|
|
/**
|
|
* This method returns `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 4.13.0
|
|
* @category Util
|
|
* @returns {boolean} Returns `false`.
|
|
* @example
|
|
*
|
|
* _.times(2, _.stubFalse);
|
|
* // => [false, false]
|
|
*/
|
|
function stubFalse() {
|
|
return false;
|
|
}
|
|
|
|
/** Detect free variable `exports`. */
|
|
var freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports;
|
|
|
|
/** Detect free variable `module`. */
|
|
var freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module;
|
|
|
|
/** Detect the popular CommonJS extension `module.exports`. */
|
|
var moduleExports = freeModule && freeModule.exports === freeExports;
|
|
|
|
/** Built-in value references. */
|
|
var Buffer = moduleExports ? root.Buffer : undefined;
|
|
|
|
/* Built-in method references for those with the same name as other `lodash` methods. */
|
|
var nativeIsBuffer = Buffer ? Buffer.isBuffer : undefined;
|
|
|
|
/**
|
|
* Checks if `value` is a buffer.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 4.3.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a buffer, else `false`.
|
|
* @example
|
|
*
|
|
* _.isBuffer(new Buffer(2));
|
|
* // => true
|
|
*
|
|
* _.isBuffer(new Uint8Array(2));
|
|
* // => false
|
|
*/
|
|
var isBuffer = nativeIsBuffer || stubFalse;
|
|
|
|
/** `Object#toString` result references. */
|
|
var argsTag$1 = '[object Arguments]',
|
|
arrayTag = '[object Array]',
|
|
boolTag = '[object Boolean]',
|
|
dateTag = '[object Date]',
|
|
errorTag = '[object Error]',
|
|
funcTag$1 = '[object Function]',
|
|
mapTag$1 = '[object Map]',
|
|
numberTag = '[object Number]',
|
|
objectTag$1 = '[object Object]',
|
|
regexpTag = '[object RegExp]',
|
|
setTag$1 = '[object Set]',
|
|
stringTag = '[object String]',
|
|
weakMapTag$1 = '[object WeakMap]';
|
|
|
|
var arrayBufferTag = '[object ArrayBuffer]',
|
|
dataViewTag$1 = '[object DataView]',
|
|
float32Tag = '[object Float32Array]',
|
|
float64Tag = '[object Float64Array]',
|
|
int8Tag = '[object Int8Array]',
|
|
int16Tag = '[object Int16Array]',
|
|
int32Tag = '[object Int32Array]',
|
|
uint8Tag = '[object Uint8Array]',
|
|
uint8ClampedTag = '[object Uint8ClampedArray]',
|
|
uint16Tag = '[object Uint16Array]',
|
|
uint32Tag = '[object Uint32Array]';
|
|
|
|
/** Used to identify `toStringTag` values of typed arrays. */
|
|
var typedArrayTags = {};
|
|
typedArrayTags[float32Tag] = typedArrayTags[float64Tag] =
|
|
typedArrayTags[int8Tag] = typedArrayTags[int16Tag] =
|
|
typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] =
|
|
typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] =
|
|
typedArrayTags[uint32Tag] = true;
|
|
typedArrayTags[argsTag$1] = typedArrayTags[arrayTag] =
|
|
typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] =
|
|
typedArrayTags[dataViewTag$1] = typedArrayTags[dateTag] =
|
|
typedArrayTags[errorTag] = typedArrayTags[funcTag$1] =
|
|
typedArrayTags[mapTag$1] = typedArrayTags[numberTag] =
|
|
typedArrayTags[objectTag$1] = typedArrayTags[regexpTag] =
|
|
typedArrayTags[setTag$1] = typedArrayTags[stringTag] =
|
|
typedArrayTags[weakMapTag$1] = false;
|
|
|
|
/**
|
|
* The base implementation of `_.isTypedArray` without Node.js optimizations.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
|
|
*/
|
|
function baseIsTypedArray(value) {
|
|
return isObjectLike(value) &&
|
|
isLength(value.length) && !!typedArrayTags[baseGetTag(value)];
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.unary` without support for storing metadata.
|
|
*
|
|
* @private
|
|
* @param {Function} func The function to cap arguments for.
|
|
* @returns {Function} Returns the new capped function.
|
|
*/
|
|
function baseUnary(func) {
|
|
return function(value) {
|
|
return func(value);
|
|
};
|
|
}
|
|
|
|
/** Detect free variable `exports`. */
|
|
var freeExports$1 = typeof exports == 'object' && exports && !exports.nodeType && exports;
|
|
|
|
/** Detect free variable `module`. */
|
|
var freeModule$1 = freeExports$1 && typeof module == 'object' && module && !module.nodeType && module;
|
|
|
|
/** Detect the popular CommonJS extension `module.exports`. */
|
|
var moduleExports$1 = freeModule$1 && freeModule$1.exports === freeExports$1;
|
|
|
|
/** Detect free variable `process` from Node.js. */
|
|
var freeProcess = moduleExports$1 && freeGlobal.process;
|
|
|
|
/** Used to access faster Node.js helpers. */
|
|
var nodeUtil = (function() {
|
|
try {
|
|
// Use `util.types` for Node.js 10+.
|
|
var types = freeModule$1 && freeModule$1.require && freeModule$1.require('util').types;
|
|
|
|
if (types) {
|
|
return types;
|
|
}
|
|
|
|
// Legacy `process.binding('util')` for Node.js < 10.
|
|
return freeProcess && freeProcess.binding && freeProcess.binding('util');
|
|
} catch (e) {}
|
|
}());
|
|
|
|
/* Node.js helper references. */
|
|
var nodeIsTypedArray = nodeUtil && nodeUtil.isTypedArray;
|
|
|
|
/**
|
|
* Checks if `value` is classified as a typed array.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 3.0.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a typed array, else `false`.
|
|
* @example
|
|
*
|
|
* _.isTypedArray(new Uint8Array);
|
|
* // => true
|
|
*
|
|
* _.isTypedArray([]);
|
|
* // => false
|
|
*/
|
|
var isTypedArray = nodeIsTypedArray ? baseUnary(nodeIsTypedArray) : baseIsTypedArray;
|
|
|
|
/** `Object#toString` result references. */
|
|
var mapTag$2 = '[object Map]',
|
|
setTag$2 = '[object Set]';
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$6 = Object.prototype;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$5 = objectProto$6.hasOwnProperty;
|
|
|
|
/**
|
|
* Checks if `value` is an empty object, collection, map, or set.
|
|
*
|
|
* Objects are considered empty if they have no own enumerable string keyed
|
|
* properties.
|
|
*
|
|
* Array-like values such as `arguments` objects, arrays, buffers, strings, or
|
|
* jQuery-like collections are considered empty if they have a `length` of `0`.
|
|
* Similarly, maps and sets are considered empty if they have a `size` of `0`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @category Lang
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is empty, else `false`.
|
|
* @example
|
|
*
|
|
* _.isEmpty(null);
|
|
* // => true
|
|
*
|
|
* _.isEmpty(true);
|
|
* // => true
|
|
*
|
|
* _.isEmpty(1);
|
|
* // => true
|
|
*
|
|
* _.isEmpty([1, 2, 3]);
|
|
* // => false
|
|
*
|
|
* _.isEmpty({ 'a': 1 });
|
|
* // => false
|
|
*/
|
|
function isEmpty(value) {
|
|
if (value == null) {
|
|
return true;
|
|
}
|
|
if (isArrayLike(value) &&
|
|
(isArray$1(value) || typeof value == 'string' || typeof value.splice == 'function' ||
|
|
isBuffer(value) || isTypedArray(value) || isArguments(value))) {
|
|
return !value.length;
|
|
}
|
|
var tag = getTag$1(value);
|
|
if (tag == mapTag$2 || tag == setTag$2) {
|
|
return !value.size;
|
|
}
|
|
if (isPrototype(value)) {
|
|
return !baseKeys(value).length;
|
|
}
|
|
for (var key in value) {
|
|
if (hasOwnProperty$5.call(value, key)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function uniqueId() {
|
|
return 'f'
|
|
+ Math.random().toString(16).replace('.', '')
|
|
+ (+new Date()).toString(16);
|
|
}
|
|
|
|
function keyBy(array, keyfunc) {
|
|
let result = {};
|
|
for(let item of array) {
|
|
if (typeof keyfunc == 'string') {
|
|
result[item[keyfunc]] = item;
|
|
} else {
|
|
result[keyfunc(item)] = item;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
function objCopy(object) {
|
|
return JSON.parse(JSON.stringify(object));
|
|
}
|
|
|
|
function castId(id) {
|
|
// TODO no-op after switching to UUIDs
|
|
return +id;
|
|
}
|
|
|
|
// like _.isEmpty, but less stupid
|
|
function isEmpty$1(object) {
|
|
if (typeof object == 'number') {
|
|
return false;
|
|
}
|
|
|
|
if (object === true) {
|
|
return false;
|
|
}
|
|
|
|
if (typeof object == 'string') {
|
|
return object.length === 0;
|
|
}
|
|
|
|
return isEmpty(object)
|
|
}
|
|
|
|
var script$4 = {
|
|
name: "PropertyField",
|
|
props: ['model', 'values'],
|
|
data() {
|
|
return {
|
|
id: uniqueId(),
|
|
fieldRefs: [],
|
|
}
|
|
},
|
|
methods: {
|
|
addValue(event) {
|
|
this.values.push(objCopy(this.model.default));
|
|
|
|
nextTick(() => {
|
|
this.fieldRefs[this.values.length-1].focus();
|
|
});
|
|
},
|
|
removeValue(vi) {
|
|
this.values.splice(vi, 1);
|
|
},
|
|
setFieldRef(el) {
|
|
if (el) {
|
|
this.fieldRefs.push(el);
|
|
}
|
|
},
|
|
},
|
|
beforeUpdate() {
|
|
this.fieldRefs = [];
|
|
}
|
|
};
|
|
|
|
const _hoisted_1 = { key: 0 };
|
|
const _hoisted_2 = { key: 2 };
|
|
|
|
function render$4(_ctx, _cache, $props, $setup, $data, $options) {
|
|
const _component_string_value = resolveComponent("string-value");
|
|
const _component_integer_value = resolveComponent("integer-value");
|
|
const _component_decimal_value = resolveComponent("decimal-value");
|
|
const _component_boolean_value = resolveComponent("boolean-value");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
($props.values.length===0)
|
|
? (openBlock(), createBlock("tr", _hoisted_1, [
|
|
createVNode("th", {
|
|
onClick: _cache[1] || (_cache[1] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, toDisplayString($props.model.name), 1 /* TEXT */),
|
|
createVNode("td", null, [
|
|
createVNode("a", {
|
|
href: "#",
|
|
onClick: _cache[2] || (_cache[2] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, "Add")
|
|
])
|
|
]))
|
|
: (openBlock(true), createBlock(Fragment, { key: 1 }, renderList($props.values, (value, vi) => {
|
|
return (openBlock(), createBlock("tr", { key: vi }, [
|
|
(vi == 0)
|
|
? (openBlock(), createBlock("th", {
|
|
key: 0,
|
|
rowspan: $props.values.length + $props.model.multiple
|
|
}, [
|
|
createVNode("label", { for: $data.id }, toDisplayString($props.model.name), 9 /* TEXT, PROPS */, ["for"])
|
|
], 8 /* PROPS */, ["rowspan"]))
|
|
: createCommentVNode("v-if", true),
|
|
createVNode("td", null, [
|
|
($props.model.data_type==='String')
|
|
? (openBlock(), createBlock(_component_string_value, {
|
|
key: 0,
|
|
ref: $options.setFieldRef,
|
|
value: value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Integer')
|
|
? (openBlock(), createBlock(_component_integer_value, {
|
|
key: 1,
|
|
ref: $options.setFieldRef,
|
|
value: value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Decimal')
|
|
? (openBlock(), createBlock(_component_decimal_value, {
|
|
key: 2,
|
|
ref: $options.setFieldRef,
|
|
value: value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Boolean')
|
|
? (openBlock(), createBlock(_component_boolean_value, {
|
|
key: 3,
|
|
ref: $options.setFieldRef,
|
|
value: value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
(vi > 0 || $props.model.optional)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 4,
|
|
href: "#",
|
|
onClick: $event => ($options.removeValue(vi)),
|
|
style: {"margin-left":"5px"}
|
|
}, "X", 8 /* PROPS */, ["onClick"]))
|
|
: createCommentVNode("v-if", true)
|
|
])
|
|
]))
|
|
}), 128 /* KEYED_FRAGMENT */)),
|
|
($props.values.length > 0 && $props.model.multiple)
|
|
? (openBlock(), createBlock("tr", _hoisted_2, [
|
|
createVNode("td", null, [
|
|
createVNode("a", {
|
|
href: "#",
|
|
onClick: _cache[3] || (_cache[3] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, "Add")
|
|
])
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
}
|
|
|
|
script$4.render = render$4;
|
|
script$4.__file = "src/components/PropertyField.vue";
|
|
|
|
/**
|
|
* A specialized version of `_.forEach` for arrays without support for
|
|
* iteratee shorthands.
|
|
*
|
|
* @private
|
|
* @param {Array} [array] The array to iterate over.
|
|
* @param {Function} iteratee The function invoked per iteration.
|
|
* @returns {Array} Returns `array`.
|
|
*/
|
|
function arrayEach(array, iteratee) {
|
|
var index = -1,
|
|
length = array == null ? 0 : array.length;
|
|
|
|
while (++index < length) {
|
|
if (iteratee(array[index], index, array) === false) {
|
|
break;
|
|
}
|
|
}
|
|
return array;
|
|
}
|
|
|
|
/**
|
|
* Creates a base function for methods like `_.forIn` and `_.forOwn`.
|
|
*
|
|
* @private
|
|
* @param {boolean} [fromRight] Specify iterating from right to left.
|
|
* @returns {Function} Returns the new base function.
|
|
*/
|
|
function createBaseFor(fromRight) {
|
|
return function(object, iteratee, keysFunc) {
|
|
var index = -1,
|
|
iterable = Object(object),
|
|
props = keysFunc(object),
|
|
length = props.length;
|
|
|
|
while (length--) {
|
|
var key = props[fromRight ? length : ++index];
|
|
if (iteratee(iterable[key], key, iterable) === false) {
|
|
break;
|
|
}
|
|
}
|
|
return object;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `baseForOwn` which iterates over `object`
|
|
* properties returned by `keysFunc` and invokes `iteratee` for each property.
|
|
* Iteratee functions may exit iteration early by explicitly returning `false`.
|
|
*
|
|
* @private
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} iteratee The function invoked per iteration.
|
|
* @param {Function} keysFunc The function to get the keys of `object`.
|
|
* @returns {Object} Returns `object`.
|
|
*/
|
|
var baseFor = createBaseFor();
|
|
|
|
/**
|
|
* The base implementation of `_.times` without support for iteratee shorthands
|
|
* or max array length checks.
|
|
*
|
|
* @private
|
|
* @param {number} n The number of times to invoke `iteratee`.
|
|
* @param {Function} iteratee The function invoked per iteration.
|
|
* @returns {Array} Returns the array of results.
|
|
*/
|
|
function baseTimes(n, iteratee) {
|
|
var index = -1,
|
|
result = Array(n);
|
|
|
|
while (++index < n) {
|
|
result[index] = iteratee(index);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/** Used as references for various `Number` constants. */
|
|
var MAX_SAFE_INTEGER$1 = 9007199254740991;
|
|
|
|
/** Used to detect unsigned integer values. */
|
|
var reIsUint = /^(?:0|[1-9]\d*)$/;
|
|
|
|
/**
|
|
* Checks if `value` is a valid array-like index.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index.
|
|
* @returns {boolean} Returns `true` if `value` is a valid index, else `false`.
|
|
*/
|
|
function isIndex(value, length) {
|
|
var type = typeof value;
|
|
length = length == null ? MAX_SAFE_INTEGER$1 : length;
|
|
|
|
return !!length &&
|
|
(type == 'number' ||
|
|
(type != 'symbol' && reIsUint.test(value))) &&
|
|
(value > -1 && value % 1 == 0 && value < length);
|
|
}
|
|
|
|
/** Used for built-in method references. */
|
|
var objectProto$7 = Object.prototype;
|
|
|
|
/** Used to check objects for own properties. */
|
|
var hasOwnProperty$6 = objectProto$7.hasOwnProperty;
|
|
|
|
/**
|
|
* Creates an array of the enumerable property names of the array-like `value`.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to query.
|
|
* @param {boolean} inherited Specify returning inherited property names.
|
|
* @returns {Array} Returns the array of property names.
|
|
*/
|
|
function arrayLikeKeys(value, inherited) {
|
|
var isArr = isArray$1(value),
|
|
isArg = !isArr && isArguments(value),
|
|
isBuff = !isArr && !isArg && isBuffer(value),
|
|
isType = !isArr && !isArg && !isBuff && isTypedArray(value),
|
|
skipIndexes = isArr || isArg || isBuff || isType,
|
|
result = skipIndexes ? baseTimes(value.length, String) : [],
|
|
length = result.length;
|
|
|
|
for (var key in value) {
|
|
if ((inherited || hasOwnProperty$6.call(value, key)) &&
|
|
!(skipIndexes && (
|
|
// Safari 9 has enumerable `arguments.length` in strict mode.
|
|
key == 'length' ||
|
|
// Node.js 0.10 has enumerable non-index properties on buffers.
|
|
(isBuff && (key == 'offset' || key == 'parent')) ||
|
|
// PhantomJS 2 has enumerable non-index properties on typed arrays.
|
|
(isType && (key == 'buffer' || key == 'byteLength' || key == 'byteOffset')) ||
|
|
// Skip index properties.
|
|
isIndex(key, length)
|
|
))) {
|
|
result.push(key);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of the own enumerable property names of `object`.
|
|
*
|
|
* **Note:** Non-object values are coerced to objects. See the
|
|
* [ES spec](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)
|
|
* for more details.
|
|
*
|
|
* @static
|
|
* @since 0.1.0
|
|
* @memberOf _
|
|
* @category Object
|
|
* @param {Object} object The object to query.
|
|
* @returns {Array} Returns the array of property names.
|
|
* @example
|
|
*
|
|
* function Foo() {
|
|
* this.a = 1;
|
|
* this.b = 2;
|
|
* }
|
|
*
|
|
* Foo.prototype.c = 3;
|
|
*
|
|
* _.keys(new Foo);
|
|
* // => ['a', 'b'] (iteration order is not guaranteed)
|
|
*
|
|
* _.keys('hi');
|
|
* // => ['0', '1']
|
|
*/
|
|
function keys(object) {
|
|
return isArrayLike(object) ? arrayLikeKeys(object) : baseKeys(object);
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.forOwn` without support for iteratee shorthands.
|
|
*
|
|
* @private
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} iteratee The function invoked per iteration.
|
|
* @returns {Object} Returns `object`.
|
|
*/
|
|
function baseForOwn(object, iteratee) {
|
|
return object && baseFor(object, iteratee, keys);
|
|
}
|
|
|
|
/**
|
|
* Creates a `baseEach` or `baseEachRight` function.
|
|
*
|
|
* @private
|
|
* @param {Function} eachFunc The function to iterate over a collection.
|
|
* @param {boolean} [fromRight] Specify iterating from right to left.
|
|
* @returns {Function} Returns the new base function.
|
|
*/
|
|
function createBaseEach(eachFunc, fromRight) {
|
|
return function(collection, iteratee) {
|
|
if (collection == null) {
|
|
return collection;
|
|
}
|
|
if (!isArrayLike(collection)) {
|
|
return eachFunc(collection, iteratee);
|
|
}
|
|
var length = collection.length,
|
|
index = fromRight ? length : -1,
|
|
iterable = Object(collection);
|
|
|
|
while ((fromRight ? index-- : ++index < length)) {
|
|
if (iteratee(iterable[index], index, iterable) === false) {
|
|
break;
|
|
}
|
|
}
|
|
return collection;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.forEach` without support for iteratee shorthands.
|
|
*
|
|
* @private
|
|
* @param {Array|Object} collection The collection to iterate over.
|
|
* @param {Function} iteratee The function invoked per iteration.
|
|
* @returns {Array|Object} Returns `collection`.
|
|
*/
|
|
var baseEach = createBaseEach(baseForOwn);
|
|
|
|
/**
|
|
* This method returns the first argument it receives.
|
|
*
|
|
* @static
|
|
* @since 0.1.0
|
|
* @memberOf _
|
|
* @category Util
|
|
* @param {*} value Any value.
|
|
* @returns {*} Returns `value`.
|
|
* @example
|
|
*
|
|
* var object = { 'a': 1 };
|
|
*
|
|
* console.log(_.identity(object) === object);
|
|
* // => true
|
|
*/
|
|
function identity(value) {
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Casts `value` to `identity` if it's not a function.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to inspect.
|
|
* @returns {Function} Returns cast function.
|
|
*/
|
|
function castFunction(value) {
|
|
return typeof value == 'function' ? value : identity;
|
|
}
|
|
|
|
/**
|
|
* Iterates over elements of `collection` and invokes `iteratee` for each element.
|
|
* The iteratee is invoked with three arguments: (value, index|key, collection).
|
|
* Iteratee functions may exit iteration early by explicitly returning `false`.
|
|
*
|
|
* **Note:** As with other "Collections" methods, objects with a "length"
|
|
* property are iterated like arrays. To avoid this behavior use `_.forIn`
|
|
* or `_.forOwn` for object iteration.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @since 0.1.0
|
|
* @alias each
|
|
* @category Collection
|
|
* @param {Array|Object} collection The collection to iterate over.
|
|
* @param {Function} [iteratee=_.identity] The function invoked per iteration.
|
|
* @returns {Array|Object} Returns `collection`.
|
|
* @see _.forEachRight
|
|
* @example
|
|
*
|
|
* _.forEach([1, 2], function(value) {
|
|
* console.log(value);
|
|
* });
|
|
* // => Logs `1` then `2`.
|
|
*
|
|
* _.forEach({ 'a': 1, 'b': 2 }, function(value, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => Logs 'a' then 'b' (iteration order is not guaranteed).
|
|
*/
|
|
function forEach(collection, iteratee) {
|
|
var func = isArray$1(collection) ? arrayEach : baseEach;
|
|
return func(collection, castFunction(iteratee));
|
|
}
|
|
|
|
var bind = function bind(fn, thisArg) {
|
|
return function wrap() {
|
|
var args = new Array(arguments.length);
|
|
for (var i = 0; i < args.length; i++) {
|
|
args[i] = arguments[i];
|
|
}
|
|
return fn.apply(thisArg, args);
|
|
};
|
|
};
|
|
|
|
/*global toString:true*/
|
|
|
|
// utils is a library of generic helper functions non-specific to axios
|
|
|
|
var toString = Object.prototype.toString;
|
|
|
|
/**
|
|
* Determine if a value is an Array
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is an Array, otherwise false
|
|
*/
|
|
function isArray$2(val) {
|
|
return toString.call(val) === '[object Array]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is undefined
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if the value is undefined, otherwise false
|
|
*/
|
|
function isUndefined(val) {
|
|
return typeof val === 'undefined';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Buffer
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Buffer, otherwise false
|
|
*/
|
|
function isBuffer$1(val) {
|
|
return val !== null && !isUndefined(val) && val.constructor !== null && !isUndefined(val.constructor)
|
|
&& typeof val.constructor.isBuffer === 'function' && val.constructor.isBuffer(val);
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is an ArrayBuffer
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is an ArrayBuffer, otherwise false
|
|
*/
|
|
function isArrayBuffer(val) {
|
|
return toString.call(val) === '[object ArrayBuffer]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a FormData
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is an FormData, otherwise false
|
|
*/
|
|
function isFormData(val) {
|
|
return (typeof FormData !== 'undefined') && (val instanceof FormData);
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a view on an ArrayBuffer
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a view on an ArrayBuffer, otherwise false
|
|
*/
|
|
function isArrayBufferView(val) {
|
|
var result;
|
|
if ((typeof ArrayBuffer !== 'undefined') && (ArrayBuffer.isView)) {
|
|
result = ArrayBuffer.isView(val);
|
|
} else {
|
|
result = (val) && (val.buffer) && (val.buffer instanceof ArrayBuffer);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a String
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a String, otherwise false
|
|
*/
|
|
function isString$1(val) {
|
|
return typeof val === 'string';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Number
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Number, otherwise false
|
|
*/
|
|
function isNumber(val) {
|
|
return typeof val === 'number';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is an Object
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is an Object, otherwise false
|
|
*/
|
|
function isObject$2(val) {
|
|
return val !== null && typeof val === 'object';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a plain Object
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @return {boolean} True if value is a plain Object, otherwise false
|
|
*/
|
|
function isPlainObject$1(val) {
|
|
if (toString.call(val) !== '[object Object]') {
|
|
return false;
|
|
}
|
|
|
|
var prototype = Object.getPrototypeOf(val);
|
|
return prototype === null || prototype === Object.prototype;
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Date
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Date, otherwise false
|
|
*/
|
|
function isDate$1(val) {
|
|
return toString.call(val) === '[object Date]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a File
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a File, otherwise false
|
|
*/
|
|
function isFile(val) {
|
|
return toString.call(val) === '[object File]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Blob
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Blob, otherwise false
|
|
*/
|
|
function isBlob(val) {
|
|
return toString.call(val) === '[object Blob]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Function
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Function, otherwise false
|
|
*/
|
|
function isFunction$2(val) {
|
|
return toString.call(val) === '[object Function]';
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a Stream
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a Stream, otherwise false
|
|
*/
|
|
function isStream(val) {
|
|
return isObject$2(val) && isFunction$2(val.pipe);
|
|
}
|
|
|
|
/**
|
|
* Determine if a value is a URLSearchParams object
|
|
*
|
|
* @param {Object} val The value to test
|
|
* @returns {boolean} True if value is a URLSearchParams object, otherwise false
|
|
*/
|
|
function isURLSearchParams(val) {
|
|
return typeof URLSearchParams !== 'undefined' && val instanceof URLSearchParams;
|
|
}
|
|
|
|
/**
|
|
* Trim excess whitespace off the beginning and end of a string
|
|
*
|
|
* @param {String} str The String to trim
|
|
* @returns {String} The String freed of excess whitespace
|
|
*/
|
|
function trim(str) {
|
|
return str.replace(/^\s*/, '').replace(/\s*$/, '');
|
|
}
|
|
|
|
/**
|
|
* Determine if we're running in a standard browser environment
|
|
*
|
|
* This allows axios to run in a web worker, and react-native.
|
|
* Both environments support XMLHttpRequest, but not fully standard globals.
|
|
*
|
|
* web workers:
|
|
* typeof window -> undefined
|
|
* typeof document -> undefined
|
|
*
|
|
* react-native:
|
|
* navigator.product -> 'ReactNative'
|
|
* nativescript
|
|
* navigator.product -> 'NativeScript' or 'NS'
|
|
*/
|
|
function isStandardBrowserEnv() {
|
|
if (typeof navigator !== 'undefined' && (navigator.product === 'ReactNative' ||
|
|
navigator.product === 'NativeScript' ||
|
|
navigator.product === 'NS')) {
|
|
return false;
|
|
}
|
|
return (
|
|
typeof window !== 'undefined' &&
|
|
typeof document !== 'undefined'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Iterate over an Array or an Object invoking a function for each item.
|
|
*
|
|
* If `obj` is an Array callback will be called passing
|
|
* the value, index, and complete array for each item.
|
|
*
|
|
* If 'obj' is an Object callback will be called passing
|
|
* the value, key, and complete object for each property.
|
|
*
|
|
* @param {Object|Array} obj The object to iterate
|
|
* @param {Function} fn The callback to invoke for each item
|
|
*/
|
|
function forEach$1(obj, fn) {
|
|
// Don't bother if no value provided
|
|
if (obj === null || typeof obj === 'undefined') {
|
|
return;
|
|
}
|
|
|
|
// Force an array if not already something iterable
|
|
if (typeof obj !== 'object') {
|
|
/*eslint no-param-reassign:0*/
|
|
obj = [obj];
|
|
}
|
|
|
|
if (isArray$2(obj)) {
|
|
// Iterate over array values
|
|
for (var i = 0, l = obj.length; i < l; i++) {
|
|
fn.call(null, obj[i], i, obj);
|
|
}
|
|
} else {
|
|
// Iterate over object keys
|
|
for (var key in obj) {
|
|
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
fn.call(null, obj[key], key, obj);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Accepts varargs expecting each argument to be an object, then
|
|
* immutably merges the properties of each object and returns result.
|
|
*
|
|
* When multiple objects contain the same key the later object in
|
|
* the arguments list will take precedence.
|
|
*
|
|
* Example:
|
|
*
|
|
* ```js
|
|
* var result = merge({foo: 123}, {foo: 456});
|
|
* console.log(result.foo); // outputs 456
|
|
* ```
|
|
*
|
|
* @param {Object} obj1 Object to merge
|
|
* @returns {Object} Result of all merge properties
|
|
*/
|
|
function merge(/* obj1, obj2, obj3, ... */) {
|
|
var result = {};
|
|
function assignValue(val, key) {
|
|
if (isPlainObject$1(result[key]) && isPlainObject$1(val)) {
|
|
result[key] = merge(result[key], val);
|
|
} else if (isPlainObject$1(val)) {
|
|
result[key] = merge({}, val);
|
|
} else if (isArray$2(val)) {
|
|
result[key] = val.slice();
|
|
} else {
|
|
result[key] = val;
|
|
}
|
|
}
|
|
|
|
for (var i = 0, l = arguments.length; i < l; i++) {
|
|
forEach$1(arguments[i], assignValue);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Extends object a by mutably adding to it the properties of object b.
|
|
*
|
|
* @param {Object} a The object to be extended
|
|
* @param {Object} b The object to copy properties from
|
|
* @param {Object} thisArg The object to bind function to
|
|
* @return {Object} The resulting value of object a
|
|
*/
|
|
function extend$1(a, b, thisArg) {
|
|
forEach$1(b, function assignValue(val, key) {
|
|
if (thisArg && typeof val === 'function') {
|
|
a[key] = bind(val, thisArg);
|
|
} else {
|
|
a[key] = val;
|
|
}
|
|
});
|
|
return a;
|
|
}
|
|
|
|
/**
|
|
* Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)
|
|
*
|
|
* @param {string} content with BOM
|
|
* @return {string} content value without BOM
|
|
*/
|
|
function stripBOM(content) {
|
|
if (content.charCodeAt(0) === 0xFEFF) {
|
|
content = content.slice(1);
|
|
}
|
|
return content;
|
|
}
|
|
|
|
var utils = {
|
|
isArray: isArray$2,
|
|
isArrayBuffer: isArrayBuffer,
|
|
isBuffer: isBuffer$1,
|
|
isFormData: isFormData,
|
|
isArrayBufferView: isArrayBufferView,
|
|
isString: isString$1,
|
|
isNumber: isNumber,
|
|
isObject: isObject$2,
|
|
isPlainObject: isPlainObject$1,
|
|
isUndefined: isUndefined,
|
|
isDate: isDate$1,
|
|
isFile: isFile,
|
|
isBlob: isBlob,
|
|
isFunction: isFunction$2,
|
|
isStream: isStream,
|
|
isURLSearchParams: isURLSearchParams,
|
|
isStandardBrowserEnv: isStandardBrowserEnv,
|
|
forEach: forEach$1,
|
|
merge: merge,
|
|
extend: extend$1,
|
|
trim: trim,
|
|
stripBOM: stripBOM
|
|
};
|
|
|
|
function encode(val) {
|
|
return encodeURIComponent(val).
|
|
replace(/%3A/gi, ':').
|
|
replace(/%24/g, '$').
|
|
replace(/%2C/gi, ',').
|
|
replace(/%20/g, '+').
|
|
replace(/%5B/gi, '[').
|
|
replace(/%5D/gi, ']');
|
|
}
|
|
|
|
/**
|
|
* Build a URL by appending params to the end
|
|
*
|
|
* @param {string} url The base of the url (e.g., http://www.google.com)
|
|
* @param {object} [params] The params to be appended
|
|
* @returns {string} The formatted url
|
|
*/
|
|
var buildURL = function buildURL(url, params, paramsSerializer) {
|
|
/*eslint no-param-reassign:0*/
|
|
if (!params) {
|
|
return url;
|
|
}
|
|
|
|
var serializedParams;
|
|
if (paramsSerializer) {
|
|
serializedParams = paramsSerializer(params);
|
|
} else if (utils.isURLSearchParams(params)) {
|
|
serializedParams = params.toString();
|
|
} else {
|
|
var parts = [];
|
|
|
|
utils.forEach(params, function serialize(val, key) {
|
|
if (val === null || typeof val === 'undefined') {
|
|
return;
|
|
}
|
|
|
|
if (utils.isArray(val)) {
|
|
key = key + '[]';
|
|
} else {
|
|
val = [val];
|
|
}
|
|
|
|
utils.forEach(val, function parseValue(v) {
|
|
if (utils.isDate(v)) {
|
|
v = v.toISOString();
|
|
} else if (utils.isObject(v)) {
|
|
v = JSON.stringify(v);
|
|
}
|
|
parts.push(encode(key) + '=' + encode(v));
|
|
});
|
|
});
|
|
|
|
serializedParams = parts.join('&');
|
|
}
|
|
|
|
if (serializedParams) {
|
|
var hashmarkIndex = url.indexOf('#');
|
|
if (hashmarkIndex !== -1) {
|
|
url = url.slice(0, hashmarkIndex);
|
|
}
|
|
|
|
url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;
|
|
}
|
|
|
|
return url;
|
|
};
|
|
|
|
function InterceptorManager() {
|
|
this.handlers = [];
|
|
}
|
|
|
|
/**
|
|
* Add a new interceptor to the stack
|
|
*
|
|
* @param {Function} fulfilled The function to handle `then` for a `Promise`
|
|
* @param {Function} rejected The function to handle `reject` for a `Promise`
|
|
*
|
|
* @return {Number} An ID used to remove interceptor later
|
|
*/
|
|
InterceptorManager.prototype.use = function use(fulfilled, rejected) {
|
|
this.handlers.push({
|
|
fulfilled: fulfilled,
|
|
rejected: rejected
|
|
});
|
|
return this.handlers.length - 1;
|
|
};
|
|
|
|
/**
|
|
* Remove an interceptor from the stack
|
|
*
|
|
* @param {Number} id The ID that was returned by `use`
|
|
*/
|
|
InterceptorManager.prototype.eject = function eject(id) {
|
|
if (this.handlers[id]) {
|
|
this.handlers[id] = null;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Iterate over all the registered interceptors
|
|
*
|
|
* This method is particularly useful for skipping over any
|
|
* interceptors that may have become `null` calling `eject`.
|
|
*
|
|
* @param {Function} fn The function to call for each interceptor
|
|
*/
|
|
InterceptorManager.prototype.forEach = function forEach(fn) {
|
|
utils.forEach(this.handlers, function forEachHandler(h) {
|
|
if (h !== null) {
|
|
fn(h);
|
|
}
|
|
});
|
|
};
|
|
|
|
var InterceptorManager_1 = InterceptorManager;
|
|
|
|
/**
|
|
* Transform the data for a request or a response
|
|
*
|
|
* @param {Object|String} data The data to be transformed
|
|
* @param {Array} headers The headers for the request or response
|
|
* @param {Array|Function} fns A single function or Array of functions
|
|
* @returns {*} The resulting transformed data
|
|
*/
|
|
var transformData = function transformData(data, headers, fns) {
|
|
/*eslint no-param-reassign:0*/
|
|
utils.forEach(fns, function transform(fn) {
|
|
data = fn(data, headers);
|
|
});
|
|
|
|
return data;
|
|
};
|
|
|
|
var isCancel = function isCancel(value) {
|
|
return !!(value && value.__CANCEL__);
|
|
};
|
|
|
|
var normalizeHeaderName = function normalizeHeaderName(headers, normalizedName) {
|
|
utils.forEach(headers, function processHeader(value, name) {
|
|
if (name !== normalizedName && name.toUpperCase() === normalizedName.toUpperCase()) {
|
|
headers[normalizedName] = value;
|
|
delete headers[name];
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Update an Error with the specified config, error code, and response.
|
|
*
|
|
* @param {Error} error The error to update.
|
|
* @param {Object} config The config.
|
|
* @param {string} [code] The error code (for example, 'ECONNABORTED').
|
|
* @param {Object} [request] The request.
|
|
* @param {Object} [response] The response.
|
|
* @returns {Error} The error.
|
|
*/
|
|
var enhanceError = function enhanceError(error, config, code, request, response) {
|
|
error.config = config;
|
|
if (code) {
|
|
error.code = code;
|
|
}
|
|
|
|
error.request = request;
|
|
error.response = response;
|
|
error.isAxiosError = true;
|
|
|
|
error.toJSON = function toJSON() {
|
|
return {
|
|
// Standard
|
|
message: this.message,
|
|
name: this.name,
|
|
// Microsoft
|
|
description: this.description,
|
|
number: this.number,
|
|
// Mozilla
|
|
fileName: this.fileName,
|
|
lineNumber: this.lineNumber,
|
|
columnNumber: this.columnNumber,
|
|
stack: this.stack,
|
|
// Axios
|
|
config: this.config,
|
|
code: this.code
|
|
};
|
|
};
|
|
return error;
|
|
};
|
|
|
|
/**
|
|
* Create an Error with the specified message, config, error code, request and response.
|
|
*
|
|
* @param {string} message The error message.
|
|
* @param {Object} config The config.
|
|
* @param {string} [code] The error code (for example, 'ECONNABORTED').
|
|
* @param {Object} [request] The request.
|
|
* @param {Object} [response] The response.
|
|
* @returns {Error} The created error.
|
|
*/
|
|
var createError = function createError(message, config, code, request, response) {
|
|
var error = new Error(message);
|
|
return enhanceError(error, config, code, request, response);
|
|
};
|
|
|
|
/**
|
|
* Resolve or reject a Promise based on response status.
|
|
*
|
|
* @param {Function} resolve A function that resolves the promise.
|
|
* @param {Function} reject A function that rejects the promise.
|
|
* @param {object} response The response.
|
|
*/
|
|
var settle = function settle(resolve, reject, response) {
|
|
var validateStatus = response.config.validateStatus;
|
|
if (!response.status || !validateStatus || validateStatus(response.status)) {
|
|
resolve(response);
|
|
} else {
|
|
reject(createError(
|
|
'Request failed with status code ' + response.status,
|
|
response.config,
|
|
null,
|
|
response.request,
|
|
response
|
|
));
|
|
}
|
|
};
|
|
|
|
var cookies = (
|
|
utils.isStandardBrowserEnv() ?
|
|
|
|
// Standard browser envs support document.cookie
|
|
(function standardBrowserEnv() {
|
|
return {
|
|
write: function write(name, value, expires, path, domain, secure) {
|
|
var cookie = [];
|
|
cookie.push(name + '=' + encodeURIComponent(value));
|
|
|
|
if (utils.isNumber(expires)) {
|
|
cookie.push('expires=' + new Date(expires).toGMTString());
|
|
}
|
|
|
|
if (utils.isString(path)) {
|
|
cookie.push('path=' + path);
|
|
}
|
|
|
|
if (utils.isString(domain)) {
|
|
cookie.push('domain=' + domain);
|
|
}
|
|
|
|
if (secure === true) {
|
|
cookie.push('secure');
|
|
}
|
|
|
|
document.cookie = cookie.join('; ');
|
|
},
|
|
|
|
read: function read(name) {
|
|
var match = document.cookie.match(new RegExp('(^|;\\s*)(' + name + ')=([^;]*)'));
|
|
return (match ? decodeURIComponent(match[3]) : null);
|
|
},
|
|
|
|
remove: function remove(name) {
|
|
this.write(name, '', Date.now() - 86400000);
|
|
}
|
|
};
|
|
})() :
|
|
|
|
// Non standard browser env (web workers, react-native) lack needed support.
|
|
(function nonStandardBrowserEnv() {
|
|
return {
|
|
write: function write() {},
|
|
read: function read() { return null; },
|
|
remove: function remove() {}
|
|
};
|
|
})()
|
|
);
|
|
|
|
/**
|
|
* Determines whether the specified URL is absolute
|
|
*
|
|
* @param {string} url The URL to test
|
|
* @returns {boolean} True if the specified URL is absolute, otherwise false
|
|
*/
|
|
var isAbsoluteURL = function isAbsoluteURL(url) {
|
|
// A URL is considered absolute if it begins with "<scheme>://" or "//" (protocol-relative URL).
|
|
// RFC 3986 defines scheme name as a sequence of characters beginning with a letter and followed
|
|
// by any combination of letters, digits, plus, period, or hyphen.
|
|
return /^([a-z][a-z\d\+\-\.]*:)?\/\//i.test(url);
|
|
};
|
|
|
|
/**
|
|
* Creates a new URL by combining the specified URLs
|
|
*
|
|
* @param {string} baseURL The base URL
|
|
* @param {string} relativeURL The relative URL
|
|
* @returns {string} The combined URL
|
|
*/
|
|
var combineURLs = function combineURLs(baseURL, relativeURL) {
|
|
return relativeURL
|
|
? baseURL.replace(/\/+$/, '') + '/' + relativeURL.replace(/^\/+/, '')
|
|
: baseURL;
|
|
};
|
|
|
|
/**
|
|
* Creates a new URL by combining the baseURL with the requestedURL,
|
|
* only when the requestedURL is not already an absolute URL.
|
|
* If the requestURL is absolute, this function returns the requestedURL untouched.
|
|
*
|
|
* @param {string} baseURL The base URL
|
|
* @param {string} requestedURL Absolute or relative URL to combine
|
|
* @returns {string} The combined full path
|
|
*/
|
|
var buildFullPath = function buildFullPath(baseURL, requestedURL) {
|
|
if (baseURL && !isAbsoluteURL(requestedURL)) {
|
|
return combineURLs(baseURL, requestedURL);
|
|
}
|
|
return requestedURL;
|
|
};
|
|
|
|
// Headers whose duplicates are ignored by node
|
|
// c.f. https://nodejs.org/api/http.html#http_message_headers
|
|
var ignoreDuplicateOf = [
|
|
'age', 'authorization', 'content-length', 'content-type', 'etag',
|
|
'expires', 'from', 'host', 'if-modified-since', 'if-unmodified-since',
|
|
'last-modified', 'location', 'max-forwards', 'proxy-authorization',
|
|
'referer', 'retry-after', 'user-agent'
|
|
];
|
|
|
|
/**
|
|
* Parse headers into an object
|
|
*
|
|
* ```
|
|
* Date: Wed, 27 Aug 2014 08:58:49 GMT
|
|
* Content-Type: application/json
|
|
* Connection: keep-alive
|
|
* Transfer-Encoding: chunked
|
|
* ```
|
|
*
|
|
* @param {String} headers Headers needing to be parsed
|
|
* @returns {Object} Headers parsed into an object
|
|
*/
|
|
var parseHeaders = function parseHeaders(headers) {
|
|
var parsed = {};
|
|
var key;
|
|
var val;
|
|
var i;
|
|
|
|
if (!headers) { return parsed; }
|
|
|
|
utils.forEach(headers.split('\n'), function parser(line) {
|
|
i = line.indexOf(':');
|
|
key = utils.trim(line.substr(0, i)).toLowerCase();
|
|
val = utils.trim(line.substr(i + 1));
|
|
|
|
if (key) {
|
|
if (parsed[key] && ignoreDuplicateOf.indexOf(key) >= 0) {
|
|
return;
|
|
}
|
|
if (key === 'set-cookie') {
|
|
parsed[key] = (parsed[key] ? parsed[key] : []).concat([val]);
|
|
} else {
|
|
parsed[key] = parsed[key] ? parsed[key] + ', ' + val : val;
|
|
}
|
|
}
|
|
});
|
|
|
|
return parsed;
|
|
};
|
|
|
|
var isURLSameOrigin = (
|
|
utils.isStandardBrowserEnv() ?
|
|
|
|
// Standard browser envs have full support of the APIs needed to test
|
|
// whether the request URL is of the same origin as current location.
|
|
(function standardBrowserEnv() {
|
|
var msie = /(msie|trident)/i.test(navigator.userAgent);
|
|
var urlParsingNode = document.createElement('a');
|
|
var originURL;
|
|
|
|
/**
|
|
* Parse a URL to discover it's components
|
|
*
|
|
* @param {String} url The URL to be parsed
|
|
* @returns {Object}
|
|
*/
|
|
function resolveURL(url) {
|
|
var href = url;
|
|
|
|
if (msie) {
|
|
// IE needs attribute set twice to normalize properties
|
|
urlParsingNode.setAttribute('href', href);
|
|
href = urlParsingNode.href;
|
|
}
|
|
|
|
urlParsingNode.setAttribute('href', href);
|
|
|
|
// urlParsingNode provides the UrlUtils interface - http://url.spec.whatwg.org/#urlutils
|
|
return {
|
|
href: urlParsingNode.href,
|
|
protocol: urlParsingNode.protocol ? urlParsingNode.protocol.replace(/:$/, '') : '',
|
|
host: urlParsingNode.host,
|
|
search: urlParsingNode.search ? urlParsingNode.search.replace(/^\?/, '') : '',
|
|
hash: urlParsingNode.hash ? urlParsingNode.hash.replace(/^#/, '') : '',
|
|
hostname: urlParsingNode.hostname,
|
|
port: urlParsingNode.port,
|
|
pathname: (urlParsingNode.pathname.charAt(0) === '/') ?
|
|
urlParsingNode.pathname :
|
|
'/' + urlParsingNode.pathname
|
|
};
|
|
}
|
|
|
|
originURL = resolveURL(window.location.href);
|
|
|
|
/**
|
|
* Determine if a URL shares the same origin as the current location
|
|
*
|
|
* @param {String} requestURL The URL to test
|
|
* @returns {boolean} True if URL shares the same origin, otherwise false
|
|
*/
|
|
return function isURLSameOrigin(requestURL) {
|
|
var parsed = (utils.isString(requestURL)) ? resolveURL(requestURL) : requestURL;
|
|
return (parsed.protocol === originURL.protocol &&
|
|
parsed.host === originURL.host);
|
|
};
|
|
})() :
|
|
|
|
// Non standard browser envs (web workers, react-native) lack needed support.
|
|
(function nonStandardBrowserEnv() {
|
|
return function isURLSameOrigin() {
|
|
return true;
|
|
};
|
|
})()
|
|
);
|
|
|
|
var xhr = function xhrAdapter(config) {
|
|
return new Promise(function dispatchXhrRequest(resolve, reject) {
|
|
var requestData = config.data;
|
|
var requestHeaders = config.headers;
|
|
|
|
if (utils.isFormData(requestData)) {
|
|
delete requestHeaders['Content-Type']; // Let the browser set it
|
|
}
|
|
|
|
var request = new XMLHttpRequest();
|
|
|
|
// HTTP basic authentication
|
|
if (config.auth) {
|
|
var username = config.auth.username || '';
|
|
var password = config.auth.password ? unescape(encodeURIComponent(config.auth.password)) : '';
|
|
requestHeaders.Authorization = 'Basic ' + btoa(username + ':' + password);
|
|
}
|
|
|
|
var fullPath = buildFullPath(config.baseURL, config.url);
|
|
request.open(config.method.toUpperCase(), buildURL(fullPath, config.params, config.paramsSerializer), true);
|
|
|
|
// Set the request timeout in MS
|
|
request.timeout = config.timeout;
|
|
|
|
// Listen for ready state
|
|
request.onreadystatechange = function handleLoad() {
|
|
if (!request || request.readyState !== 4) {
|
|
return;
|
|
}
|
|
|
|
// The request errored out and we didn't get a response, this will be
|
|
// handled by onerror instead
|
|
// With one exception: request that using file: protocol, most browsers
|
|
// will return status as 0 even though it's a successful request
|
|
if (request.status === 0 && !(request.responseURL && request.responseURL.indexOf('file:') === 0)) {
|
|
return;
|
|
}
|
|
|
|
// Prepare the response
|
|
var responseHeaders = 'getAllResponseHeaders' in request ? parseHeaders(request.getAllResponseHeaders()) : null;
|
|
var responseData = !config.responseType || config.responseType === 'text' ? request.responseText : request.response;
|
|
var response = {
|
|
data: responseData,
|
|
status: request.status,
|
|
statusText: request.statusText,
|
|
headers: responseHeaders,
|
|
config: config,
|
|
request: request
|
|
};
|
|
|
|
settle(resolve, reject, response);
|
|
|
|
// Clean up request
|
|
request = null;
|
|
};
|
|
|
|
// Handle browser request cancellation (as opposed to a manual cancellation)
|
|
request.onabort = function handleAbort() {
|
|
if (!request) {
|
|
return;
|
|
}
|
|
|
|
reject(createError('Request aborted', config, 'ECONNABORTED', request));
|
|
|
|
// Clean up request
|
|
request = null;
|
|
};
|
|
|
|
// Handle low level network errors
|
|
request.onerror = function handleError() {
|
|
// Real errors are hidden from us by the browser
|
|
// onerror should only fire if it's a network error
|
|
reject(createError('Network Error', config, null, request));
|
|
|
|
// Clean up request
|
|
request = null;
|
|
};
|
|
|
|
// Handle timeout
|
|
request.ontimeout = function handleTimeout() {
|
|
var timeoutErrorMessage = 'timeout of ' + config.timeout + 'ms exceeded';
|
|
if (config.timeoutErrorMessage) {
|
|
timeoutErrorMessage = config.timeoutErrorMessage;
|
|
}
|
|
reject(createError(timeoutErrorMessage, config, 'ECONNABORTED',
|
|
request));
|
|
|
|
// Clean up request
|
|
request = null;
|
|
};
|
|
|
|
// Add xsrf header
|
|
// This is only done if running in a standard browser environment.
|
|
// Specifically not if we're in a web worker, or react-native.
|
|
if (utils.isStandardBrowserEnv()) {
|
|
// Add xsrf header
|
|
var xsrfValue = (config.withCredentials || isURLSameOrigin(fullPath)) && config.xsrfCookieName ?
|
|
cookies.read(config.xsrfCookieName) :
|
|
undefined;
|
|
|
|
if (xsrfValue) {
|
|
requestHeaders[config.xsrfHeaderName] = xsrfValue;
|
|
}
|
|
}
|
|
|
|
// Add headers to the request
|
|
if ('setRequestHeader' in request) {
|
|
utils.forEach(requestHeaders, function setRequestHeader(val, key) {
|
|
if (typeof requestData === 'undefined' && key.toLowerCase() === 'content-type') {
|
|
// Remove Content-Type if data is undefined
|
|
delete requestHeaders[key];
|
|
} else {
|
|
// Otherwise add header to the request
|
|
request.setRequestHeader(key, val);
|
|
}
|
|
});
|
|
}
|
|
|
|
// Add withCredentials to request if needed
|
|
if (!utils.isUndefined(config.withCredentials)) {
|
|
request.withCredentials = !!config.withCredentials;
|
|
}
|
|
|
|
// Add responseType to request if needed
|
|
if (config.responseType) {
|
|
try {
|
|
request.responseType = config.responseType;
|
|
} catch (e) {
|
|
// Expected DOMException thrown by browsers not compatible XMLHttpRequest Level 2.
|
|
// But, this can be suppressed for 'json' type as it can be parsed by default 'transformResponse' function.
|
|
if (config.responseType !== 'json') {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Handle progress if needed
|
|
if (typeof config.onDownloadProgress === 'function') {
|
|
request.addEventListener('progress', config.onDownloadProgress);
|
|
}
|
|
|
|
// Not all browsers support upload events
|
|
if (typeof config.onUploadProgress === 'function' && request.upload) {
|
|
request.upload.addEventListener('progress', config.onUploadProgress);
|
|
}
|
|
|
|
if (config.cancelToken) {
|
|
// Handle cancellation
|
|
config.cancelToken.promise.then(function onCanceled(cancel) {
|
|
if (!request) {
|
|
return;
|
|
}
|
|
|
|
request.abort();
|
|
reject(cancel);
|
|
// Clean up request
|
|
request = null;
|
|
});
|
|
}
|
|
|
|
if (!requestData) {
|
|
requestData = null;
|
|
}
|
|
|
|
// Send the request
|
|
request.send(requestData);
|
|
});
|
|
};
|
|
|
|
var DEFAULT_CONTENT_TYPE = {
|
|
'Content-Type': 'application/x-www-form-urlencoded'
|
|
};
|
|
|
|
function setContentTypeIfUnset(headers, value) {
|
|
if (!utils.isUndefined(headers) && utils.isUndefined(headers['Content-Type'])) {
|
|
headers['Content-Type'] = value;
|
|
}
|
|
}
|
|
|
|
function getDefaultAdapter() {
|
|
var adapter;
|
|
if (typeof XMLHttpRequest !== 'undefined') {
|
|
// For browsers use XHR adapter
|
|
adapter = xhr;
|
|
} else if (typeof process !== 'undefined' && Object.prototype.toString.call(process) === '[object process]') {
|
|
// For node use HTTP adapter
|
|
adapter = xhr;
|
|
}
|
|
return adapter;
|
|
}
|
|
|
|
var defaults = {
|
|
adapter: getDefaultAdapter(),
|
|
|
|
transformRequest: [function transformRequest(data, headers) {
|
|
normalizeHeaderName(headers, 'Accept');
|
|
normalizeHeaderName(headers, 'Content-Type');
|
|
if (utils.isFormData(data) ||
|
|
utils.isArrayBuffer(data) ||
|
|
utils.isBuffer(data) ||
|
|
utils.isStream(data) ||
|
|
utils.isFile(data) ||
|
|
utils.isBlob(data)
|
|
) {
|
|
return data;
|
|
}
|
|
if (utils.isArrayBufferView(data)) {
|
|
return data.buffer;
|
|
}
|
|
if (utils.isURLSearchParams(data)) {
|
|
setContentTypeIfUnset(headers, 'application/x-www-form-urlencoded;charset=utf-8');
|
|
return data.toString();
|
|
}
|
|
if (utils.isObject(data)) {
|
|
setContentTypeIfUnset(headers, 'application/json;charset=utf-8');
|
|
return JSON.stringify(data);
|
|
}
|
|
return data;
|
|
}],
|
|
|
|
transformResponse: [function transformResponse(data) {
|
|
/*eslint no-param-reassign:0*/
|
|
if (typeof data === 'string') {
|
|
try {
|
|
data = JSON.parse(data);
|
|
} catch (e) { /* Ignore */ }
|
|
}
|
|
return data;
|
|
}],
|
|
|
|
/**
|
|
* A timeout in milliseconds to abort a request. If set to 0 (default) a
|
|
* timeout is not created.
|
|
*/
|
|
timeout: 0,
|
|
|
|
xsrfCookieName: 'XSRF-TOKEN',
|
|
xsrfHeaderName: 'X-XSRF-TOKEN',
|
|
|
|
maxContentLength: -1,
|
|
maxBodyLength: -1,
|
|
|
|
validateStatus: function validateStatus(status) {
|
|
return status >= 200 && status < 300;
|
|
}
|
|
};
|
|
|
|
defaults.headers = {
|
|
common: {
|
|
'Accept': 'application/json, text/plain, */*'
|
|
}
|
|
};
|
|
|
|
utils.forEach(['delete', 'get', 'head'], function forEachMethodNoData(method) {
|
|
defaults.headers[method] = {};
|
|
});
|
|
|
|
utils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {
|
|
defaults.headers[method] = utils.merge(DEFAULT_CONTENT_TYPE);
|
|
});
|
|
|
|
var defaults_1 = defaults;
|
|
|
|
/**
|
|
* Throws a `Cancel` if cancellation has been requested.
|
|
*/
|
|
function throwIfCancellationRequested(config) {
|
|
if (config.cancelToken) {
|
|
config.cancelToken.throwIfRequested();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Dispatch a request to the server using the configured adapter.
|
|
*
|
|
* @param {object} config The config that is to be used for the request
|
|
* @returns {Promise} The Promise to be fulfilled
|
|
*/
|
|
var dispatchRequest = function dispatchRequest(config) {
|
|
throwIfCancellationRequested(config);
|
|
|
|
// Ensure headers exist
|
|
config.headers = config.headers || {};
|
|
|
|
// Transform request data
|
|
config.data = transformData(
|
|
config.data,
|
|
config.headers,
|
|
config.transformRequest
|
|
);
|
|
|
|
// Flatten headers
|
|
config.headers = utils.merge(
|
|
config.headers.common || {},
|
|
config.headers[config.method] || {},
|
|
config.headers
|
|
);
|
|
|
|
utils.forEach(
|
|
['delete', 'get', 'head', 'post', 'put', 'patch', 'common'],
|
|
function cleanHeaderConfig(method) {
|
|
delete config.headers[method];
|
|
}
|
|
);
|
|
|
|
var adapter = config.adapter || defaults_1.adapter;
|
|
|
|
return adapter(config).then(function onAdapterResolution(response) {
|
|
throwIfCancellationRequested(config);
|
|
|
|
// Transform response data
|
|
response.data = transformData(
|
|
response.data,
|
|
response.headers,
|
|
config.transformResponse
|
|
);
|
|
|
|
return response;
|
|
}, function onAdapterRejection(reason) {
|
|
if (!isCancel(reason)) {
|
|
throwIfCancellationRequested(config);
|
|
|
|
// Transform response data
|
|
if (reason && reason.response) {
|
|
reason.response.data = transformData(
|
|
reason.response.data,
|
|
reason.response.headers,
|
|
config.transformResponse
|
|
);
|
|
}
|
|
}
|
|
|
|
return Promise.reject(reason);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Config-specific merge-function which creates a new config-object
|
|
* by merging two configuration objects together.
|
|
*
|
|
* @param {Object} config1
|
|
* @param {Object} config2
|
|
* @returns {Object} New object resulting from merging config2 to config1
|
|
*/
|
|
var mergeConfig = function mergeConfig(config1, config2) {
|
|
// eslint-disable-next-line no-param-reassign
|
|
config2 = config2 || {};
|
|
var config = {};
|
|
|
|
var valueFromConfig2Keys = ['url', 'method', 'data'];
|
|
var mergeDeepPropertiesKeys = ['headers', 'auth', 'proxy', 'params'];
|
|
var defaultToConfig2Keys = [
|
|
'baseURL', 'transformRequest', 'transformResponse', 'paramsSerializer',
|
|
'timeout', 'timeoutMessage', 'withCredentials', 'adapter', 'responseType', 'xsrfCookieName',
|
|
'xsrfHeaderName', 'onUploadProgress', 'onDownloadProgress', 'decompress',
|
|
'maxContentLength', 'maxBodyLength', 'maxRedirects', 'transport', 'httpAgent',
|
|
'httpsAgent', 'cancelToken', 'socketPath', 'responseEncoding'
|
|
];
|
|
var directMergeKeys = ['validateStatus'];
|
|
|
|
function getMergedValue(target, source) {
|
|
if (utils.isPlainObject(target) && utils.isPlainObject(source)) {
|
|
return utils.merge(target, source);
|
|
} else if (utils.isPlainObject(source)) {
|
|
return utils.merge({}, source);
|
|
} else if (utils.isArray(source)) {
|
|
return source.slice();
|
|
}
|
|
return source;
|
|
}
|
|
|
|
function mergeDeepProperties(prop) {
|
|
if (!utils.isUndefined(config2[prop])) {
|
|
config[prop] = getMergedValue(config1[prop], config2[prop]);
|
|
} else if (!utils.isUndefined(config1[prop])) {
|
|
config[prop] = getMergedValue(undefined, config1[prop]);
|
|
}
|
|
}
|
|
|
|
utils.forEach(valueFromConfig2Keys, function valueFromConfig2(prop) {
|
|
if (!utils.isUndefined(config2[prop])) {
|
|
config[prop] = getMergedValue(undefined, config2[prop]);
|
|
}
|
|
});
|
|
|
|
utils.forEach(mergeDeepPropertiesKeys, mergeDeepProperties);
|
|
|
|
utils.forEach(defaultToConfig2Keys, function defaultToConfig2(prop) {
|
|
if (!utils.isUndefined(config2[prop])) {
|
|
config[prop] = getMergedValue(undefined, config2[prop]);
|
|
} else if (!utils.isUndefined(config1[prop])) {
|
|
config[prop] = getMergedValue(undefined, config1[prop]);
|
|
}
|
|
});
|
|
|
|
utils.forEach(directMergeKeys, function merge(prop) {
|
|
if (prop in config2) {
|
|
config[prop] = getMergedValue(config1[prop], config2[prop]);
|
|
} else if (prop in config1) {
|
|
config[prop] = getMergedValue(undefined, config1[prop]);
|
|
}
|
|
});
|
|
|
|
var axiosKeys = valueFromConfig2Keys
|
|
.concat(mergeDeepPropertiesKeys)
|
|
.concat(defaultToConfig2Keys)
|
|
.concat(directMergeKeys);
|
|
|
|
var otherKeys = Object
|
|
.keys(config1)
|
|
.concat(Object.keys(config2))
|
|
.filter(function filterAxiosKeys(key) {
|
|
return axiosKeys.indexOf(key) === -1;
|
|
});
|
|
|
|
utils.forEach(otherKeys, mergeDeepProperties);
|
|
|
|
return config;
|
|
};
|
|
|
|
/**
|
|
* Create a new instance of Axios
|
|
*
|
|
* @param {Object} instanceConfig The default config for the instance
|
|
*/
|
|
function Axios(instanceConfig) {
|
|
this.defaults = instanceConfig;
|
|
this.interceptors = {
|
|
request: new InterceptorManager_1(),
|
|
response: new InterceptorManager_1()
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Dispatch a request
|
|
*
|
|
* @param {Object} config The config specific for this request (merged with this.defaults)
|
|
*/
|
|
Axios.prototype.request = function request(config) {
|
|
/*eslint no-param-reassign:0*/
|
|
// Allow for axios('example/url'[, config]) a la fetch API
|
|
if (typeof config === 'string') {
|
|
config = arguments[1] || {};
|
|
config.url = arguments[0];
|
|
} else {
|
|
config = config || {};
|
|
}
|
|
|
|
config = mergeConfig(this.defaults, config);
|
|
|
|
// Set config.method
|
|
if (config.method) {
|
|
config.method = config.method.toLowerCase();
|
|
} else if (this.defaults.method) {
|
|
config.method = this.defaults.method.toLowerCase();
|
|
} else {
|
|
config.method = 'get';
|
|
}
|
|
|
|
// Hook up interceptors middleware
|
|
var chain = [dispatchRequest, undefined];
|
|
var promise = Promise.resolve(config);
|
|
|
|
this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
|
|
chain.unshift(interceptor.fulfilled, interceptor.rejected);
|
|
});
|
|
|
|
this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
|
|
chain.push(interceptor.fulfilled, interceptor.rejected);
|
|
});
|
|
|
|
while (chain.length) {
|
|
promise = promise.then(chain.shift(), chain.shift());
|
|
}
|
|
|
|
return promise;
|
|
};
|
|
|
|
Axios.prototype.getUri = function getUri(config) {
|
|
config = mergeConfig(this.defaults, config);
|
|
return buildURL(config.url, config.params, config.paramsSerializer).replace(/^\?/, '');
|
|
};
|
|
|
|
// Provide aliases for supported request methods
|
|
utils.forEach(['delete', 'get', 'head', 'options'], function forEachMethodNoData(method) {
|
|
/*eslint func-names:0*/
|
|
Axios.prototype[method] = function(url, config) {
|
|
return this.request(mergeConfig(config || {}, {
|
|
method: method,
|
|
url: url,
|
|
data: (config || {}).data
|
|
}));
|
|
};
|
|
});
|
|
|
|
utils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {
|
|
/*eslint func-names:0*/
|
|
Axios.prototype[method] = function(url, data, config) {
|
|
return this.request(mergeConfig(config || {}, {
|
|
method: method,
|
|
url: url,
|
|
data: data
|
|
}));
|
|
};
|
|
});
|
|
|
|
var Axios_1 = Axios;
|
|
|
|
/**
|
|
* A `Cancel` is an object that is thrown when an operation is canceled.
|
|
*
|
|
* @class
|
|
* @param {string=} message The message.
|
|
*/
|
|
function Cancel(message) {
|
|
this.message = message;
|
|
}
|
|
|
|
Cancel.prototype.toString = function toString() {
|
|
return 'Cancel' + (this.message ? ': ' + this.message : '');
|
|
};
|
|
|
|
Cancel.prototype.__CANCEL__ = true;
|
|
|
|
var Cancel_1 = Cancel;
|
|
|
|
/**
|
|
* A `CancelToken` is an object that can be used to request cancellation of an operation.
|
|
*
|
|
* @class
|
|
* @param {Function} executor The executor function.
|
|
*/
|
|
function CancelToken(executor) {
|
|
if (typeof executor !== 'function') {
|
|
throw new TypeError('executor must be a function.');
|
|
}
|
|
|
|
var resolvePromise;
|
|
this.promise = new Promise(function promiseExecutor(resolve) {
|
|
resolvePromise = resolve;
|
|
});
|
|
|
|
var token = this;
|
|
executor(function cancel(message) {
|
|
if (token.reason) {
|
|
// Cancellation has already been requested
|
|
return;
|
|
}
|
|
|
|
token.reason = new Cancel_1(message);
|
|
resolvePromise(token.reason);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Throws a `Cancel` if cancellation has been requested.
|
|
*/
|
|
CancelToken.prototype.throwIfRequested = function throwIfRequested() {
|
|
if (this.reason) {
|
|
throw this.reason;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Returns an object that contains a new `CancelToken` and a function that, when called,
|
|
* cancels the `CancelToken`.
|
|
*/
|
|
CancelToken.source = function source() {
|
|
var cancel;
|
|
var token = new CancelToken(function executor(c) {
|
|
cancel = c;
|
|
});
|
|
return {
|
|
token: token,
|
|
cancel: cancel
|
|
};
|
|
};
|
|
|
|
var CancelToken_1 = CancelToken;
|
|
|
|
/**
|
|
* Syntactic sugar for invoking a function and expanding an array for arguments.
|
|
*
|
|
* Common use case would be to use `Function.prototype.apply`.
|
|
*
|
|
* ```js
|
|
* function f(x, y, z) {}
|
|
* var args = [1, 2, 3];
|
|
* f.apply(null, args);
|
|
* ```
|
|
*
|
|
* With `spread` this example can be re-written.
|
|
*
|
|
* ```js
|
|
* spread(function(x, y, z) {})([1, 2, 3]);
|
|
* ```
|
|
*
|
|
* @param {Function} callback
|
|
* @returns {Function}
|
|
*/
|
|
var spread = function spread(callback) {
|
|
return function wrap(arr) {
|
|
return callback.apply(null, arr);
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Determines whether the payload is an error thrown by Axios
|
|
*
|
|
* @param {*} payload The value to test
|
|
* @returns {boolean} True if the payload is an error thrown by Axios, otherwise false
|
|
*/
|
|
var isAxiosError = function isAxiosError(payload) {
|
|
return (typeof payload === 'object') && (payload.isAxiosError === true);
|
|
};
|
|
|
|
/**
|
|
* Create an instance of Axios
|
|
*
|
|
* @param {Object} defaultConfig The default config for the instance
|
|
* @return {Axios} A new instance of Axios
|
|
*/
|
|
function createInstance(defaultConfig) {
|
|
var context = new Axios_1(defaultConfig);
|
|
var instance = bind(Axios_1.prototype.request, context);
|
|
|
|
// Copy axios.prototype to instance
|
|
utils.extend(instance, Axios_1.prototype, context);
|
|
|
|
// Copy context to instance
|
|
utils.extend(instance, context);
|
|
|
|
return instance;
|
|
}
|
|
|
|
// Create the default instance to be exported
|
|
var axios = createInstance(defaults_1);
|
|
|
|
// Expose Axios class to allow class inheritance
|
|
axios.Axios = Axios_1;
|
|
|
|
// Factory for creating new instances
|
|
axios.create = function create(instanceConfig) {
|
|
return createInstance(mergeConfig(axios.defaults, instanceConfig));
|
|
};
|
|
|
|
// Expose Cancel & CancelToken
|
|
axios.Cancel = Cancel_1;
|
|
axios.CancelToken = CancelToken_1;
|
|
axios.isCancel = isCancel;
|
|
|
|
// Expose all/spread
|
|
axios.all = function all(promises) {
|
|
return Promise.all(promises);
|
|
};
|
|
axios.spread = spread;
|
|
|
|
// Expose isAxiosError
|
|
axios.isAxiosError = isAxiosError;
|
|
|
|
var axios_1 = axios;
|
|
|
|
// Allow use of default import syntax in TypeScript
|
|
var _default = axios;
|
|
axios_1.default = _default;
|
|
|
|
var axios$1 = axios_1;
|
|
|
|
var script$5 = {
|
|
props: ['model_id', 'schema', 'objects'],
|
|
name: "NewObjectForm",
|
|
data() {
|
|
const model = this.schema.obj_models.find((m) => m.id === this.model_id);
|
|
let properties = this.schema.prop_models.filter((m) => m.object === model.id);
|
|
let relations = this.schema.rel_models.filter((m) => m.object === model.id);
|
|
|
|
properties.sort((a, b) => a.name.localeCompare(b.name));
|
|
relations.sort((a, b) => a.name.localeCompare(b.name));
|
|
|
|
let values = {};
|
|
properties.forEach((p) => {
|
|
if (p.optional) {
|
|
values[p.id] = [];
|
|
} else {
|
|
values[p.id] = [objCopy(p.default)];
|
|
}
|
|
});
|
|
|
|
properties = keyBy(properties, 'id');
|
|
relations = keyBy(relations, 'id');
|
|
|
|
let model_names = {};
|
|
this.schema.obj_models.forEach((m) => {
|
|
model_names[m.id] = m.name;
|
|
});
|
|
|
|
return {
|
|
model,
|
|
properties,
|
|
relations,
|
|
haveRelations: !isEmpty$1(relations),
|
|
model_names,
|
|
values,
|
|
name: '',
|
|
relationRefs: [],
|
|
}
|
|
},
|
|
methods: {
|
|
/** Get values in the raw format without grouping */
|
|
collectData() {
|
|
if (isEmpty$1(this.name)) {
|
|
throw new Error("Name is required");
|
|
}
|
|
|
|
let values = [];
|
|
forEach(objCopy(this.values), (vv, k) => {
|
|
for (let v of vv) {
|
|
values.push({
|
|
model_id: castId(k),
|
|
value: v
|
|
});
|
|
}
|
|
});
|
|
|
|
let relations = [];
|
|
for (let rref of this.relationRefs) {
|
|
for (let r of rref.collectData()) {
|
|
relations.push(r);
|
|
}
|
|
}
|
|
|
|
return {
|
|
model_id: this.model_id, // string is fine
|
|
name: this.name,
|
|
values,
|
|
relations,
|
|
};
|
|
},
|
|
|
|
trySave() {
|
|
let data;
|
|
try {
|
|
data = this.collectData();
|
|
} catch (e) {
|
|
alert(e.message);
|
|
return;
|
|
}
|
|
console.log('Try save', data);
|
|
|
|
axios$1({
|
|
method: 'post',
|
|
url: '/object/create',
|
|
data: data
|
|
})
|
|
.then(function (response) {
|
|
location.href = '/objects';
|
|
})
|
|
.catch(function (error) {
|
|
// TODO show error toast instead
|
|
alert(error.response ?
|
|
error.response.data :
|
|
error);
|
|
});
|
|
},
|
|
|
|
setRelationRef(el) {
|
|
if (el) {
|
|
this.relationRefs.push(el);
|
|
}
|
|
},
|
|
},
|
|
beforeUpdate() {
|
|
this.relationRefs = [];
|
|
},
|
|
mounted() {
|
|
this.$el.parentNode
|
|
.classList.add('EditForm');
|
|
}
|
|
};
|
|
|
|
const _hoisted_1$1 = /*#__PURE__*/createVNode("th", null, [
|
|
/*#__PURE__*/createVNode("label", { for: "field-name" }, "Name")
|
|
], -1 /* HOISTED */);
|
|
const _hoisted_2$1 = { key: 0 };
|
|
const _hoisted_3 = /*#__PURE__*/createVNode("h3", null, "Relations", -1 /* HOISTED */);
|
|
|
|
function render$5(_ctx, _cache, $props, $setup, $data, $options) {
|
|
const _component_property = resolveComponent("property");
|
|
const _component_new_relation = resolveComponent("new-relation");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
createVNode("h2", null, "New " + toDisplayString($data.model.name), 1 /* TEXT */),
|
|
createVNode("p", null, [
|
|
createVNode("input", {
|
|
type: "button",
|
|
value: "Save",
|
|
onClick: _cache[1] || (_cache[1] = (...args) => ($options.trySave && $options.trySave(...args)))
|
|
})
|
|
]),
|
|
createVNode("table", null, [
|
|
createVNode("tr", null, [
|
|
_hoisted_1$1,
|
|
createVNode("td", null, [
|
|
withDirectives(createVNode("input", {
|
|
type: "text",
|
|
id: "field-name",
|
|
"onUpdate:modelValue": _cache[2] || (_cache[2] = $event => ($data.name = $event))
|
|
}, null, 512 /* NEED_PATCH */), [
|
|
[vModelText, $data.name]
|
|
])
|
|
])
|
|
]),
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.properties, (property, pi) => {
|
|
return (openBlock(), createBlock(_component_property, {
|
|
model: property,
|
|
values: $data.values[property.id],
|
|
key: pi
|
|
}, null, 8 /* PROPS */, ["model", "values"]))
|
|
}), 128 /* KEYED_FRAGMENT */))
|
|
]),
|
|
($data.haveRelations)
|
|
? (openBlock(), createBlock("div", _hoisted_2$1, [
|
|
_hoisted_3,
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.relations, (relation) => {
|
|
return (openBlock(), createBlock(_component_new_relation, {
|
|
ref: $options.setRelationRef,
|
|
model_id: relation.id,
|
|
objects: $props.objects,
|
|
schema: $props.schema
|
|
}, null, 8 /* PROPS */, ["model_id", "objects", "schema"]))
|
|
}), 256 /* UNKEYED_FRAGMENT */))
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
}
|
|
|
|
script$5.render = render$5;
|
|
script$5.__file = "src/components/NewObjectForm.vue";
|
|
|
|
var script$6 = {
|
|
props: ['model_id', 'schema', 'objects'],
|
|
name: "NewRelationForm",
|
|
data() {
|
|
const model = this.schema.rel_models.find((m) => m.id === this.model_id);
|
|
if(!model) throw Error("Relation model not exist");
|
|
|
|
let properties = this.schema.prop_models.filter((m) => m.object === model.id);
|
|
|
|
properties.sort((a, b) => a.name.localeCompare(b.name));
|
|
|
|
if (isEmpty$1(properties)) {
|
|
properties = null;
|
|
} else {
|
|
properties = keyBy(properties, 'id');
|
|
}
|
|
|
|
let related_model = this.schema.obj_models.find((m) => m.id === model.related);
|
|
|
|
if(!related_model) throw Error("Related model not exist");
|
|
|
|
let choices = {};
|
|
this.objects.forEach((obj) => {
|
|
if (obj.model === model.related) {
|
|
choices[obj.id] = obj.name;
|
|
}
|
|
});
|
|
|
|
let instances = [];
|
|
if (!model.optional) {
|
|
// TODO avoid duplicated code
|
|
let values = {};
|
|
forEach(this.properties, (p) => {
|
|
if (p.optional) {
|
|
values[p.id] = [];
|
|
} else {
|
|
values[p.id] = [objCopy(p.default)];
|
|
}
|
|
});
|
|
instances.push({
|
|
related: '',
|
|
values
|
|
});
|
|
}
|
|
|
|
return {
|
|
model,
|
|
related_model,
|
|
properties,
|
|
object_names: choices,
|
|
instances,
|
|
}
|
|
},
|
|
methods: {
|
|
collectData() {
|
|
let relations = [];
|
|
forEach(objCopy(this.instances), (instance) => {
|
|
if (isEmpty$1(instance.related)) {
|
|
if (!this.model.optional) {
|
|
throw new Error(`Relation "${this.model.name}" is required`)
|
|
}
|
|
return; // continue
|
|
}
|
|
|
|
let values = [];
|
|
forEach(instance.values, (vv, prop_model_id) => {
|
|
|
|
for (let v of vv) {
|
|
values.push({
|
|
model_id: castId(prop_model_id),
|
|
value: v
|
|
});
|
|
}
|
|
});
|
|
|
|
relations.push({
|
|
model_id: this.model.id,
|
|
related_id: castId(instance.related),
|
|
values
|
|
});
|
|
});
|
|
return relations;
|
|
},
|
|
|
|
addInstance() {
|
|
let values = {};
|
|
forEach(this.properties, (p) => {
|
|
if (p.optional) {
|
|
values[p.id] = [];
|
|
} else {
|
|
values[p.id] = [objCopy(p.default)];
|
|
}
|
|
});
|
|
this.instances.push({
|
|
related: '',
|
|
values
|
|
});
|
|
},
|
|
|
|
removeInstance(ri) {
|
|
this.instances.splice(ri, 1);
|
|
}
|
|
}
|
|
};
|
|
|
|
const _withId = /*#__PURE__*/withScopeId("data-v-0f98a4a0");
|
|
|
|
pushScopeId("data-v-0f98a4a0");
|
|
const _hoisted_1$2 = { key: 1 };
|
|
const _hoisted_2$2 = /*#__PURE__*/createVNode("br", null, null, -1 /* HOISTED */);
|
|
popScopeId();
|
|
|
|
const render$6 = /*#__PURE__*/_withId((_ctx, _cache, $props, $setup, $data, $options) => {
|
|
const _component_property = resolveComponent("property");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.instances, (instance, ri) => {
|
|
return (openBlock(), createBlock("div", {
|
|
class: "new-relation",
|
|
key: ri
|
|
}, [
|
|
createVNode("b", null, [
|
|
createTextVNode(toDisplayString($data.model.name) + " -> " + toDisplayString($data.related_model.name) + " ", 1 /* TEXT */),
|
|
withDirectives(createVNode("select", {
|
|
"onUpdate:modelValue": $event => (instance.related = $event)
|
|
}, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.object_names, (name, id) => {
|
|
return (openBlock(), createBlock("option", { value: id }, toDisplayString(name), 9 /* TEXT, PROPS */, ["value"]))
|
|
}), 256 /* UNKEYED_FRAGMENT */))
|
|
], 8 /* PROPS */, ["onUpdate:modelValue"]), [
|
|
[vModelSelect, instance.related]
|
|
])
|
|
]),
|
|
($data.model.multiple || $data.model.optional && $data.instances.length > 0)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 0,
|
|
href: "#",
|
|
style: {"margin-left":"5px"},
|
|
onClick: $event => ($options.removeInstance(ri))
|
|
}, "X", 8 /* PROPS */, ["onClick"]))
|
|
: createCommentVNode("v-if", true),
|
|
($data.properties)
|
|
? (openBlock(), createBlock("table", _hoisted_1$2, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.properties, (property, id) => {
|
|
return (openBlock(), createBlock(_component_property, {
|
|
model: property,
|
|
values: instance.values[id],
|
|
key: id
|
|
}, null, 8 /* PROPS */, ["model", "values"]))
|
|
}), 128 /* KEYED_FRAGMENT */))
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
]))
|
|
}), 128 /* KEYED_FRAGMENT */)),
|
|
($data.model.multiple || $data.model.optional && $data.instances.length==0)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 0,
|
|
href: "#",
|
|
onClick: _cache[1] || (_cache[1] = (...args) => ($options.addInstance && $options.addInstance(...args)))
|
|
}, "Add " + toDisplayString($data.model.name) + " -> " + toDisplayString($data.related_model.name), 1 /* TEXT */))
|
|
: createCommentVNode("v-if", true),
|
|
_hoisted_2$2
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
});
|
|
|
|
script$6.render = render$6;
|
|
script$6.__scopeId = "data-v-0f98a4a0";
|
|
script$6.__file = "src/components/NewRelationForm.vue";
|
|
|
|
var script$7 = {
|
|
props: ['object', 'schema', 'objects'],
|
|
name: "EditObjectForm",
|
|
data() {
|
|
let object = this.object;
|
|
|
|
const model = this.schema.obj_models.find((m) => m.id === object.model);
|
|
let properties = this.schema.prop_models.filter((m) => m.object === model.id);
|
|
let relations = this.schema.rel_models.filter((m) => m.object === model.id);
|
|
|
|
properties.sort((a, b) => a.name.localeCompare(b.name));
|
|
relations.sort((a, b) => a.name.localeCompare(b.name));
|
|
|
|
let values = {};
|
|
properties.forEach((p) => {
|
|
let existing = object.values[p.id] || [];
|
|
|
|
if (existing.length) {
|
|
values[p.id] = existing;
|
|
} else {
|
|
if (p.optional) {
|
|
values[p.id] = [];
|
|
} else {
|
|
values[p.id] = [
|
|
// this is the format used for values
|
|
{
|
|
id: null,
|
|
// it can also have model: ... here
|
|
value: objCopy(p.default)
|
|
}
|
|
];
|
|
}
|
|
}
|
|
});
|
|
|
|
properties = keyBy(properties, 'id');
|
|
relations = keyBy(relations, 'id');
|
|
|
|
let model_names = {};
|
|
this.schema.obj_models.forEach((m) => {
|
|
model_names[m.id] = m.name;
|
|
});
|
|
|
|
return {
|
|
model,
|
|
properties,
|
|
relations,
|
|
haveRelations: !isEmpty$1(relations),
|
|
model_names,
|
|
values,
|
|
name: object.name,
|
|
relationRefs: [],
|
|
}
|
|
},
|
|
methods: {
|
|
/** Get values in the raw format without grouping */
|
|
collectData() {
|
|
if (isEmpty$1(this.name)) {
|
|
throw new Error("Name is required");
|
|
}
|
|
|
|
let values = [];
|
|
forEach(objCopy(this.values), (vv, prop_model_id) => {
|
|
for (let v of vv) {
|
|
v.model_id = castId(prop_model_id);
|
|
values.push(v);
|
|
}
|
|
});
|
|
|
|
let relations = [];
|
|
for (let rref of this.relationRefs) {
|
|
for (let r of rref.collectData()) {
|
|
relations.push(r);
|
|
}
|
|
}
|
|
|
|
return {
|
|
model_id: this.object.model, // string is fine
|
|
id: this.object.id,
|
|
name: this.name,
|
|
values,
|
|
relations,
|
|
};
|
|
},
|
|
|
|
trySave() {
|
|
let data;
|
|
try {
|
|
data = this.collectData();
|
|
} catch (e) {
|
|
alert(e.message);
|
|
return;
|
|
}
|
|
console.log('Try save', data);
|
|
|
|
axios$1({
|
|
method: 'post',
|
|
url: '/object/update',
|
|
data: data
|
|
})
|
|
.then(function (response) {
|
|
location.href = '/object/detail/'+this.object.id;
|
|
})
|
|
.catch(function (error) {
|
|
// TODO show error toast instead
|
|
alert(error.response ?
|
|
error.response.data :
|
|
error);
|
|
});
|
|
},
|
|
|
|
setRelationRef(el) {
|
|
if (el) {
|
|
this.relationRefs.push(el);
|
|
}
|
|
},
|
|
},
|
|
beforeUpdate() {
|
|
this.relationRefs = [];
|
|
},
|
|
mounted() {
|
|
this.$el.parentNode
|
|
.classList.add('EditForm');
|
|
}
|
|
};
|
|
|
|
const _hoisted_1$3 = /*#__PURE__*/createVNode("th", null, [
|
|
/*#__PURE__*/createVNode("label", { for: "field-name" }, "Name")
|
|
], -1 /* HOISTED */);
|
|
const _hoisted_2$3 = { key: 0 };
|
|
const _hoisted_3$1 = /*#__PURE__*/createVNode("h3", null, "Relations", -1 /* HOISTED */);
|
|
|
|
function render$7(_ctx, _cache, $props, $setup, $data, $options) {
|
|
const _component_edit_property = resolveComponent("edit-property");
|
|
const _component_edit_relation = resolveComponent("edit-relation");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
createVNode("h2", null, "Edit " + toDisplayString($data.model.name), 1 /* TEXT */),
|
|
createVNode("p", null, [
|
|
createVNode("input", {
|
|
type: "button",
|
|
value: "Save",
|
|
onClick: _cache[1] || (_cache[1] = (...args) => (_ctx.trySave && _ctx.trySave(...args)))
|
|
})
|
|
]),
|
|
createVNode("table", null, [
|
|
createVNode("tr", null, [
|
|
_hoisted_1$3,
|
|
createVNode("td", null, [
|
|
withDirectives(createVNode("input", {
|
|
type: "text",
|
|
id: "field-name",
|
|
"onUpdate:modelValue": _cache[2] || (_cache[2] = $event => ($data.name = $event))
|
|
}, null, 512 /* NEED_PATCH */), [
|
|
[vModelText, $data.name]
|
|
])
|
|
])
|
|
]),
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.properties, (property, pi) => {
|
|
return (openBlock(), createBlock(_component_edit_property, {
|
|
model: property,
|
|
values: $data.values[property.id],
|
|
key: pi
|
|
}, null, 8 /* PROPS */, ["model", "values"]))
|
|
}), 128 /* KEYED_FRAGMENT */))
|
|
]),
|
|
($data.haveRelations)
|
|
? (openBlock(), createBlock("div", _hoisted_2$3, [
|
|
_hoisted_3$1,
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.relations, (relation) => {
|
|
return (openBlock(), createBlock(_component_edit_relation, {
|
|
ref: $options.setRelationRef,
|
|
model_id: relation.id,
|
|
objects: $props.objects,
|
|
initialInstances: $props.object.relations[relation.id],
|
|
schema: $props.schema
|
|
}, null, 8 /* PROPS */, ["model_id", "objects", "initialInstances", "schema"]))
|
|
}), 256 /* UNKEYED_FRAGMENT */))
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
}
|
|
|
|
script$7.render = render$7;
|
|
script$7.__file = "src/components/EditObjectForm.vue";
|
|
|
|
var script$8 = {
|
|
props: ['model_id', 'schema', 'objects', 'initialInstances'],
|
|
name: "EditRelationForm",
|
|
data() {
|
|
const model = this.schema.rel_models.find((m) => m.id === this.model_id);
|
|
if(!model) throw Error("Relation model not exist");
|
|
|
|
let properties = this.schema.prop_models.filter((m) => m.object === model.id);
|
|
|
|
properties.sort((a, b) => a.name.localeCompare(b.name));
|
|
|
|
if (isEmpty$1(properties)) {
|
|
properties = null;
|
|
} else {
|
|
properties = keyBy(properties, 'id');
|
|
}
|
|
|
|
let related_model = this.schema.obj_models.find((m) => m.id === model.related);
|
|
|
|
if(!related_model) throw Error("Related model not exist");
|
|
|
|
let choices = {};
|
|
this.objects.forEach((obj) => {
|
|
if (obj.model === model.related) {
|
|
choices[obj.id] = obj.name;
|
|
}
|
|
});
|
|
|
|
return {
|
|
model,
|
|
related_model,
|
|
properties,
|
|
object_names: choices,
|
|
instances: objCopy(this.initialInstances),
|
|
}
|
|
},
|
|
methods: {
|
|
collectData() {
|
|
console.log('relation->collect', this.instances);
|
|
let relations = [];
|
|
forEach(objCopy(this.instances), (instance) => {
|
|
console.log('a instance', instance);
|
|
if (isEmpty$1(instance.related)) {
|
|
if (!this.model.optional) {
|
|
throw new Error(`Relation "${this.model.name}" is required`)
|
|
}
|
|
console.log("empty related", instance.related);
|
|
return; // continue
|
|
}
|
|
|
|
let values = [];
|
|
forEach(instance.values, (vv, prop_model_id) => {
|
|
for (let v of vv) {
|
|
v.model_id = castId(prop_model_id);
|
|
values.push(v);
|
|
}
|
|
});
|
|
instance.model_id = this.model.id;
|
|
instance.values = values;
|
|
relations.push(instance);
|
|
});
|
|
console.log('collected', relations);
|
|
return relations;
|
|
},
|
|
|
|
addInstance() {
|
|
console.log('Add instance');
|
|
let values = {};
|
|
forEach(this.properties, (p) => {
|
|
if (p.optional) {
|
|
values[p.id] = [];
|
|
} else {
|
|
values[p.id] = [{id: null, value: objCopy(p.default)}];
|
|
}
|
|
});
|
|
this.instances.push({
|
|
id: null,
|
|
related: '',
|
|
values
|
|
});
|
|
},
|
|
|
|
removeInstance(ri) {
|
|
this.instances.splice(ri, 1);
|
|
}
|
|
}
|
|
};
|
|
|
|
const _withId$1 = /*#__PURE__*/withScopeId("data-v-760e133a");
|
|
|
|
pushScopeId("data-v-760e133a");
|
|
const _hoisted_1$4 = { key: 1 };
|
|
const _hoisted_2$4 = /*#__PURE__*/createVNode("br", null, null, -1 /* HOISTED */);
|
|
popScopeId();
|
|
|
|
const render$8 = /*#__PURE__*/_withId$1((_ctx, _cache, $props, $setup, $data, $options) => {
|
|
const _component_edit_property = resolveComponent("edit-property");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.instances, (instance, ri) => {
|
|
return (openBlock(), createBlock("div", {
|
|
class: "new-relation",
|
|
key: ri
|
|
}, [
|
|
createVNode("b", null, [
|
|
createTextVNode(toDisplayString($data.model.name) + " -> " + toDisplayString($data.related_model.name) + " ", 1 /* TEXT */),
|
|
withDirectives(createVNode("select", {
|
|
"onUpdate:modelValue": $event => (instance.related = $event)
|
|
}, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.object_names, (name, id) => {
|
|
return (openBlock(), createBlock("option", { value: id }, toDisplayString(name), 9 /* TEXT, PROPS */, ["value"]))
|
|
}), 256 /* UNKEYED_FRAGMENT */))
|
|
], 8 /* PROPS */, ["onUpdate:modelValue"]), [
|
|
[vModelSelect, instance.related]
|
|
])
|
|
]),
|
|
($data.model.multiple || $data.model.optional && $data.instances.length > 0)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 0,
|
|
href: "#",
|
|
style: {"margin-left":"5px"},
|
|
onClick: $event => ($options.removeInstance(ri))
|
|
}, "X", 8 /* PROPS */, ["onClick"]))
|
|
: createCommentVNode("v-if", true),
|
|
($data.properties)
|
|
? (openBlock(), createBlock("table", _hoisted_1$4, [
|
|
(openBlock(true), createBlock(Fragment, null, renderList($data.properties, (property, id) => {
|
|
return (openBlock(), createBlock(_component_edit_property, {
|
|
model: property,
|
|
values: instance.values[id],
|
|
key: id
|
|
}, null, 8 /* PROPS */, ["model", "values"]))
|
|
}), 128 /* KEYED_FRAGMENT */))
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
]))
|
|
}), 128 /* KEYED_FRAGMENT */)),
|
|
($data.model.multiple || $data.model.optional && $data.instances.length==0)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 0,
|
|
href: "#",
|
|
onClick: _cache[1] || (_cache[1] = (...args) => ($options.addInstance && $options.addInstance(...args)))
|
|
}, "Add " + toDisplayString($data.model.name) + " -> " + toDisplayString($data.related_model.name), 1 /* TEXT */))
|
|
: createCommentVNode("v-if", true),
|
|
_hoisted_2$4
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
});
|
|
|
|
script$8.render = render$8;
|
|
script$8.__scopeId = "data-v-760e133a";
|
|
script$8.__file = "src/components/EditRelationForm.vue";
|
|
|
|
var script$9 = {
|
|
name: "EditPropertyField",
|
|
props: ['model', 'values'],
|
|
data() {
|
|
return {
|
|
id: uniqueId(),
|
|
fieldRefs: [],
|
|
}
|
|
},
|
|
methods: {
|
|
addValue(event) {
|
|
this.values.push({
|
|
id: null,
|
|
value: objCopy(this.model.default)
|
|
});
|
|
|
|
nextTick(() => {
|
|
this.fieldRefs[this.values.length-1].focus();
|
|
});
|
|
},
|
|
removeValue(vi) {
|
|
this.values.splice(vi, 1);
|
|
},
|
|
setFieldRef(el) {
|
|
if (el) {
|
|
this.fieldRefs.push(el);
|
|
}
|
|
},
|
|
},
|
|
beforeUpdate() {
|
|
this.fieldRefs = [];
|
|
}
|
|
};
|
|
|
|
const _hoisted_1$5 = { key: 0 };
|
|
const _hoisted_2$5 = { key: 2 };
|
|
|
|
function render$9(_ctx, _cache, $props, $setup, $data, $options) {
|
|
const _component_string_value = resolveComponent("string-value");
|
|
const _component_integer_value = resolveComponent("integer-value");
|
|
const _component_decimal_value = resolveComponent("decimal-value");
|
|
const _component_boolean_value = resolveComponent("boolean-value");
|
|
|
|
return (openBlock(), createBlock(Fragment, null, [
|
|
($props.values.length===0)
|
|
? (openBlock(), createBlock("tr", _hoisted_1$5, [
|
|
createVNode("th", {
|
|
onClick: _cache[1] || (_cache[1] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, toDisplayString($props.model.name), 1 /* TEXT */),
|
|
createVNode("td", null, [
|
|
createVNode("a", {
|
|
href: "#",
|
|
onClick: _cache[2] || (_cache[2] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, "Add")
|
|
])
|
|
]))
|
|
: (openBlock(true), createBlock(Fragment, { key: 1 }, renderList($props.values, (instance, vi) => {
|
|
return (openBlock(), createBlock("tr", { key: vi }, [
|
|
(vi == 0)
|
|
? (openBlock(), createBlock("th", {
|
|
key: 0,
|
|
rowspan: $props.values.length + $props.model.multiple
|
|
}, [
|
|
createVNode("label", { for: $data.id }, toDisplayString($props.model.name), 9 /* TEXT, PROPS */, ["for"])
|
|
], 8 /* PROPS */, ["rowspan"]))
|
|
: createCommentVNode("v-if", true),
|
|
createVNode("td", null, [
|
|
($props.model.data_type==='String')
|
|
? (openBlock(), createBlock(_component_string_value, {
|
|
key: 0,
|
|
ref: $options.setFieldRef,
|
|
value: instance.value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Integer')
|
|
? (openBlock(), createBlock(_component_integer_value, {
|
|
key: 1,
|
|
ref: $options.setFieldRef,
|
|
value: instance.value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Decimal')
|
|
? (openBlock(), createBlock(_component_decimal_value, {
|
|
key: 2,
|
|
ref: $options.setFieldRef,
|
|
value: instance.value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
($props.model.data_type==='Boolean')
|
|
? (openBlock(), createBlock(_component_boolean_value, {
|
|
key: 3,
|
|
ref: $options.setFieldRef,
|
|
value: instance.value,
|
|
id: vi===0?$data.id:null
|
|
}, null, 8 /* PROPS */, ["value", "id"]))
|
|
: createCommentVNode("v-if", true),
|
|
(vi > 0 || $props.model.optional)
|
|
? (openBlock(), createBlock("a", {
|
|
key: 4,
|
|
href: "#",
|
|
onClick: $event => ($options.removeValue(vi)),
|
|
style: {"margin-left":"5px"}
|
|
}, "X", 8 /* PROPS */, ["onClick"]))
|
|
: createCommentVNode("v-if", true)
|
|
])
|
|
]))
|
|
}), 128 /* KEYED_FRAGMENT */)),
|
|
($props.values.length > 0 && $props.model.multiple)
|
|
? (openBlock(), createBlock("tr", _hoisted_2$5, [
|
|
createVNode("td", null, [
|
|
createVNode("a", {
|
|
href: "#",
|
|
onClick: _cache[3] || (_cache[3] = (...args) => ($options.addValue && $options.addValue(...args)))
|
|
}, "Add")
|
|
])
|
|
]))
|
|
: createCommentVNode("v-if", true)
|
|
], 64 /* STABLE_FRAGMENT */))
|
|
}
|
|
|
|
script$9.render = render$9;
|
|
script$9.__file = "src/components/EditPropertyField.vue";
|
|
|
|
function registerComponents(app) {
|
|
app.component('string-value', script);
|
|
app.component('integer-value', script$3);
|
|
app.component('decimal-value', script$1);
|
|
app.component('boolean-value', script$2);
|
|
}
|
|
|
|
window.onLoad = function (callback) {
|
|
document.addEventListener('DOMContentLoaded', callback);
|
|
};
|
|
|
|
window.Yopa = {
|
|
newObjectForm(opts) {
|
|
// Opts: model_id, schema, objects (named objects for relations)
|
|
let app = window.app = createApp(script$5, opts);
|
|
registerComponents(app);
|
|
app.component('new-relation', script$6);
|
|
app.component('property', script$4);
|
|
let instance = app.mount('#new-object-form');
|
|
|
|
// ...
|
|
return instance;
|
|
},
|
|
editObjectForm(opts) {
|
|
// Opts: model_id, schema, objects (named objects for relations)
|
|
let app = window.app = createApp(script$7, opts);
|
|
registerComponents(app);
|
|
app.component('edit-relation', script$8);
|
|
app.component('edit-property', script$9);
|
|
let instance = app.mount('#edit-object-form');
|
|
|
|
// ...
|
|
return instance;
|
|
}
|
|
};
|
|
|
|
onLoad(() => {
|
|
setTimeout(() => {
|
|
let toasts = document.getElementsByClassName('toast');
|
|
if (toasts.length) {
|
|
toasts[0].style.display = 'none';
|
|
}
|
|
}, 3000);
|
|
});
|
|
|
|
}());
|
|
//# sourceMappingURL=bundle.js.map
|
|
|