Skip to content

Commit ade07c6

Browse files
committed
fix(compiler-core): fix keep-alive when used in templates
fix #715
1 parent 6d10a6c commit ade07c6

File tree

2 files changed

+46
-24
lines changed

2 files changed

+46
-24
lines changed

packages/compiler-core/__tests__/transforms/transformElement.spec.ts

+18-10
Original file line numberDiff line numberDiff line change
@@ -346,18 +346,26 @@ describe('compiler: element transform', () => {
346346

347347
test('should handle <KeepAlive>', () => {
348348
function assert(tag: string) {
349-
const { root, node } = parseWithElementTransform(
350-
`<${tag}><span /></${tag}>`
351-
)
349+
const root = parse(`<div><${tag}><span /></${tag}></div>`)
350+
transform(root, {
351+
nodeTransforms: [transformElement, transformText]
352+
})
352353
expect(root.components.length).toBe(0)
353354
expect(root.helpers).toContain(KEEP_ALIVE)
354-
expect(node.callee).toBe(CREATE_VNODE)
355-
expect(node.arguments).toMatchObject([
356-
KEEP_ALIVE,
357-
`null`,
358-
// keep-alive should not compile content to slots
359-
[{ type: NodeTypes.ELEMENT, tag: 'span' }]
360-
])
355+
const node = (root.children[0] as any).children[0].codegenNode
356+
expect(node.type).toBe(NodeTypes.JS_SEQUENCE_EXPRESSION)
357+
expect(node.expressions[1]).toMatchObject({
358+
type: NodeTypes.JS_CALL_EXPRESSION,
359+
callee: CREATE_BLOCK, // should be forced into a block
360+
arguments: [
361+
KEEP_ALIVE,
362+
`null`,
363+
// keep-alive should not compile content to slots
364+
[{ type: NodeTypes.ELEMENT, tag: 'span' }],
365+
// should get a dynamic slots flag to force updates
366+
genFlagText(PatchFlags.DYNAMIC_SLOTS)
367+
]
368+
})
361369
}
362370

363371
assert(`keep-alive`)

packages/compiler-core/src/transforms/transformElement.ts

+28-14
Original file line numberDiff line numberDiff line change
@@ -67,7 +67,7 @@ export const transformElement: NodeTransform = (node, context) => {
6767
// updates inside get proper isSVG flag at runtime. (#639, #643)
6868
// This is technically web-specific, but splitting the logic out of core
6969
// leads to too much unnecessary complexity.
70-
const shouldUseBlock =
70+
let shouldUseBlock =
7171
!isComponent && (tag === 'svg' || tag === 'foreignObject')
7272

7373
const nodeType = isComponent
@@ -101,21 +101,35 @@ export const transformElement: NodeTransform = (node, context) => {
101101
args.push(`null`)
102102
}
103103

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+
}
112123
}
113124

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) {
119133
const { slots, hasDynamicSlots } = buildSlots(node, context)
120134
args.push(slots)
121135
if (hasDynamicSlots) {

0 commit comments

Comments
 (0)