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>
4.8 KiB
title, impact, impactDescription, type, tags
| title | impact | impactDescription | type | tags | ||||||
|---|---|---|---|---|---|---|---|---|---|---|
| RouterView Transitions Always Apply Despite Missing appear Prop | LOW | Initial page load with RouterView triggers transition animation even without the appear prop due to async navigation | gotcha |
|
RouterView Transitions Always Apply Despite Missing appear Prop
Impact: LOW - When using <Transition> with Vue Router's <RouterView>, the enter transition animation runs on initial page load even if you haven't added the appear prop. This differs from normal Transition behavior where appear is required for initial render animations. This happens because Vue Router's navigations are asynchronous, causing the component to mount after the initial render.
Task Checklist
- Be aware that RouterView transitions always animate on initial load
- If you want NO animation on initial load, you need to handle this explicitly
- Don't add
appearprop expecting it to change behavior - it's already effectively enabled - Consider whether initial animation is desired for your UX
Expected Behavior (Normal Transition):
<template>
<!-- Without appear: No animation on initial render -->
<Transition name="fade">
<div v-if="show">Content</div>
</Transition>
<!-- With appear: Animates on initial render -->
<Transition name="fade" appear>
<div v-if="show">Content</div>
</Transition>
</template>
RouterView Behavior (Different!):
<template>
<!-- RouterView transitions ALWAYS animate on initial load -->
<!-- The appear prop has no effect here -->
<RouterView v-slot="{ Component }">
<Transition name="fade">
<component :is="Component" />
</Transition>
</RouterView>
</template>
<!--
On initial page load:
1. Vue renders the app
2. Router resolves the route (async)
3. Component mounts AFTER initial render
4. Enter transition triggers (as if toggled from v-if="false" to v-if="true")
-->
Why This Happens
Vue Router navigations are asynchronous. The sequence is:
- Vue application mounts with empty RouterView
- Router resolves the initial route
- Route component is inserted into RouterView
- This insertion triggers the enter transition
Since the component wasn't present in the initial render and is "inserted" afterward, Vue treats it as a normal enter transition, not an initial render.
If You Want to Disable Initial Animation
<template>
<RouterView v-slot="{ Component }">
<Transition :name="isInitialLoad ? '' : 'fade'" mode="out-in">
<component :is="Component" />
</Transition>
</RouterView>
</template>
<script setup>
import { ref, onMounted } from 'vue'
import { useRouter } from 'vue-router'
const isInitialLoad = ref(true)
const router = useRouter()
// After first navigation completes, enable transitions
router.isReady().then(() => {
// Small delay to ensure initial render is complete
setTimeout(() => {
isInitialLoad.value = false
}, 0)
})
</script>
Alternative: Use CSS to Skip First Animation
<template>
<RouterView v-slot="{ Component }">
<Transition name="fade" mode="out-in">
<component :is="Component" :class="{ 'skip-initial': isInitialLoad }" />
</Transition>
</RouterView>
</template>
<script setup>
import { ref } from 'vue'
import { useRouter } from 'vue-router'
const isInitialLoad = ref(true)
const router = useRouter()
router.isReady().then(() => {
isInitialLoad.value = false
})
</script>
<style>
.fade-enter-active,
.fade-leave-active {
transition: opacity 0.3s ease;
}
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
/* Skip animation on initial load */
.skip-initial.fade-enter-active {
transition: none;
}
</style>
Standard RouterView Transition Pattern
If you're fine with initial animation (often desired), use the standard pattern:
<template>
<RouterView v-slot="{ Component, route }">
<Transition :name="route.meta.transition || 'fade'" mode="out-in">
<component :is="Component" :key="route.path" />
</Transition>
</RouterView>
</template>
<style>
.fade-enter-active,
.fade-leave-active {
transition: opacity 0.3s ease;
}
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
/* Route-specific transitions via meta */
.slide-enter-active,
.slide-leave-active {
transition: transform 0.3s ease;
}
.slide-enter-from {
transform: translateX(100%);
}
.slide-leave-to {
transform: translateX(-100%);
}
</style>
// router.js
const routes = [
{
path: '/',
component: Home
},
{
path: '/about',
component: About,
meta: { transition: 'slide' } // Custom transition for this route
}
]