import { ComponentInternalInstance, FunctionalComponent, Data } from './component' import { VNode, normalizeVNode, createVNode, Comment, cloneVNode, Fragment, VNodeArrayChildren, isVNode } from './vnode' import { handleError, ErrorCodes } from './errorHandling' import { PatchFlags, ShapeFlags, isOn } from '@vue/shared' import { warn } from './warning' // mark the current rendering instance for asset resolution (e.g. // resolveComponent, resolveDirective) during render export let currentRenderingInstance: ComponentInternalInstance | null = null export function setCurrentRenderingInstance( instance: ComponentInternalInstance | null ) { 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: boolean = false export function markAttrsAccessed() { accessedAttrs = true } export function renderComponentRoot( instance: ComponentInternalInstance ): VNode { const { type: Component, parent, vnode, proxy, withProxy, props, slots, attrs, emit, renderCache } = instance let result currentRenderingInstance = instance if (__DEV__) { accessedAttrs = false } try { let fallthroughAttrs if (vnode.shapeFlag & ShapeFlags.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( instance.render!.call(proxyToUse, proxyToUse!, renderCache) ) fallthroughAttrs = attrs } else { // functional const render = Component as FunctionalComponent // in dev, mark attrs accessed if optional props (attrs === props) if (__DEV__ && attrs === props) { markAttrsAccessed() } result = normalizeVNode( render.length > 1 ? render( props, __DEV__ ? { get attrs() { markAttrsAccessed() return attrs }, slots, emit } : { attrs, slots, emit } ) : render(props, null as any /* we know it doesn't need it */) ) fallthroughAttrs = Component.props ? attrs : getFallthroughAttrs(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: ((root: VNode) => void) | undefined = undefined if (__DEV__) { ;[root, setRoot] = getChildRoot(result) } if ( Component.inheritAttrs !== false && fallthroughAttrs && Object.keys(fallthroughAttrs).length ) { if ( root.shapeFlag & ShapeFlags.ELEMENT || root.shapeFlag & ShapeFlags.COMPONENT ) { root = cloneVNode(root, fallthroughAttrs) } else if (__DEV__ && !accessedAttrs && root.type !== Comment) { const allAttrs = Object.keys(attrs) const eventAttrs: string[] = [] const extraAttrs: string[] = [] for (let i = 0, l = allAttrs.length; i < l; i++) { const key = allAttrs[i] if (isOn(key)) { // remove `on`, lowercase first letter to reflect event casing accurately eventAttrs.push(key[2].toLowerCase() + key.slice(3)) } else { extraAttrs.push(key) } } if (extraAttrs.length) { warn( `Extraneous non-props attributes (` + `${extraAttrs.join(', ')}) ` + `were passed to component but could not be automatically inherited ` + `because component renders fragment or text root nodes.` ) } if (eventAttrs.length) { warn( `Extraneous non-emits event listeners (` + `${eventAttrs.join(', ')}) ` + `were passed to component but could not be automatically inherited ` + `because component renders fragment or text root nodes. ` + `If the listener is intended to be a component custom event listener only, ` + `declare it using the "emits" option.` ) } } } // inherit scopeId const parentScopeId = parent && parent.type.__scopeId if (parentScopeId) { root = cloneVNode(root, { [parentScopeId]: '' }) } // inherit directives if (vnode.dirs) { if (__DEV__ && !isElementRoot(root)) { warn( `Runtime directive used on component with non-element root node. ` + `The directives will not function as intended.` ) } root.dirs = vnode.dirs } // inherit transition data if (vnode.transition) { if (__DEV__ && !isElementRoot(root)) { warn( `Component inside renders non-element root node ` + `that cannot be animated.` ) } root.transition = vnode.transition } // inherit ref if (Component.inheritRef && vnode.ref != null) { root.ref = vnode.ref } if (__DEV__ && setRoot) { setRoot(root) } else { result = root } } catch (err) { handleError(err, instance, ErrorCodes.RENDER_FUNCTION) result = createVNode(Comment) } currentRenderingInstance = null return result } const getChildRoot = ( vnode: VNode ): [VNode, ((root: VNode) => void) | undefined] => { if (vnode.type !== Fragment) { return [vnode, undefined] } const rawChildren = vnode.children as VNodeArrayChildren const dynamicChildren = vnode.dynamicChildren as VNodeArrayChildren const children = rawChildren.filter(child => { return !(isVNode(child) && child.type === Comment) }) if (children.length !== 1) { return [vnode, undefined] } const childRoot = children[0] const index = rawChildren.indexOf(childRoot) const dynamicIndex = dynamicChildren ? dynamicChildren.indexOf(childRoot) : null const setRoot = (updatedRoot: VNode) => { rawChildren[index] = updatedRoot if (dynamicIndex !== null) dynamicChildren[dynamicIndex] = updatedRoot } return [normalizeVNode(childRoot), setRoot] } const getFallthroughAttrs = (attrs: Data): Data | undefined => { let res: Data | undefined for (const key in attrs) { if (key === 'class' || key === 'style' || isOn(key)) { ;(res || (res = {}))[key] = attrs[key] } } return res } const isElementRoot = (vnode: VNode) => { return ( vnode.shapeFlag & ShapeFlags.COMPONENT || vnode.shapeFlag & ShapeFlags.ELEMENT || vnode.type === Comment // potential v-if branch switch ) } export function shouldUpdateComponent( prevVNode: VNode, nextVNode: VNode, parentComponent: ComponentInternalInstance | null, optimized?: boolean ): boolean { const { props: prevProps, children: prevChildren } = prevVNode const { props: nextProps, children: nextChildren, patchFlag } = nextVNode // Parent component's render function was hot-updated. Since this may have // caused the child component's slots content to have changed, we need to // force the child to update as well. if (__DEV__ && (prevChildren || nextChildren) && parentComponent) { let parent: ComponentInternalInstance | null = parentComponent do { if (parent.hmrUpdated) { return true } } while ((parent = parent.parent)) } // force child update for runtime directive or transition on component vnode. if (nextVNode.dirs || nextVNode.transition) { return true } if (patchFlag > 0) { if (patchFlag & PatchFlags.DYNAMIC_SLOTS) { // slot content that references values that might have changed, // e.g. in a v-for return true } if (patchFlag & PatchFlags.FULL_PROPS) { if (!prevProps) { return !!nextProps } // presence of this flag indicates props are always non-null return hasPropsChanged(prevProps, nextProps!) } else if (patchFlag & PatchFlags.PROPS) { const dynamicProps = nextVNode.dynamicProps! for (let i = 0; i < dynamicProps.length; i++) { const key = dynamicProps[i] if (nextProps![key] !== prevProps![key]) { return true } } } } else if (!optimized) { // 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 as any).$stable) { return true } } if (prevProps === nextProps) { return false } if (!prevProps) { return !!nextProps } if (!nextProps) { return true } return hasPropsChanged(prevProps, nextProps) } return false } function hasPropsChanged(prevProps: Data, nextProps: Data): boolean { 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]) { return true } } return false } export function updateHOCHostEl( { vnode, parent }: ComponentInternalInstance, el: typeof vnode.el // HostNode ) { while (parent && parent.subTree === vnode) { ;(vnode = parent.vnode).el = el parent = parent.parent } }