Skip to content

Commit

Permalink
fix(keep-alive): fix keep-alive with scopeId/fallthrough attrs
Browse files Browse the repository at this point in the history
fix #1511
  • Loading branch information
yyx990803 committed Jul 6, 2020
1 parent 6dd59ee commit d86b01b
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 3 deletions.
29 changes: 28 additions & 1 deletion packages/runtime-core/__tests__/components/KeepAlive.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,8 @@ import {
ComponentPublicInstance,
Ref,
cloneVNode,
provide
provide,
withScopeId
} from '@vue/runtime-test'
import { KeepAliveProps } from '../../src/components/KeepAlive'

Expand Down Expand Up @@ -655,4 +656,30 @@ describe('KeepAlive', () => {
expect(spyMounted).toHaveBeenCalledTimes(3)
expect(spyUnmounted).toHaveBeenCalledTimes(4)
})

// #1513
test('should work with cloned root due to scopeId / fallthrough attrs', async () => {
const viewRef = ref('one')
const instanceRef = ref<any>(null)
const withId = withScopeId('foo')
const App = {
__scopeId: 'foo',
render: withId(() => {
return h(KeepAlive, null, {
default: () => h(views[viewRef.value], { ref: instanceRef })
})
})
}
render(h(App), root)
expect(serializeInner(root)).toBe(`<div foo>one</div>`)
instanceRef.value.msg = 'changed'
await nextTick()
expect(serializeInner(root)).toBe(`<div foo>changed</div>`)
viewRef.value = 'two'
await nextTick()
expect(serializeInner(root)).toBe(`<div foo>two</div>`)
viewRef.value = 'one'
await nextTick()
expect(serializeInner(root)).toBe(`<div foo>changed</div>`)
})
})
27 changes: 25 additions & 2 deletions packages/runtime-core/src/components/KeepAlive.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,13 @@ import {
} from '../component'
import { VNode, cloneVNode, isVNode, VNodeProps } from '../vnode'
import { warn } from '../warning'
import { onBeforeUnmount, injectHook, onUnmounted } from '../apiLifecycle'
import {
onBeforeUnmount,
injectHook,
onUnmounted,
onBeforeMount,
onBeforeUpdate
} from '../apiLifecycle'
import {
isString,
isArray,
Expand Down Expand Up @@ -173,6 +179,16 @@ const KeepAliveImpl = {
}
)

// cache sub tree in beforeMount/Update (i.e. right after the render)
let pendingCacheKey: CacheKey | null = null
const cacheSubtree = () => {
if (pendingCacheKey) {
cache.set(pendingCacheKey, instance.subTree)
}
}
onBeforeMount(cacheSubtree)
onBeforeUpdate(cacheSubtree)

onBeforeUnmount(() => {
cache.forEach(cached => {
const { subTree, suspense } = instance
Expand All @@ -189,6 +205,8 @@ const KeepAliveImpl = {
})

return () => {
pendingCacheKey = null

if (!slots.default) {
return null
}
Expand Down Expand Up @@ -227,7 +245,12 @@ const KeepAliveImpl = {
if (vnode.el) {
vnode = cloneVNode(vnode)
}
cache.set(key, vnode)
// #1513 it's possible for the returned vnode to be cloned due to attr
// fallthrough or scopeId, so the vnode here may not be the final vnode
// that is mounted. Instead of caching it directly, we store the pending
// key and cache `instance.subTree` (the normalized vnode) in
// beforeMount/beforeUpdate hooks.
pendingCacheKey = key

if (cachedVNode) {
// copy over mounted state
Expand Down

0 comments on commit d86b01b

Please sign in to comment.