magic-ui
components
ui

marquee

An infinite scrolling component that can be used to display text, images, or videos.

animated
flex
hover
View Docs

Source Code

Files
marquee
1import { cn } from "@/lib/utils";
2import { Marquee } from "@/components/magicui/marquee";
3 
4const reviews = [
5  {
6    name: "Jack",
7    username: "@jack",
8    body: "I've never seen anything like this before. It's amazing. I love it.",
9    img: "https://avatar.vercel.sh/jack",
10  },
11  {
12    name: "Jill",
13    username: "@jill",
14    body: "I don't know what to say. I'm speechless. This is amazing.",
15    img: "https://avatar.vercel.sh/jill",
16  },
17  {
18    name: "John",
19    username: "@john",
20    body: "I'm at a loss for words. This is amazing. I love it.",
21    img: "https://avatar.vercel.sh/john",
22  },
23  {
24    name: "Jane",
25    username: "@jane",
26    body: "I'm at a loss for words. This is amazing. I love it.",
27    img: "https://avatar.vercel.sh/jane",
28  },
29  {
30    name: "Jenny",
31    username: "@jenny",
32    body: "I'm at a loss for words. This is amazing. I love it.",
33    img: "https://avatar.vercel.sh/jenny",
34  },
35  {
36    name: "James",
37    username: "@james",
38    body: "I'm at a loss for words. This is amazing. I love it.",
39    img: "https://avatar.vercel.sh/james",
40  },
41];
42 
43const firstRow = reviews.slice(0, reviews.length / 2);
44const secondRow = reviews.slice(reviews.length / 2);
45 
46const ReviewCard = ({
47  img,
48  name,
49  username,
50  body,
51}: {
52  img: string;
53  name: string;
54  username: string;
55  body: string;
56}) => {
57  return (
58    <figure
59      className={cn(
60        "relative w-64 cursor-pointer overflow-hidden rounded-xl border p-4",
61        // light styles
62        "border-gray-950/[.1] bg-gray-950/[.01] hover:bg-gray-950/[.05]",
63        // dark styles
64        "dark:border-gray-50/[.1] dark:bg-gray-50/[.10] dark:hover:bg-gray-50/[.15]",
65      )}
66    >
67      <div className="flex flex-row items-center gap-2">
68        <img className="rounded-full" width="32" height="32" alt="" src={img} />
69        <div className="flex flex-col">
70          <figcaption className="text-sm font-medium dark:text-white">
71            {name}
72          </figcaption>
73          <p className="text-xs font-medium dark:text-white/40">{username}</p>
74        </div>
75      </div>
76      <blockquote className="mt-2 text-sm">{body}</blockquote>
77    </figure>
78  );
79};
80 
81export function MarqueeDemo() {
82  return (
83    <div className="relative flex h-[500px] w-full flex-col items-center justify-center overflow-hidden rounded-lg border bg-background md:shadow-xl">
84      <Marquee pauseOnHover className="[--duration:20s]">
85        {firstRow.map((review) => (
86          <ReviewCard key={review.username} {...review} />
87        ))}
88      </Marquee>
89      <Marquee reverse pauseOnHover className="[--duration:20s]">
90        {secondRow.map((review) => (
91          <ReviewCard key={review.username} {...review} />
92        ))}
93      </Marquee>
94      <div className="pointer-events-none absolute inset-y-0 left-0 w-1/3 bg-gradient-to-r from-white dark:from-background"></div>
95      <div className="pointer-events-none absolute inset-y-0 right-0 w-1/3 bg-gradient-to-l from-white dark:from-background"></div>
96    </div>
97  );
98}