Skip to content

Commit f219bed

Browse files
authored
perf: improve scoped slots change detection accuracy (#9371)
Ensure that state mutations that only affect parent scope only trigger parent update and does not affect child components with only scoped slots.
1 parent 770c6ed commit f219bed

File tree

8 files changed

+233
-167
lines changed

8 files changed

+233
-167
lines changed

flow/compiler.js

+1
Original file line numberDiff line numberDiff line change
@@ -119,6 +119,7 @@ declare type ASTElement = {
119119
transitionMode?: string | null;
120120
slotName?: ?string;
121121
slotTarget?: ?string;
122+
slotTargetDynamic?: boolean;
122123
slotScope?: ?string;
123124
scopedSlots?: { [name: string]: ASTElement };
124125

src/compiler/codegen/index.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -354,11 +354,12 @@ function genScopedSlots (
354354
slots: { [key: string]: ASTElement },
355355
state: CodegenState
356356
): string {
357+
const hasDynamicKeys = Object.keys(slots).some(key => slots[key].slotTargetDynamic)
357358
return `scopedSlots:_u([${
358359
Object.keys(slots).map(key => {
359360
return genScopedSlot(key, slots[key], state)
360361
}).join(',')
361-
}])`
362+
}]${hasDynamicKeys ? `,true` : ``})`
362363
}
363364

364365
function genScopedSlot (

src/compiler/parser/index.js

+11-7
Original file line numberDiff line numberDiff line change
@@ -586,6 +586,7 @@ function processSlotContent (el) {
586586
const slotTarget = getBindingAttr(el, 'slot')
587587
if (slotTarget) {
588588
el.slotTarget = slotTarget === '""' ? '"default"' : slotTarget
589+
el.slotTargetDynamic = !!(el.attrsMap[':slot'] || el.attrsMap['v-bind:slot'])
589590
// preserve slot as an attribute for native shadow DOM compat
590591
// only for non-scoped slots.
591592
if (el.tag !== 'template' && !el.slotScope) {
@@ -607,8 +608,10 @@ function processSlotContent (el) {
607608
)
608609
}
609610
}
610-
el.slotTarget = getSlotName(slotBinding)
611-
el.slotScope = slotBinding.value
611+
const { name, dynamic } = getSlotName(slotBinding)
612+
el.slotTarget = name
613+
el.slotTargetDynamic = dynamic
614+
el.slotScope = slotBinding.value || `_` // force it into a scoped slot for perf
612615
}
613616
} else {
614617
// v-slot on component, denotes default slot
@@ -637,10 +640,11 @@ function processSlotContent (el) {
637640
}
638641
// add the component's children to its default slot
639642
const slots = el.scopedSlots || (el.scopedSlots = {})
640-
const target = getSlotName(slotBinding)
641-
const slotContainer = slots[target] = createASTElement('template', [], el)
643+
const { name, dynamic } = getSlotName(slotBinding)
644+
const slotContainer = slots[name] = createASTElement('template', [], el)
645+
slotContainer.slotTargetDynamic = dynamic
642646
slotContainer.children = el.children
643-
slotContainer.slotScope = slotBinding.value
647+
slotContainer.slotScope = slotBinding.value || `_`
644648
// remove children as they are returned from scopedSlots now
645649
el.children = []
646650
// mark el non-plain so data gets generated
@@ -664,9 +668,9 @@ function getSlotName (binding) {
664668
}
665669
return dynamicKeyRE.test(name)
666670
// dynamic [name]
667-
? name.slice(1, -1)
671+
? { name: name.slice(1, -1), dynamic: true }
668672
// static name
669-
: `"${name}"`
673+
: { name: `"${name}"`, dynamic: false }
670674
}
671675

672676
// handle <slot/> outlets

src/core/instance/lifecycle.js

+15-5
Original file line numberDiff line numberDiff line change
@@ -224,12 +224,22 @@ export function updateChildComponent (
224224
}
225225

226226
// determine whether component has slot children
227-
// we need to do this before overwriting $options._renderChildren
228-
const hasChildren = !!(
227+
// we need to do this before overwriting $options._renderChildren.
228+
229+
// check if there are dynamic scopedSlots (hand-written or compiled but with
230+
// dynamic slot names). Static scoped slots compiled from template has the
231+
// "$stable" marker.
232+
const hasDynamicScopedSlot = !!(
233+
(parentVnode.data.scopedSlots && !parentVnode.data.scopedSlots.$stable) ||
234+
(vm.$scopedSlots !== emptyObject && !vm.$scopedSlots.$stable)
235+
)
236+
// Any static slot children from the parent may have changed during parent's
237+
// update. Dynamic scoped slots may also have changed. In such cases, a forced
238+
// update is necessary to ensure correctness.
239+
const needsForceUpdate = !!(
229240
renderChildren || // has new static slots
230241
vm.$options._renderChildren || // has old static slots
231-
parentVnode.data.scopedSlots || // has new scoped slots
232-
vm.$scopedSlots !== emptyObject // has old scoped slots
242+
hasDynamicScopedSlot
233243
)
234244

235245
vm.$options._parentVnode = parentVnode
@@ -268,7 +278,7 @@ export function updateChildComponent (
268278
updateComponentListeners(vm, listeners, oldListeners)
269279

270280
// resolve slots + force update if has children
271-
if (hasChildren) {
281+
if (needsForceUpdate) {
272282
vm.$slots = resolveSlots(renderChildren, parentVnode.context)
273283
vm.$forceUpdate()
274284
}

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

+4-3
Original file line numberDiff line numberDiff line change
@@ -51,13 +51,14 @@ function isWhitespace (node: VNode): boolean {
5151

5252
export function resolveScopedSlots (
5353
fns: ScopedSlotsData, // see flow/vnode
54+
hasDynamicKeys?: boolean,
5455
res?: Object
55-
): { [key: string]: Function } {
56-
res = res || {}
56+
): { [key: string]: Function, $stable: boolean } {
57+
res = res || { $stable: !hasDynamicKeys }
5758
for (let i = 0; i < fns.length; i++) {
5859
const slot = fns[i]
5960
if (Array.isArray(slot)) {
60-
resolveScopedSlots(slot, res)
61+
resolveScopedSlots(slot, hasDynamicKeys, res)
6162
} else {
6263
res[slot.key] = slot.fn
6364
}

src/core/vdom/helpers/normalize-scoped-slots.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ export function normalizeScopedSlots (
1414
} else {
1515
res = {}
1616
for (const key in slots) {
17-
if (slots[key]) {
17+
if (slots[key] && key[0] !== '$') {
1818
res[key] = normalizeScopedSlot(slots[key])
1919
}
2020
}
@@ -26,6 +26,7 @@ export function normalizeScopedSlots (
2626
}
2727
}
2828
res._normalized = true
29+
res.$stable = slots && slots.$stable
2930
return res
3031
}
3132

0 commit comments

Comments
 (0)