Intersection Observer
You'll use the Intersection Observer API to observe when the element is in the viewport and trigger the animation based on that.
Scroll Animation
Example of how to use Scroll animation with frame motion.
Scroll Animation
scroll-animation.tsx
import Modal from '@/components/animations/modal' import React from 'react' const Scroll = () => { return ( <div className="p-5"> <ScrollAnimation animate={{ opacity: 1, y: 0 }} initial={{ opacity: 0, y: 50 }} threshold={0.5} // Animation triggers when 50% of the element is in view > <div className="w-52 h-52 rounded-full from-orange-500 to-yellow-500 bg-gradient-to-t" /> </ScrollAnimation> </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; threshold?: number; } interface ControlProps { opacity: number; y: number; } const ScrollAnimation = ({ children, animate, initial, threshold = 0.5 }: ScrollProps) => { const [count, setCount] = useState(0); const controls = useAnimation(); const ref = useRef<HTMLDivElement>(null); useEffect(() => { const observer = new IntersectionObserver( ([entry]) => { if (entry.isIntersecting) { controls.start({ opacity: animate.opacity, y: animate.y }); } }, { threshold } ); if (ref.current) { observer.observe(ref.current); } return () => { if (ref.current) { observer.unobserve(ref.current); } }; }, [controls, animate, threshold]); return ( <Preview SetCount={setCount} isRefreshing={true} hideIcon animeName='Scroll Animation'> <motion.div key={count} ref={ref} animate={controls} initial={{ opacity: initial.opacity, y: initial.y }} > {children} </motion.div> </Preview> ); }; export default ScrollAnimation; 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> ) };