"use client";

import { useEffect, useRef, useState } from "react";
import { cn } from "@/lib/utils";

type Props = {
  children: React.ReactNode;
  /** Délai avant de jouer l'animation, en ms. Pratique pour les staggers. */
  delay?: number;
  /** Distance verticale de l'animation en px. 12 par défaut, valeur sobre. */
  offsetY?: number;
  /** Durée de l'animation en ms. */
  duration?: number;
  /** Class additionnelle. */
  className?: string;
  /** Force le rendu visible immédiat (utile pour des cas de SSR critique). */
  immediate?: boolean;
};

/**
 * Apparition sobre au défilement : fade + translation verticale légère.
 * Une seule fois par élément (unobserve dès la première intersection).
 * Respecte `prefers-reduced-motion`.
 */
export function Reveal({
  children,
  delay = 0,
  offsetY = 12,
  duration = 700,
  className,
  immediate = false,
}: Props) {
  const ref = useRef<HTMLDivElement>(null);
  const [visible, setVisible] = useState(immediate);

  useEffect(() => {
    if (immediate) return;
    const node = ref.current;
    if (!node) return;

    // Réduire ou désactiver les animations selon préférence système.
    if (typeof window !== "undefined" && window.matchMedia("(prefers-reduced-motion: reduce)").matches) {
      setVisible(true);
      return;
    }

    const observer = new IntersectionObserver(
      ([entry]) => {
        if (entry.isIntersecting) {
          setVisible(true);
          observer.unobserve(node);
        }
      },
      { threshold: 0.15, rootMargin: "0px 0px -40px 0px" },
    );
    observer.observe(node);
    return () => observer.disconnect();
  }, [immediate]);

  return (
    <div
      ref={ref}
      className={cn("will-change-transform", className)}
      style={{
        opacity: visible ? 1 : 0,
        transform: visible ? "translateY(0)" : `translateY(${offsetY}px)`,
        transition: `opacity ${duration}ms cubic-bezier(0.22, 1, 0.36, 1), transform ${duration}ms cubic-bezier(0.22, 1, 0.36, 1)`,
        transitionDelay: visible ? `${delay}ms` : "0ms",
      }}
    >
      {children}
    </div>
  );
}
