AgentSkillsCN

vue-best-practices

Vue 3与Vue.js的最佳实践:针对TypeScript、vue-tsc、Volar以及组件开发模式。适用于使用TypeScript编写、审查或重构Vue 3组件的场景,包括配置Volar/vueCompilerOptions、提取组件类型、运用defineModel/withDefaults、搭建Pinia Store测试环境,或调试Vue工具链相关问题。可触发的场景包括:Vue组件、属性提取、包装组件、模板类型检查、strictTemplates、vueCompilerOptions、Volar 3、CSS模块、透传属性、defineModel、withDefaults、深度监听、vue-router带类型参数、Pinia模拟、HMR SSR、moduleResolution打包器、useTemplateRef、onWatcherCleanup、useId、泛型组件、响应式属性解构。

SKILL.md
--- frontmatter
name: vue-best-practices
description: "Vue 3 and Vue.js best practices for TypeScript, vue-tsc, Volar, and component patterns. Use when writing, reviewing, or refactoring Vue 3 components with TypeScript, configuring Volar/vueCompilerOptions, extracting component types, working with defineModel/withDefaults, setting up Pinia store tests, or debugging Vue tooling issues. Triggers on Vue components, props extraction, wrapper components, template type checking, strictTemplates, vueCompilerOptions, Volar 3, CSS modules, fallthrough attributes, defineModel, withDefaults, deep watch, vue-router typed params, Pinia mocking, HMR SSR, moduleResolution bundler, useTemplateRef, onWatcherCleanup, useId, generic components, reactive props destructure."
license: MIT
metadata:
  author: ejirocodes
  version: '1.0.0'

Vue 3 Best Practices

Quick Reference

TopicWhen to UseReference
TypeScriptProps extraction, generic components, useTemplateRef, JSDoc, reactive props destructuretypescript.md
VolarIDE config, strictTemplates, CSS modules, directive comments, Volar 3.0 migrationvolar.md
ComponentsdefineModel, deep watch, onWatcherCleanup, useId, deferred teleportcomponents.md
ToolingmoduleResolution, HMR SSR, duplicate plugin detectiontooling.md
TestingPinia store mocking, setup stores, Vue Router typed paramstesting.md

Essential Patterns

Extract Component Props

typescript
import type { ComponentProps } from 'vue-component-type-helpers'
import MyButton from './MyButton.vue'

type Props = ComponentProps<typeof MyButton>

Reactive Props Destructure (Vue 3.5+)

vue
<script setup lang="ts">
// Destructured props are reactive - preferred in Vue 3.5+
const { name, count = 0 } = defineProps<{ name: string; count?: number }>()
</script>

useTemplateRef (Vue 3.5+)

vue
<script setup lang="ts">
import { useTemplateRef, onMounted } from 'vue'

const inputRef = useTemplateRef('input')  // Auto-typed
onMounted(() => inputRef.value?.focus())
</script>
<template><input ref="input" /></template>

onWatcherCleanup (Vue 3.5+)

typescript
import { watch, onWatcherCleanup } from 'vue'

watch(query, async (q) => {
  const controller = new AbortController()
  onWatcherCleanup(() => controller.abort())
  await fetch(`/api?q=${q}`, { signal: controller.signal })
})

defineModel with Required

typescript
// Returns Ref<Item> instead of Ref<Item | undefined>
const model = defineModel<Item>({ required: true })

Deep Watch with Numeric Depth

typescript
// Vue 3.5+ - watch array mutations without full traversal
watch(items, handler, { deep: 1 })

Pinia Store Test Setup

typescript
import { createTestingPinia } from '@pinia/testing'
import { vi } from 'vitest'

mount(Component, {
  global: {
    plugins: [createTestingPinia({ createSpy: vi.fn })]
  }
})

Common Mistakes

  1. Using InstanceType<typeof Component>['$props'] - Use ComponentProps instead
  2. Missing createSpy in createTestingPinia - Required in @pinia/testing 1.0+
  3. Using withDefaults with union types - Use Reactive Props Destructure
  4. strictTemplates in wrong tsconfig - Add to tsconfig.app.json, not root
  5. ts_ls with Volar 3.0 - Use vtsls instead (Neovim)
  6. deep: true on large structures - Use numeric depth for performance
  7. Watching destructured props directly - Wrap in getter: watch(() => count, ...)
  8. Random IDs in SSR - Use useId() for hydration-safe IDs