Back to Airi

Async Component Best Practices

.agents/skills/vue-best-practices/references/component-async.md

0.10.12.2 KB
Original Source

Async Component Best Practices

Impact: MEDIUM - Async components should reduce JavaScript cost without degrading perceived performance. Focus on hydration timing in SSR and stable loading UX.

Task List

  • Use lazy hydration strategies for non-critical SSR component trees
  • Import only the hydration helpers you actually use
  • Keep loadingComponent delay near the default 200ms unless real UX data suggests otherwise
  • Configure delay and timeout together for predictable loading behavior

Use Lazy Hydration Strategies in SSR

In Vue 3.5+, async components can delay hydration until idle time, visibility, media query match, or user interaction.

BAD:

vue
<script setup lang="ts">
import { defineAsyncComponent } from 'vue'

const AsyncComments = defineAsyncComponent({
  loader: () => import('./Comments.vue')
})
</script>

GOOD:

vue
<script setup lang="ts">
import {
  defineAsyncComponent,
  hydrateOnIdle,
  hydrateOnVisible
} from 'vue'

const AsyncComments = defineAsyncComponent({
  loader: () => import('./Comments.vue'),
  hydrate: hydrateOnVisible({ rootMargin: '100px' })
})

const AsyncFooter = defineAsyncComponent({
  loader: () => import('./Footer.vue'),
  hydrate: hydrateOnIdle(5000)
})
</script>

Prevent Loading Spinner Flicker

Avoid showing loading UI immediately for components that usually resolve quickly.

BAD:

vue
<script setup lang="ts">
import { defineAsyncComponent } from 'vue'

import LoadingSpinner from './LoadingSpinner.vue'

const AsyncDashboard = defineAsyncComponent({
  loader: () => import('./Dashboard.vue'),
  loadingComponent: LoadingSpinner,
  delay: 0
})
</script>

GOOD:

vue
<script setup lang="ts">
import { defineAsyncComponent } from 'vue'

import ErrorDisplay from './ErrorDisplay.vue'
import LoadingSpinner from './LoadingSpinner.vue'

const AsyncDashboard = defineAsyncComponent({
  loader: () => import('./Dashboard.vue'),
  loadingComponent: LoadingSpinner,
  errorComponent: ErrorDisplay,
  delay: 200,
  timeout: 30000
})
</script>

Delay Guidelines

ScenarioRecommended Delay
Small component, fast network200ms
Known heavy component100ms
Background or non-critical UI300-500ms