diff --git a/swc-plugins/app/(home)/page.tsx b/swc-plugins/app/(home)/page.tsx index c8c992d..7deb1d1 100644 --- a/swc-plugins/app/(home)/page.tsx +++ b/swc-plugins/app/(home)/page.tsx @@ -1,83 +1,33 @@ -"use client"; - -import { useState } from "react"; - +import { Logo } from "@/components/logo"; +import { RuntimeVersionSelector } from "@/components/runtime-version-selector"; import { Button } from "@/components/ui/button"; -import { - Select, - SelectContent, - SelectItem, - SelectTrigger, - SelectValue, -} from "@/components/ui/select"; -import { apiClient } from "@/lib/trpc/web-client"; +import { Metadata } from "next"; import Link from "next/link"; -import { useRouter } from "next/navigation"; - -export default function Home() { - const [runtimes] = apiClient.runtime.list.useSuspenseQuery(); - - const [selectedRuntime, setSelectedRuntime] = useState(); - - return ( -
-
- - - {selectedRuntime && } -
-
- - - +import { FC } from "react"; + +export const metadata: Metadata = { + title: "SWC Plugins", + description: "A collection of SWC plugins, ready to use in your project.", +}; + +const Home: FC = () => ( +
+
+ +
+

+ SWC Plugins +

+

+ A collection of SWC plugins, ready to use in your project. +

+ +
- ); -} - -function VersionSelector({ runtimeId }: { runtimeId: bigint }) { - const router = useRouter(); - const versions = apiClient.runtime.listVersions.useQuery({ - runtimeId, - }); +
+); - return ( - - ); -} +export default Home; diff --git a/swc-plugins/app/layout.tsx b/swc-plugins/app/layout.tsx index 17b0df6..fddd209 100644 --- a/swc-plugins/app/layout.tsx +++ b/swc-plugins/app/layout.tsx @@ -1,42 +1,27 @@ import { Dynamic } from "@/components/dynamic"; import { Toaster } from "@/components/ui/toaster"; +import { fontBody, fontHeading } from "@/lib/fonts"; import { cn } from "@/lib/utils"; import { SessionProvider } from "next-auth/react"; -import { Manrope } from "next/font/google"; import NextTopLoader from "nextjs-toploader"; -import { PropsWithChildren } from "react"; +import { FC, PropsWithChildren } from "react"; import { ClientProviders } from "./client-providers"; import "./globals.css"; -const fontHeading = Manrope({ - subsets: ["latin"], - display: "swap", - variable: "--font-heading", -}); +const RootLayout: FC = ({ children }) => ( + + + + + + {children} + + + + + +); -const fontBody = Manrope({ - subsets: ["latin"], - display: "swap", - variable: "--font-body", -}); - -export default function RootLayout({ children }: PropsWithChildren) { - return ( - - - - - - -
- {children} -
-
-
- - - - ); -} +export default RootLayout; diff --git a/swc-plugins/app/versions/layout.tsx b/swc-plugins/app/versions/layout.tsx new file mode 100644 index 0000000..66a7fb4 --- /dev/null +++ b/swc-plugins/app/versions/layout.tsx @@ -0,0 +1,28 @@ +import Link from "next/link"; +import { FC, ReactNode } from "react"; +import { Logo } from "../../components/logo"; +import { RuntimeVersionSelector } from "../../components/runtime-version-selector"; + +type ResultsLayoutProps = { + children: ReactNode; +}; + +const ResultsLayout: FC = ({ children }) => { + return ( + <> + +
{children}
+ + ); +}; + +export default ResultsLayout; diff --git a/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-header.tsx b/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-header.tsx new file mode 100644 index 0000000..572d8dc --- /dev/null +++ b/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-header.tsx @@ -0,0 +1,47 @@ +"use client"; + +import { Checkbox } from "@/components/ui/checkbox"; +import { apiClient } from "@/lib/trpc/web-client"; +import { parseAsBoolean, useQueryState } from "next-usequerystate"; +import { FC } from "react"; + +type CompatRangeHeaderProps = { + compatRangeId: string; +}; + +export const CompatRangeHeader: FC = ({ + compatRangeId, +}) => { + const [includePrerelease, setIncludePrerelease] = useQueryState( + "includePrerelease", + parseAsBoolean.withDefault(false) + ); + const [compatRange] = apiClient.compatRange.get.useSuspenseQuery({ + id: BigInt(compatRangeId), + includePrerelease, + }); + + const handleCheckedChange = (checked: boolean) => { + setIncludePrerelease(checked); + }; + + return ( +
+

+

swc_core

+ + @{compatRange.from} -{" "} + {compatRange.to} + +

+ +
+ + +
+
+ ); +}; diff --git a/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-tables.tsx b/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-tables.tsx new file mode 100644 index 0000000..b6eb50a --- /dev/null +++ b/swc-plugins/app/versions/range/[compatRangeId]/components/compat-range-tables.tsx @@ -0,0 +1,81 @@ +"use client"; + +import { TableContainer } from "@/components/table-container"; +import { + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, +} from "@/components/ui/table"; +import { apiClient } from "@/lib/trpc/web-client"; +import { parseAsBoolean, useQueryState } from "next-usequerystate"; +import { FC } from "react"; + +type CompatRangeTablesProps = { + compatRangeId: string; +}; + +export const CompatRangeTables: FC = ({ + compatRangeId, +}) => { + const [includePrerelease] = useQueryState( + "includePrerelease", + parseAsBoolean.withDefault(false) + ); + const [compatRange] = apiClient.compatRange.get.useSuspenseQuery({ + id: BigInt(compatRangeId), + includePrerelease, + }); + + return ( + <> + + + + + Runtime + Minimum Version + Maximum Version + + + + {compatRange.runtimes.map((runtime) => ( + + {runtime.name} + + {runtime.minVersion} + + + {runtime.maxVersion} + + + ))} + +
+
+ + + + + + Plugin + Minimum Version + Maximum Version + + + + {compatRange.plugins.map((plugin) => ( + + {plugin.name} + {plugin.minVersion} + {plugin.maxVersion} + + ))} + +
+
+ + ); +}; diff --git a/swc-plugins/app/versions/range/[compatRangeId]/page.tsx b/swc-plugins/app/versions/range/[compatRangeId]/page.tsx index 60fab93..9e3c8af 100644 --- a/swc-plugins/app/versions/range/[compatRangeId]/page.tsx +++ b/swc-plugins/app/versions/range/[compatRangeId]/page.tsx @@ -1,93 +1,27 @@ -"use client"; - -import { Checkbox } from "@/components/ui/checkbox"; -import { - Table, - TableBody, - TableCaption, - TableCell, - TableHead, - TableHeader, - TableRow, -} from "@/components/ui/table"; -import { apiClient } from "@/lib/trpc/web-client"; -import { useState } from "react"; - -export default function Page({ - params: { compatRangeId }, -}: { - params: { compatRangeId: string }; -}) { - const [includePrerelease, setIncludePrerelease] = useState(false); - const [compatRange] = apiClient.compatRange.get.useSuspenseQuery({ - id: BigInt(compatRangeId), - includePrerelease, - }); - - return ( -
-
-

- swc_core - - @{compatRange.from} - {compatRange.to} - -

- -
- { - setIncludePrerelease(!!v); - }} - /> - -
-
- - - Runtime Version Ranges - - - Runtime - Minimum Version - Maximum Version - - - - {compatRange.runtimes.map((runtime) => ( - - {runtime.name} - {runtime.minVersion} - {runtime.maxVersion} - - ))} - -
- -

Plugins

- - - Compatible Plugins - - - Plugin - Minimum Version - Maximum Version - - - - {compatRange.plugins.map((plugin) => ( - - {plugin.name} - {plugin.minVersion} - {plugin.maxVersion} - - ))} - -
-
- ); -} +import { Metadata } from "next"; +import { FC } from "react"; +import { CompatRangeHeader } from "./components/compat-range-header"; +import { CompatRangeTables } from "./components/compat-range-tables"; export const dynamic = "force-dynamic"; +export const metadata: Metadata = { + title: "Compat Range", + description: "Compat ranges for swc_core", +}; + +type CompatRangePageProps = { + params: { + compatRangeId: string; + }; +}; + +const CompatRangePage: FC = ({ + params: { compatRangeId }, +}) => ( +
+ + +
+); + +export default CompatRangePage; diff --git a/swc-plugins/app/versions/range/components/range-table.tsx b/swc-plugins/app/versions/range/components/range-table.tsx new file mode 100644 index 0000000..1879055 --- /dev/null +++ b/swc-plugins/app/versions/range/components/range-table.tsx @@ -0,0 +1,51 @@ +"use client"; + +import { TableContainer } from "@/components/table-container"; +import { + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, +} from "@/components/ui/table"; +import { useRouter } from "next/navigation"; + +type RangeTableProps = { + ranges: { id: bigint; from: string; to: string }[]; +}; + +export const RangeTable = ({ ranges }: RangeTableProps) => { + const router = useRouter(); + + const handleClick = (id: bigint) => { + router.push(`/versions/range/${id}`); + }; + + return ( + + + + + Runtime + Minimum Version + Maximum Version + + + + {ranges.map((range) => ( + handleClick(range.id)} + > + swc_core + {range.from} + {range.to} + + ))} + +
+
+ ); +}; diff --git a/swc-plugins/app/versions/range/page.tsx b/swc-plugins/app/versions/range/page.tsx index 0b91129..ac2ba51 100644 --- a/swc-plugins/app/versions/range/page.tsx +++ b/swc-plugins/app/versions/range/page.tsx @@ -1,26 +1,19 @@ import { createCaller } from "@/lib/server"; -import Link from "next/link"; +import { Metadata } from "next"; +import { RangeTable } from "./components/range-table"; -export default async function Page() { +export const dynamic = "force-dynamic"; +export const fetchCache = "force-no-store"; +export const metadata: Metadata = { + title: "Compat Ranges", + description: "A list of compat ranges for SWC plugins.", +}; + +const RangePage = async () => { const api = await createCaller(); const ranges = await api.compatRange.list(); - return ( -
-

Compat Ranges

-
    - {ranges.map((range) => ( -
  • - - swc_core@{range.from} -{" "} - {range.to} - -
  • - ))} -
-
- ); -} + return ; +}; -export const dynamic = "force-dynamic"; -export const fetchCache = "force-no-store"; +export default RangePage; diff --git a/swc-plugins/components/logo/index.tsx b/swc-plugins/components/logo/index.tsx new file mode 100644 index 0000000..0574529 --- /dev/null +++ b/swc-plugins/components/logo/index.tsx @@ -0,0 +1,15 @@ +import { FC } from "react"; + +import { cn } from "@/lib/utils"; +import Image from "next/image"; +import SWCLogo from "./swc.svg"; + +export const Logo: FC<{ className?: string }> = ({ className }) => ( + SWC Logo +); diff --git a/swc-plugins/components/logo/swc.svg b/swc-plugins/components/logo/swc.svg new file mode 100644 index 0000000..b0ca9e3 --- /dev/null +++ b/swc-plugins/components/logo/swc.svg @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/swc-plugins/components/runtime-version-selector.tsx b/swc-plugins/components/runtime-version-selector.tsx new file mode 100644 index 0000000..ab1071a --- /dev/null +++ b/swc-plugins/components/runtime-version-selector.tsx @@ -0,0 +1,52 @@ +"use client"; + +import { apiClient } from "@/lib/trpc/web-client"; +import { useRouter } from "next/navigation"; +import { FC, useState } from "react"; +import { Select } from "./select"; + +export const RuntimeVersionSelector: FC = () => { + const [runtimes] = apiClient.runtime.list.useSuspenseQuery(); + const [selectedRuntime, setSelectedRuntime] = useState(); + const [selectedVersion, setSelectedVersion] = useState(); + const router = useRouter(); + const versions = apiClient.runtime.listVersions.useQuery({ + runtimeId: selectedRuntime ?? BigInt(0), + }); + + const handleRuntimeChange = (runtimeId: string) => { + setSelectedRuntime(BigInt(runtimeId)); + }; + + const handleVersionChange = (version: string) => { + const selected = versions.data?.find((v) => v.version === version); + setSelectedVersion(version); + router.push(`/versions/range/${selected?.compatRangeId}`); + }; + + return ( +
+ ({ + value: version.version, + label: version.version, + })) ?? [] + } + type="version" + /> +
+ ); +}; diff --git a/swc-plugins/components/select.tsx b/swc-plugins/components/select.tsx new file mode 100644 index 0000000..4582757 --- /dev/null +++ b/swc-plugins/components/select.tsx @@ -0,0 +1,186 @@ +"use client"; + +import { useMeasure } from "@react-hookz/web"; +import { useCommandState } from "cmdk"; +import { CheckIcon, ChevronsUpDown, PlusIcon } from "lucide-react"; +import type { ComponentProps, FC, ReactNode } from "react"; +import { useCallback, useId, useState } from "react"; +import { cn } from "../lib/utils"; +import { Button } from "./ui/button"; +import { + Command, + CommandEmpty, + CommandGroup, + CommandInput, + CommandItem, + CommandList, +} from "./ui/command"; +import { Label } from "./ui/label"; +import { Popover, PopoverContent, PopoverTrigger } from "./ui/popover"; + +type SelectProperties = Omit< + ComponentProps, + "open" | "setOpen" +> & { + readonly label?: string; + readonly caption?: string; + readonly value?: string[] | string; + readonly data: readonly { + readonly value: string; + readonly label: string; + }[]; + readonly renderItem?: (item: SelectProperties["data"][number]) => ReactNode; + readonly disabled?: boolean; + readonly type?: string; + readonly trigger?: ReactNode; + readonly onChange?: (value: string) => void; + readonly onCreate?: (value: string) => void; + readonly loading?: boolean; + readonly exactSearch?: boolean; + readonly className?: string; +}; + +const CreateEmptyState = ({ + onCreate, +}: { + readonly onCreate: SelectProperties["onCreate"]; +}) => { + const search = useCommandState((state) => state.search); + + return ( +
+ +
+ ); +}; + +export const Select: FC = ({ + label, + value, + caption, + data, + disabled, + onChange, + type = "item", + renderItem, + trigger, + loading, + onCreate, + exactSearch, + className, + ...properties +}) => { + const id = useId(); + const [open, setOpen] = useState(false); + const selected = data.find((item) => item.value === value); + const [measurements, ref] = useMeasure(); + + const handleSelect = useCallback( + (newValue: string) => { + setOpen(false); + onChange?.(newValue); + }, + [onChange] + ); + + const handleCreate = (newValue: string) => { + setOpen(false); + onCreate?.(newValue); + }; + + return ( + (disabled ? setOpen(false) : setOpen(newOpen))} + > + + {trigger ?? ( +
+ {label ? : null} + + {caption ? ( +

{caption}

+ ) : null} +
+ )} +
+ + + + + {onCreate ? ( + + + + ) : ( + No results found. + )} + + {data.map((item) => { + const active = Array.isArray(value) + ? value.includes(item.value) + : value === item.value; + + return ( + handleSelect(item.value)} + className="flex items-center gap-2" + > +
+ {renderItem ? renderItem(item) : item.label} +
+ +
+ ); + })} +
+
+
+
+
+ ); +}; diff --git a/swc-plugins/components/table-container.tsx b/swc-plugins/components/table-container.tsx new file mode 100644 index 0000000..f9e61f5 --- /dev/null +++ b/swc-plugins/components/table-container.tsx @@ -0,0 +1,29 @@ +import { cn } from "@/lib/utils"; +import type { FC, ReactNode } from "react"; + +type TableContainerProperties = { + readonly title: string; + readonly children: ReactNode; + readonly className?: string; +}; + +export const TableContainer: FC = ({ + title, + children, + className, +}) => ( +
+

+ {title} +

+
+ {children} +
+
+); diff --git a/swc-plugins/lib/fonts.ts b/swc-plugins/lib/fonts.ts new file mode 100644 index 0000000..4542b74 --- /dev/null +++ b/swc-plugins/lib/fonts.ts @@ -0,0 +1,13 @@ +import { Manrope } from "next/font/google"; + +export const fontHeading = Manrope({ + subsets: ["latin"], + display: "swap", + variable: "--font-heading", +}); + +export const fontBody = Manrope({ + subsets: ["latin"], + display: "swap", + variable: "--font-body", +}); diff --git a/swc-plugins/package.json b/swc-plugins/package.json index e8d26f1..52a4761 100644 --- a/swc-plugins/package.json +++ b/swc-plugins/package.json @@ -41,6 +41,7 @@ "@radix-ui/react-toggle": "^1.1.0", "@radix-ui/react-toggle-group": "^1.1.0", "@radix-ui/react-tooltip": "^1.0.7", + "@react-hookz/web": "^24.0.4", "@simplewebauthn/browser": "9.0.1", "@simplewebauthn/server": "9.0.3", "@tanstack/react-query": "^5.45.1", @@ -60,6 +61,7 @@ "next": "14.2.4", "next-auth": "5.0.0-beta.19", "next-themes": "^0.2.1", + "next-usequerystate": "^1.17.8", "nextjs-toploader": "^1.6.4", "node-fetch": "^3.3.2", "react": "18.2.0", @@ -67,6 +69,7 @@ "react-dom": "18.2.0", "react-hook-form": "^7.52.0", "react-resizable-panels": "^2.1.1", + "react-virtuoso": "^4.10.1", "recharts": "^2.12.7", "semver": "^7.6.3", "sharp": "^0.33.2",