Создание анимаций переходов между страницами сайта

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.
Разработка и обслуживание любых видов сайтов:
Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Создание анимаций переходов между страницами сайта
Средняя
~2-3 рабочих дня
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1214
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1161
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    852
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1041
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    823
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    815

Создание анимаций переходов между страницами сайта

Анимации переходов делают навигацию воспринимаемой как непрерывный процесс, а не серию телепортаций. Правильно реализованные переходы сокращают ощущаемое время загрузки и дают пользователю пространственный контекст («я ушёл вглубь» vs «я вернулся назад»). Неправильно — раздражают задержками и конфликтуют с логикой маршрутизатора.

Выбор подхода по стеку

Стек Подход
React + React Router framer-motion + AnimatePresence
Next.js App Router View Transitions API или framer-motion
Vue / Nuxt <Transition> + <TransitionGroup>
Astro View Transitions API нативно
Многостраничный сайт (MPA) View Transitions API

Framer Motion + React Router

Базовая схема: оборачиваем <Routes> в <AnimatePresence>, каждый экран — <motion.div> с вариантами анимации:

import { AnimatePresence, motion } from 'framer-motion';
import { useLocation, Routes, Route } from 'react-router-dom';

const pageVariants = {
  initial:  { opacity: 0, x: 20 },
  animate:  { opacity: 1, x: 0 },
  exit:     { opacity: 0, x: -20 },
};

const pageTransition = {
  type: 'tween',
  ease: 'anticipate',
  duration: 0.25,
};

export const AnimatedRoutes: React.FC = () => {
  const location = useLocation();

  return (
    <AnimatePresence mode="wait" initial={false}>
      <Routes location={location} key={location.pathname}>
        <Route path="/"        element={<PageWrapper><Home /></PageWrapper>} />
        <Route path="/about"   element={<PageWrapper><About /></PageWrapper>} />
        <Route path="/catalog" element={<PageWrapper><Catalog /></PageWrapper>} />
      </Routes>
    </AnimatePresence>
  );
};

const PageWrapper: React.FC<{ children: React.ReactNode }> = ({ children }) => (
  <motion.div
    variants={pageVariants}
    initial="initial"
    animate="animate"
    exit="exit"
    transition={pageTransition}
  >
    {children}
  </motion.div>
);

mode="wait" гарантирует, что старая страница полностью ушла до появления новой. mode="sync" — обе анимации одновременно (быстрее, но может выглядеть хаотично).

Направленные переходы (вперёд/назад)

Для иерархической навигации (каталог → товар → корзина) переход должен быть направленным: вперёд — слайд вправо, назад — слайд влево.

const useNavigationDirection = () => {
  const [direction, setDirection] = useState(0);
  const location = useLocation();
  const prevLocation = useRef(location);
  const navHistory = useRef<string[]>([location.pathname]);

  useEffect(() => {
    const currentPath = location.pathname;
    const history = navHistory.current;
    const prevIndex = history.lastIndexOf(prevLocation.current.pathname);
    const currentIndex = history.indexOf(currentPath);

    if (currentIndex > prevIndex) setDirection(1);   // вперёд
    else setDirection(-1);                           // назад

    if (currentIndex === -1) {
      navHistory.current = [...history, currentPath];
    }

    prevLocation.current = location;
  }, [location]);

  return direction;
};

// Вариант анимации с направлением
const variants = {
  initial:  (dir: number) => ({ x: dir > 0 ? '100%' : '-100%', opacity: 0 }),
  animate:  { x: 0, opacity: 1 },
  exit:     (dir: number) => ({ x: dir > 0 ? '-100%' : '100%', opacity: 0 }),
};

View Transitions API (нативный браузерный подход)

Поддерживается в Chrome 111+, Safari 18+. Для MPA и Next.js — минимальный код:

// Обёртка навигации
async function navigateTo(url) {
  if (!document.startViewTransition) {
    window.location.href = url;
    return;
  }

  const transition = document.startViewTransition(async () => {
    const html = await fetch(url).then(r => r.text());
    const doc = new DOMParser().parseFromString(html, 'text/html');
    document.querySelector('main').replaceWith(doc.querySelector('main'));
    history.pushState({}, '', url);
  });

  await transition.ready;
}

CSS для управления анимацией:

/* Дефолтный cross-fade переопределяем */
@keyframes slide-from-right {
  from { transform: translateX(100%); }
}

@keyframes slide-to-left {
  to { transform: translateX(-100%); }
}

::view-transition-old(root) {
  animation: 250ms ease slide-to-left;
}

::view-transition-new(root) {
  animation: 250ms ease slide-from-right;
}

/* Для конкретных элементов — shared element transition */
.product-image {
  view-transition-name: product-hero;
}

Shared element transitions — hero-анимации, где конкретный элемент (карточка товара) плавно «превращается» в hero-изображение на странице товара. Это самая впечатляющая возможность View Transitions API.

Скелетон-экраны вместо спиннеров

При переходе данные часто грузятся асинхронно. Спиннер показывает «загрузка» — скелетон показывает структуру страницы:

const ProductSkeleton: React.FC = () => (
  <motion.div
    initial={{ opacity: 0 }}
    animate={{ opacity: 1 }}
    className="skeleton-wrapper"
  >
    <div className="skeleton skeleton--image" />
    <div className="skeleton skeleton--title" />
    <div className="skeleton skeleton--text" />
    <div className="skeleton skeleton--text skeleton--short" />
  </motion.div>
);
.skeleton {
  background: linear-gradient(90deg, #f0f0f0 25%, #e0e0e0 50%, #f0f0f0 75%);
  background-size: 200% 100%;
  animation: shimmer 1.5s infinite;
  border-radius: 4px;
}

@keyframes shimmer {
  0%   { background-position: 200% 0; }
  100% { background-position: -200% 0; }
}

Отмена анимаций при быстрой навигации

Если пользователь кликает быстро, анимации должны прерываться, а не накапливаться в очередь:

// framer-motion делает это автоматически через AnimatePresence
// Для кастомных анимаций используем useAnimation:
const controls = useAnimation();

const navigate = async (to: string) => {
  await controls.start('exit'); // ждём выхода
  router.push(to);
};

// Или просто сокращаем duration до 150-200ms
// что делает прерывание незаметным для пользователя

Доступность

Анимации могут быть неприятны людям с вестибулярными расстройствами. Обязательный медиа-запрос:

@media (prefers-reduced-motion: reduce) {
  ::view-transition-group(*),
  ::view-transition-image-pair(*),
  ::view-transition-old(*),
  ::view-transition-new(*) {
    animation: none !important;
  }
}

В framer-motion:

const shouldReduceMotion = useReducedMotion();

const transition = shouldReduceMotion
  ? { duration: 0 }
  : { duration: 0.25, ease: 'easeInOut' };

Сроки

Задача Время
Базовые fade/slide переходы (framer-motion) 0.5 дня
Направленные переходы вперёд/назад 1 день
View Transitions API + shared elements 1–2 дня
Скелетон-экраны для 3–5 шаблонов 1 день