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>
3.8 KiB
3.8 KiB
title, impact, impactDescription, type, tags
| title | impact | impactDescription | type | tags | |||||||
|---|---|---|---|---|---|---|---|---|---|---|---|
| Register Lifecycle Hooks Synchronously During Setup | HIGH | Asynchronously registered lifecycle hooks will never execute | capability |
|
Register Lifecycle Hooks Synchronously During Setup
Impact: HIGH - Lifecycle hooks registered asynchronously (e.g., inside setTimeout, after await) will never be called because Vue cannot associate them with the component instance. This leads to silent failures where expected initialization or cleanup code never runs.
In Vue 3's Composition API, lifecycle hooks like onMounted, onUnmounted, onUpdated, etc. must be registered synchronously during component setup. The hook registration doesn't need to be lexically inside setup() or <script setup>, but the call stack must be synchronous and originate from within setup.
Task Checklist
- Register all lifecycle hooks at the top level of setup() or
<script setup> - Never register hooks inside setTimeout, setInterval, or Promise callbacks
- When calling composables that use lifecycle hooks, call them synchronously
- Hooks CAN be in external functions if called synchronously from setup
Incorrect:
// WRONG: Hook registered asynchronously - will NEVER execute
import { onMounted } from 'vue'
export default {
async setup() {
// After await, we're in a different call stack
const data = await fetchInitialData()
// This hook will NOT be registered!
onMounted(() => {
console.log('This will never run')
})
}
}
// WRONG: Hook registered in setTimeout - will NEVER execute
import { onMounted } from 'vue'
export default {
setup() {
setTimeout(() => {
// This is asynchronous - hook won't be registered!
onMounted(() => {
initializeChart()
})
}, 100)
}
}
// WRONG: Hook registered in Promise callback
import { onMounted } from 'vue'
export default {
setup() {
fetchConfig().then(() => {
// Asynchronous! This will silently fail
onMounted(() => {
applyConfig()
})
})
}
}
Correct:
// CORRECT: Hook registered synchronously at top level
import { onMounted, ref } from 'vue'
export default {
setup() {
const data = ref(null)
// Register hook synchronously FIRST
onMounted(async () => {
// Async operations are fine INSIDE the hook
data.value = await fetchInitialData()
initializeChart()
})
return { data }
}
}
<!-- CORRECT: <script setup> - hooks at top level -->
<script setup>
import { onMounted, onUnmounted, ref } from 'vue'
const isReady = ref(false)
// These are synchronous during script setup execution
onMounted(() => {
isReady.value = true
})
onUnmounted(() => {
cleanup()
})
</script>
// CORRECT: Hook in external function called synchronously from setup
import { onMounted, onUnmounted } from 'vue'
function useWindowResize(callback) {
// This is fine - it's called synchronously from setup
onMounted(() => {
window.addEventListener('resize', callback)
})
onUnmounted(() => {
window.removeEventListener('resize', callback)
})
}
export default {
setup() {
// Composable called synchronously - hooks will be registered
useWindowResize(handleResize)
}
}
Multiple Hooks Are Allowed
// CORRECT: You can register the same hook multiple times
import { onMounted } from 'vue'
export default {
setup() {
// Both will run, in order of registration
onMounted(() => {
initializeA()
})
onMounted(() => {
initializeB()
})
}
}