import { ElementNode, ObjectExpression, createObjectExpression, NodeTypes, createObjectProperty, createSimpleExpression, createFunctionExpression, DirectiveNode, ElementTypes, ExpressionNode, Property, TemplateChildNode, SourceLocation, createConditionalExpression, ConditionalExpression, JSChildNode, SimpleExpressionNode, FunctionExpression, CallExpression, createCallExpression, createArrayExpression, SlotsExpression } from '../ast' import { TransformContext, NodeTransform } from '../transform' import { createCompilerError, ErrorCodes } from '../errors' import { findDir, isTemplateNode, assert, isVSlot, hasScopeRef } from '../utils' import { CREATE_SLOTS, RENDER_LIST, WITH_CTX } from '../runtimeHelpers' import { parseForExpression, createForLoopParams } from './vFor' const isStaticExp = (p: JSChildNode): p is SimpleExpressionNode => p.type === NodeTypes.SIMPLE_EXPRESSION && p.isStatic const defaultFallback = createSimpleExpression(`undefined`, false) // A NodeTransform that: // 1. Tracks scope identifiers for scoped slots so that they don't get prefixed // by transformExpression. This is only applied in non-browser builds with // { prefixIdentifiers: true }. // 2. Track v-slot depths so that we know a slot is inside another slot. // Note the exit callback is executed before buildSlots() on the same node, // so only nested slots see positive numbers. export const trackSlotScopes: NodeTransform = (node, context) => { if ( node.type === NodeTypes.ELEMENT && (node.tagType === ElementTypes.COMPONENT || node.tagType === ElementTypes.TEMPLATE) ) { // We are only checking non-empty v-slot here // since we only care about slots that introduce scope variables. const vSlot = findDir(node, 'slot') if (vSlot) { const slotProps = vSlot.exp if (!__BROWSER__ && context.prefixIdentifiers) { slotProps && context.addIdentifiers(slotProps) } context.scopes.vSlot++ return () => { if (!__BROWSER__ && context.prefixIdentifiers) { slotProps && context.removeIdentifiers(slotProps) } context.scopes.vSlot-- } } } } // A NodeTransform that tracks scope identifiers for scoped slots with v-for. // This transform is only applied in non-browser builds with { prefixIdentifiers: true } export const trackVForSlotScopes: NodeTransform = (node, context) => { let vFor if ( isTemplateNode(node) && node.props.some(isVSlot) && (vFor = findDir(node, 'for')) ) { const result = (vFor.parseResult = parseForExpression( vFor.exp as SimpleExpressionNode, context )) if (result) { const { value, key, index } = result const { addIdentifiers, removeIdentifiers } = context value && addIdentifiers(value) key && addIdentifiers(key) index && addIdentifiers(index) return () => { value && removeIdentifiers(value) key && removeIdentifiers(key) index && removeIdentifiers(index) } } } } export type SlotFnBuilder = ( slotProps: ExpressionNode | undefined, slotChildren: TemplateChildNode[], loc: SourceLocation ) => FunctionExpression const buildClientSlotFn: SlotFnBuilder = (props, children, loc) => createFunctionExpression( props, children, false /* newline */, true /* isSlot */, children.length ? children[0].loc : loc ) // Instead of being a DirectiveTransform, v-slot processing is called during // transformElement to build the slots object for a component. export function buildSlots( node: ElementNode, context: TransformContext, buildSlotFn: SlotFnBuilder = buildClientSlotFn ): { slots: SlotsExpression hasDynamicSlots: boolean } { context.helper(WITH_CTX) const { children, loc } = node const slotsProperties: Property[] = [] const dynamicSlots: (ConditionalExpression | CallExpression)[] = [] const buildDefaultSlotProperty = ( props: ExpressionNode | undefined, children: TemplateChildNode[] ) => createObjectProperty(`default`, buildSlotFn(props, children, loc)) // If the slot is inside a v-for or another v-slot, force it to be dynamic // since it likely uses a scope variable. let hasDynamicSlots = context.scopes.vSlot > 0 || context.scopes.vFor > 0 // with `prefixIdentifiers: true`, this can be further optimized to make // it dynamic only when the slot actually uses the scope variables. if (!__BROWSER__ && context.prefixIdentifiers) { hasDynamicSlots = hasScopeRef(node, context.identifiers) } // 1. Check for slot with slotProps on component itself. // const onComponentSlot = findDir(node, 'slot', true) if (onComponentSlot) { const { arg, exp } = onComponentSlot slotsProperties.push( createObjectProperty( arg || createSimpleExpression('default', true), buildSlotFn(exp, children, loc) ) ) } // 2. Iterate through children and check for template slots //