From c121a12f03a07f5d144ccbd4ccbdcb884687b6cf Mon Sep 17 00:00:00 2001 From: Arghya Das Date: Fri, 27 Dec 2024 09:37:46 +0530 Subject: [PATCH] feat: added spotlight effect on border in magic card --- content/docs/components/magic-card.mdx | 16 +++++++------- public/r/styles/default/magic-card.json | 2 +- registry/default/example/magic-card-demo.tsx | 6 +++-- registry/default/magicui/magic-card.tsx | 23 +++++++++++++++++--- 4 files changed, 33 insertions(+), 14 deletions(-) diff --git a/content/docs/components/magic-card.mdx b/content/docs/components/magic-card.mdx index c6afc5f95..44b98747a 100644 --- a/content/docs/components/magic-card.mdx +++ b/content/docs/components/magic-card.mdx @@ -44,14 +44,14 @@ npx shadcn@latest add "https://magicui.design/r/magic-card" ### MagicCard -| Prop name | Type | Default | Description | -| -------------------- | --------------- | --------- | ------------------------------------------- | -| children | React.ReactNode | - | The content to be rendered inside the card | -| className | string | - | Additional CSS classes to apply to the card | -| gradientSize | number | 200 | Size of the gradient effect | -| gradientColor | string | "#262626" | Color of the gradient effect | -| gradientOpacity | number | - | Opacity of the gradient effect | -| gradientTransparency | number | 80 | Transparency level of the gradient effect | +| Prop name | Type | Default | Description | +| --------------- | --------------- | --------- | ------------------------------------------- | +| children | React.ReactNode | - | The content to be rendered inside the card | +| className | string | - | Additional CSS classes to apply to the card | +| gradientSize | number | 200 | Size of the gradient effect | +| gradientColor | string | "#262626" | Color of the gradient effect | +| gradientOpacity | number | - | Opacity of the gradient effect | +| gradientColors | string[] | - | Colors of the gradient effect | ## Credits diff --git a/public/r/styles/default/magic-card.json b/public/r/styles/default/magic-card.json index 8d93efa66..7fa1d6f96 100644 --- a/public/r/styles/default/magic-card.json +++ b/public/r/styles/default/magic-card.json @@ -7,7 +7,7 @@ "files": [ { "path": "magicui/magic-card.tsx", - "content": "\"use client\";\n\nimport React, { useCallback, useEffect, useRef } from \"react\";\nimport { motion, useMotionTemplate, useMotionValue } from \"framer-motion\";\n\nimport { cn } from \"@/lib/utils\";\n\nexport interface MagicCardProps extends React.HTMLAttributes {\n gradientSize?: number;\n gradientColor?: string;\n gradientOpacity?: number;\n}\n\nexport function MagicCard({\n children,\n className,\n gradientSize = 200,\n gradientColor = \"#262626\",\n gradientOpacity = 0.8,\n}: MagicCardProps) {\n const cardRef = useRef(null);\n const mouseX = useMotionValue(-gradientSize);\n const mouseY = useMotionValue(-gradientSize);\n\n const handleMouseMove = useCallback(\n (e: MouseEvent) => {\n if (cardRef.current) {\n const { left, top } = cardRef.current.getBoundingClientRect();\n const clientX = e.clientX;\n const clientY = e.clientY;\n mouseX.set(clientX - left);\n mouseY.set(clientY - top);\n }\n },\n [mouseX, mouseY],\n );\n\n const handleMouseOut = useCallback(\n (e: MouseEvent) => {\n if (!e.relatedTarget) {\n document.removeEventListener(\"mousemove\", handleMouseMove);\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }\n },\n [handleMouseMove, mouseX, gradientSize, mouseY],\n );\n\n const handleMouseEnter = useCallback(() => {\n document.addEventListener(\"mousemove\", handleMouseMove);\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }, [handleMouseMove, mouseX, gradientSize, mouseY]);\n\n useEffect(() => {\n document.addEventListener(\"mousemove\", handleMouseMove);\n document.addEventListener(\"mouseout\", handleMouseOut);\n document.addEventListener(\"mouseenter\", handleMouseEnter);\n\n return () => {\n document.removeEventListener(\"mousemove\", handleMouseMove);\n document.removeEventListener(\"mouseout\", handleMouseOut);\n document.removeEventListener(\"mouseenter\", handleMouseEnter);\n };\n }, [handleMouseEnter, handleMouseMove, handleMouseOut]);\n\n useEffect(() => {\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }, [gradientSize, mouseX, mouseY]);\n\n return (\n \n
{children}
\n \n \n );\n}\n", + "content": "\"use client\";\n\nimport React, { useCallback, useEffect, useRef } from \"react\";\nimport { motion, useMotionTemplate, useMotionValue } from \"framer-motion\";\n\nimport { cn } from \"@/lib/utils\";\n\nexport interface MagicCardProps extends React.HTMLAttributes {\n gradientSize?: number;\n gradientColor?: string;\n gradientOpacity?: number;\n gradientColors?: [string, string, string];\n}\n\nexport function MagicCard({\n children,\n className,\n gradientSize = 200,\n gradientColor = \"#262626\",\n gradientOpacity = 0.8,\n gradientColors = [\"#FF0080\", \"#4096FF\", \"#78FF78\"],\n}: MagicCardProps) {\n const cardRef = useRef(null);\n const mouseX = useMotionValue(-gradientSize);\n const mouseY = useMotionValue(-gradientSize);\n\n const handleMouseMove = useCallback(\n (e: MouseEvent) => {\n if (cardRef.current) {\n const { left, top } = cardRef.current.getBoundingClientRect();\n const clientX = e.clientX;\n const clientY = e.clientY;\n mouseX.set(clientX - left);\n mouseY.set(clientY - top);\n }\n },\n [mouseX, mouseY]\n );\n\n const handleMouseOut = useCallback(\n (e: MouseEvent) => {\n if (!e.relatedTarget) {\n document.removeEventListener(\"mousemove\", handleMouseMove);\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }\n },\n [handleMouseMove, mouseX, gradientSize, mouseY]\n );\n\n const handleMouseEnter = useCallback(() => {\n document.addEventListener(\"mousemove\", handleMouseMove);\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }, [handleMouseMove, mouseX, gradientSize, mouseY]);\n\n useEffect(() => {\n document.addEventListener(\"mousemove\", handleMouseMove);\n document.addEventListener(\"mouseout\", handleMouseOut);\n document.addEventListener(\"mouseenter\", handleMouseEnter);\n\n return () => {\n document.removeEventListener(\"mousemove\", handleMouseMove);\n document.removeEventListener(\"mouseout\", handleMouseOut);\n document.removeEventListener(\"mouseenter\", handleMouseEnter);\n };\n }, [handleMouseEnter, handleMouseMove, handleMouseOut]);\n\n useEffect(() => {\n mouseX.set(-gradientSize);\n mouseY.set(-gradientSize);\n }, [gradientSize, mouseX, mouseY]);\n\n return (\n \n
\n
{children}
\n \n \n
\n );\n}\n", "type": "registry:ui", "target": "" } diff --git a/registry/default/example/magic-card-demo.tsx b/registry/default/example/magic-card-demo.tsx index 7128bff3b..a87e53aa7 100644 --- a/registry/default/example/magic-card-demo.tsx +++ b/registry/default/example/magic-card-demo.tsx @@ -11,14 +11,16 @@ export default function MagicCardDemo() { } > Magic Card diff --git a/registry/default/magicui/magic-card.tsx b/registry/default/magicui/magic-card.tsx index e90ece6ff..ebc3e27f1 100644 --- a/registry/default/magicui/magic-card.tsx +++ b/registry/default/magicui/magic-card.tsx @@ -9,6 +9,7 @@ export interface MagicCardProps extends React.HTMLAttributes { gradientSize?: number; gradientColor?: string; gradientOpacity?: number; + gradientColors?: [string, string, string]; } export function MagicCard({ @@ -17,6 +18,7 @@ export function MagicCard({ gradientSize = 200, gradientColor = "#262626", gradientOpacity = 0.8, + gradientColors = ["#FF0080", "#4096FF", "#78FF78"], }: MagicCardProps) { const cardRef = useRef(null); const mouseX = useMotionValue(-gradientSize); @@ -73,13 +75,14 @@ export function MagicCard({
-
{children}
+
+
{children}
+
); }