.agents/skills/vue-best-practices/references/state-management.md
Impact: HIGH - Use the lightest state solution that fits your app architecture. SPA-only apps can use lightweight global composables, while SSR/Nuxt apps should default to Pinia for request-safe isolation and predictable tooling.
createGlobalState: Small non-SSR apps needing shared app state.BAD:
// store/cart.ts
import { reactive } from 'vue'
export const cart = reactive({
items: [] as Array<{ id: string, qty: number }>
})
GOOD:
// composables/useCartStore.ts
import { reactive, readonly } from 'vue'
let _store: ReturnType<typeof createCartStore> | null = null
function createCartStore() {
const state = reactive({
items: [] as Array<{ id: string, qty: number }>
})
function addItem(id: string, qty = 1) {
const existing = state.items.find(item => item.id === id)
if (existing) {
existing.qty += qty
return
}
state.items.push({ id, qty })
}
return {
state: readonly(state),
addItem
}
}
export function useCartStore() {
if (!_store)
_store = createCartStore()
return _store
}
Module singletons live for the runtime lifetime. In SSR this can leak state between requests.
BAD:
// shared singleton reused across requests
const cartStore = useCartStore()
export function useServerCart() {
return cartStore
}
GOOD:
piniadependency required.
// stores/cart.ts
import { defineStore } from 'pinia'
export const useCartStore = defineStore('cart', {
state: () => ({
items: [] as Array<{ id: string, qty: number }>
}),
actions: {
addItem(id: string, qty = 1) {
const existing = this.items.find(item => item.id === id)
if (existing) {
existing.qty += qty
return
}
this.items.push({ id, qty })
}
}
})
createGlobalState for Small SPA Global State
@vueuse/coredependency required.
If the app is non-SSR and already uses VueUse, createGlobalState removes singleton boilerplate.
import { createGlobalState } from '@vueuse/core'
import { computed, ref } from 'vue'
export const useAuthState = createGlobalState(() => {
const token = ref<string | null>(null)
const isAuthenticated = computed(() => token.value !== null)
function setToken(next: string | null) {
token.value = next
}
return {
token,
isAuthenticated,
setToken
}
})