Skip to content

Commit 604e081

Browse files
committed
fix: ensure functionalContext is cloned during slot clones
fix #7106
1 parent 3932a45 commit 604e081

File tree

6 files changed

+63
-15
lines changed

6 files changed

+63
-15
lines changed

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ export function resolveSlots (
2020
}
2121
// named slots should only be respected if the vnode was rendered in the
2222
// same context.
23-
if ((child.context === context || child.functionalContext === context) &&
23+
if ((child.context === context || child.fnContext === context) &&
2424
data && data.slot != null
2525
) {
2626
const name = child.data.slot

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

+4-4
Original file line numberDiff line numberDiff line change
@@ -49,8 +49,8 @@ function FunctionalRenderContext (
4949
this._c = (a, b, c, d) => {
5050
const vnode: ?VNode = createElement(contextVm, a, b, c, d, needNormalization)
5151
if (vnode) {
52-
vnode.functionalScopeId = options._scopeId
53-
vnode.functionalContext = parent
52+
vnode.fnScopeId = options._scopeId
53+
vnode.fnContext = parent
5454
}
5555
return vnode
5656
}
@@ -91,8 +91,8 @@ export function createFunctionalComponent (
9191
const vnode = options.render.call(null, renderContext._c, renderContext)
9292

9393
if (vnode instanceof VNode) {
94-
vnode.functionalContext = contextVm
95-
vnode.functionalOptions = options
94+
vnode.fnContext = contextVm
95+
vnode.fnOptions = options
9696
if (data.slot) {
9797
(vnode.data || (vnode.data = {})).slot = data.slot
9898
}

src/core/vdom/patch.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -294,7 +294,7 @@ export function createPatchFunction (backend) {
294294
// of going through the normal attribute patching process.
295295
function setScope (vnode) {
296296
let i
297-
if (isDef(i = vnode.functionalScopeId)) {
297+
if (isDef(i = vnode.fnScopeId)) {
298298
nodeOps.setAttribute(vnode.elm, i, '')
299299
} else {
300300
let ancestor = vnode
@@ -308,7 +308,7 @@ export function createPatchFunction (backend) {
308308
// for slot content they should also get the scopeId from the host instance.
309309
if (isDef(i = activeInstance) &&
310310
i !== vnode.context &&
311-
i !== vnode.functionalContext &&
311+
i !== vnode.fnContext &&
312312
isDef(i = i.$options._scopeId)
313313
) {
314314
nodeOps.setAttribute(vnode.elm, i, '')

src/core/vdom/vnode.js

+9-6
Original file line numberDiff line numberDiff line change
@@ -24,9 +24,9 @@ export default class VNode {
2424
asyncMeta: Object | void;
2525
isAsyncPlaceholder: boolean;
2626
ssrContext: Object | void;
27-
functionalContext: Component | void; // real context vm for functional nodes
28-
functionalOptions: ?ComponentOptions; // for SSR caching
29-
functionalScopeId: ?string; // functioanl scope id support
27+
fnContext: Component | void; // real context vm for functional nodes
28+
fnOptions: ?ComponentOptions; // for SSR caching
29+
fnScopeId: ?string; // functioanl scope id support
3030

3131
constructor (
3232
tag?: string,
@@ -45,9 +45,9 @@ export default class VNode {
4545
this.elm = elm
4646
this.ns = undefined
4747
this.context = context
48-
this.functionalContext = undefined
49-
this.functionalOptions = undefined
50-
this.functionalScopeId = undefined
48+
this.fnContext = undefined
49+
this.fnOptions = undefined
50+
this.fnScopeId = undefined
5151
this.key = data && data.key
5252
this.componentOptions = componentOptions
5353
this.componentInstance = undefined
@@ -101,6 +101,9 @@ export function cloneVNode (vnode: VNode, deep?: boolean): VNode {
101101
cloned.isStatic = vnode.isStatic
102102
cloned.key = vnode.key
103103
cloned.isComment = vnode.isComment
104+
cloned.fnContext = vnode.fnContext
105+
cloned.fnOptions = vnode.fnOptions
106+
cloned.fnScopeId = vnode.fnScopeId
104107
cloned.isCloned = true
105108
if (deep) {
106109
if (vnode.children) {

src/server/render.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -252,8 +252,8 @@ function renderElement (el, isRoot, context) {
252252
el.data.attrs[SSR_ATTR] = 'true'
253253
}
254254

255-
if (el.functionalOptions) {
256-
registerComponentForCache(el.functionalOptions, write)
255+
if (el.fnOptions) {
256+
registerComponentForCache(el.fnOptions, write)
257257
}
258258

259259
const startTag = renderStartingTag(el, context)

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

+45
Original file line numberDiff line numberDiff line change
@@ -779,4 +779,49 @@ describe('Component slot', () => {
779779

780780
expect(vm.$el.innerHTML).toBe('<div class="foo"><div class="bar">fallback</div></div>')
781781
})
782+
783+
// #7106
784+
it('should not lose functional slot across renders', done => {
785+
const One = {
786+
data: () => ({
787+
foo: true
788+
}),
789+
render (h) {
790+
this.foo
791+
return h('div', this.$slots.slot)
792+
}
793+
}
794+
795+
const Two = {
796+
render (h) {
797+
return h('span', this.$slots.slot)
798+
}
799+
}
800+
801+
const Three = {
802+
functional: true,
803+
render: (h, { children }) => h('span', children)
804+
}
805+
806+
const vm = new Vue({
807+
template: `
808+
<div>
809+
<one ref="one">
810+
<two slot="slot">
811+
<three slot="slot">hello</three>
812+
</two>
813+
</one>
814+
</div>
815+
`,
816+
components: { One, Two, Three }
817+
}).$mount()
818+
819+
expect(vm.$el.textContent).toBe('hello')
820+
// trigger re-render of <one>
821+
vm.$refs.one.foo = false
822+
waitForUpdate(() => {
823+
// should still be there
824+
expect(vm.$el.textContent).toBe('hello')
825+
}).then(done)
826+
})
782827
})

0 commit comments

Comments
 (0)