Files
better-auth/docs/components/landing/people-say.tsx
2025-02-21 12:00:21 +03:00

252 lines
8.2 KiB
TypeScript

import { cn } from "@/lib/utils";
import { Icons } from "../icons";
import Link from "next/link";
const testimonials = [
{
name: "Dev Ed",
profession: "Content Creator",
link: "https://x.com/edgarasben/status/1856336936505590160",
description:
"This has been the best auth experience by a mileee, auto generated my drizzle schemas for users, sessions etc, full type safe and dead simple api, well done @better_auth 👏👏",
avatar:
"https://pbs.twimg.com/profile_images/1620476753398452224/fcozbw1J_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Lazar Nikolov",
profession: "Software Engineer & Educator",
link: "https://x.com/NikolovLazar/status/1888992999872331985",
description:
"I didn't know @better_auth was THAT good. I'm implementing it in TanStack Start and I can't believe how good the DX is. This is my favorite stack now (along with @DrizzleORM and @shadcn ui).",
avatar:
"https://pbs.twimg.com/profile_images/1838045335282745344/sCjDaZ2-_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Theo - t3.gg",
profession: "CEO of t3.chat",
link: "https://x.com/theo/status/1879769267866120341",
description: "Very exciting project and a super easy rec",
avatar:
"https://pbs.twimg.com/profile_images/1799982162831396865/Fnol01I1_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Jonathan Wilke",
profession: "Creator of Supastarter",
link: "https://x.com/jonathan_wilke/status/1853086900279562329",
description: "fuck, @better_auth is just so good",
avatar:
"https://pbs.twimg.com/profile_images/1884529433979068416/AhfbeVEh_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Sébastien Chopin",
profession: "Creator of Nuxt & NuxtLabs",
link: "https://x.com/Atinux/status/1853751424561336322",
description:
"When @better_auth meets @nuxt_hub to build full-stack Nuxt apps on Cloudflare (using D1 & KV).",
avatar:
"https://pbs.twimg.com/profile_images/1862505215624142848/7tSrng8r_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Dax",
profession: "Creator of SST",
link: "https://x.com/thdxr/status/1866222656468705426",
description:
"between better-auth and openauth one of those options should cover how you want to do things for 95% of cases. the problem of defaulting to SaaS for auth in js is finally fixed...",
avatar:
"https://pbs.twimg.com/profile_images/1602333093485891584/mmVqjFNI_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Dagmawi Babi",
profession: "Developer",
link: "https://x.com/DagmawiBabi/status/1845966382703280458",
description:
"@better_auth exceeded all expectations, and it's just getting started",
avatar:
"https://pbs.twimg.com/profile_images/1853424779392380928/NMpggRqG_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Tech Nerd",
profession: "Developer",
link: "https://x.com/TechNerd556/status/1863523931614822784",
description:
"Using @better_auth with custom components feels like having someone hand you the remote while you're comfortably on the sofa. The ease I'm feeling rn is insane Auth done in under 5 minutes 🤌⚡️.",
avatar:
"https://pbs.twimg.com/profile_images/1826246307326902273/Ee4nlPjH_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Omar McAdam",
profession: "Creator of AugmentedHQ",
link: "https://x.com/McPizza0/status/1879526862046839249",
description:
"if you're building a code project in 2025 use @better_auth. It has everything you need now and everything you'll need at scale. dont take this suggestion lightly..",
avatar:
"https://pbs.twimg.com/profile_images/1889741029470801920/6J25rfKo_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Guillermo Rauch",
profession: "CEO of Vercel",
link: "https://x.com/rauchg/status/1871628287962906846",
description: "Great project & maintainer",
avatar:
"https://pbs.twimg.com/profile_images/1783856060249595904/8TfcCN0r_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Nizzy",
profession: "engineer @calcom",
link: "https://x.com/NizzyABI/status/1889178812459422162",
description:
"i cant believe how easy @better_auth is compared to @authjs all i had to do was connect it to my drizzle schema and create a sign up page w the auth :)))",
avatar:
"https://pbs.twimg.com/profile_images/1884987569961570304/TP3OWz64_400x400.jpg",
image: "",
social: <Icons.x />,
},
{
name: "Vybhav Bhargav",
description: "better-auth is a work of art.",
profession: "Founding engineer @glyfspace",
avatar:
"https://pbs.twimg.com/profile_images/1690579403070451712/zs46gsPl_400x400.jpg",
link: "https://x.com/vybhavab/status/1891589126513684669",
social: <Icons.x />,
},
{
name: "Shreyas Mididoddi",
description:
"dang!! Didn't think I would but absolutely loving @better_auth",
profession: "Senior Developer",
avatar:
"https://pbs.twimg.com/profile_images/1690579403070451712/zs46gsPl_400x400.jpg",
link: "https://x.com/vybhavab/status/1891589126513684669",
social: <Icons.x />,
},
];
type TestimonialProps = (typeof testimonials)[number];
const PeopleSay = ({
reverse = false,
testimonials,
}: {
reverse?: boolean;
testimonials: TestimonialProps[];
}) => {
const animeSeconds = testimonials.length * 10;
return (
<div className="mx-auto max-w-full">
<div
className={`[--anime-duration:${animeSeconds}s] px-10 mx-auto w-full`}
>
<div
style={{
animationDuration: `${animeSeconds}s`,
}}
className={cn(
"scroller flex flex-nowrap w-max min-w-full duration-[1000s] hover:[animation-play-state:paused] overflow-hidden relative gap-5 justify-around shrink-0",
reverse ? "animate-hrtl-scroll-reverse " : "animate-hrtl-scroll",
)}
>
{testimonials.map((testimonial, indx) => {
return (
<div
key={indx}
className={cn(
"flex flex-col justify-between h-[220px] rounded-none border-[1.2px] border-black/20 shrink-0 grow-0 w-[450px] dark:border-white/10",
)}
>
<p className="py-5 px-5 text-md font-extralight tracking-tight sm:text-xl md:text-lg text-pretty text-text-primary dark:text-dark-text-primary">
&quot;{testimonial.description}.&quot;
</p>
<div className="flex overflow-hidden h-[28%] gap-1 w-full border-t-[1.2px]">
<div className="flex gap-3 items-center py-3 px-4 w-3/4">
<img
src={testimonial.avatar}
className="w-10 rounded-full h-10"
alt="avatar"
/>
<div className="flex flex-col flex-1 gap-0 justify-start items-start">
<h5 className="text-base font-medium md:text-md">
{testimonial.name}
</h5>
<p className="text-sm md:text-base text-black/30 mt-[-4px] text-text-tertiary dark:text-white/50 dark:text-dark-text-tertiary">
{testimonial.profession}
</p>
</div>
</div>
<div className="w-[1px] bg-black/20 dark:bg-white/20" />
<div className="flex justify-center items-center mx-auto max-w-full">
<Link href={testimonial.link} target="_blank">
{testimonial.social}
</Link>
</div>
</div>
</div>
);
})}
</div>
</div>
</div>
);
};
export const Testimonial = () => {
return (
<div className="mx-auto overflow-hidden py-5 max-w-full">
<div className="flex flex-col gap-3">
<div
style={{
maskImage:
"linear-gradient(to left, transparent 0%, black 20%, black 80%, transparent 95%)",
}}
className="flex overflow-hidden relative gap-5 justify-around shrink-0"
>
<PeopleSay
reverse
testimonials={Array(10)
.fill(testimonials.slice(0, Math.floor(testimonials.length / 2)))
.flat()}
/>
</div>
<div
style={{
maskImage:
"linear-gradient(to left, transparent 0%, black 20%, black 80%, transparent 95%)",
}}
className="flex overflow-hidden relative gap-5 justify-around shrink-0"
>
<PeopleSay
testimonials={Array(10)
.fill(
testimonials.slice(
Math.floor(testimonials.length / 2) + 1,
testimonials.length - 1,
),
)
.flat()}
/>
</div>
</div>
</div>
);
};