2019-09-28 00:29:20 +00:00
|
|
|
import {
|
|
|
|
ElementNode,
|
|
|
|
ObjectExpression,
|
|
|
|
createObjectExpression,
|
|
|
|
NodeTypes,
|
2019-09-28 02:25:32 +00:00
|
|
|
createObjectProperty,
|
|
|
|
createSimpleExpression,
|
|
|
|
createFunctionExpression,
|
|
|
|
DirectiveNode,
|
|
|
|
ElementTypes,
|
|
|
|
ExpressionNode,
|
|
|
|
Property,
|
2019-10-01 13:27:34 +00:00
|
|
|
TemplateChildNode,
|
2019-10-02 21:18:11 +00:00
|
|
|
SourceLocation,
|
|
|
|
createConditionalExpression,
|
|
|
|
ConditionalExpression,
|
|
|
|
JSChildNode,
|
2019-10-03 03:10:41 +00:00
|
|
|
SimpleExpressionNode,
|
|
|
|
FunctionExpression,
|
|
|
|
CallExpression,
|
|
|
|
createCallExpression,
|
2019-10-16 19:30:21 +00:00
|
|
|
createArrayExpression,
|
|
|
|
IfBranchNode
|
2019-09-28 00:29:20 +00:00
|
|
|
} from '../ast'
|
2019-09-28 04:19:24 +00:00
|
|
|
import { TransformContext, NodeTransform } from '../transform'
|
2019-09-28 00:29:20 +00:00
|
|
|
import { createCompilerError, ErrorCodes } from '../errors'
|
2019-10-16 19:30:21 +00:00
|
|
|
import {
|
|
|
|
findDir,
|
|
|
|
isTemplateNode,
|
|
|
|
assert,
|
|
|
|
isVSlot,
|
|
|
|
isSimpleIdentifier
|
|
|
|
} from '../utils'
|
2019-10-05 21:18:25 +00:00
|
|
|
import { CREATE_SLOTS, RENDER_LIST } from '../runtimeHelpers'
|
2019-10-03 03:10:41 +00:00
|
|
|
import { parseForExpression, createForLoopParams } from './vFor'
|
2019-10-16 19:30:21 +00:00
|
|
|
import { isObject } from '@vue/shared'
|
2019-09-28 00:29:20 +00:00
|
|
|
|
2019-10-02 21:18:11 +00:00
|
|
|
const isStaticExp = (p: JSChildNode): p is SimpleExpressionNode =>
|
|
|
|
p.type === NodeTypes.SIMPLE_EXPRESSION && p.isStatic
|
|
|
|
|
|
|
|
const defaultFallback = createSimpleExpression(`undefined`, false)
|
|
|
|
|
2019-10-03 20:27:46 +00:00
|
|
|
// 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.
|
2019-09-28 04:19:24 +00:00
|
|
|
export const trackSlotScopes: NodeTransform = (node, context) => {
|
|
|
|
if (
|
|
|
|
node.type === NodeTypes.ELEMENT &&
|
|
|
|
(node.tagType === ElementTypes.COMPONENT ||
|
|
|
|
node.tagType === ElementTypes.TEMPLATE)
|
|
|
|
) {
|
2019-10-03 20:27:46 +00:00
|
|
|
// We are only checking non-empty v-slot here
|
|
|
|
// since we only care about slots that introduce scope variables.
|
2019-10-03 03:10:41 +00:00
|
|
|
const vSlot = findDir(node, 'slot')
|
|
|
|
if (vSlot) {
|
|
|
|
const slotProps = vSlot.exp
|
2019-10-03 20:27:46 +00:00
|
|
|
if (!__BROWSER__ && context.prefixIdentifiers) {
|
|
|
|
slotProps && context.addIdentifiers(slotProps)
|
|
|
|
}
|
|
|
|
context.scopes.vSlot++
|
2019-09-28 20:02:08 +00:00
|
|
|
return () => {
|
2019-10-03 20:27:46 +00:00
|
|
|
if (!__BROWSER__ && context.prefixIdentifiers) {
|
|
|
|
slotProps && context.removeIdentifiers(slotProps)
|
|
|
|
}
|
|
|
|
context.scopes.vSlot--
|
2019-10-03 03:10:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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)
|
2019-09-28 04:19:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instead of being a DirectiveTransform, v-slot processing is called during
|
|
|
|
// transformElement to build the slots object for a component.
|
2019-09-28 00:29:20 +00:00
|
|
|
export function buildSlots(
|
2019-10-03 03:10:41 +00:00
|
|
|
node: ElementNode,
|
2019-09-28 00:29:20 +00:00
|
|
|
context: TransformContext
|
2019-10-01 01:17:12 +00:00
|
|
|
): {
|
2019-10-03 03:10:41 +00:00
|
|
|
slots: ObjectExpression | CallExpression
|
2019-10-02 21:18:11 +00:00
|
|
|
hasDynamicSlots: boolean
|
2019-10-01 01:17:12 +00:00
|
|
|
} {
|
2019-10-03 03:10:41 +00:00
|
|
|
const { children, loc } = node
|
|
|
|
const slotsProperties: Property[] = []
|
|
|
|
const dynamicSlots: (ConditionalExpression | CallExpression)[] = []
|
2019-10-03 20:27:46 +00:00
|
|
|
|
|
|
|
// If the slot is inside a v-for or another v-slot, force it to be dynamic
|
|
|
|
// since it likely uses a scope variable.
|
2019-10-16 19:30:21 +00:00
|
|
|
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)
|
|
|
|
}
|
2019-09-28 02:25:32 +00:00
|
|
|
|
|
|
|
// 1. Check for default slot with slotProps on component itself.
|
|
|
|
// <Comp v-slot="{ prop }"/>
|
2019-10-03 03:10:41 +00:00
|
|
|
const explicitDefaultSlot = findDir(node, 'slot', true)
|
2019-09-28 02:25:32 +00:00
|
|
|
if (explicitDefaultSlot) {
|
|
|
|
const { arg, exp, loc } = explicitDefaultSlot
|
|
|
|
if (arg) {
|
|
|
|
context.onError(
|
2019-10-09 14:27:24 +00:00
|
|
|
createCompilerError(ErrorCodes.X_V_SLOT_NAMED_SLOT_ON_COMPONENT, loc)
|
2019-09-28 02:25:32 +00:00
|
|
|
)
|
|
|
|
}
|
2019-10-03 03:10:41 +00:00
|
|
|
slotsProperties.push(buildDefaultSlot(exp, children, loc))
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
2019-09-28 00:29:20 +00:00
|
|
|
|
2019-09-28 02:25:32 +00:00
|
|
|
// 2. Iterate through children and check for template slots
|
|
|
|
// <template v-slot:foo="{ prop }">
|
|
|
|
let hasTemplateSlots = false
|
2019-10-01 13:27:34 +00:00
|
|
|
let extraneousChild: TemplateChildNode | undefined = undefined
|
2019-09-28 02:25:32 +00:00
|
|
|
const seenSlotNames = new Set<string>()
|
|
|
|
for (let i = 0; i < children.length; i++) {
|
2019-10-02 21:18:11 +00:00
|
|
|
const slotElement = children[i]
|
2019-09-28 05:35:49 +00:00
|
|
|
let slotDir
|
2019-10-02 21:18:11 +00:00
|
|
|
|
2019-09-28 02:25:32 +00:00
|
|
|
if (
|
2019-10-03 03:10:41 +00:00
|
|
|
!isTemplateNode(slotElement) ||
|
|
|
|
!(slotDir = findDir(slotElement, 'slot', true))
|
2019-09-28 02:25:32 +00:00
|
|
|
) {
|
2019-10-02 21:18:11 +00:00
|
|
|
// not a <template v-slot>, skip.
|
2019-10-03 03:10:41 +00:00
|
|
|
if (slotElement.type !== NodeTypes.COMMENT && !extraneousChild) {
|
|
|
|
extraneousChild = slotElement
|
|
|
|
}
|
2019-10-02 21:18:11 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if (explicitDefaultSlot) {
|
|
|
|
// already has on-component default slot - this is incorrect usage.
|
|
|
|
context.onError(
|
2019-10-09 14:27:24 +00:00
|
|
|
createCompilerError(ErrorCodes.X_V_SLOT_MIXED_SLOT_USAGE, slotDir.loc)
|
2019-10-02 21:18:11 +00:00
|
|
|
)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
hasTemplateSlots = true
|
|
|
|
const { children: slotChildren, loc: slotLoc } = slotElement
|
|
|
|
const {
|
|
|
|
arg: slotName = createSimpleExpression(`default`, true),
|
|
|
|
exp: slotProps,
|
|
|
|
loc: dirLoc
|
|
|
|
} = slotDir
|
|
|
|
|
|
|
|
// check if name is dynamic.
|
2019-10-02 22:03:42 +00:00
|
|
|
let staticSlotName: string | undefined
|
2019-10-02 21:18:11 +00:00
|
|
|
if (isStaticExp(slotName)) {
|
|
|
|
staticSlotName = slotName ? slotName.content : `default`
|
|
|
|
} else {
|
|
|
|
hasDynamicSlots = true
|
|
|
|
}
|
|
|
|
|
|
|
|
const slotFunction = createFunctionExpression(
|
|
|
|
slotProps,
|
|
|
|
slotChildren,
|
|
|
|
false,
|
|
|
|
slotChildren.length ? slotChildren[0].loc : slotLoc
|
|
|
|
)
|
|
|
|
|
2019-10-03 03:10:41 +00:00
|
|
|
// check if this slot is conditional (v-if/v-for)
|
2019-10-02 22:03:42 +00:00
|
|
|
let vIf: DirectiveNode | undefined
|
|
|
|
let vElse: DirectiveNode | undefined
|
2019-10-03 03:10:41 +00:00
|
|
|
let vFor: DirectiveNode | undefined
|
2019-10-02 22:03:42 +00:00
|
|
|
if ((vIf = findDir(slotElement, 'if'))) {
|
2019-10-02 21:18:11 +00:00
|
|
|
hasDynamicSlots = true
|
2019-10-03 03:10:41 +00:00
|
|
|
dynamicSlots.push(
|
|
|
|
createConditionalExpression(
|
|
|
|
vIf.exp!,
|
|
|
|
buildDynamicSlot(slotName, slotFunction),
|
|
|
|
defaultFallback
|
2019-09-28 05:35:49 +00:00
|
|
|
)
|
2019-10-02 21:18:11 +00:00
|
|
|
)
|
2019-10-02 22:03:42 +00:00
|
|
|
} else if (
|
2019-10-03 03:10:41 +00:00
|
|
|
(vElse = findDir(slotElement, /^else(-if)?$/, true /* allowEmpty */))
|
2019-10-02 22:03:42 +00:00
|
|
|
) {
|
2019-10-03 03:10:41 +00:00
|
|
|
// find adjacent v-if
|
|
|
|
let j = i
|
|
|
|
let prev
|
|
|
|
while (j--) {
|
|
|
|
prev = children[j]
|
|
|
|
if (prev.type !== NodeTypes.COMMENT) {
|
|
|
|
break
|
2019-10-02 21:18:11 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-03 03:10:41 +00:00
|
|
|
if (prev && isTemplateNode(prev) && findDir(prev, 'if')) {
|
|
|
|
// remove node
|
|
|
|
children.splice(i, 1)
|
|
|
|
i--
|
|
|
|
__DEV__ && assert(dynamicSlots.length > 0)
|
|
|
|
// attach this slot to previous conditional
|
|
|
|
let conditional = dynamicSlots[
|
|
|
|
dynamicSlots.length - 1
|
|
|
|
] as ConditionalExpression
|
2019-10-02 22:03:42 +00:00
|
|
|
while (
|
|
|
|
conditional.alternate.type === NodeTypes.JS_CONDITIONAL_EXPRESSION
|
2019-09-28 05:35:49 +00:00
|
|
|
) {
|
2019-10-02 22:03:42 +00:00
|
|
|
conditional = conditional.alternate
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
2019-10-02 22:03:42 +00:00
|
|
|
conditional.alternate = vElse.exp
|
|
|
|
? createConditionalExpression(
|
|
|
|
vElse.exp,
|
2019-10-03 03:10:41 +00:00
|
|
|
buildDynamicSlot(slotName, slotFunction),
|
2019-10-02 22:03:42 +00:00
|
|
|
defaultFallback
|
|
|
|
)
|
2019-10-03 03:10:41 +00:00
|
|
|
: buildDynamicSlot(slotName, slotFunction)
|
2019-10-02 21:18:11 +00:00
|
|
|
} else {
|
2019-10-03 03:10:41 +00:00
|
|
|
context.onError(
|
2019-10-09 14:27:24 +00:00
|
|
|
createCompilerError(ErrorCodes.X_V_ELSE_NO_ADJACENT_IF, vElse.loc)
|
2019-10-03 03:10:41 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
} else if ((vFor = findDir(slotElement, 'for'))) {
|
|
|
|
hasDynamicSlots = true
|
|
|
|
const parseResult =
|
|
|
|
vFor.parseResult ||
|
|
|
|
parseForExpression(vFor.exp as SimpleExpressionNode, context)
|
|
|
|
if (parseResult) {
|
|
|
|
// Render the dynamic slots as an array and add it to the createSlot()
|
|
|
|
// args. The runtime knows how to handle it appropriately.
|
|
|
|
dynamicSlots.push(
|
|
|
|
createCallExpression(context.helper(RENDER_LIST), [
|
|
|
|
parseResult.source,
|
|
|
|
createFunctionExpression(
|
|
|
|
createForLoopParams(parseResult),
|
|
|
|
buildDynamicSlot(slotName, slotFunction),
|
|
|
|
true
|
2019-10-02 22:03:42 +00:00
|
|
|
)
|
2019-10-03 03:10:41 +00:00
|
|
|
])
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
context.onError(
|
2019-10-09 14:27:24 +00:00
|
|
|
createCompilerError(ErrorCodes.X_V_FOR_MALFORMED_EXPRESSION, vFor.loc)
|
2019-09-28 05:35:49 +00:00
|
|
|
)
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
2019-10-02 21:18:11 +00:00
|
|
|
} else {
|
|
|
|
// check duplicate static names
|
|
|
|
if (staticSlotName) {
|
|
|
|
if (seenSlotNames.has(staticSlotName)) {
|
|
|
|
context.onError(
|
2019-10-09 14:27:24 +00:00
|
|
|
createCompilerError(
|
|
|
|
ErrorCodes.X_V_SLOT_DUPLICATE_SLOT_NAMES,
|
|
|
|
dirLoc
|
|
|
|
)
|
2019-10-02 21:18:11 +00:00
|
|
|
)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
seenSlotNames.add(staticSlotName)
|
|
|
|
}
|
2019-10-03 03:10:41 +00:00
|
|
|
slotsProperties.push(createObjectProperty(slotName, slotFunction))
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-28 05:35:49 +00:00
|
|
|
if (hasTemplateSlots && extraneousChild) {
|
2019-09-28 02:25:32 +00:00
|
|
|
context.onError(
|
|
|
|
createCompilerError(
|
2019-10-09 14:27:24 +00:00
|
|
|
ErrorCodes.X_V_SLOT_EXTRANEOUS_NON_SLOT_CHILDREN,
|
2019-09-28 05:35:49 +00:00
|
|
|
extraneousChild.loc
|
2019-09-28 02:25:32 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!explicitDefaultSlot && !hasTemplateSlots) {
|
|
|
|
// implicit default slot.
|
2019-10-03 03:10:41 +00:00
|
|
|
slotsProperties.push(buildDefaultSlot(undefined, children, loc))
|
|
|
|
}
|
|
|
|
|
|
|
|
let slots: ObjectExpression | CallExpression = createObjectExpression(
|
2019-10-03 18:08:14 +00:00
|
|
|
slotsProperties.concat(
|
|
|
|
createObjectProperty(`_compiled`, createSimpleExpression(`true`, false))
|
|
|
|
),
|
2019-10-03 03:10:41 +00:00
|
|
|
loc
|
|
|
|
)
|
|
|
|
if (dynamicSlots.length) {
|
|
|
|
slots = createCallExpression(context.helper(CREATE_SLOTS), [
|
|
|
|
slots,
|
|
|
|
createArrayExpression(dynamicSlots)
|
|
|
|
])
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
|
|
|
|
2019-10-01 01:17:12 +00:00
|
|
|
return {
|
2019-10-03 03:10:41 +00:00
|
|
|
slots,
|
2019-10-02 21:18:11 +00:00
|
|
|
hasDynamicSlots
|
2019-10-01 01:17:12 +00:00
|
|
|
}
|
2019-09-28 02:25:32 +00:00
|
|
|
}
|
|
|
|
|
2019-10-02 21:18:11 +00:00
|
|
|
function buildDefaultSlot(
|
2019-09-28 02:25:32 +00:00
|
|
|
slotProps: ExpressionNode | undefined,
|
2019-10-01 13:27:34 +00:00
|
|
|
children: TemplateChildNode[],
|
2019-09-28 02:25:32 +00:00
|
|
|
loc: SourceLocation
|
|
|
|
): Property {
|
|
|
|
return createObjectProperty(
|
2019-10-03 03:10:41 +00:00
|
|
|
`default`,
|
2019-09-28 02:25:32 +00:00
|
|
|
createFunctionExpression(
|
|
|
|
slotProps,
|
|
|
|
children,
|
2019-10-01 20:48:20 +00:00
|
|
|
false,
|
2019-09-28 02:25:32 +00:00
|
|
|
children.length ? children[0].loc : loc
|
2019-10-01 16:25:13 +00:00
|
|
|
)
|
2019-09-28 02:25:32 +00:00
|
|
|
)
|
2019-09-28 00:29:20 +00:00
|
|
|
}
|
2019-10-03 03:10:41 +00:00
|
|
|
|
|
|
|
function buildDynamicSlot(
|
|
|
|
name: ExpressionNode,
|
|
|
|
fn: FunctionExpression
|
|
|
|
): ObjectExpression {
|
|
|
|
return createObjectExpression([
|
|
|
|
createObjectProperty(`name`, name),
|
|
|
|
createObjectProperty(`fn`, fn)
|
|
|
|
])
|
|
|
|
}
|
2019-10-16 19:30:21 +00:00
|
|
|
|
|
|
|
function hasScopeRef(
|
|
|
|
node: TemplateChildNode | IfBranchNode | SimpleExpressionNode | undefined,
|
|
|
|
ids: TransformContext['identifiers']
|
|
|
|
): boolean {
|
|
|
|
if (!node) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
switch (node.type) {
|
|
|
|
case NodeTypes.ELEMENT:
|
|
|
|
for (let i = 0; i < node.props.length; i++) {
|
|
|
|
const p = node.props[i]
|
|
|
|
if (
|
|
|
|
p.type === NodeTypes.DIRECTIVE &&
|
|
|
|
(hasScopeRef(p.arg, ids) || hasScopeRef(p.exp, ids))
|
|
|
|
) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.FOR:
|
|
|
|
if (hasScopeRef(node.source, ids)) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.IF:
|
|
|
|
return node.branches.some(b => hasScopeRef(b, ids))
|
|
|
|
case NodeTypes.IF_BRANCH:
|
|
|
|
if (hasScopeRef(node.condition, ids)) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.SIMPLE_EXPRESSION:
|
|
|
|
return (
|
|
|
|
!node.isStatic &&
|
|
|
|
isSimpleIdentifier(node.content) &&
|
|
|
|
!!ids[node.content]
|
|
|
|
)
|
|
|
|
case NodeTypes.COMPOUND_EXPRESSION:
|
|
|
|
return node.children.some(c => isObject(c) && hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.INTERPOLATION:
|
|
|
|
return hasScopeRef(node.content, ids)
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|