generated from Nicholai/astro-template
54 lines
1.2 KiB
TypeScript
54 lines
1.2 KiB
TypeScript
import React, { useEffect, useRef, useState } from 'react';
|
|
|
|
interface RevealProps {
|
|
children: React.ReactNode;
|
|
className?: string;
|
|
delay?: number;
|
|
duration?: number;
|
|
threshold?: number;
|
|
}
|
|
|
|
export const Reveal: React.FC<RevealProps> = ({
|
|
children,
|
|
className = "",
|
|
delay = 0,
|
|
duration = 1000,
|
|
threshold = 0.1
|
|
}) => {
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
const [isVisible, setIsVisible] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const observer = new IntersectionObserver(
|
|
([entry]) => {
|
|
if (entry.isIntersecting) {
|
|
setIsVisible(true);
|
|
observer.disconnect();
|
|
}
|
|
},
|
|
{ threshold, rootMargin: "50px" }
|
|
);
|
|
|
|
if (ref.current) {
|
|
observer.observe(ref.current);
|
|
}
|
|
|
|
return () => observer.disconnect();
|
|
}, [threshold]);
|
|
|
|
return (
|
|
<div
|
|
ref={ref}
|
|
className={className}
|
|
style={{
|
|
opacity: isVisible ? 1 : 0,
|
|
transform: isVisible ? 'translateY(0) scale(1)' : 'translateY(30px) scale(0.98)',
|
|
transition: `opacity ${duration}ms ease-out, transform ${duration}ms cubic-bezier(0.16, 1, 0.3, 1)`,
|
|
transitionDelay: `${delay}ms`,
|
|
willChange: 'opacity, transform'
|
|
}}
|
|
>
|
|
{children}
|
|
</div>
|
|
);
|
|
}; |