SVG-графика поддаётся анимированию. В этой статье мы покажем, как анимировать отрисовку контуров.

Сначала мы разберём свойства SVG-анимации и синтаксис библиотеки Anime.js, а затем анимируем отрисовку котов. Если вы уже знакомы с теорией, можете сразу перейти к практике.

Для анимации нам понадобятся:

  • элемент path — контур;
  • атрибуты обводки: stroke и stroke-dashoffset;
  • библиотека Anime.js.

Контур и обводка в SVG

path — это способ описать траекторию внутри векторного изображения. Далее эту траекторию можно отобразить, задав ей обводку. Как выглядит path:

<svg xmlns="http://www.w3.org/2000/svg" width="205" height="118" viewBox="0 0 205 118" fill="none">
  <path d="M3 55.5286C8.43655 49.2385 14.7452 43.6721 21.6476 39.052C23.1876 38.0129 25.4513 37.1037 26.6945 38.4768C27.381 39.2375 27.3995 40.3693 27.384 V45.43 C8.43 48.57" stroke="white" stroke-width="5" stroke-linecap="round" stroke-linejoin="round"/>
</svg>

Обводка задаётся с помощью атрибута stroke или аналогичного CSS-свойства. За толщину отвечает stroke-width, за скругление — stroke-linecap: round;.

stroke-dashoffset задаёт смещение обводки. Этот атрибут можно анимировать. Например, с его помощью создаётся примитивный индикатор загрузки.

Синтаксис библиотеки Anime.js

Мы будем использовать Anime.js, потому что с ней легко настроить анимацию — библиотека берёт на себя часть вычислений. Например, когда мы не можем посчитать длину контура, anime.setDashoffset делает это за нас.

У библиотеки простой синтаксис. В примере ниже мы в первой строке находим все контуры, а внутри anime() описываем параметры анимации. Последняя строка указывает на изменяемое свойство, а остальные являются параметрами и не затрагивают конкретных атрибутов или свойств элемента.

const paths = document.querySelectorAll('svg path');
anime({
  targets: paths,
  duration: 2000,
  delay: 200,
  easing: 'easeInOutExpo',
  strokeDashoffset: [anime.setDashoffset, 0]
});

targets — элементы, у которых будет меняться значение атрибута.

duration — длительность анимации в миллисекундах.

delay — задержка перед началом анимации.

easing — временная функция. Здесь для примера приводим easeInOutExpo, а полный список временных функций вы найдёте на сайте библиотеки.

strokeDashoffset — изменяемый атрибут. Передаём туда массив с исходным и конечным значениями атрибута.

Мы не сможем вычислить длину каждого контура в SVG с котами, поэтому используем anime.setDashoffset, чтобы получить длину.

Можно воздействовать на любые свойства и атрибуты, которые поддаются анимированию. Например, opacity, fill, background-color или transform:

const circles = document.querySelectorAll('svg circle');
anime({
  targets: circles,
  duration: 1000,
  easing: 'easeInOutExpo',
  opacity: [0, 1],
  scale: [0, 1]
});

В результате получится анимация плавного появления окружности из прозрачности.

💫 Научитесь создавать анимации

Приходите на наши курсы — мы показываем там самые популярные и интересные техники для создания эффектов на CSS и JavaScript.

Практика: создаём SVG-анимацию контуров

Соберём анимацию — отрисуем три изображения с котами. Используем для этого заготовку — SVG-файл.

Экспортируем каждое изображение в формате SVG и поместим в <div> с классом wrapper.

<div class='wrapper'>
  //Сюда помещаем все <svg>
</div>

Найдём все контуры и анимируем так, как показывали выше:

const paths = document.querySelectorAll('svg path');

anime({
  targets: paths,
  duration: 2000,
  delay: 200,
  easing: 'easeInOutExpo',
  strokeDashoffset: [anime.setDashoffset, 0]
});

Последовательная отрисовка картинок

Сейчас все изображения рисуются одновременно. Сделаем так, чтобы картинки появлялись последовательно. Используем для этого timeline — временной отрезок, на котором расставляется анимация в определённом порядке и с нужной задержкой.

Синтаксис timeline:

const timeline = anime.timeline({
  duration: 3000,
  easing: 'easeInOutExpo'
});

timeline.add({
  targets: firstImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0]
});

timeline.add({
  targets: secondImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0]
});

timeline.add({
  targets: thirdImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0]
});

timeline создаётся один раз. Он принимает параметры, общие для каждой анимации, которая будет на нём располагаться — в нашем случае duration и easing. Затем при помощи timeline.add() добавляются анимации — они будут выполняться последовательно.

💡 Некоторые параметры можно переопределять — например, delay, easing, duration. А вот loop и direction переопределить нельзя. Параметр loop отвечает за бесконечное проигрывание, direction — за направление анимации.

Задержка следующей анимации

Чтобы начать анимацию не сразу после окончания предыдущей, передайте второй аргумент со временем в миллисекундах:

timeline.add({
  targets: firstImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0]
});

timeline.add({
  targets: secondImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0]
}, ‘+=500’);

Вторая анимация начнётся спустя полсекунды после окончания предыдущей.

Если вместо знака + поставить -, анимация начнётся заранее — когда предыдущей останется проигрывать 500 миллисекунд. Так можно управлять композицией.

Стейджер

Напоследок рассмотрим ещё одну возможность управлять анимацией контура — стейджер. Каждый кот состоит из нескольких контуров — сделаем так, чтобы эти контуры рисовались последовательно.

Стейджер реализуется при помощи конструкции anime.stagger(<время задержки>).

timeline.add({
  targets: firstImagePaths,
  strokeDashoffset: [anime.setDashoffset, 0],
  delay: anime.stagger(300)
});

Теперь каждый последующий контур начнёт анимироваться спустя 300 миллисекунд после начала анимирования предыдущего.

Итог

Мы рассмотрели лишь один из видов анимаций векторной графики, но есть и другие. Например, если совместить разобранную выше анимацию с SVG-масками, можно добиться такого эффекта:

Можно анимировать форму фигур. Такая анимация называется морфингом.

Источник

Ещё один эффект достигается при помощи SVG-фильтров:

Материалы по теме