"use client"; import { useRef, useEffect, useState, type ReactNode } from "react"; interface ScrollFadeInProps { children: ReactNode; className?: string; delay?: number; } /** * Wraps children in a fade-in-up animation triggered by scroll. * Unlike Framer Motion's initial={{ opacity: 0 }}, this does NOT * render opacity:0 in SSR HTML — content is visible by default * and only hidden after JS hydration for the animation effect. */ export default function ScrollFadeIn({ children, className = "", delay = 0 }: ScrollFadeInProps) { const ref = useRef(null); const [isVisible, setIsVisible] = useState(false); const [hasMounted, setHasMounted] = useState(false); useEffect(() => { setHasMounted(true); const el = ref.current; if (!el) return; const observer = new IntersectionObserver( ([entry]) => { if (entry.isIntersecting) { setIsVisible(true); observer.unobserve(el); } }, { threshold: 0.1 } ); observer.observe(el); return () => observer.disconnect(); }, []); return (
{children}
); }