A Vue plugin which aims to make complex route-to-route transitions much less complex to manage. It leverages Vue's excellent transition system and router. It Uses Promises to determine animation lifecycle throughout the component tree.
This simple demo app showcases different components animating together.
$ npm install vue-stage --save
or
$ yarn add vue-stage
vue-stage
contains two pieces: The plugin and the top-level <Stage />
component.
// main.js
import Vue from 'vue'
import VueStage from 'vue-stage'
Vue.use( VueStage )
The plugin applies new "stage" lifecycle methods and data to each Vue instance in your app via a Global Mixin.
// App.vue
<template>
<Stage />
</template>
<script>
import { Stage } from 'vue-stage'
export default {
name: 'App',
components: { Stage }
}
</script>
It accepts four function props to help relay lifecycle upward to the main App.vue
:
beforeEnter
, afterEnter
, beforeLeave
, afterLeave
// App.vue
...
<Stage :beforeLeave="beforeLeave" />
...
methods: {
beforeLeave() {
// Fired before views start leaving the stage
}
}
...
These methods are useful when the App needs to perform tasks when views change, like updating the scroll position.
The Stage
component is a simple wrapper around the default <router-view>
. It acts as the top-level stage lifecycle manager using Vue's transition JS Hooks and <keep-alive>
. This can be easily refactored into your own template if required, see the section below on extending.
Once applied, each Vue instance in your app has these data props applied to it:
isEnteringStage
: animating inisLeavingStage
: animating outisOnStage
: currently visibleisActiveOnStage
: currently visible, but is not animating out (useful withv-if
/v-show
)
These are updated internally, and can be used to conditionally render template markup or determine the current animation state.
Named methods are automatically called on instances where they are defined. They only need to be specified on an instance if they are required.
stageEnter
stageLeave
: Particularly important, the parent view will wait for this to resolve before unmounting.
These are the primary animation methods.
They must return a Promise
if the parent view should wait for their animation to complete before proceeding. Elements can be directly animated here with GSAP or other libraries (see JS animation example), or the returned Promise
's resolve()
can be stored for firing later (e.g. if using CSS transition
or animation
, see CSS animation example).
stageEnter
and stageLeave
can alternatively be defined as data props; with a duration in milliseconds as their value. This replaces the method, automatically returning a promise which resolves after the specified duration.
data() {
return {
stageLeave: 2000
}
},
methods: {
...
}
stageBeforeEnter
: before animating instageAfterEnter
: after inbound animation is completestageBeforeLeave
: before animating outstageAfterLeave
: after outbound animation is complete
Useful if the component needs to perform additional work before or after animating.
The simple demo app showcases both JS and CSS animations, and illustrates how a highly nested component can affect view-level lifecycle.
<template>
<div class="ComponentName">
<div ref="animated" />
</div>
</template>
import TweenMax from 'gsap'
export default {
name: 'ComponentName',
methods: {
stageEnter() {
let onComplete
const promise = new Promise( resolve => { onComplete = resolve } )
TweenMax.fromTo( this.$refs.animated, 1, { scale: 0 }, { scale: 1, onComplete } )
return promise
},
stageLeave() {
let onComplete
const promise = new Promise( resolve => { onComplete = resolve } )
TweenMax.to( this.$refs.animated, 1, { scale: 0, onComplete } )
return promise
}
}
}
<template>
<transition name="fade" v-on:after-leave="onLeaveComplete" v-on:after-enter="onEnterComplete">
<p v-show="isActiveOnStage">Oh no</p>
</transition>
</template>
export default {
name: 'ComponentName',
data () {
return {
onEnterComplete: () => {},
onLeaveComplete: () => {}
}
},
methods: {
stageLeave () {
return new Promise(resolve => { this.onLeaveComplete = resolve })
},
stageEnter () {
return new Promise(resolve => { this.onEnterComplete = resolve })
}
}
}
.fade-enter-active, .fade-leave-active {
transition: opacity 1s;
}
.fade-enter, .fade-leave-to {
opacity: 0;
}
By default components are manually destroyed after they've finished leaving the stage. In certain situations you may want to keep the instances alive as if you've wrapped them in a <keep-alive>
. This can be set when installing the plugin: Vue.use(VueStage, { destroyAfterLeave: false })
. Note that this will keep all instances in the tree alive when set.
The plugin provides console warning messaging for certain situations:
-
If your Promise has not resolved after a certain timeout (10s by default):
This timeout can be adjusted if you're using long-running animations, but it can't be turned off entirely. To change this timeout, apply it when installing the plugin:Vue.use(VueStage, { promiseTimeout: 15000 })
-
If you've defined a
stageEnter
orstageLeave
method, but have not returned a promise from it:
Components don't always need to return a promise (e.g. in cases where the named method is only used for lifecycle/convenience purposes). This warning is to remind you that the parent view won't wait for the component to finish animating. This can be snoozed locally for an individual component by setting the data propstageWarningsSnoozed
totrue
, or globally for all components when installing the plugin:Vue.use(VueStage, { promiseWarnings: false })
.
Tip: Specify a name
option on your components. Error messages list the name option, pointing you directly to the component that's throwing it:
export default {
name: 'ComponentName',
methods: { ...
In cases where using the provided <Stage />
component is undesirable (i.e. when not using Vue Router), this can be easily factored into your own component. In it's most minimal form vue-stage
requires this structure:
<template>
<transition appear mode="out-in" v-on:enter="enter" v-on:leave="leave">
<keep-alive>
<!-- Wraps Vue Router's router-view by default -->
<!-- <router-view ref="view" /> -->
<div ref="view">
...
</div>
</keep-alive>
</transition>
</template>
export default {
methods: {
enter (el, done) {
this.$refs.view.$_stageEnter( done )
},
leave (el, done) {
this.$refs.view.$_stageLeave( done )
}
}
}
- The
<Stage />
relies on Vue's transition system to relay view-levelenter
andleave
lifecycle events through to the top level<router-view />
components (main page views). It does so using transition JS Hooks, only callingdone()
once all children have resolved their promises. - These methods traverse the entire component tree for each top-level view, calling the appropriate lifecycle method on descendants, which then propagate upward their own animation promises. Once all promises for the view's tree are resolved, it completes the animation at the view level.
- The
out-in
transition mode is important here, as we need the incoming view to wait for the outgoing view's entire tree to finish animating before performing the view-level mounting operation. <keep-alive>
is critical here working with CSS transitions. By default Vue immediately destroys outward transitioning elements, they're ephemeral - only visually changing, and so they are immediately removed from the virtual DOM. This makes toggling a transition withv-if
orv-show
impossible, as the component has already been scrapped in memory. Components are kept alive and then manually destroyed once they've finished leaving the stage.
- Test browser support
- Add documentation around Promise polyfill
- Cancel promises so they don't resolve if overridden by an opposite transition
- Cancel animations and immediately run inverse transition
- Add
waitForChildren
local component flag:- Any component should be able to set this flag locally, causing it to wait for
it's descendants' promises to resolve before running it's own
stageLeave()
- Any component should be able to set this flag locally, causing it to wait for
it's descendants' promises to resolve before running it's own
- Add
waitForParent
local component flag:- Any component should be able to set this flag locally, causing it to wait for
it's parent to finish entering before running it's own
stageEnter()
- Any component should be able to set this flag locally, causing it to wait for
it's parent to finish entering before running it's own