Skip to content

Commit c7c13c2

Browse files
committed
fix(scoped-slots): ensure $scopedSlots calls always return Arrays
Also allow render functions to return an Array of a single element. Close #8056
1 parent d747469 commit c7c13c2

File tree

6 files changed

+67
-15
lines changed

6 files changed

+67
-15
lines changed

src/core/instance/render-helpers/resolve-slots.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -55,10 +55,11 @@ export function resolveScopedSlots (
5555
): { [key: string]: Function } {
5656
res = res || {}
5757
for (let i = 0; i < fns.length; i++) {
58-
if (Array.isArray(fns[i])) {
59-
resolveScopedSlots(fns[i], res)
58+
const slot = fns[i]
59+
if (Array.isArray(slot)) {
60+
resolveScopedSlots(slot, res)
6061
} else {
61-
res[fns[i].key] = fns[i].fn
62+
res[slot.key] = slot.fn
6263
}
6364
}
6465
return res

src/core/instance/render.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import {
1111
import { createElement } from '../vdom/create-element'
1212
import { installRenderHelpers } from './render-helpers/index'
1313
import { resolveSlots } from './render-helpers/resolve-slots'
14+
import { normalizeScopedSlots } from '../vdom/helpers/normalize-scoped-slots'
1415
import VNode, { createEmptyVNode } from '../vdom/vnode'
1516

1617
import { isUpdatingChildComponent } from './lifecycle'
@@ -63,7 +64,7 @@ export function renderMixin (Vue: Class<Component>) {
6364
const { render, _parentVnode } = vm.$options
6465

6566
if (_parentVnode) {
66-
vm.$scopedSlots = _parentVnode.data.scopedSlots || emptyObject
67+
vm.$scopedSlots = normalizeScopedSlots(_parentVnode.data.scopedSlots)
6768
}
6869

6970
// set parent vnode. this allows render functions to have access
@@ -89,6 +90,10 @@ export function renderMixin (Vue: Class<Component>) {
8990
vnode = vm._vnode
9091
}
9192
}
93+
// if the returned array contains only a single node, allow it
94+
if (Array.isArray(vnode) && vnode.length === 1) {
95+
vnode = vnode[0]
96+
}
9297
// return empty vnode in case the render function errored out
9398
if (!(vnode instanceof VNode)) {
9499
if (process.env.NODE_ENV !== 'production' && Array.isArray(vnode)) {

src/core/vdom/create-functional-component.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import { createElement } from './create-element'
55
import { resolveInject } from '../instance/inject'
66
import { normalizeChildren } from '../vdom/helpers/normalize-children'
77
import { resolveSlots } from '../instance/render-helpers/resolve-slots'
8+
import { normalizeScopedSlots } from '../vdom/helpers/normalize-scoped-slots'
89
import { installRenderHelpers } from '../instance/render-helpers/index'
910

1011
import {
@@ -56,7 +57,7 @@ export function FunctionalRenderContext (
5657
this.$options = options
5758
// pre-resolve slots for renderSlot()
5859
this.$slots = this.slots()
59-
this.$scopedSlots = data.scopedSlots || emptyObject
60+
this.$scopedSlots = normalizeScopedSlots(data.scopedSlots)
6061
}
6162

6263
if (options._scopeId) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
/* @flow */
2+
3+
import { emptyObject } from 'core/util/index'
4+
5+
export function normalizeScopedSlots (slots: { [key: string]: Function }) {
6+
if (!slots) {
7+
return emptyObject
8+
} else if (slots._normalized) {
9+
return slots
10+
} else {
11+
const res = {}
12+
for (const key in slots) {
13+
res[key] = normalizeScopedSlot(slots[key])
14+
}
15+
res._normalized = true
16+
return res
17+
}
18+
}
19+
20+
function normalizeScopedSlot(fn: Function) {
21+
return scope => {
22+
const res = fn(scope)
23+
return Array.isArray(res) ? res : res ? [res] : res
24+
}
25+
}

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

+28-8
Original file line numberDiff line numberDiff line change
@@ -395,11 +395,9 @@ describe('Component scoped slot', () => {
395395
return { msg: 'hello' }
396396
},
397397
render (h) {
398-
return h('div', [
399-
this.$scopedSlots.item({
400-
text: this.msg
401-
})
402-
])
398+
return h('div', this.$scopedSlots.item({
399+
text: this.msg
400+
}))
403401
}
404402
}
405403
}
@@ -425,16 +423,38 @@ describe('Component scoped slot', () => {
425423
return { msg: 'hello' }
426424
},
427425
render (h) {
428-
return h('div', [
429-
this.$scopedSlots.default({ msg: this.msg })
430-
])
426+
return h('div', this.$scopedSlots.default({ msg: this.msg }))
431427
}
432428
}
433429
}
434430
}).$mount()
435431
expect(vm.$el.innerHTML).toBe('<span>hello</span>')
436432
})
437433

434+
it('render function usage (default, as root)', () => {
435+
const vm = new Vue({
436+
render (h) {
437+
return h('test', [
438+
props => h('span', [props.msg])
439+
])
440+
},
441+
components: {
442+
test: {
443+
data () {
444+
return { msg: 'hello' }
445+
},
446+
render (h) {
447+
const res = this.$scopedSlots.default({ msg: this.msg })
448+
// all scoped slots should be normalized into arrays
449+
expect(Array.isArray(res)).toBe(true)
450+
return res
451+
}
452+
}
453+
}
454+
}).$mount()
455+
expect(vm.$el.outerHTML).toBe('<span>hello</span>')
456+
})
457+
438458
// #4779
439459
it('should support dynamic slot target', done => {
440460
const Child = {

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

+2-2
Original file line numberDiff line numberDiff line change
@@ -327,11 +327,11 @@ describe('Component slot', () => {
327327

328328
it('warn if user directly returns array', () => {
329329
new Vue({
330-
template: '<test><div></div></test>',
330+
template: '<test><div slot="foo"></div><div slot="foo"></div></test>',
331331
components: {
332332
test: {
333333
render () {
334-
return this.$slots.default
334+
return this.$slots.foo
335335
}
336336
}
337337
}

0 commit comments

Comments
 (0)