mirror of
https://github.com/supabase/supabase.git
synced 2026-06-09 19:50:28 +08:00
This PR migrates the JS config for Tailwind into a CSS config. As such, all variables have been defined as CSS variables and they're using the specialized Tailwind syntax for generating utility classes. Beside the migration, these changes were also added: - Added `tailwind.config.css` to few packages to make the Tailwind Intellisense work. - Migrated away from Radix style color classes to our defined classes, the values will remain the same. - Most of the CSS is generated by scripts, they'll be removed in next PRs. * Removed redundant `border-light` classes from several components since it was undefined. * Removed redundant `text-strong` classes from several components since it was undefined. How to test: - Open all apps, compare the UI (mainly colors) to builds from #45417 and try to find a difference. <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Style** * Harmonized color variable usages and updated UI color references (affects palettes, charts, gradients, hero illustrations, and scrollbars). * Tweaked border, tab, and selection visuals across components. * **New Features** * Added a suite of theme animations and refined typography presets used by site prose and docs. * **Refactor** * Overhauled Tailwind/theme configuration and color token generation for more consistent theming. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
237 lines
8.5 KiB
TypeScript
237 lines
8.5 KiB
TypeScript
import { AnimatePresence } from 'framer-motion'
|
|
import { Pencil, Users, Users2 } from 'lucide-react'
|
|
import Link from 'next/link'
|
|
import React, { useEffect, useState } from 'react'
|
|
import { Swiper, SwiperSlide } from 'swiper/react'
|
|
import { cn, LogoLoader } from 'ui'
|
|
|
|
import RepoCard from './RepoCard'
|
|
|
|
interface TabProps {
|
|
label: string
|
|
isActive: boolean
|
|
icon?: string
|
|
onClick: VoidFunction
|
|
}
|
|
|
|
const Tab = ({ isActive, label, icon, onClick }: TabProps) => (
|
|
<button
|
|
onClick={onClick}
|
|
className={`rounded-full px-4 md:px-3 py-2 md:py-1 nowrap flex group gap-1 transition-all ${
|
|
isActive ? 'bg-surface-300' : 'text-foreground-lighter bg-surface-200 hover:bg-overlay-hover'
|
|
}`}
|
|
aria-selected={isActive}
|
|
role="tab"
|
|
>
|
|
<div className="flex flex-nowrap text-sm lg:text-base gap-1 lg:gap-2 items-center">
|
|
{icon && (
|
|
<svg
|
|
width="16"
|
|
height="16"
|
|
viewBox="0 0 16 16"
|
|
fill="none"
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
className={cn('text-foreground-light shrink-0', isActive && 'text-brand')}
|
|
>
|
|
<path
|
|
d={icon}
|
|
stroke="currentColor"
|
|
strokeMiterlimit="10"
|
|
strokeLinecap="round"
|
|
strokeLinejoin="round"
|
|
/>
|
|
</svg>
|
|
)}
|
|
<span className="whitespace-nowrap tracking-tight lg:tracking-normal">{label}</span>
|
|
</div>
|
|
</button>
|
|
)
|
|
|
|
export interface RepoTab {
|
|
label: string
|
|
icon?: string
|
|
repos?: string[]
|
|
}
|
|
|
|
interface Props {
|
|
tabs: RepoTab[]
|
|
}
|
|
|
|
enum SWIPER_STATE {
|
|
START = 'START',
|
|
MIDDLE = 'MIDDLE',
|
|
END = 'END',
|
|
}
|
|
|
|
const Repos = ({ tabs }: Props) => {
|
|
const [repos, setRepos] = useState<any[] | null>(null)
|
|
const [activeTab, setActiveTab] = useState(0)
|
|
const [apiSwiper, setApiSwiper] = useState(undefined)
|
|
const [swiperState, setSwiperState] = useState<SWIPER_STATE>(SWIPER_STATE.START)
|
|
|
|
useEffect(() => {
|
|
async function fetchOctoData() {
|
|
const { Octokit } = await import('@octokit/core')
|
|
const octokit = new Octokit()
|
|
const res = await octokit.request('GET /orgs/{org}/repos', {
|
|
org: 'supabase',
|
|
type: 'public',
|
|
per_page: 200,
|
|
page: 1,
|
|
})
|
|
|
|
setRepos(res.data)
|
|
}
|
|
fetchOctoData()
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
if (!apiSwiper) return
|
|
// @ts-ignore
|
|
apiSwiper.slideTo(activeTab)
|
|
}, [activeTab])
|
|
|
|
const handleTabClick = (tabIndex: number) => {
|
|
setActiveTab(tabIndex)
|
|
}
|
|
|
|
const activeTabRepos = repos
|
|
? repos
|
|
?.filter((repo) => tabs[activeTab].repos?.includes(repo.name))
|
|
?.sort((a, b) => (a.stargazers_count < b.stargazers_count ? 1 : -1))
|
|
: []
|
|
|
|
return (
|
|
<div className="flex flex-col gap-8 xl:gap-10">
|
|
<div className="flex flex-wrap mx-auto items-center justify-center gap-y-6 gap-x-8 text-sm">
|
|
<Link
|
|
href="https://github.com/supabase/supabase/blob/master/DEVELOPERS.md"
|
|
className="text-foreground-lighter hover:underline flex gap-1 items-center"
|
|
target="_blank"
|
|
>
|
|
<GitHubIcon />
|
|
How to contribute
|
|
</Link>
|
|
<Link
|
|
href="https://github.com/supabase/.github/blob/main/CODE_OF_CONDUCT.md"
|
|
className="text-foreground-lighter hover:underline flex gap-1 items-center"
|
|
target="_blank"
|
|
>
|
|
<GitHubIcon />
|
|
Code of Conduct
|
|
</Link>
|
|
<Link
|
|
href="/open-source/contributing/supasquad"
|
|
className="text-foreground-lighter hover:underline flex gap-1.5 items-center"
|
|
>
|
|
<Users className="w-4 h-4" />
|
|
SupaSquad
|
|
</Link>
|
|
<Link
|
|
href="https://docs.google.com/forms/d/e/1FAIpQLSfJoQ6_uWymc4DJok2YVY8K_jp27S6HrnOIKmtHuDhBCWetDg/viewform?pli=1"
|
|
className="text-foreground-lighter hover:underline flex gap-1.5 items-center"
|
|
target="_blank"
|
|
>
|
|
<Pencil className="w-4 h-4" />
|
|
Community Content Program
|
|
</Link>
|
|
</div>
|
|
<div className="w-full gap-2 flex flex-col items-center">
|
|
<div className="relative flex border justify-center h-fit max-w-full w-full md:w-auto overflow-hidden items-center rounded-full bg-surface-100 [&_.swiper-wrapper]:w-full [&_.swiper-slide]:w-fit">
|
|
<Swiper
|
|
// @ts-ignore
|
|
onSwiper={setApiSwiper}
|
|
style={{ padding: 10 }}
|
|
initialSlide={0}
|
|
spaceBetween={10}
|
|
grabCursor
|
|
slidesPerView="auto"
|
|
onSlideChange={(slider) =>
|
|
setSwiperState(
|
|
slider.isEnd
|
|
? SWIPER_STATE.END
|
|
: slider.isBeginning
|
|
? SWIPER_STATE.START
|
|
: SWIPER_STATE.MIDDLE
|
|
)
|
|
}
|
|
className="relative flex md:hidden! justify-center max-w-full w-full overflow-hidden items-center rounded-full bg-surface-100 p-2"
|
|
>
|
|
<div
|
|
className={cn(
|
|
'not-sr-only absolute inset-0 left-auto bg-linear-to-r from-transparent to-background-surface-100 w-10 z-20 pointer-events-none opacity-0 transition-opacity',
|
|
swiperState !== SWIPER_STATE.END && 'opacity-100'
|
|
)}
|
|
/>
|
|
<div
|
|
className={cn(
|
|
'not-sr-only absolute inset-0 right-auto bg-linear-to-l from-transparent to-background-surface-100 w-10 z-20 pointer-events-none opacity-0 transition-opacity',
|
|
swiperState !== SWIPER_STATE.START && 'opacity-100'
|
|
)}
|
|
/>
|
|
{tabs.map((tab, index) => (
|
|
<SwiperSlide key={tab.label}>
|
|
<Tab
|
|
isActive={index === activeTab}
|
|
label={tab.label}
|
|
icon={tab.icon}
|
|
onClick={() => handleTabClick(index)}
|
|
/>
|
|
</SwiperSlide>
|
|
))}
|
|
</Swiper>
|
|
<div className="hidden md:flex flex-nowrap overflow-x-scroll items-center p-2 md:p-1 gap-2 no-scrollbar">
|
|
{tabs.map((tab, index) => (
|
|
<Tab
|
|
key={tab.label}
|
|
isActive={index === activeTab}
|
|
label={tab.label}
|
|
icon={tab.icon}
|
|
onClick={() => handleTabClick(index)}
|
|
/>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div className="relative w-full h-fit grid md:grid-cols-2 lg:grid-cols-3 gap-4">
|
|
<AnimatePresence mode="wait">
|
|
{repos === null ? (
|
|
<div className="col-span-full flex justify-center items-center min-h-[300px]">
|
|
<LogoLoader />
|
|
</div>
|
|
) : (
|
|
activeTabRepos?.map((repo: any, i: number) => (
|
|
<RepoCard
|
|
key={`${activeTab}-${repo.name}`}
|
|
repo={repo}
|
|
activeTab={activeTab}
|
|
index={i}
|
|
/>
|
|
))
|
|
)}
|
|
</AnimatePresence>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
const GitHubIcon = () => (
|
|
<svg
|
|
width="24"
|
|
height="24"
|
|
viewBox="0 0 24 24"
|
|
fill="none"
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
className="fill-foreground-lighter grouop-hover:fill-foreground w-4 h-4"
|
|
>
|
|
<path
|
|
fillRule="evenodd"
|
|
clipRule="evenodd"
|
|
d="M12 3.33215C7.09969 3.33215 3.12744 7.31061 3.12744 12.2198C3.12744 16.1459 5.66943 19.4775 9.19538 20.6523C9.63901 20.7339 9.80049 20.4597 9.80049 20.2237C9.80049 20.0135 9.7934 19.4536 9.78896 18.7127C7.32061 19.2495 6.79979 17.5211 6.79979 17.5211C6.39698 16.4937 5.81494 16.2204 5.81494 16.2204C5.00931 15.6703 5.87616 15.681 5.87616 15.681C6.76608 15.7431 7.23455 16.5966 7.23455 16.5966C8.02598 17.9541 9.31161 17.562 9.81646 17.3348C9.89809 16.7608 10.127 16.3695 10.3808 16.1477C8.41105 15.9232 6.33931 15.1602 6.33931 11.7549C6.33931 10.7851 6.68534 9.99101 7.25229 9.36993C7.16091 9.14545 6.85658 8.24134 7.33925 7.0187C7.33925 7.0187 8.08454 6.77914 9.7792 7.92903C10.503 7.73162 11.2498 7.63108 12 7.63002C12.7542 7.63357 13.5128 7.73206 14.2217 7.92903C15.9155 6.77914 16.659 7.01781 16.659 7.01781C17.1434 8.24134 16.8382 9.14545 16.7477 9.36993C17.3155 9.99101 17.6598 10.7851 17.6598 11.7549C17.6598 15.169 15.5845 15.9205 13.6086 16.1406C13.9271 16.4147 14.2102 16.9569 14.2102 17.7864C14.2102 18.9736 14.1995 19.9327 14.1995 20.2237C14.1995 20.4615 14.3592 20.7383 14.8099 20.6514C16.5767 20.0588 18.1126 18.9259 19.2005 17.4129C20.2884 15.8999 20.8733 14.0833 20.8726 12.2198C20.8726 7.31061 16.8994 3.33215 12 3.33215Z"
|
|
fill="currentColor"
|
|
/>
|
|
</svg>
|
|
)
|
|
|
|
export default Repos
|