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>
191 lines
4.8 KiB
Markdown
191 lines
4.8 KiB
Markdown
---
|
|
title: Avoid Expensive Operations in Updated Hook
|
|
impact: MEDIUM
|
|
impactDescription: Heavy computations in updated hook cause performance bottlenecks and potential infinite loops
|
|
type: capability
|
|
tags: [vue3, vue2, lifecycle, updated, performance, optimization, reactivity]
|
|
---
|
|
|
|
# Avoid Expensive Operations in Updated Hook
|
|
|
|
**Impact: MEDIUM** - The `updated` hook runs after every reactive state change that causes a re-render. Placing expensive operations, API calls, or state mutations here can cause severe performance degradation, infinite loops, and dropped frames below the optimal 60fps threshold.
|
|
|
|
Use `updated`/`onUpdated` sparingly for post-DOM-update operations that cannot be handled by watchers or computed properties. For most reactive data handling, prefer watchers (`watch`/`watchEffect`) which provide more control over what triggers the callback.
|
|
|
|
## Task Checklist
|
|
|
|
- [ ] Never perform API calls in updated hook
|
|
- [ ] Never mutate reactive state inside updated (causes infinite loops)
|
|
- [ ] Use conditional checks to verify updates are relevant before acting
|
|
- [ ] Prefer `watch` or `watchEffect` for reacting to specific data changes
|
|
- [ ] Use throttling/debouncing if updated operations are expensive
|
|
- [ ] Reserve updated for low-level DOM synchronization tasks
|
|
|
|
**Incorrect:**
|
|
```javascript
|
|
// WRONG: API call in updated - fires on EVERY re-render
|
|
export default {
|
|
data() {
|
|
return { items: [], lastUpdate: null }
|
|
},
|
|
updated() {
|
|
// This runs after every single state change!
|
|
fetch('/api/sync', {
|
|
method: 'POST',
|
|
body: JSON.stringify(this.items)
|
|
})
|
|
}
|
|
}
|
|
```
|
|
|
|
```javascript
|
|
// WRONG: State mutation in updated - INFINITE LOOP
|
|
export default {
|
|
data() {
|
|
return { renderCount: 0 }
|
|
},
|
|
updated() {
|
|
// This causes another update, which triggers updated again!
|
|
this.renderCount++ // INFINITE LOOP!
|
|
}
|
|
}
|
|
```
|
|
|
|
```javascript
|
|
// WRONG: Heavy computation on every update
|
|
export default {
|
|
updated() {
|
|
// Expensive operation runs on every keystroke, every state change
|
|
this.processedData = this.heavyComputation(this.rawData)
|
|
this.analytics = this.calculateMetrics(this.allData)
|
|
}
|
|
}
|
|
```
|
|
|
|
**Correct:**
|
|
```javascript
|
|
// CORRECT: Use watcher for specific data changes
|
|
export default {
|
|
data() {
|
|
return { items: [] }
|
|
},
|
|
watch: {
|
|
// Only fires when items actually changes
|
|
items: {
|
|
handler(newItems) {
|
|
this.syncToServer(newItems)
|
|
},
|
|
deep: true
|
|
}
|
|
},
|
|
methods: {
|
|
syncToServer: debounce(function(items) {
|
|
fetch('/api/sync', {
|
|
method: 'POST',
|
|
body: JSON.stringify(items)
|
|
})
|
|
}, 500)
|
|
}
|
|
}
|
|
```
|
|
|
|
```vue
|
|
<!-- CORRECT: Composition API with targeted watchers -->
|
|
<script setup>
|
|
import { ref, watch, onUpdated } from 'vue'
|
|
import { useDebounceFn } from '@vueuse/core'
|
|
|
|
const items = ref([])
|
|
const scrollContainer = ref(null)
|
|
|
|
// Watch specific data - not all updates
|
|
watch(items, (newItems) => {
|
|
syncToServer(newItems)
|
|
}, { deep: true })
|
|
|
|
const syncToServer = useDebounceFn((items) => {
|
|
fetch('/api/sync', { method: 'POST', body: JSON.stringify(items) })
|
|
}, 500)
|
|
|
|
// Only use onUpdated for DOM synchronization
|
|
onUpdated(() => {
|
|
// Scroll to bottom only if content changed height
|
|
if (scrollContainer.value) {
|
|
scrollContainer.value.scrollTop = scrollContainer.value.scrollHeight
|
|
}
|
|
})
|
|
</script>
|
|
```
|
|
|
|
```javascript
|
|
// CORRECT: Conditional check in updated hook
|
|
export default {
|
|
data() {
|
|
return {
|
|
content: '',
|
|
lastSyncedContent: ''
|
|
}
|
|
},
|
|
updated() {
|
|
// Only act if specific condition is met
|
|
if (this.content !== this.lastSyncedContent) {
|
|
this.syncContent()
|
|
this.lastSyncedContent = this.content
|
|
}
|
|
},
|
|
methods: {
|
|
syncContent: debounce(function() {
|
|
// Sync logic
|
|
}, 300)
|
|
}
|
|
}
|
|
```
|
|
|
|
## Valid Use Cases for Updated Hook
|
|
|
|
```javascript
|
|
// CORRECT: Low-level DOM synchronization
|
|
export default {
|
|
updated() {
|
|
// Sync third-party library with Vue's DOM
|
|
this.thirdPartyWidget.refresh()
|
|
|
|
// Update scroll position after content change
|
|
this.$nextTick(() => {
|
|
this.maintainScrollPosition()
|
|
})
|
|
}
|
|
}
|
|
```
|
|
|
|
## Prefer Computed Properties for Derived Data
|
|
|
|
```javascript
|
|
// WRONG: Calculating derived data in updated
|
|
export default {
|
|
data() {
|
|
return { numbers: [1, 2, 3, 4, 5] }
|
|
},
|
|
updated() {
|
|
this.sum = this.numbers.reduce((a, b) => a + b, 0) // Causes another update!
|
|
}
|
|
}
|
|
|
|
// CORRECT: Use computed property instead
|
|
export default {
|
|
data() {
|
|
return { numbers: [1, 2, 3, 4, 5] }
|
|
},
|
|
computed: {
|
|
sum() {
|
|
return this.numbers.reduce((a, b) => a + b, 0)
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
## Reference
|
|
- [Vue.js Lifecycle Hooks](https://vuejs.org/guide/essentials/lifecycle.html)
|
|
- [Vue.js Watchers](https://vuejs.org/guide/essentials/watchers.html)
|
|
- [Vue.js Computed Properties](https://vuejs.org/guide/essentials/computed.html)
|