aboutsummaryrefslogtreecommitdiff
path: root/src/components/ui/wooble-card.tsx
diff options
context:
space:
mode:
authorsotech117 <michael_foiani@brown.edu>2025-09-23 01:46:55 -0400
committersotech117 <michael_foiani@brown.edu>2025-09-23 01:46:55 -0400
commit3a712982307391ae1196f50e252cb37ed5f67ccb (patch)
tree279aa2d97198a08aef2eb1c80bc5008763eb98da /src/components/ui/wooble-card.tsx
parentc5547dc1ba827a4256fbe1ed08bda61f8c660815 (diff)
add hero and get-started page
Diffstat (limited to 'src/components/ui/wooble-card.tsx')
-rw-r--r--src/components/ui/wooble-card.tsx78
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>
+ );
+};