import type VNode from 'core/vdom/vnode' import type { Component } from 'types/component' /** * Runtime helper for resolving raw children VNodes into a slot object. */ export function resolveSlots( children: Array | null | undefined, context: Component | null ): { [key: string]: Array } { if (!children || !children.length) { return {} } const slots: Record = {} for (let i = 0, l = children.length; i < l; i++) { const child = children[i] const data = child.data // remove slot attribute if the node is resolved as a Vue slot node if (data && data.attrs && data.attrs.slot) { delete data.attrs.slot } // named slots should only be respected if the vnode was rendered in the // same context. if ( (child.context === context || child.fnContext === context) && data && data.slot != null ) { const name = data.slot const slot = slots[name] || (slots[name] = []) if (child.tag === 'template') { slot.push.apply(slot, child.children || []) } else { slot.push(child) } } else { ;(slots.default || (slots.default = [])).push(child) } } // ignore slots that contains only whitespace for (const name in slots) { if (slots[name].every(isWhitespace)) { delete slots[name] } } return slots } function isWhitespace(node: VNode): boolean { return (node.isComment && !node.asyncFactory) || node.text === ' ' }