Async Components with keep-alive Ref Issues
Rule
When using <keep-alive>, <component>, and defineAsyncComponent together, be aware that template refs can become undefined when the component is re-activated after being deactivated.
Why This Matters
This is a known Vue issue where the ref binding works correctly on first activation but becomes undefined on subsequent activations. This can cause runtime errors when trying to access component methods or properties through refs.
Problem Scenario
<script setup>
import { ref, defineAsyncComponent } from 'vue'
const AsyncWidget = defineAsyncComponent(() =>
import('./Widget.vue')
)
const currentComponent = ref(AsyncWidget)
const widgetRef = ref(null)
function callWidgetMethod() {
// May be undefined after component reactivation!
widgetRef.value?.doSomething()
}
</script>
<template>
<keep-alive>
<component :is="currentComponent" ref="widgetRef" />
</keep-alive>
</template>Workarounds
Option 1: Use onActivated to re-establish ref access
<script setup>
import { ref, defineAsyncComponent, onActivated, nextTick } from 'vue'
const AsyncWidget = defineAsyncComponent(() =>
import('./Widget.vue')
)
const currentComponent = ref(AsyncWidget)
const widgetRef = ref(null)
// Use a computed or method that waits for ref to be available
async function callWidgetMethod() {
await nextTick()
if (widgetRef.value) {
widgetRef.value.doSomething()
}
}
</script>Option 2: Avoid mixing all three patterns
If possible, use one of these alternatives:
<!-- Option A: Don't use keep-alive with async components -->
<template>
<component :is="currentComponent" ref="widgetRef" />
</template>
<!-- Option B: Use static component with keep-alive -->
<script setup>
import Widget from './Widget.vue' // Regular import
</script>
<template>
<keep-alive>
<component :is="Widget" ref="widgetRef" />
</keep-alive>
</template>Option 3: Use provide/inject instead of refs
<!-- Parent.vue -->
<script setup>
import { provide, ref } from 'vue'
const sharedState = ref({ /* shared data */ })
provide('widgetState', sharedState)
</script>
<!-- Widget.vue (async component) -->
<script setup>
import { inject } from 'vue'
const widgetState = inject('widgetState')
</script>Key Points
- This is a known issue when combining
<keep-alive>,<component :is>, anddefineAsyncComponent - Refs may become undefined after component deactivation/reactivation
- Use
nextTickand null checks when accessing refs - Consider alternative patterns like provide/inject for cross-component communication
- Test thoroughly when using this combination