All components
Infinite Moving Cards
carouselsAceternity UI component.
responsive · 480px
Install
Same command in any shadcn project — React (Vite/CRA), Next.js, Remix, Astro, and more:
$
npx shadcn@latest add https://your-domain/r/infinite-moving-cards.jsonUsage
"use client";
import { InfiniteMovingCards } from "@/registry/aceternity-ui/infinite-moving-cards";
const testimonials = [
{
quote: "This component library is absolutely stunning. The animations are smooth and the design is top-notch.",
name: "Alice Johnson",
title: "Senior Frontend Engineer",
},
{
quote: "I've never seen such beautiful UI components. They integrate seamlessly into any project.",
name: "Bob Martinez",
title: "Product Designer",
},
{
quote: "The infinite scroll effect is mesmerizing. Our users love it and engagement has increased significantly.",
name: "Carol Chen",
title: "Head of Product",
},
{
quote: "Building with these components saves us hours every week. The quality is simply unmatched.",
name: "David Kim",
title: "Full Stack Developer",
},
{
quote: "Our design system transformed overnight thanks to these incredible ready-to-use components.",
name: "Emma Wilson",
title: "CTO at Acme Corp",
},
];
export default function Demo() {
return (
<div className="w-full max-w-5xl flex flex-col items-center justify-center py-10">
<InfiniteMovingCards items={testimonials} direction="left" speed="slow" />
</div>
);
}Component source
"use client";
import { cn } from "@/lib/utils";
import React, { useEffect, useState } from "react";
export const InfiniteMovingCards = ({
items,
direction = "left",
speed = "fast",
pauseOnHover = true,
className,
}: {
items: {
quote: string;
name: string;
title: string;
}[];
direction?: "left" | "right";
speed?: "fast" | "normal" | "slow";
pauseOnHover?: boolean;
className?: string;
}) => {
const containerRef = React.useRef<HTMLDivElement>(null);
const scrollerRef = React.useRef<HTMLUListElement>(null);
useEffect(() => {
addAnimation();
}, []);
const [start, setStart] = useState(false);
function addAnimation() {
if (containerRef.current && scrollerRef.current) {
const scrollerContent = Array.from(scrollerRef.current.children);
scrollerContent.forEach((item) => {
const duplicatedItem = item.cloneNode(true);
if (scrollerRef.current) {
scrollerRef.current.appendChild(duplicatedItem);
}
});
getDirection();
getSpeed();
setStart(true);
}
}
const getDirection = () => {
if (containerRef.current) {
if (direction === "left") {
containerRef.current.style.setProperty(
"--animation-direction",
"forwards",
);
} else {
containerRef.current.style.setProperty(
"--animation-direction",
"reverse",
);
}
}
};
const getSpeed = () => {
if (containerRef.current) {
if (speed === "fast") {
containerRef.current.style.setProperty("--animation-duration", "20s");
} else if (speed === "normal") {
containerRef.current.style.setProperty("--animation-duration", "40s");
} else {
containerRef.current.style.setProperty("--animation-duration", "80s");
}
}
};
return (
<div
ref={containerRef}
className={cn(
"scroller relative z-20 max-w-7xl overflow-hidden [mask-image:linear-gradient(to_right,transparent,white_20%,white_80%,transparent)]",
className,
)}
>
<ul
ref={scrollerRef}
className={cn(
"flex w-max min-w-full shrink-0 flex-nowrap gap-4 py-4",
start && "animate-scroll",
pauseOnHover && "hover:[animation-play-state:paused]",
)}
>
{items.map((item, idx) => (
<li
className="relative w-[350px] max-w-full shrink-0 rounded-2xl border border-b-0 border-zinc-200 bg-[linear-gradient(180deg,#fafafa,#f5f5f5)] px-8 py-6 md:w-[450px] dark:border-zinc-700 dark:bg-[linear-gradient(180deg,#27272a,#18181b)]"
key={item.name}
>
<blockquote>
<div
aria-hidden="true"
className="user-select-none pointer-events-none absolute -top-0.5 -left-0.5 -z-1 h-[calc(100%_+_4px)] w-[calc(100%_+_4px)]"
></div>
<span className="relative z-20 text-sm leading-[1.6] font-normal text-neutral-800 dark:text-gray-100">
{item.quote}
</span>
<div className="relative z-20 mt-6 flex flex-row items-center">
<span className="flex flex-col gap-1">
<span className="text-sm leading-[1.6] font-normal text-neutral-500 dark:text-gray-400">
{item.name}
</span>
<span className="text-sm leading-[1.6] font-normal text-neutral-500 dark:text-gray-400">
{item.title}
</span>
</span>
</div>
</blockquote>
</li>
))}
</ul>
</div>
);
};Source: Aceternity UI