Skip to content

Commit 050bb33

Browse files
committed
feat: scoped CSS support for functional components
1 parent ea0d227 commit 050bb33

File tree

5 files changed

+69
-16
lines changed

5 files changed

+69
-16
lines changed

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

+29-7
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import { installRenderHelpers } from '../instance/render-helpers/index'
88

99
import {
1010
isDef,
11+
isTrue,
1112
camelize,
1213
emptyObject,
1314
validateProp
@@ -28,14 +29,35 @@ function FunctionalRenderContext (
2829
this.listeners = data.on || emptyObject
2930
this.injections = resolveInject(options.inject, parent)
3031
this.slots = () => resolveSlots(children, parent)
32+
33+
// ensure the createElement function in functional components
34+
// gets a unique context - this is necessary for correct named slot check
35+
const contextVm = Object.create(parent)
36+
const isCompiled = isTrue(options._compiled)
37+
const needNormalization = !isCompiled
38+
3139
// support for compiled functional template
32-
if (options._compiled) {
40+
if (isCompiled) {
41+
// exposing constructor and $options for renderStatic() because it needs
42+
// to cache the rendered trees on shared options
3343
this.constructor = Ctor
3444
this.$options = options
35-
this._c = parent._c
45+
// pre-resolve slots for renderSlot()
3646
this.$slots = this.slots()
3747
this.$scopedSlots = data.scopedSlots || emptyObject
3848
}
49+
50+
if (options._scopeId) {
51+
this._c = (a, b, c, d) => {
52+
const vnode: ?VNode = createElement(contextVm, a, b, c, d, needNormalization)
53+
if (vnode) {
54+
vnode.fnScopeId = options._scopeId
55+
}
56+
return vnode
57+
}
58+
} else {
59+
this._c = (a, b, c, d) => createElement(contextVm, a, b, c, d, needNormalization)
60+
}
3961
}
4062

4163
installRenderHelpers(FunctionalRenderContext.prototype)
@@ -58,25 +80,25 @@ export function createFunctionalComponent (
5880
if (isDef(data.attrs)) mergeProps(props, data.attrs)
5981
if (isDef(data.props)) mergeProps(props, data.props)
6082
}
61-
// ensure the createElement function in functional components
62-
// gets a unique context - this is necessary for correct named slot check
63-
const _contextVm = Object.create(contextVm)
64-
const h = (a, b, c, d) => createElement(_contextVm, a, b, c, d, true)
83+
6584
const renderContext = new FunctionalRenderContext(
6685
data,
6786
props,
6887
children,
6988
contextVm,
7089
Ctor
7190
)
72-
const vnode = options.render.call(null, h, renderContext)
91+
92+
const vnode = options.render.call(null, renderContext._c, renderContext)
93+
7394
if (vnode instanceof VNode) {
7495
vnode.functionalContext = contextVm
7596
vnode.functionalOptions = options
7697
if (data.slot) {
7798
(vnode.data || (vnode.data = {})).slot = data.slot
7899
}
79100
}
101+
80102
return vnode
81103
}
82104

src/core/vdom/patch.js

+10-5
Original file line numberDiff line numberDiff line change
@@ -281,16 +281,21 @@ export function createPatchFunction (backend) {
281281
// of going through the normal attribute patching process.
282282
function setScope (vnode) {
283283
let i
284-
let ancestor = vnode
285-
while (ancestor) {
286-
if (isDef(i = ancestor.context) && isDef(i = i.$options._scopeId)) {
287-
nodeOps.setAttribute(vnode.elm, i, '')
284+
if (isDef(i = vnode.fnScopeId)) {
285+
nodeOps.setAttribute(vnode.elm, i, '')
286+
} else {
287+
let ancestor = vnode
288+
while (ancestor) {
289+
if (isDef(i = ancestor.context) && isDef(i = i.$options._scopeId)) {
290+
nodeOps.setAttribute(vnode.elm, i, '')
291+
}
292+
ancestor = ancestor.parent
288293
}
289-
ancestor = ancestor.parent
290294
}
291295
// for slot content they should also get the scopeId from the host instance.
292296
if (isDef(i = activeInstance) &&
293297
i !== vnode.context &&
298+
i !== vnode.functionalContext &&
294299
isDef(i = i.$options._scopeId)
295300
) {
296301
nodeOps.setAttribute(vnode.elm, i, '')

src/core/vdom/vnode.js

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ export default class VNode {
2323
asyncMeta: Object | void;
2424
isAsyncPlaceholder: boolean;
2525
ssrContext: Object | void;
26+
fnScopeId: ?string;
2627

2728
constructor (
2829
tag?: string,

src/server/render.js

+8-4
Original file line numberDiff line numberDiff line change
@@ -342,11 +342,15 @@ function renderStartingTag (node: VNode, context) {
342342
) {
343343
markup += ` ${(scopeId: any)}`
344344
}
345-
while (isDef(node)) {
346-
if (isDef(scopeId = node.context.$options._scopeId)) {
347-
markup += ` ${scopeId}`
345+
if (isDef(node.fnScopeId)) {
346+
markup += ` ${node.fnScopeId}`
347+
} else {
348+
while (isDef(node)) {
349+
if (isDef(scopeId = node.context.$options._scopeId)) {
350+
markup += ` ${scopeId}`
351+
}
352+
node = node.parent
348353
}
349-
node = node.parent
350354
}
351355
return markup + '>'
352356
}

test/unit/features/options/_scopeId.spec.js

+21
Original file line numberDiff line numberDiff line change
@@ -68,4 +68,25 @@ describe('Options _scopeId', () => {
6868
expect(child.$el.hasAttribute('data-2')).toBe(true)
6969
}).then(done)
7070
})
71+
72+
it('should work on functional components', () => {
73+
const child = {
74+
functional: true,
75+
_scopeId: 'child',
76+
render (h) {
77+
return h('div', { class: 'child' }, 'child')
78+
}
79+
}
80+
const vm = new Vue({
81+
_scopeId: 'parent',
82+
components: { child },
83+
template: '<div><child></child></div>'
84+
}).$mount()
85+
86+
expect(vm.$el.hasAttribute('parent')).toBe(true)
87+
const childEl = vm.$el.querySelector('.child')
88+
expect(childEl.hasAttribute('child')).toBe(true)
89+
// functional component with scopeId will not inherit parent scopeId
90+
expect(childEl.hasAttribute('parent')).toBe(false)
91+
})
7192
})

0 commit comments

Comments
 (0)