Создание анимаций переходов между страницами сайта
Анимации переходов делают навигацию воспринимаемой как непрерывный процесс, а не серию телепортаций. Правильно реализованные переходы сокращают ощущаемое время загрузки и дают пользователю пространственный контекст («я ушёл вглубь» 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 день |







