mirror of
https://github.com/DeveloLongScript/MHSF.git
synced 2026-05-07 23:54:59 -05:00
60 lines
1.6 KiB
TypeScript
60 lines
1.6 KiB
TypeScript
|
|
import React, { ComponentPropsWithoutRef, CSSProperties } from "react";
|
||
|
|
|
||
|
|
import { cn } from "@/lib/utils";
|
||
|
|
|
||
|
|
interface RippleProps extends ComponentPropsWithoutRef<"div"> {
|
||
|
|
mainCircleSize?: number;
|
||
|
|
mainCircleOpacity?: number;
|
||
|
|
numCircles?: number;
|
||
|
|
}
|
||
|
|
|
||
|
|
export const Ripple = React.memo(function Ripple({
|
||
|
|
mainCircleSize = 210,
|
||
|
|
mainCircleOpacity = 0.24,
|
||
|
|
numCircles = 8,
|
||
|
|
className,
|
||
|
|
...props
|
||
|
|
}: RippleProps) {
|
||
|
|
return (
|
||
|
|
<div
|
||
|
|
className={cn(
|
||
|
|
"pointer-events-none absolute inset-0 select-none [mask-image:linear-gradient(to_bottom,white,transparent)]",
|
||
|
|
className,
|
||
|
|
)}
|
||
|
|
{...props}
|
||
|
|
>
|
||
|
|
{Array.from({ length: numCircles }, (_, i) => {
|
||
|
|
const size = mainCircleSize + i * 70;
|
||
|
|
const opacity = mainCircleOpacity - i * 0.03;
|
||
|
|
const animationDelay = `${i * 0.06}s`;
|
||
|
|
const borderStyle = i === numCircles - 1 ? "dashed" : "solid";
|
||
|
|
const borderOpacity = 5 + i * 5;
|
||
|
|
|
||
|
|
return (
|
||
|
|
<div
|
||
|
|
key={i}
|
||
|
|
className={`absolute animate-ripple rounded-full border bg-foreground/25 shadow-xl`}
|
||
|
|
style={
|
||
|
|
{
|
||
|
|
"--i": i,
|
||
|
|
width: `${size}px`,
|
||
|
|
height: `${size}px`,
|
||
|
|
opacity,
|
||
|
|
animationDelay,
|
||
|
|
borderStyle,
|
||
|
|
borderWidth: "1px",
|
||
|
|
borderColor: `hsl(var(--foreground), ${borderOpacity / 100})`,
|
||
|
|
top: "50%",
|
||
|
|
left: "50%",
|
||
|
|
transform: "translate(-50%, -50%) scale(1)",
|
||
|
|
} as CSSProperties
|
||
|
|
}
|
||
|
|
/>
|
||
|
|
);
|
||
|
|
})}
|
||
|
|
</div>
|
||
|
|
);
|
||
|
|
});
|
||
|
|
|
||
|
|
Ripple.displayName = "Ripple";
|