mirror of
https://github.com/LukeHagar/better-auth.git
synced 2025-12-07 04:19:22 +00:00
239 lines
7.5 KiB
TypeScript
239 lines
7.5 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: "/people-say/dev-ed.png",
|
|
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: "/people-say/lazar-nikolov.png",
|
|
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: "/people-say/theo.png",
|
|
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: "/people-say/jonathan-wilke.png",
|
|
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: "/people-say/sebastien-chopin.png",
|
|
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: "/people-say/dax.png",
|
|
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: "/people-say/dagmawi-babi.png",
|
|
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: "/people-say/tech-nerd.png",
|
|
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: "/people-say/omar-mcadam.png",
|
|
image: "",
|
|
social: <Icons.x />,
|
|
},
|
|
{
|
|
name: "Guillermo Rauch",
|
|
profession: "CEO of Vercel",
|
|
link: "https://x.com/rauchg/status/1871628287962906846",
|
|
description: "Great project & maintainer",
|
|
avatar: "/people-say/guillermo-rauch.png",
|
|
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: "/people-say/nizzy.png",
|
|
image: "",
|
|
social: <Icons.x />,
|
|
},
|
|
|
|
{
|
|
name: "Vybhav Bhargav",
|
|
profession: "Founding engineer @glyfspace",
|
|
link: "https://x.com/vybhavab/status/1891589126513684669",
|
|
description: "better-auth is a work of art.",
|
|
avatar: "/people-say/vybhav-bhargav.png",
|
|
social: <Icons.x />,
|
|
},
|
|
{
|
|
name: "Shreyas Mididoddi",
|
|
profession: "Senior Developer",
|
|
link: "https://x.com/Shreyassanthu77/status/1887544964881784927",
|
|
description:
|
|
"dang!! Didn't think I would but absolutely loving @better_auth",
|
|
avatar: "/people-say/shreyas-mididddi.png",
|
|
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="max-w-full mx-auto">
|
|
<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="px-5 py-5 tracking-tight text-md font-extralight sm:text-xl md:text-lg text-pretty text-text-primary dark:text-dark-text-primary">
|
|
"{testimonial.description}."
|
|
</p>
|
|
<div className="flex overflow-hidden h-[28%] gap-1 w-full border-t-[1.2px]">
|
|
<div className="flex items-center w-3/4 gap-3 px-4 py-3">
|
|
<img
|
|
src={testimonial.avatar}
|
|
className="w-10 h-10 rounded-full"
|
|
alt="avatar"
|
|
/>
|
|
<div className="flex flex-col items-start justify-start flex-1 gap-0">
|
|
<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 items-center justify-center max-w-full mx-auto">
|
|
<Link href={testimonial.link} target="_blank">
|
|
{testimonial.social}
|
|
</Link>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export const Testimonial = () => {
|
|
return (
|
|
<div className="max-w-full py-5 mx-auto overflow-hidden">
|
|
<div className="flex flex-col gap-3">
|
|
<div
|
|
style={{
|
|
maskImage:
|
|
"linear-gradient(to left, transparent 0%, black 20%, black 80%, transparent 95%)",
|
|
}}
|
|
className="relative flex justify-around gap-5 overflow-hidden 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="relative flex justify-around gap-5 overflow-hidden shrink-0"
|
|
>
|
|
<PeopleSay
|
|
testimonials={Array(10)
|
|
.fill(
|
|
testimonials.slice(
|
|
Math.floor(testimonials.length / 2) + 1,
|
|
testimonials.length - 1,
|
|
),
|
|
)
|
|
.flat()}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|