Skip to content

Commit e09c26b

Browse files
authoredOct 24, 2023
fix(compiler-ssr): proper scope analysis for ssr vnode slot fallback (#7184)
close #7095
1 parent 7374e93 commit e09c26b

File tree

4 files changed

+58
-33
lines changed

4 files changed

+58
-33
lines changed
 

‎packages/compiler-core/src/transforms/vSlot.ts

+13-6
Original file line numberDiff line numberDiff line change
@@ -100,11 +100,12 @@ export const trackVForSlotScopes: NodeTransform = (node, context) => {
100100

101101
export type SlotFnBuilder = (
102102
slotProps: ExpressionNode | undefined,
103+
vForExp: ExpressionNode | undefined,
103104
slotChildren: TemplateChildNode[],
104105
loc: SourceLocation
105106
) => FunctionExpression
106107

107-
const buildClientSlotFn: SlotFnBuilder = (props, children, loc) =>
108+
const buildClientSlotFn: SlotFnBuilder = (props, _vForExp, children, loc) =>
108109
createFunctionExpression(
109110
props,
110111
children,
@@ -149,7 +150,7 @@ export function buildSlots(
149150
slotsProperties.push(
150151
createObjectProperty(
151152
arg || createSimpleExpression('default', true),
152-
buildSlotFn(exp, children, loc)
153+
buildSlotFn(exp, undefined, children, loc)
153154
)
154155
)
155156
}
@@ -201,11 +202,17 @@ export function buildSlots(
201202
hasDynamicSlots = true
202203
}
203204

204-
const slotFunction = buildSlotFn(slotProps, slotChildren, slotLoc)
205+
const vFor = findDir(slotElement, 'for')
206+
const slotFunction = buildSlotFn(
207+
slotProps,
208+
vFor?.exp,
209+
slotChildren,
210+
slotLoc
211+
)
212+
205213
// check if this slot is conditional (v-if/v-for)
206214
let vIf: DirectiveNode | undefined
207215
let vElse: DirectiveNode | undefined
208-
let vFor: DirectiveNode | undefined
209216
if ((vIf = findDir(slotElement, 'if'))) {
210217
hasDynamicSlots = true
211218
dynamicSlots.push(
@@ -257,7 +264,7 @@ export function buildSlots(
257264
createCompilerError(ErrorCodes.X_V_ELSE_NO_ADJACENT_IF, vElse.loc)
258265
)
259266
}
260-
} else if ((vFor = findDir(slotElement, 'for'))) {
267+
} else if (vFor) {
261268
hasDynamicSlots = true
262269
const parseResult =
263270
vFor.parseResult ||
@@ -306,7 +313,7 @@ export function buildSlots(
306313
props: ExpressionNode | undefined,
307314
children: TemplateChildNode[]
308315
) => {
309-
const fn = buildSlotFn(props, children, loc)
316+
const fn = buildSlotFn(props, undefined, children, loc)
310317
if (__COMPAT__ && context.compatConfig) {
311318
fn.isNonScopedSlot = true
312319
}

‎packages/compiler-ssr/__tests__/ssrComponent.spec.ts

+11-8
Original file line numberDiff line numberDiff line change
@@ -181,27 +181,30 @@ describe('ssr: components', () => {
181181
})
182182

183183
test('v-for slot', () => {
184-
expect(
185-
compile(`<foo>
186-
<template v-for="key in names" v-slot:[key]="{ msg }">{{ msg + key + bar }}</template>
187-
</foo>`).code
188-
).toMatchInlineSnapshot(`
184+
const { code } = compile(`<foo>
185+
<template v-for="(key, index) in names" v-slot:[key]="{ msg }">{{ msg + key + index + bar }}</template>
186+
</foo>`)
187+
expect(code).not.toMatch(`_ctx.msg`)
188+
expect(code).not.toMatch(`_ctx.key`)
189+
expect(code).not.toMatch(`_ctx.index`)
190+
expect(code).toMatch(`_ctx.bar`)
191+
expect(code).toMatchInlineSnapshot(`
189192
"const { resolveComponent: _resolveComponent, withCtx: _withCtx, toDisplayString: _toDisplayString, createTextVNode: _createTextVNode, renderList: _renderList, createSlots: _createSlots } = require(\\"vue\\")
190193
const { ssrRenderComponent: _ssrRenderComponent, ssrInterpolate: _ssrInterpolate } = require(\\"vue/server-renderer\\")
191194
192195
return function ssrRender(_ctx, _push, _parent, _attrs) {
193196
const _component_foo = _resolveComponent(\\"foo\\")
194197
195198
_push(_ssrRenderComponent(_component_foo, _attrs, _createSlots({ _: 2 /* DYNAMIC */ }, [
196-
_renderList(_ctx.names, (key) => {
199+
_renderList(_ctx.names, (key, index) => {
197200
return {
198201
name: key,
199202
fn: _withCtx(({ msg }, _push, _parent, _scopeId) => {
200203
if (_push) {
201-
_push(\`\${_ssrInterpolate(msg + key + _ctx.bar)}\`)
204+
_push(\`\${_ssrInterpolate(msg + key + index + _ctx.bar)}\`)
202205
} else {
203206
return [
204-
_createTextVNode(_toDisplayString(msg + _ctx.key + _ctx.bar), 1 /* TEXT */)
207+
_createTextVNode(_toDisplayString(msg + key + index + _ctx.bar), 1 /* TEXT */)
205208
]
206209
}
207210
})

‎packages/compiler-ssr/src/transforms/ssrTransformComponent.ts

+16-5
Original file line numberDiff line numberDiff line change
@@ -125,8 +125,10 @@ export const ssrTransformComponent: NodeTransform = (node, context) => {
125125
// fallback in case the child is render-fn based). Store them in an array
126126
// for later use.
127127
if (clonedNode.children.length) {
128-
buildSlots(clonedNode, context, (props, children) => {
129-
vnodeBranches.push(createVNodeSlotBranch(props, children, context))
128+
buildSlots(clonedNode, context, (props, vFor, children) => {
129+
vnodeBranches.push(
130+
createVNodeSlotBranch(props, vFor, children, context)
131+
)
130132
return createFunctionExpression(undefined)
131133
})
132134
}
@@ -150,7 +152,7 @@ export const ssrTransformComponent: NodeTransform = (node, context) => {
150152
const wipEntries: WIPSlotEntry[] = []
151153
wipMap.set(node, wipEntries)
152154

153-
const buildSSRSlotFn: SlotFnBuilder = (props, children, loc) => {
155+
const buildSSRSlotFn: SlotFnBuilder = (props, _vForExp, children, loc) => {
154156
const param0 = (props && stringifyExpression(props)) || `_`
155157
const fn = createFunctionExpression(
156158
[param0, `_push`, `_parent`, `_scopeId`],
@@ -277,6 +279,7 @@ const vnodeDirectiveTransforms = {
277279

278280
function createVNodeSlotBranch(
279281
props: ExpressionNode | undefined,
282+
vForExp: ExpressionNode | undefined,
280283
children: TemplateChildNode[],
281284
parentContext: TransformContext
282285
): ReturnStatement {
@@ -303,8 +306,8 @@ function createVNodeSlotBranch(
303306
tag: 'template',
304307
tagType: ElementTypes.TEMPLATE,
305308
isSelfClosing: false,
306-
// important: provide v-slot="props" on the wrapper for proper
307-
// scope analysis
309+
// important: provide v-slot="props" and v-for="exp" on the wrapper for
310+
// proper scope analysis
308311
props: [
309312
{
310313
type: NodeTypes.DIRECTIVE,
@@ -313,6 +316,14 @@ function createVNodeSlotBranch(
313316
arg: undefined,
314317
modifiers: [],
315318
loc: locStub
319+
},
320+
{
321+
type: NodeTypes.DIRECTIVE,
322+
name: 'for',
323+
exp: vForExp,
324+
arg: undefined,
325+
modifiers: [],
326+
loc: locStub
316327
}
317328
],
318329
children,

‎packages/compiler-ssr/src/transforms/ssrTransformSuspense.ts

+18-14
Original file line numberDiff line numberDiff line change
@@ -36,20 +36,24 @@ export function ssrTransformSuspense(
3636
wipSlots: []
3737
}
3838
wipMap.set(node, wipEntry)
39-
wipEntry.slotsExp = buildSlots(node, context, (_props, children, loc) => {
40-
const fn = createFunctionExpression(
41-
[],
42-
undefined, // no return, assign body later
43-
true, // newline
44-
false, // suspense slots are not treated as normal slots
45-
loc
46-
)
47-
wipEntry.wipSlots.push({
48-
fn,
49-
children
50-
})
51-
return fn
52-
}).slots
39+
wipEntry.slotsExp = buildSlots(
40+
node,
41+
context,
42+
(_props, _vForExp, children, loc) => {
43+
const fn = createFunctionExpression(
44+
[],
45+
undefined, // no return, assign body later
46+
true, // newline
47+
false, // suspense slots are not treated as normal slots
48+
loc
49+
)
50+
wipEntry.wipSlots.push({
51+
fn,
52+
children
53+
})
54+
return fn
55+
}
56+
).slots
5357
}
5458
}
5559
}

0 commit comments

Comments
 (0)
Please sign in to comment.