@@ -67,7 +67,7 @@ export const transformElement: NodeTransform = (node, context) => {
67
67
// updates inside get proper isSVG flag at runtime. (#639, #643)
68
68
// This is technically web-specific, but splitting the logic out of core
69
69
// leads to too much unnecessary complexity.
70
- const shouldUseBlock =
70
+ let shouldUseBlock =
71
71
! isComponent && ( tag === 'svg' || tag === 'foreignObject' )
72
72
73
73
const nodeType = isComponent
@@ -101,21 +101,35 @@ export const transformElement: NodeTransform = (node, context) => {
101
101
args . push ( `null` )
102
102
}
103
103
104
- if ( __DEV__ && nodeType === KEEP_ALIVE && node . children . length > 1 ) {
105
- context . onError (
106
- createCompilerError ( ErrorCodes . X_KEEP_ALIVE_INVALID_CHILDREN , {
107
- start : node . children [ 0 ] . loc . start ,
108
- end : node . children [ node . children . length - 1 ] . loc . end ,
109
- source : ''
110
- } )
111
- )
104
+ if ( nodeType === KEEP_ALIVE ) {
105
+ // Although a built-in component, we compile KeepAlive with raw children
106
+ // instead of slot functions so that it can be used inside Transition
107
+ // or other Transition-wrapping HOCs.
108
+ // To ensure correct updates with block optimizations, we need to:
109
+ // 1. Force keep-alive into a block. This avoids its children being
110
+ // collected by a parent block.
111
+ shouldUseBlock = true
112
+ // 2. Force keep-alive to always be updated, since it uses raw children.
113
+ patchFlag |= PatchFlags . DYNAMIC_SLOTS
114
+ if ( __DEV__ && node . children . length > 1 ) {
115
+ context . onError (
116
+ createCompilerError ( ErrorCodes . X_KEEP_ALIVE_INVALID_CHILDREN , {
117
+ start : node . children [ 0 ] . loc . start ,
118
+ end : node . children [ node . children . length - 1 ] . loc . end ,
119
+ source : ''
120
+ } )
121
+ )
122
+ }
112
123
}
113
124
114
- // Portal & KeepAlive should have normal children instead of slots
115
- // Portal is not a real component has dedicated handling in the renderer
116
- // KeepAlive should not track its own deps so that it can be used inside
117
- // Transition
118
- if ( isComponent && nodeType !== PORTAL && nodeType !== KEEP_ALIVE ) {
125
+ const shouldBuildAsSlots =
126
+ isComponent &&
127
+ // Portal is not a real component has dedicated handling in the renderer
128
+ nodeType !== PORTAL &&
129
+ // explained above.
130
+ nodeType !== KEEP_ALIVE
131
+
132
+ if ( shouldBuildAsSlots ) {
119
133
const { slots, hasDynamicSlots } = buildSlots ( node , context )
120
134
args . push ( slots )
121
135
if ( hasDynamicSlots ) {
0 commit comments