Skip to content

leonsilicon/vue-animate-height

Repository files navigation

Vue Animate Height

npm version

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.

Demo

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.

Usage

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>

Props

Additional props will be forwarded to the wrapper div to make adding attributes like aria-* easier.

height

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".

duration

Type: number
Default: 250

Duration of the animation in milliseconds.

delay

Type: number
Default: 0

Animation delay in milliseconds.

easing

Type: string
Default: "ease"

CSS easing function to be applied to the animation.

contentClass

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.

animationStateClasses

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.

applyInlineTransitions

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.

animateOpacity

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.

Emits

animation-start

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 event v-on:animationstart!

animation-end

Emitted when the animation ends.

Passes a payload object containing newHeight, the pixel value of the height at which the animation ended.

Accessibility

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>

Reduced Motion

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.

Gotchas

Bounded flexboxes

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.

About

Lightweight Vue component for animating height using CSS transitions.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published