for (const key in propsOptions) { // 循环 props 选项 keys.push(key) // 校验 props 并获取 props 值 const value = validateProp(key, propsOptions, propsData, vm) /* istanbul ignore else */ // 开发环境下的校验告警 if (__DEV__) { const hyphenatedKey = hyphenate(key) if ( isReservedAttribute(hyphenatedKey) || config.isReservedAttr(hyphenatedKey) ) { warn( `"${hyphenatedKey}" is a reserved attribute and cannot be used as component prop.`, vm ) } defineReactive(props, key, value, () => { if (!isRoot && !isUpdatingChildComponent) { warn( `Avoid mutating a prop directly since the value will be ` + `overwritten whenever the parent component re-renders. ` + `Instead, use a data or computed property based on the prop's ` + `value. Prop being mutated: "${key}"`, vm ) } }) } else { // 定义一个响应式的 props defineReactive(props, key, value) } // 当前key 不在组件的原型上 // 代理到组件中,确保能够访问 props if (!(key in vm)) { proxy(vm, `_props`, key) } } toggleObserving(true) //不是根组件 启用响应式观察 }
functionmakeReactive(target: any, shallow: boolean) { // if trying to observe a readonly proxy, return the readonly version. if (!isReadonly(target)) { // 检查目标对象是否为只读代理,如果是,直接返回只读版本 if (__DEV__) { // 如果目标对象是数组,发出警告,因为 Vue 2 不支持在 watch 或 watchEffect 中跟踪数组的变化 if (isArray(target)) { warn( `Avoid using Array as root value for ${ shallow ? `shallowReactive()` : `reactive()` } as it cannot be tracked in watch() or watchEffect(). Use ${ shallow ? `shallowRef()` : `ref()` } instead. This is a Vue-2-only limitation.` ) } const existingOb = target && target.__ob__// 检查目标对象是否已经有响应式对象 if (existingOb && existingOb.shallow !== shallow) { // 如果目标对象已经有响应式对象,检查它是否和 shallow 参数相同 warn( `Target is already a ${ existingOb.shallow ? `` : `non-` }shallow reactive object, and cannot be converted to ${ shallow ? `` : `non-` }shallow.` ) } } // // 调用 observe 函数创建响应式对象 const ob = observe( target, shallow, isServerRendering() /* ssr mock reactivity */ ) if (__DEV__ && !ob) { // 在开发环境下,如果创建响应式对象失败,发出相应的警告 if (target == null || isPrimitive(target)) { warn(`value cannot be made reactive: ${String(target)}`) } if (isCollectionType(target)) { // 如果目标对象是集合类型(如 Map 或 Set),发出警告,因为 Vue 2 不支持响应式集合类型 warn( `Vue 2 does not support reactive collection types such as Map or Set.` ) } } } }