A port of Stanko's excellent React Animate Height for Vue.js.
Most of the below documentation is derived from React Animate Height's documentation.
Vue Animate Height is a lightweight Vue component for animating height using CSS transitions.
It slides an element up or down and animates it to any specific height.
The content's opacity can also be animated as well (see the animateOpacity
prop below).
CSS classes are applied during specific animation states. For more information, check out the animationStateClasses
prop.
Live demo: leonzalion.github.io/vue-animate-height
To build the examples locally, run:
pnpm install
pnpm dev:docs
Then, open localhost:3000
in your browser of choice.
Install the package from npm:
npm install vue-animate-height
Then, import it and use it in your Vue app:
<script setup>
import { ref } from 'vue';
import AnimateHeight from 'vue-animate-height';
const height = ref(0);
const toggle = () => height.value = height.value === 0 ? 'auto' : 0;
</script>
<template>
<div>
<button
:aria-expanded="height !== 0"
aria-controls='example-panel'
@click='toggle'
>
{{ height === 0 ? 'Open' : 'Close' }}
</button>
<AnimateHeight
id='example-panel'
:duration="500"
:height="height"
>
<h1>Your content goes here</h1>
<p>Put as many Vue or HTML components here as your heart desires</p>
</AnimateHeight>
</div>
</template>
Additional props will be forwarded to the wrapper div
to make adding attributes like aria-*
easier.
Type: number
, string
as percentage value (e.g. "50%"
) or "auto"
Required: true
When changed, element height will be animated to that height.
To slide up, use 0
. To slide down, use "auto"
.
Type: number
Default: 250
Duration of the animation in milliseconds.
Type: number
Default: 0
Animation delay in milliseconds.
Type: string
Default: "ease"
CSS easing function to be applied to the animation.
Type: string
Required: false
CSS class to be applied to content wrapper element.
Please note that you shouldn't apply properties that mess with the layout (like display
, height
...), as these might break height calculations.
Type: object
Default:
{
animating: 'vah-animating',
animatingUp: 'vah-animating--up',
animatingDown: 'vah-animating--down',
static: 'vah-static',
animatingToHeightZero: 'vah-animating--to-height-zero',
animatingToHeightAuto: 'vah-animating--to-height-auto',
animatingToHeightSpecific: 'vah-animating--to-height-specific',
staticHeightZero: 'vah-static--height-zero',
staticHeightAuto: 'vah-static--height-auto',
staticHeightSpecific: 'vah-static--height-specific',
}
Please note that this prop will be merged with the default object and cached when the component is created, so changing it afterwards will have no effect.
Type: boolean
Default: true
If this flag is set to false
, only CSS classes will be applied to the element and inline transition styles will not be present.
Type: boolean
Default: false
If set to true
, the content will fade in (and fade out) while the height is animated.
aria-hidden
Type: boolean
Required: false
By default, this library will set aria-hidden
to true
when height is zero. If you wish to override it, you can pass this prop yourself.
Emitted when the animation starts.
Passes a payload object containing newHeight
, the pixel value of the height at which the animation will end.
Do not confuse
v-on:animation-start
with the native HTML eventv-on:animationstart
!
Emitted when the animation ends.
Passes a payload object containing newHeight
, the pixel value of the height at which the animation ended.
When the height
prop is 0
, this library will hide the content using display: hidden
. It will also apply aria-hidden="true"
which you can override by passing aria-hidden
prop yourself.
When using a button to toggle height, make sure you add aria-expanded
and aria-controls
to make everything accessible:
<!-- The id of `aria-controls` has to match the id passed to AnimateHeight -->
<button
:aria-expanded="height !== 0"
aria-controls='example-panel'
@click="toggleHeight"
...
>
Toggle
</button>
<AnimateHeight id='example-panel'>
Content
</AnimateHeight>
The component checks for prefers-reduced-motion
and disables animations if it set to true. Please note that component doesn't listen for potential changes of the prefers-reduced-motion
option.
If AnimateHeight
is a flex child and its parent has a fixed height, the animation won't work. To fix this, you just need to add the following CSS rule to the AnimateHeight
instance:
flex-shrink: 0;
You can do this by passing a class
or by using inline CSS in the style
prop:
<AnimateHeight style="flex-shrink: 0">
Check out issue #89 of React Animate Height for an example and more details.