Skip to content

Commit 2807fd2

Browse files
committed
fix: template v-slot should work with v-else conditions
1 parent 5851961 commit 2807fd2

File tree

3 files changed

+53
-41
lines changed

3 files changed

+53
-41
lines changed

src/compiler/codegen/index.js

+4-30
Original file line numberDiff line numberDiff line change
@@ -366,53 +366,27 @@ function genScopedSlots (
366366
})
367367
return `scopedSlots:_u([${
368368
Object.keys(slots).map(key => {
369-
return genScopedSlot(key, slots[key], state)
369+
return genScopedSlot(slots[key], state)
370370
}).join(',')
371371
}]${hasDynamicKeys ? `,true` : ``})`
372372
}
373373

374374
function genScopedSlot (
375-
key: string,
376375
el: ASTElement,
377376
state: CodegenState
378377
): string {
379378
if (el.if && !el.ifProcessed) {
380-
return genIfScopedSlot(key, el, state)
379+
return genIf(el, state, genScopedSlot, `null`)
381380
}
382381
if (el.for && !el.forProcessed) {
383-
return genForScopedSlot(key, el, state)
382+
return genFor(el, state, genScopedSlot)
384383
}
385384
const fn = `function(${String(el.slotScope)}){` +
386385
`return ${el.tag === 'template'
387386
? genChildren(el, state) || 'undefined'
388387
: genElement(el, state)
389388
}}`
390-
return `{key:${key},fn:${fn}}`
391-
}
392-
393-
function genIfScopedSlot (
394-
key: string,
395-
el: any,
396-
state: CodegenState
397-
): string {
398-
el.ifProcessed = true
399-
return `(${el.if})?${genScopedSlot(key, el, state)}:null`
400-
}
401-
402-
function genForScopedSlot (
403-
key: string,
404-
el: any,
405-
state: CodegenState
406-
): string {
407-
const exp = el.for
408-
const alias = el.alias
409-
const iterator1 = el.iterator1 ? `,${el.iterator1}` : ''
410-
const iterator2 = el.iterator2 ? `,${el.iterator2}` : ''
411-
el.forProcessed = true // avoid recursion
412-
return `_l((${exp}),` +
413-
`function(${alias}${iterator1}${iterator2}){` +
414-
`return ${genScopedSlot(key, el, state)}` +
415-
'})'
389+
return `{key:${el.slotTarget || `"default"`},fn:${fn}}`
416390
}
417391

418392
export function genChildren (

src/compiler/parser/index.js

+31-11
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,7 @@ export function parse (
107107
}
108108

109109
function closeElement (element) {
110+
trimEndingWhitespace(element)
110111
if (!inVPre && !element.processed) {
111112
element = processElement(element, options)
112113
}
@@ -133,14 +134,25 @@ export function parse (
133134
if (currentParent && !element.forbidden) {
134135
if (element.elseif || element.else) {
135136
processIfConditions(element, currentParent)
136-
} else if (element.slotScope) { // scoped slot
137-
const name = element.slotTarget || '"default"'
138-
;(currentParent.scopedSlots || (currentParent.scopedSlots = {}))[name] = element
139137
} else {
138+
if (element.slotScope) {
139+
// scoped slot
140+
// keep it in the children list so that v-else(-if) conditions can
141+
// find it as the prev node.
142+
const name = element.slotTarget || '"default"'
143+
;(currentParent.scopedSlots || (currentParent.scopedSlots = {}))[name] = element
144+
}
140145
currentParent.children.push(element)
141146
element.parent = currentParent
142147
}
143148
}
149+
150+
// final children cleanup
151+
// filter out scoped slots
152+
element.children = element.children.filter(c => !c.slotScope)
153+
// remove trailing whitespace node again
154+
trimEndingWhitespace(element)
155+
144156
// check pre state
145157
if (element.pre) {
146158
inVPre = false
@@ -154,6 +166,20 @@ export function parse (
154166
}
155167
}
156168

169+
function trimEndingWhitespace (el) {
170+
// remove trailing whitespace node
171+
if (!inPre) {
172+
let lastNode
173+
while (
174+
(lastNode = el.children[el.children.length - 1]) &&
175+
lastNode.type === 3 &&
176+
lastNode.text === ' '
177+
) {
178+
el.children.pop()
179+
}
180+
}
181+
}
182+
157183
function checkRootConstraints (el) {
158184
if (el.tag === 'slot' || el.tag === 'template') {
159185
warnOnce(
@@ -268,13 +294,6 @@ export function parse (
268294

269295
end (tag, start, end) {
270296
const element = stack[stack.length - 1]
271-
if (!inPre) {
272-
// remove trailing whitespace node
273-
const lastNode = element.children[element.children.length - 1]
274-
if (lastNode && lastNode.type === 3 && lastNode.text === ' ') {
275-
element.children.pop()
276-
}
277-
}
278297
// pop stack
279298
stack.length -= 1
280299
currentParent = stack[stack.length - 1]
@@ -658,8 +677,9 @@ function processSlotContent (el) {
658677
const slots = el.scopedSlots || (el.scopedSlots = {})
659678
const { name, dynamic } = getSlotName(slotBinding)
660679
const slotContainer = slots[name] = createASTElement('template', [], el)
680+
slotContainer.slotTarget = name
661681
slotContainer.slotTargetDynamic = dynamic
662-
slotContainer.children = el.children
682+
slotContainer.children = el.children.filter(c => !c.slotScope)
663683
slotContainer.slotScope = slotBinding.value || `_`
664684
// remove children as they are returned from scopedSlots now
665685
el.children = []

test/unit/features/component/component-scoped-slot.spec.js

+18
Original file line numberDiff line numberDiff line change
@@ -827,6 +827,24 @@ describe('Component scoped slot', () => {
827827
expect(vm.$el.innerHTML.replace(/\s+/g, ' ')).toMatch(`b from foo one a from foo two `)
828828
}).then(done)
829829
})
830+
831+
it('should work with v-if/v-else', done => {
832+
const vm = new Vue({
833+
data: { flag: true },
834+
template: `
835+
<foo>
836+
<template v-if="flag" v-slot:one="one">a {{ one }} </template>
837+
<template v-else v-slot:two="two">b {{ two }} </template>
838+
</foo>
839+
`,
840+
components: { Foo }
841+
}).$mount()
842+
expect(vm.$el.innerHTML).toBe(`a from foo one `)
843+
vm.flag = false
844+
waitForUpdate(() => {
845+
expect(vm.$el.innerHTML).toBe(`b from foo two `)
846+
}).then(done)
847+
})
830848
})
831849

832850
// 2.6 scoped slot perf optimization

0 commit comments

Comments
 (0)