feat: add AnimatedList component and its demo example to MagicuiPage
This commit is contained in:
parent
e626f57aac
commit
efba35272d
@ -1,3 +1,4 @@
|
||||
import { AnimatedListDemo } from '@/components/magicui/example/animated-list-example';
|
||||
import { BentoDemo } from '@/components/magicui/example/bento-grid-example';
|
||||
import { HeroVideoDialogDemoTopInBottomOut } from '@/components/magicui/example/hero-video-dialog-example';
|
||||
import { MarqueeDemoVertical } from '@/components/magicui/example/marquee-example';
|
||||
@ -13,6 +14,7 @@ export default async function MagicuiPage() {
|
||||
<HeroVideoDialogDemoTopInBottomOut />
|
||||
<BentoDemo />
|
||||
<MarqueeDemoVertical />
|
||||
<AnimatedListDemo />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
72
src/components/magicui/animated-list.tsx
Normal file
72
src/components/magicui/animated-list.tsx
Normal file
@ -0,0 +1,72 @@
|
||||
"use client";
|
||||
|
||||
import { cn } from "@/lib/utils";
|
||||
import { AnimatePresence, motion } from "motion/react";
|
||||
import React, {
|
||||
ComponentPropsWithoutRef,
|
||||
useEffect,
|
||||
useMemo,
|
||||
useState,
|
||||
} from "react";
|
||||
|
||||
export function AnimatedListItem({ children }: { children: React.ReactNode }) {
|
||||
const animations = {
|
||||
initial: { scale: 0, opacity: 0 },
|
||||
animate: { scale: 1, opacity: 1, originY: 0 },
|
||||
exit: { scale: 0, opacity: 0 },
|
||||
transition: { type: "spring", stiffness: 350, damping: 40 },
|
||||
};
|
||||
|
||||
return (
|
||||
<motion.div {...animations} layout className="mx-auto w-full">
|
||||
{children}
|
||||
</motion.div>
|
||||
);
|
||||
}
|
||||
|
||||
export interface AnimatedListProps extends ComponentPropsWithoutRef<"div"> {
|
||||
children: React.ReactNode;
|
||||
delay?: number;
|
||||
}
|
||||
|
||||
export const AnimatedList = React.memo(
|
||||
({ children, className, delay = 1000, ...props }: AnimatedListProps) => {
|
||||
const [index, setIndex] = useState(0);
|
||||
const childrenArray = useMemo(
|
||||
() => React.Children.toArray(children),
|
||||
[children],
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
if (index < childrenArray.length - 1) {
|
||||
const timeout = setTimeout(() => {
|
||||
setIndex((prevIndex) => (prevIndex + 1) % childrenArray.length);
|
||||
}, delay);
|
||||
|
||||
return () => clearTimeout(timeout);
|
||||
}
|
||||
}, [index, delay, childrenArray.length]);
|
||||
|
||||
const itemsToShow = useMemo(() => {
|
||||
const result = childrenArray.slice(0, index + 1).reverse();
|
||||
return result;
|
||||
}, [index, childrenArray]);
|
||||
|
||||
return (
|
||||
<div
|
||||
className={cn(`flex flex-col items-center gap-4`, className)}
|
||||
{...props}
|
||||
>
|
||||
<AnimatePresence>
|
||||
{itemsToShow.map((item) => (
|
||||
<AnimatedListItem key={(item as React.ReactElement).key}>
|
||||
{item}
|
||||
</AnimatedListItem>
|
||||
))}
|
||||
</AnimatePresence>
|
||||
</div>
|
||||
);
|
||||
},
|
||||
);
|
||||
|
||||
AnimatedList.displayName = "AnimatedList";
|
106
src/components/magicui/example/animated-list-example.tsx
Normal file
106
src/components/magicui/example/animated-list-example.tsx
Normal file
@ -0,0 +1,106 @@
|
||||
"use client";
|
||||
|
||||
import { cn } from "@/lib/utils";
|
||||
import { AnimatedList } from "@/components/magicui/animated-list";
|
||||
|
||||
interface Item {
|
||||
name: string;
|
||||
description: string;
|
||||
icon: string;
|
||||
color: string;
|
||||
time: string;
|
||||
}
|
||||
|
||||
let notifications = [
|
||||
{
|
||||
name: "Payment received",
|
||||
description: "Magic UI",
|
||||
time: "15m ago",
|
||||
|
||||
icon: "💸",
|
||||
color: "#00C9A7",
|
||||
},
|
||||
{
|
||||
name: "User signed up",
|
||||
description: "Magic UI",
|
||||
time: "10m ago",
|
||||
icon: "👤",
|
||||
color: "#FFB800",
|
||||
},
|
||||
{
|
||||
name: "New message",
|
||||
description: "Magic UI",
|
||||
time: "5m ago",
|
||||
icon: "💬",
|
||||
color: "#FF3D71",
|
||||
},
|
||||
{
|
||||
name: "New event",
|
||||
description: "Magic UI",
|
||||
time: "2m ago",
|
||||
icon: "🗞️",
|
||||
color: "#1E86FF",
|
||||
},
|
||||
];
|
||||
|
||||
notifications = Array.from({ length: 10 }, () => notifications).flat();
|
||||
|
||||
const Notification = ({ name, description, icon, color, time }: Item) => {
|
||||
return (
|
||||
<figure
|
||||
className={cn(
|
||||
"relative mx-auto min-h-fit w-full max-w-[400px] cursor-pointer overflow-hidden rounded-2xl p-4",
|
||||
// animation styles
|
||||
"transition-all duration-200 ease-in-out hover:scale-[103%]",
|
||||
// light styles
|
||||
"bg-white [box-shadow:0_0_0_1px_rgba(0,0,0,.03),0_2px_4px_rgba(0,0,0,.05),0_12px_24px_rgba(0,0,0,.05)]",
|
||||
// dark styles
|
||||
"transform-gpu dark:bg-transparent dark:backdrop-blur-md dark:[border:1px_solid_rgba(255,255,255,.1)] dark:[box-shadow:0_-20px_80px_-20px_#ffffff1f_inset]",
|
||||
)}
|
||||
>
|
||||
<div className="flex flex-row items-center gap-3">
|
||||
<div
|
||||
className="flex size-10 items-center justify-center rounded-2xl"
|
||||
style={{
|
||||
backgroundColor: color,
|
||||
}}
|
||||
>
|
||||
<span className="text-lg">{icon}</span>
|
||||
</div>
|
||||
<div className="flex flex-col overflow-hidden">
|
||||
<figcaption className="flex flex-row items-center whitespace-pre text-lg font-medium dark:text-white ">
|
||||
<span className="text-sm sm:text-lg">{name}</span>
|
||||
<span className="mx-1">·</span>
|
||||
<span className="text-xs text-gray-500">{time}</span>
|
||||
</figcaption>
|
||||
<p className="text-sm font-normal dark:text-white/60">
|
||||
{description}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</figure>
|
||||
);
|
||||
};
|
||||
|
||||
export function AnimatedListDemo({
|
||||
className,
|
||||
}: {
|
||||
className?: string;
|
||||
}) {
|
||||
return (
|
||||
<div
|
||||
className={cn(
|
||||
"relative flex h-[500px] w-full flex-col overflow-hidden p-2",
|
||||
className,
|
||||
)}
|
||||
>
|
||||
<AnimatedList>
|
||||
{notifications.map((item, idx) => (
|
||||
<Notification {...item} key={idx} />
|
||||
))}
|
||||
</AnimatedList>
|
||||
|
||||
<div className="pointer-events-none absolute inset-x-0 bottom-0 h-1/4 bg-gradient-to-t from-background"></div>
|
||||
</div>
|
||||
);
|
||||
}
|
Loading…
Reference in New Issue
Block a user