Files
agent-skills/skills/vue-best-practices/reference/prop-destructured-watch-getter.md
Jason Woltje f5792c40be feat: Complete fleet — 94 skills across 10+ domains
Pulled ALL skills from 15 source repositories:
- anthropics/skills: 16 (docs, design, MCP, testing)
- obra/superpowers: 14 (TDD, debugging, agents, planning)
- coreyhaines31/marketingskills: 25 (marketing, CRO, SEO, growth)
- better-auth/skills: 5 (auth patterns)
- vercel-labs/agent-skills: 5 (React, design, Vercel)
- antfu/skills: 16 (Vue, Vite, Vitest, pnpm, Turborepo)
- Plus 13 individual skills from various repos

Mosaic Stack is not limited to coding — the Orchestrator and
subagents serve coding, business, design, marketing, writing,
logistics, analysis, and more.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-16 16:27:42 -06:00

139 lines
3.9 KiB
Markdown

---
title: Wrap Destructured Props in Getters for watch and Composables
impact: MEDIUM
impactDescription: Passing destructured prop values directly to watch or composables loses reactivity
type: gotcha
tags: [vue3, props, reactivity, watch, composables, destructuring]
---
# Wrap Destructured Props in Getters for watch and Composables
**Impact: MEDIUM** - When you destructure props with `defineProps`, the destructured variables are reactive in templates but passing them directly to `watch()` or external composables will pass the current value, not a reactive source. The watcher or composable won't track future changes.
Vue 3.5+ automatically transforms destructured props for template reactivity, but external functions still need getter wrappers.
## Task Checklist
- [ ] Wrap destructured props in arrow functions when passing to `watch()`
- [ ] Use getter functions when passing destructured props to composables
- [ ] Verify composables use `toValue()` to normalize getter/ref inputs
- [ ] Consider using `props.propertyName` directly if getter syntax feels awkward
**Incorrect:**
```vue
<script setup>
import { watch } from 'vue'
import { useDebounce } from './composables'
const { searchQuery, userId } = defineProps(['searchQuery', 'userId'])
// WRONG: Passing value, not reactive source
// This captures the initial value only - changes won't trigger the watcher
watch(searchQuery, (newValue) => {
console.log('Query changed:', newValue) // Never fires after initial!
})
// WRONG: Composable receives static value
const debouncedQuery = useDebounce(searchQuery, 300) // Won't update
</script>
```
**Correct:**
```vue
<script setup>
import { watch } from 'vue'
import { useDebounce } from './composables'
const { searchQuery, userId } = defineProps(['searchQuery', 'userId'])
// CORRECT: Wrap in getter function to maintain reactivity
watch(() => searchQuery, (newValue) => {
console.log('Query changed:', newValue) // Fires on every change
})
// CORRECT: Pass getter to composable
const debouncedQuery = useDebounce(() => searchQuery, 300)
</script>
```
## Alternative: Use props Object Directly
```vue
<script setup>
import { watch } from 'vue'
const props = defineProps(['searchQuery', 'userId'])
// Also correct: Watch via props object (no getter needed)
watch(() => props.searchQuery, (newValue) => {
console.log('Query changed:', newValue)
})
// Watch multiple props
watch(
() => [props.searchQuery, props.userId],
([newQuery, newUserId]) => {
console.log('Props changed:', newQuery, newUserId)
}
)
</script>
```
## Writing Composables That Accept Props
When creating composables that should work with destructured props:
```javascript
// composables/useDebounce.js
import { ref, watch, toValue } from 'vue'
export function useDebounce(source, delay = 300) {
const debounced = ref(toValue(source)) // toValue handles both getter and ref
let timeout
watch(
// toValue normalizes getter functions and refs
() => toValue(source),
(newValue) => {
clearTimeout(timeout)
timeout = setTimeout(() => {
debounced.value = newValue
}, delay)
}
)
return debounced
}
```
```vue
<!-- Usage -->
<script setup>
const { query } = defineProps(['query'])
// Works with getter
const debouncedQuery = useDebounce(() => query, 300)
</script>
```
## Vue 3.5+ Reactive Destructuring
Vue 3.5+ added reactive props destructuring. The compiler transforms:
```javascript
const { foo } = defineProps(['foo'])
```
Into something like:
```javascript
const __props = defineProps(['foo'])
// foo accesses __props.foo reactively in templates
```
However, external function calls still need getters because JavaScript itself passes values, not references.
## Reference
- [Vue.js Props - Reactive Props Destructure](https://vuejs.org/guide/components/props.html#reactive-props-destructure)
- [Vue.js Watchers - Watch Source Types](https://vuejs.org/guide/essentials/watchers.html#watch-source-types)