Files
policy-ui/.claude/skills/vue/references/components.md
HaimKortovich a2eb1f3789 Add nuxt-skills and update auto quotes to use new policy API structure
- Add nuxt-skills (vue, nuxt, nuxt-ui) to .claude/skills/
- Create useCustomerSelection() composable for managing insured/buyer selection
- Create usePolicyApi() composable for policy API operations
- Update auto quote components to use insured/buyer instead of client
- Update vehicle fields: remove valorVehiculo, add market_value, requested_value, rc_limits
- Make chassis_number and engine_number optional
- Update auto quote types and composables to match new API structure
- Update auto quote page to submit to policy API with new structure
2026-04-27 14:56:53 -05:00

324 lines
7.5 KiB
Markdown

# Vue Components
Patterns for Vue 3 components using Composition API with `<script setup>`.
## Quick Reference
| Pattern | Syntax |
| --------------------- | --------------------------------------------------------------- |
| Props (destructured) | `const { name = 'default' } = defineProps<{ name?: string }>()` |
| Props (template-only) | `defineProps<{ name: string }>()` |
| Emits | `const emit = defineEmits<{ click: [id: number] }>()` |
| Two-way binding | `const model = defineModel<string>()` |
| Slots shorthand | `<template #header>` not `<template v-slot:header>` |
## Naming
**Files:** PascalCase (`UserProfile.vue`) OR kebab-case (`user-profile.vue`) - be consistent
**Component names in code:** Always PascalCase
**Composition:** General → Specific: `SearchButtonClear.vue` not `ClearSearchButton.vue`
## Props
**Destructure with defaults (Vue 3.5+)** when used in script or need defaults:
```ts
const { count = 0, message = 'Hello' } = defineProps<{
count?: number
message?: string
required: boolean
}>()
// Use directly - maintains reactivity
console.log(count + 1)
// ⚠️ When passing to watchers/functions, wrap in getter:
watch(() => count, (newVal) => { ... }) // ✅ Correct
watch(count, (newVal) => { ... }) // ❌ Won't work
```
**Non-destructured** only if props ONLY used in template:
```ts
defineProps<{ count: number }>()
// Template: {{ count }}
```
**Same-name shorthand (Vue 3.4+):** `:count` instead of `:count="count"`
```vue
<MyComponent :count :user :items />
<!-- Same as: :count="count" :user="user" :items="items" -->
```
[Reactive destructuring docs](https://vuejs.org/guide/components/props#reactive-props-destructure)
## Emits
Type-safe event definitions:
```ts
const emit = defineEmits<{
update: [id: number, value: string] // multiple args
close: [] // no args
}>()
// Usage
emit('update', 123, 'new value')
emit('close')
```
**Template syntax:** kebab-case (`@update-item`) vs camelCase in script (`updateItem`)
## Slots
**Always use shorthand:** `<template #header>` not `<template v-slot:header>`
**Always explicit `<template>` tags** for all slots
```vue
<template>
<Card>
<template #header>
<h2>Title</h2>
</template>
<template #default>
Content
</template>
</Card>
</template>
```
## defineModel() - Two-Way Binding
Replaces manual `modelValue` prop + `update:modelValue` emit.
### Basic
```vue
<script setup lang="ts">
const title = defineModel<string>()
</script>
<template>
<input v-model="title">
</template>
```
### With Options
```vue
<script setup lang="ts">
const [title, modifiers] = defineModel<string>({
default: 'default value',
required: true,
get: (value) => value.trim(),
set: (value) => {
if (modifiers.capitalize) {
return value.charAt(0).toUpperCase() + value.slice(1)
}
return value
},
})
</script>
```
**⚠️ Warning:** When using `default` without parent providing a value, parent and child can de-sync (parent `undefined`, child has default). Always provide matching defaults in parent or make prop required.
**Prevent double-emit with `required: true`:**
```ts
// ❌ Without required - emits twice (undefined then value)
const model = defineModel<Item>()
// ✅ With required - single emit
const model = defineModel<Item>({ required: true })
```
Use `required: true` when the model should always have a value to avoid the double-emit issue during initialization.
### Multiple Models
Default assumes `modelValue` prop. For multiple bindings, use explicit names:
```vue
<script setup lang="ts">
const firstName = defineModel<string>('firstName')
const age = defineModel<number>('age')
</script>
<!-- Usage -->
<UserForm v-model:first-name="user.firstName" v-model:age="user.age" />
```
[v-model modifiers docs](https://vuejs.org/guide/components/v-model#handling-v-model-modifiers)
## Reusable Templates
For typed, scoped template snippets within a component:
```vue
<script setup lang="ts">
import { createReusableTemplate } from '@vueuse/core'
const [DefineItem, UseItem] = createReusableTemplate<{
item: SearchItem
icon: string
color?: 'red' | 'green' | 'blue'
}>()
</script>
<template>
<DefineItem v-slot="{ item, icon, color }">
<div :class="color">
<Icon :name="icon" />
{{ item.name }}
</div>
</DefineItem>
<!-- Reuse multiple times -->
<UseItem v-for="item in items" :key="item.id" :item :icon="getIcon(item)" />
</template>
```
## Template Refs (Vue 3.5+)
Use `useTemplateRef()` for type-safe template references with IDE support:
```vue
<script setup lang="ts">
import { useTemplateRef, onMounted } from 'vue'
const input = useTemplateRef<HTMLInputElement>('my-input')
onMounted(() => {
input.value?.focus()
})
</script>
<template>
<input ref="my-input">
</template>
```
**Benefits over `ref()`:**
- Type-safe with generics
- Better IDE autocomplete and refactoring
- Explicit ref name as string literal
**Dynamic refs:**
```vue
<script setup lang="ts">
const items = ref(['a', 'b', 'c'])
const itemRefs = useTemplateRef<HTMLElement>('item')
// Access refs after mount
onMounted(() => {
console.log(itemRefs.value) // Array of elements
})
</script>
<template>
<div v-for="item in items" :key="item" ref="item">
{{ item }}
</div>
</template>
```
**Component refs with generics:**
For generic components, use `ComponentExposed` from `vue-component-type-helpers`:
```ts
import type { ComponentExposed } from 'vue-component-type-helpers'
import MyGenericComponent from './MyGenericComponent.vue'
// Get exposed methods/properties with correct generic types
const compRef = useTemplateRef<ComponentExposed<typeof MyGenericComponent>>('comp')
onMounted(() => {
compRef.value?.someExposedMethod() // Typed!
})
```
Install: `pnpm add -D vue-component-type-helpers`
## SSR Hydration (Vue 3.5+)
**Suppress hydration mismatches** for values that differ between server/client:
```vue
<template>
<!-- Client-side only values -->
<span data-allow-mismatch>{{ new Date().toLocaleString() }}</span>
<!-- Specific mismatch types -->
<span data-allow-mismatch="text">{{ timestamp }}</span>
<span data-allow-mismatch="children">
<ClientOnly>...</ClientOnly>
</span>
<span data-allow-mismatch="style">...</span>
<span data-allow-mismatch="class">...</span>
<span data-allow-mismatch="attribute">...</span>
</template>
```
**Generate SSR-stable IDs:**
```vue
<script setup lang="ts">
import { useId } from 'vue'
const id = useId() // Stable across server/client renders
</script>
<template>
<label :for="id">Name</label>
<input :id="id">
</template>
```
## Deferred Teleport (Vue 3.5+)
Teleport to elements rendered later in the same cycle:
```vue
<template>
<!-- This renders first -->
<Teleport defer to="#late-div">
<span>Deferred content</span>
</Teleport>
<!-- This renders after, but Teleport waits -->
<div id="late-div"></div>
</template>
```
Without `defer`, teleport to `#late-div` would fail since it doesn't exist yet.
## Common Mistakes
**Using `const props =` with destructured values:**
```ts
// ❌ Wrong
const props = defineProps<{ count: number }>()
const { count } = props // Loses reactivity
```
**Forgetting TypeScript types:**
```ts
// ❌ Wrong
const emit = defineEmits(['update'])
// ✅ Correct
const emit = defineEmits<{ update: [id: number] }>()
```
**Components >300 lines:** Split into smaller components or extract logic to composables