diff options
author | sotech117 <michael_foiani@brown.edu> | 2025-09-23 01:46:55 -0400 |
---|---|---|
committer | sotech117 <michael_foiani@brown.edu> | 2025-09-23 01:46:55 -0400 |
commit | 3a712982307391ae1196f50e252cb37ed5f67ccb (patch) | |
tree | 279aa2d97198a08aef2eb1c80bc5008763eb98da /src/components/ui/wooble-card.tsx | |
parent | c5547dc1ba827a4256fbe1ed08bda61f8c660815 (diff) |
add hero and get-started page
Diffstat (limited to 'src/components/ui/wooble-card.tsx')
-rw-r--r-- | src/components/ui/wooble-card.tsx | 78 |
1 files changed, 78 insertions, 0 deletions
diff --git a/src/components/ui/wooble-card.tsx b/src/components/ui/wooble-card.tsx new file mode 100644 index 0000000..05c059a --- /dev/null +++ b/src/components/ui/wooble-card.tsx @@ -0,0 +1,78 @@ +"use client"; +import { cn } from "@/lib/utils"; +import { motion } from "motion/react"; +import React, { useState } from "react"; + +export const WobbleCard = ({ + children, + containerClassName, + className, +}: { + children: React.ReactNode; + containerClassName?: string; + className?: string; +}) => { + const [mousePosition, setMousePosition] = useState({ x: 0, y: 0 }); + const [isHovering, setIsHovering] = useState(false); + + const handleMouseMove = (event: React.MouseEvent<HTMLElement>) => { + const { clientX, clientY } = event; + const rect = event.currentTarget.getBoundingClientRect(); + const x = (clientX - (rect.left + rect.width / 2)) / 20; + const y = (clientY - (rect.top + rect.height / 2)) / 20; + setMousePosition({ x, y }); + }; + return ( + <motion.section + onMouseMove={handleMouseMove} + onMouseEnter={() => setIsHovering(true)} + onMouseLeave={() => { + setIsHovering(false); + setMousePosition({ x: 0, y: 0 }); + }} + style={{ + transform: isHovering + ? `translate3d(${mousePosition.x}px, ${mousePosition.y}px, 0) scale3d(1, 1, 1)` + : "translate3d(0px, 0px, 0) scale3d(1, 1, 1)", + transition: "transform 0.1s ease-out", + }} + className={cn( + "mx-auto w-full bg-indigo-800 relative rounded-2xl overflow-hidden", + containerClassName + )} + > + <div + className="relative h-full [background-image:radial-gradient(88%_100%_at_top,rgba(255,255,255,0.5),rgba(255,255,255,0))] sm:mx-0 sm:rounded-2xl overflow-hidden" + style={{ + boxShadow: + "0 10px 32px rgba(34, 42, 53, 0.12), 0 1px 1px rgba(0, 0, 0, 0.05), 0 0 0 1px rgba(34, 42, 53, 0.05), 0 4px 6px rgba(34, 42, 53, 0.08), 0 24px 108px rgba(47, 48, 55, 0.10)", + }} + > + <motion.div + style={{ + transform: isHovering + ? `translate3d(${-mousePosition.x}px, ${-mousePosition.y}px, 0) scale3d(1.03, 1.03, 1)` + : "translate3d(0px, 0px, 0) scale3d(1, 1, 1)", + transition: "transform 0.1s ease-out", + }} + className={cn("h-full px-4 py-20 sm:px-10", className)} + > + <Noise /> + {children} + </motion.div> + </div> + </motion.section> + ); +}; + +const Noise = () => { + return ( + <div + className="absolute inset-0 w-full h-full scale-[1.2] transform opacity-10 [mask-image:radial-gradient(#fff,transparent,75%)]" + style={{ + backgroundImage: "url(/noise.webp)", + backgroundSize: "30%", + }} + ></div> + ); +}; |