Observer + Direction
You’ll need to track the scroll direction to determine whether the user is scrolling down or up. Based on the scroll direction, you can trigger the appropriate animation.
Two Way Scroll Animation
Example of how to use Two Way Scroll animation with frame motion.
Two Way Scroll Animation
two-way-scroll-animation.tsx
import Modal from '@/components/animations/modal' import React from 'react' const Scroll = () => { return ( <div className="p-5"> <TwoWayScrollAnimation animate={{ opacity: 1, y: 0 }} // Animation when scrolling down reverse={{ opacity: 0, y: 50 }} // Reverse animation when scrolling up initial={{ opacity: 0, y: 50 }} // Initial state threshold={0.5} // Trigger when 50% of the element is in view > <div className="w-52 h-52 rounded-full from-purple-500 to-pink-500 bg-gradient-to-t" /> </TwoWayScrollAnimation> </div> ) } export default Scroll "use client"; import React, { useEffect, useRef, useState } from 'react'; import { motion, useAnimation } from 'framer-motion'; import { Preview } from '../common/display'; interface ScrollProps { children: React.ReactNode; initial: ControlProps; animate: ControlProps; reverse: ControlProps; threshold?: number; } interface ControlProps { opacity: number; y: number; } const TwoWayScrollAnimation = ({ children, animate, initial, reverse, threshold = 0.5 }: ScrollProps) => { const [count, setCount] = useState(0); const controls = useAnimation(); const ref = useRef<HTMLDivElement>(null); const [scrollDirection, setScrollDirection] = useState<'up' | 'down'>('down'); const [lastScrollTop, setLastScrollTop] = useState(0); useEffect(() => { const handleScroll = () => { const scrollTop = window.pageYOffset || document.documentElement.scrollTop; if (scrollTop > lastScrollTop) { setScrollDirection('down'); } else { setScrollDirection('up'); } setLastScrollTop(scrollTop <= 0 ? 0 : scrollTop); }; window.addEventListener('scroll', handleScroll); return () => { window.removeEventListener('scroll', handleScroll); }; }, [lastScrollTop]); useEffect(() => { const observer = new IntersectionObserver( ([entry]) => { if (entry.isIntersecting) { if (scrollDirection === 'down') { controls.start({ opacity: animate.opacity, y: animate.y }); } else { controls.start({ opacity: reverse.opacity, y: reverse.y }); } } }, { threshold } ); if (ref.current) { observer.observe(ref.current); } return () => { if (ref.current) { observer.unobserve(ref.current); } }; }, [controls, animate, reverse, threshold, scrollDirection]); return ( <Preview SetCount={setCount} isRefreshing={true} hideIcon animeName='Two Way Scroll Animation'> <motion.div key={count} ref={ref} animate={controls} initial={{ opacity: initial.opacity, y: initial.y }} > {children} </motion.div> </Preview> ); }; export default TwoWayScrollAnimation; interface PreviewerProps { children: React.ReactNode; SetCount: (count: number) => void; isRefreshing: boolean; animeName:string hideIcon?: boolean; } export const Preview = ({ children, SetCount, isRefreshing, animeName, hideIcon=false }: PreviewerProps) => { const [count, setCount] = useState(0); const [isLoading, setIsLoading] = useState(false); const handleClick = () => { setIsLoading(true); setCount(count + 1); SetCount(count + 1); setTimeout(() => { setIsLoading(false); }, 400); // Timeout duration matches the animation duration }; return ( <div className="w-full h-full flex flex-col"> <div className="w-full flex-1 flex justify-center items-center"> {children} </div> {isRefreshing && <div className="h-[50px] w-full px-6 rounded-b-[24px] flex justify-between"> <span>{animeName}</span>{!hideIcon && <RotateCw onClick={handleClick} className={isLoading ? 'animate-spin duration-200' : 'animate-none'} />} </div> } </div> ) };