All components
SVG Mask Reveal
effectsAceternity UI component.
responsive · 640px
Install
Same command in any shadcn project — React (Vite/CRA), Next.js, Remix, Astro, and more:
$
npx shadcn@latest add https://your-domain/r/svg-mask-effect.jsonUsage
import { MaskContainer } from "@/registry/aceternity-ui/svg-mask-effect";
export default function Demo() {
return (
<MaskContainer
revealText={
<p className="mx-auto max-w-4xl text-center text-4xl font-bold text-foreground">
Move your cursor to reveal the hidden text beneath.
</p>
}
className="h-[28rem] rounded-xl"
>
<span className="text-white dark:text-black">
The spotlight follows your cursor.{" "}
<span className="text-violet-400">Hover here</span> to expand the mask
and reveal what lies beneath the surface.
</span>
</MaskContainer>
);
}Component source
"use client";
import { useState, useEffect, useRef } from "react";
import { motion } from "motion/react";
import { cn } from "@/lib/utils";
export const MaskContainer = ({
children,
revealText,
size = 10,
revealSize = 600,
className,
}: {
children?: string | React.ReactNode;
revealText?: string | React.ReactNode;
size?: number;
revealSize?: number;
className?: string;
}) => {
const [isHovered, setIsHovered] = useState(false);
const [mousePosition, setMousePosition] = useState<any>({ x: null, y: null });
const containerRef = useRef<any>(null);
const updateMousePosition = (e: any) => {
const rect = containerRef.current.getBoundingClientRect();
setMousePosition({ x: e.clientX - rect.left, y: e.clientY - rect.top });
};
useEffect(() => {
containerRef.current.addEventListener("mousemove", updateMousePosition);
return () => {
if (containerRef.current) {
containerRef.current.removeEventListener(
"mousemove",
updateMousePosition,
);
}
};
}, []);
let maskSize = isHovered ? revealSize : size;
return (
<motion.div
ref={containerRef}
className={cn("relative h-screen", className)}
animate={{
backgroundColor: isHovered ? "var(--slate-900)" : "var(--white)",
}}
transition={{
backgroundColor: { duration: 0.3 },
}}
>
<motion.div
className="absolute flex h-full w-full items-center justify-center bg-black text-6xl [mask-image:url(/mask.svg)] [mask-repeat:no-repeat] [mask-size:40px] dark:bg-white"
animate={{
maskPosition: `${mousePosition.x - maskSize / 2}px ${
mousePosition.y - maskSize / 2
}px`,
maskSize: `${maskSize}px`,
}}
transition={{
maskSize: { duration: 0.3, ease: "easeInOut" },
maskPosition: { duration: 0.15, ease: "linear" },
}}
>
<div className="absolute inset-0 z-0 h-full w-full bg-black opacity-50 dark:bg-white" />
<div
onMouseEnter={() => {
setIsHovered(true);
}}
onMouseLeave={() => {
setIsHovered(false);
}}
className="relative z-20 mx-auto max-w-4xl text-center text-4xl font-bold"
>
{children}
</div>
</motion.div>
<div className="flex h-full w-full items-center justify-center">
{revealText}
</div>
</motion.div>
);
};Dependencies
motion
Source: Aceternity UI