mirror of
				https://github.com/walkxcode/dashboard-icons.git
				synced 2025-11-04 02:28:57 +01:00 
			
		
		
		
	feat: Add website (#1157)
Co-authored-by: Bjorn Lammers <bjorn@lammers.media>
This commit is contained in:
		
							
								
								
									
										142
									
								
								web/src/app/globals.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								web/src/app/globals.css
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,142 @@
 | 
			
		||||
@import "tailwindcss";
 | 
			
		||||
@import "tw-animate-css";
 | 
			
		||||
 | 
			
		||||
@custom-variant dark (&:is(.dark *));
 | 
			
		||||
 | 
			
		||||
@theme inline {
 | 
			
		||||
	--color-background: var(--background);
 | 
			
		||||
	--color-foreground: var(--foreground);
 | 
			
		||||
	--font-sans: var(--font-geist-sans);
 | 
			
		||||
	--font-mono: var(--font-geist-mono);
 | 
			
		||||
	--color-sidebar-ring: var(--sidebar-ring);
 | 
			
		||||
	--color-sidebar-border: var(--sidebar-border);
 | 
			
		||||
	--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
 | 
			
		||||
	--color-sidebar-accent: var(--sidebar-accent);
 | 
			
		||||
	--color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
 | 
			
		||||
	--color-sidebar-primary: var(--sidebar-primary);
 | 
			
		||||
	--color-sidebar-foreground: var(--sidebar-foreground);
 | 
			
		||||
	--color-sidebar: var(--sidebar);
 | 
			
		||||
	--color-chart-5: var(--chart-5);
 | 
			
		||||
	--color-chart-4: var(--chart-4);
 | 
			
		||||
	--color-chart-3: var(--chart-3);
 | 
			
		||||
	--color-chart-2: var(--chart-2);
 | 
			
		||||
	--color-chart-1: var(--chart-1);
 | 
			
		||||
	--color-ring: var(--ring);
 | 
			
		||||
	--color-input: var(--input);
 | 
			
		||||
	--color-border: var(--border);
 | 
			
		||||
	--color-destructive: var(--destructive);
 | 
			
		||||
	--color-accent-foreground: var(--accent-foreground);
 | 
			
		||||
	--color-accent: var(--accent);
 | 
			
		||||
	--color-muted-foreground: var(--muted-foreground);
 | 
			
		||||
	--color-muted: var(--muted);
 | 
			
		||||
	--color-secondary-foreground: var(--secondary-foreground);
 | 
			
		||||
	--color-secondary: var(--secondary);
 | 
			
		||||
	--color-primary-foreground: var(--primary-foreground);
 | 
			
		||||
	--color-primary: var(--primary);
 | 
			
		||||
	--color-popover-foreground: var(--popover-foreground);
 | 
			
		||||
	--color-popover: var(--popover);
 | 
			
		||||
	--color-card-foreground: var(--card-foreground);
 | 
			
		||||
	--color-card: var(--card);
 | 
			
		||||
	--radius-sm: calc(var(--radius) - 4px);
 | 
			
		||||
	--radius-md: calc(var(--radius) - 2px);
 | 
			
		||||
	--radius-lg: var(--radius);
 | 
			
		||||
	--radius-xl: calc(var(--radius) + 4px);
 | 
			
		||||
	--animate-accordion-down: accordion-down 0.2s ease-out;
 | 
			
		||||
	--animate-accordion-up: accordion-up 0.2s ease-out;
 | 
			
		||||
 | 
			
		||||
	@keyframes accordion-down {
 | 
			
		||||
		from {
 | 
			
		||||
			height: 0;
 | 
			
		||||
		}
 | 
			
		||||
		to {
 | 
			
		||||
			height: var(--radix-accordion-content-height);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@keyframes accordion-up {
 | 
			
		||||
		from {
 | 
			
		||||
			height: var(--radix-accordion-content-height);
 | 
			
		||||
		}
 | 
			
		||||
		to {
 | 
			
		||||
			height: 0;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
:root {
 | 
			
		||||
	--radius: 0.3rem;
 | 
			
		||||
	--background: oklch(1 0 0);
 | 
			
		||||
	--foreground: oklch(0.141 0.005 285.823);
 | 
			
		||||
	--card: oklch(1 0 0);
 | 
			
		||||
	--card-foreground: oklch(0.141 0.005 285.823);
 | 
			
		||||
	--popover: oklch(1 0 0);
 | 
			
		||||
	--popover-foreground: oklch(0.141 0.005 285.823);
 | 
			
		||||
	--primary: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--primary-foreground: oklch(0.971 0.013 17.38);
 | 
			
		||||
	--secondary: oklch(0.967 0.001 286.375);
 | 
			
		||||
	--secondary-foreground: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--muted: oklch(0.967 0.001 286.375);
 | 
			
		||||
	--muted-foreground: oklch(0.552 0.016 285.938);
 | 
			
		||||
	--accent: oklch(0.967 0.001 286.375);
 | 
			
		||||
	--accent-foreground: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--destructive: oklch(0.577 0.245 27.325);
 | 
			
		||||
	--border: oklch(0.92 0.004 286.32);
 | 
			
		||||
	--input: oklch(0.92 0.004 286.32);
 | 
			
		||||
	--ring: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--chart-1: oklch(0.646 0.222 41.116);
 | 
			
		||||
	--chart-2: oklch(0.6 0.118 184.704);
 | 
			
		||||
	--chart-3: oklch(0.398 0.07 227.392);
 | 
			
		||||
	--chart-4: oklch(0.828 0.189 84.429);
 | 
			
		||||
	--chart-5: oklch(0.769 0.188 70.08);
 | 
			
		||||
	--sidebar: oklch(0.985 0 0);
 | 
			
		||||
	--sidebar-foreground: oklch(0.141 0.005 285.823);
 | 
			
		||||
	--sidebar-primary: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--sidebar-primary-foreground: oklch(0.971 0.013 17.38);
 | 
			
		||||
	--sidebar-accent: oklch(0.967 0.001 286.375);
 | 
			
		||||
	--sidebar-accent-foreground: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--sidebar-border: oklch(0.92 0.004 286.32);
 | 
			
		||||
	--sidebar-ring: oklch(0.637 0.237 25.331);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.dark {
 | 
			
		||||
	--background: oklch(0.141 0.005 285.823);
 | 
			
		||||
	--foreground: oklch(0.985 0 0);
 | 
			
		||||
	--card: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--card-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--popover: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--popover-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--primary: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--primary-foreground: oklch(0.971 0.013 17.38);
 | 
			
		||||
	--secondary: oklch(0.274 0.006 286.033);
 | 
			
		||||
	--secondary-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--muted: oklch(0.274 0.006 286.033);
 | 
			
		||||
	--muted-foreground: oklch(0.705 0.015 286.067);
 | 
			
		||||
	--accent: oklch(0.274 0.006 286.033);
 | 
			
		||||
	--accent-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--destructive: oklch(0.704 0.191 22.216);
 | 
			
		||||
	--border: oklch(1 0 0 / 10%);
 | 
			
		||||
	--input: oklch(1 0 0 / 15%);
 | 
			
		||||
	--ring: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--chart-1: oklch(0.488 0.243 264.376);
 | 
			
		||||
	--chart-2: oklch(0.696 0.17 162.48);
 | 
			
		||||
	--chart-3: oklch(0.769 0.188 70.08);
 | 
			
		||||
	--chart-4: oklch(0.627 0.265 303.9);
 | 
			
		||||
	--chart-5: oklch(0.645 0.246 16.439);
 | 
			
		||||
	--sidebar: oklch(0.21 0.006 285.885);
 | 
			
		||||
	--sidebar-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--sidebar-primary: oklch(0.637 0.237 25.331);
 | 
			
		||||
	--sidebar-primary-foreground: oklch(0.971 0.013 17.38);
 | 
			
		||||
	--sidebar-accent: oklch(0.274 0.006 286.033);
 | 
			
		||||
	--sidebar-accent-foreground: oklch(0.985 0 0);
 | 
			
		||||
	--sidebar-border: oklch(1 0 0 / 10%);
 | 
			
		||||
	--sidebar-ring: oklch(0.637 0.237 25.331);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@layer base {
 | 
			
		||||
	* {
 | 
			
		||||
		@apply border-border outline-ring/50;
 | 
			
		||||
	}
 | 
			
		||||
	body {
 | 
			
		||||
		@apply bg-background text-foreground;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								web/src/app/icon.svg
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1
									
								
								web/src/app/icon.svg
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
<svg xmlns="http://www.w3.org/2000/svg" xml:space="preserve" viewBox="0 82.9 512 346.2"><path d="M169.3 102.7c28.8 0 52.2 23.4 52.2 52.2v66.8c1.8-.6 3.6-1 5.5-1 5.9 0 11.1 2.9 14.2 7.4v-73.2c0-39.7-32.3-72-72-72-24.8 0-46.8 12.6-59.7 31.8 5.6 3.6 10.9 7.5 16 11.8 9.4-14.3 25.5-23.8 43.8-23.8m115.6 118.1c1.9 0 3.8.4 5.5 1V155c0-28.8 23.4-52.2 52.2-52.2 18.3 0 34.4 9.5 43.8 23.8 5.1-4.2 10.4-8.2 16-11.8C389.5 95.6 367.5 83 342.7 83c-39.7 0-72 32.3-72 72v73.2c3.1-4.5 8.3-7.4 14.2-7.4m-69.3 104.8c-35.7 8.6-66 28.1-88.1 54-12.2 14.3-21.9 30.6-28.6 48l46.6.3 265.5 1.2v-.1c-29.2-77.8-112.5-123.4-195.4-103.4m11.5-61.9c-9.7 0-17.5 7.8-17.5 17.5s7.8 17.5 17.5 17.5 17.5-7.8 17.5-17.5-7.8-17.5-17.5-17.5m57.8 35.1c9.7 0 17.5-7.8 17.5-17.5s-7.8-17.5-17.5-17.5-17.5 7.8-17.5 17.5 7.8 17.5 17.5 17.5m162.5-75.6 26.7-108.8c-22.6 2.8-43.5 11.1-61.5 23.4-6.3 4.3-12.3 9.2-17.8 14.4-26.4 25.4-42.9 61.1-42.9 100.6 0 30.7 9.9 59.1 26.7 82.1 9.2 12.7 20.6 23.7 33.4 32.6l9.3-37.8c47.9-14.3 84.1-55.7 90.7-106.5zM133.5 334.9c16.8-23 26.7-51.4 26.7-82.1 0-39.5-16.5-75.2-42.9-100.6-5.5-5.3-11.5-10.1-17.8-14.4-17.9-12.3-38.8-20.6-61.5-23.4l26.7 108.8H0c6.6 50.8 42.8 92.2 90.7 106.5l9.3 37.8c12.9-8.9 24.2-19.9 33.5-32.6" style="fill:#fa5252"/></svg>
 | 
			
		||||
| 
		 After Width: | Height: | Size: 1.2 KiB  | 
							
								
								
									
										99
									
								
								web/src/app/icons/[icon]/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								web/src/app/icons/[icon]/page.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
			
		||||
import { IconDetails } from "@/components/icon-details"
 | 
			
		||||
import { BASE_URL } from "@/constants"
 | 
			
		||||
import { getAllIcons, getAuthorData } from "@/lib/api"
 | 
			
		||||
import type { Metadata, ResolvingMetadata } from "next"
 | 
			
		||||
import { notFound } from "next/navigation"
 | 
			
		||||
 | 
			
		||||
export const dynamicParams = false
 | 
			
		||||
 | 
			
		||||
export async function generateStaticParams() {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	return Object.keys(iconsData).map((icon) => ({
 | 
			
		||||
		icon,
 | 
			
		||||
	}))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const dynamic = "force-static"
 | 
			
		||||
 | 
			
		||||
type Props = {
 | 
			
		||||
	params: Promise<{ icon: string }>
 | 
			
		||||
	searchParams: Promise<{ [key: string]: string | string[] | undefined }>
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export async function generateMetadata({ params, searchParams }: Props, parent: ResolvingMetadata): Promise<Metadata> {
 | 
			
		||||
	const { icon } = await params
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	if (!iconsData[icon]) {
 | 
			
		||||
		notFound()
 | 
			
		||||
	}
 | 
			
		||||
	const previousImages = (await parent).openGraph?.images || []
 | 
			
		||||
	const authorData = await getAuthorData(iconsData[icon].update.author.id)
 | 
			
		||||
	const authorName = authorData.name || authorData.login
 | 
			
		||||
	const updateDate = new Date(iconsData[icon].update.timestamp)
 | 
			
		||||
 | 
			
		||||
	console.debug(`Generated metadata for ${icon} by ${authorName} (${authorData.html_url}) updated at ${updateDate.toLocaleString()}`)
 | 
			
		||||
 | 
			
		||||
	const iconImageUrl = `${BASE_URL}/png/${icon}.png`
 | 
			
		||||
	const pageUrl = `${BASE_URL}/icons/${icon}`
 | 
			
		||||
 | 
			
		||||
	return {
 | 
			
		||||
		title: `${icon} icon · Dashboard Icons`,
 | 
			
		||||
		description: `Download and use the ${icon} icon from Dashboard Icons for your applications`,
 | 
			
		||||
		keywords: [`${icon} icon`, "dashboard icon", "free icon", "open source icon", "application icon"],
 | 
			
		||||
		authors: [
 | 
			
		||||
			{
 | 
			
		||||
				name: "homarr",
 | 
			
		||||
				url: "https://homarr.dev",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				name: authorName,
 | 
			
		||||
				url: authorData.html_url,
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
		openGraph: {
 | 
			
		||||
			title: `${icon} icon · Dashboard Icons`,
 | 
			
		||||
			description: `Download and use the ${icon} icon from Dashboard Icons for your applications`,
 | 
			
		||||
			type: "article",
 | 
			
		||||
			url: pageUrl,
 | 
			
		||||
			images: [
 | 
			
		||||
				{
 | 
			
		||||
					url: iconImageUrl,
 | 
			
		||||
					width: 512,
 | 
			
		||||
					height: 512,
 | 
			
		||||
					alt: `${icon} icon`,
 | 
			
		||||
					type: "image/png",
 | 
			
		||||
				},
 | 
			
		||||
				...previousImages,
 | 
			
		||||
			],
 | 
			
		||||
			authors: [authorName, "homarr"],
 | 
			
		||||
			publishedTime: updateDate.toISOString(),
 | 
			
		||||
			modifiedTime: updateDate.toISOString(),
 | 
			
		||||
		},
 | 
			
		||||
		twitter: {
 | 
			
		||||
			card: "summary_large_image",
 | 
			
		||||
			title: `${icon} icon · Dashboard Icons`,
 | 
			
		||||
			description: `Download and use the ${icon} icon from Dashboard Icons for your applications`,
 | 
			
		||||
			images: [iconImageUrl],
 | 
			
		||||
			creator: "@ajnavocado",
 | 
			
		||||
		},
 | 
			
		||||
		alternates: {
 | 
			
		||||
			canonical: pageUrl,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default async function IconPage({ params }: { params: Promise<{ icon: string }> }) {
 | 
			
		||||
	const { icon } = await params
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	const originalIconData = iconsData[icon]
 | 
			
		||||
 | 
			
		||||
	if (!originalIconData) {
 | 
			
		||||
		notFound()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Pass originalIconData directly, assuming IconDetails can handle it
 | 
			
		||||
	const iconData = originalIconData
 | 
			
		||||
 | 
			
		||||
	const authorData = await getAuthorData(originalIconData.update.author.id)
 | 
			
		||||
	return <IconDetails icon={icon} iconData={originalIconData} authorData={authorData} />
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										87
									
								
								web/src/app/icons/components.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								web/src/app/icons/components.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,87 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Input } from "@/components/ui/input"
 | 
			
		||||
import { BASE_URL } from "@/constants"
 | 
			
		||||
import type { IconSearchProps, IconWithName } from "@/types/icons"
 | 
			
		||||
import { Search } from "lucide-react"
 | 
			
		||||
import Image from "next/image"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { useState } from "react"
 | 
			
		||||
 | 
			
		||||
export function IconSearch({ icons, initialQuery = "" }: IconSearchProps) {
 | 
			
		||||
	const [searchQuery, setSearchQuery] = useState(initialQuery)
 | 
			
		||||
	const [filteredIcons, setFilteredIcons] = useState<IconWithName[]>(() => {
 | 
			
		||||
		if (!initialQuery.trim()) return icons
 | 
			
		||||
 | 
			
		||||
		const q = initialQuery.toLowerCase()
 | 
			
		||||
		return icons.filter(({ name, data }) => {
 | 
			
		||||
			if (name.toLowerCase().includes(q)) return true
 | 
			
		||||
			if (data.aliases.some((alias) => alias.toLowerCase().includes(q))) return true
 | 
			
		||||
			if (data.categories.some((category) => category.toLowerCase().includes(q))) return true
 | 
			
		||||
 | 
			
		||||
			return false
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	const handleSearch = (query: string) => {
 | 
			
		||||
		setSearchQuery(query)
 | 
			
		||||
 | 
			
		||||
		if (!query.trim()) {
 | 
			
		||||
			setFilteredIcons(icons)
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		const q = query.toLowerCase()
 | 
			
		||||
		const filtered = icons.filter(({ name, data }) => {
 | 
			
		||||
			if (name.toLowerCase().includes(q)) return true
 | 
			
		||||
			if (data.aliases.some((alias) => alias.toLowerCase().includes(q))) return true
 | 
			
		||||
			if (data.categories.some((category) => category.toLowerCase().includes(q))) return true
 | 
			
		||||
 | 
			
		||||
			return false
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
		setFilteredIcons(filtered)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<>
 | 
			
		||||
			<div className="relative w-full max-w-md">
 | 
			
		||||
				<Search className="absolute left-2.5 top-2.5 h-4 w-4 text-muted-foreground" />
 | 
			
		||||
				<Input
 | 
			
		||||
					type="search"
 | 
			
		||||
					placeholder="Search icons by name, aliases, or categories..."
 | 
			
		||||
					className="w-full pl-8"
 | 
			
		||||
					value={searchQuery}
 | 
			
		||||
					onChange={(e) => handleSearch(e.target.value)}
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
 | 
			
		||||
			{filteredIcons.length === 0 ? (
 | 
			
		||||
				<div className="text-center py-12">
 | 
			
		||||
					<h2 className="text-xl font-semibold">No icons found</h2>
 | 
			
		||||
					<p className="text-muted-foreground mt-2">Try a different search term.</p>
 | 
			
		||||
				</div>
 | 
			
		||||
			) : (
 | 
			
		||||
				<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-6 xl:grid-cols-8 gap-4 mt-8">
 | 
			
		||||
					{filteredIcons.map(({ name, data }) => (
 | 
			
		||||
						<Link
 | 
			
		||||
							key={name}
 | 
			
		||||
							href={`/icons/${name}`}
 | 
			
		||||
							className="group flex flex-col items-center p-4 rounded-lg border border-border hover:border-primary hover:bg-accent transition-colors"
 | 
			
		||||
						>
 | 
			
		||||
							<div className="relative h-16 w-16 mb-2">
 | 
			
		||||
								<Image
 | 
			
		||||
									src={`${BASE_URL}/${data.base}/${name}.${data.base}`}
 | 
			
		||||
									alt={`${name} icon`}
 | 
			
		||||
									fill
 | 
			
		||||
									className="object-contain p-1 group-hover:scale-110 transition-transform"
 | 
			
		||||
								/>
 | 
			
		||||
							</div>
 | 
			
		||||
							<span className="text-sm text-center truncate w-full">{name.replace(/-/g, " ")}</span>
 | 
			
		||||
						</Link>
 | 
			
		||||
					))}
 | 
			
		||||
				</div>
 | 
			
		||||
			)}
 | 
			
		||||
		</>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										127
									
								
								web/src/app/icons/components/icon-search.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								web/src/app/icons/components/icon-search.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,127 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { IconSubmissionContent } from "@/components/icon-submission-form"
 | 
			
		||||
import { Input } from "@/components/ui/input"
 | 
			
		||||
import { BASE_URL } from "@/constants"
 | 
			
		||||
import type { IconSearchProps } from "@/types/icons"
 | 
			
		||||
import { Search } from "lucide-react"
 | 
			
		||||
import Image from "next/image"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { usePathname, useRouter, useSearchParams } from "next/navigation"
 | 
			
		||||
import { useCallback, useEffect, useRef, useState } from "react"
 | 
			
		||||
 | 
			
		||||
export function IconSearch({ icons }: IconSearchProps) {
 | 
			
		||||
	const searchParams = useSearchParams()
 | 
			
		||||
	const initialQuery = searchParams.get("q")
 | 
			
		||||
	const router = useRouter()
 | 
			
		||||
	const pathname = usePathname()
 | 
			
		||||
	const [searchQuery, setSearchQuery] = useState(initialQuery ?? "")
 | 
			
		||||
	const timeoutRef = useRef<NodeJS.Timeout | null>(null)
 | 
			
		||||
	const [filteredIcons, setFilteredIcons] = useState(() => {
 | 
			
		||||
		if (!initialQuery?.trim()) return icons
 | 
			
		||||
 | 
			
		||||
		const q = initialQuery.toLowerCase()
 | 
			
		||||
		return icons.filter(({ name, data }) => {
 | 
			
		||||
			if (name.toLowerCase().includes(q)) return true
 | 
			
		||||
			if (data.aliases.some((alias) => alias.toLowerCase().includes(q))) return true
 | 
			
		||||
			if (data.categories.some((category) => category.toLowerCase().includes(q))) return true
 | 
			
		||||
 | 
			
		||||
			return false
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
	const filterIcons = useCallback(
 | 
			
		||||
		(query: string) => {
 | 
			
		||||
			if (!query.trim()) {
 | 
			
		||||
				return icons
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			const q = query.toLowerCase()
 | 
			
		||||
			return icons.filter(({ name, data }) => {
 | 
			
		||||
				if (name.toLowerCase().includes(q)) return true
 | 
			
		||||
				if (data.aliases.some((alias) => alias.toLowerCase().includes(q))) return true
 | 
			
		||||
				if (data.categories.some((category) => category.toLowerCase().includes(q))) return true
 | 
			
		||||
 | 
			
		||||
				return false
 | 
			
		||||
			})
 | 
			
		||||
		},
 | 
			
		||||
		[icons],
 | 
			
		||||
	)
 | 
			
		||||
	const updateResults = useCallback(
 | 
			
		||||
		(query: string) => {
 | 
			
		||||
			setFilteredIcons(filterIcons(query))
 | 
			
		||||
			const params = new URLSearchParams()
 | 
			
		||||
			if (query) params.set("q", query)
 | 
			
		||||
 | 
			
		||||
			const newUrl = query ? `${pathname}?${params.toString()}` : pathname
 | 
			
		||||
 | 
			
		||||
			router.push(newUrl, { scroll: false })
 | 
			
		||||
		},
 | 
			
		||||
		[filterIcons, pathname, router],
 | 
			
		||||
	)
 | 
			
		||||
	const handleSearch = useCallback(
 | 
			
		||||
		(query: string) => {
 | 
			
		||||
			setSearchQuery(query)
 | 
			
		||||
			if (timeoutRef.current) {
 | 
			
		||||
				clearTimeout(timeoutRef.current)
 | 
			
		||||
			}
 | 
			
		||||
			timeoutRef.current = setTimeout(() => {
 | 
			
		||||
				updateResults(query)
 | 
			
		||||
			}, 100)
 | 
			
		||||
		},
 | 
			
		||||
		[updateResults],
 | 
			
		||||
	)
 | 
			
		||||
	useEffect(() => {
 | 
			
		||||
		return () => {
 | 
			
		||||
			if (timeoutRef.current) {
 | 
			
		||||
				clearTimeout(timeoutRef.current)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	if (!searchParams) return null
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<>
 | 
			
		||||
			<div className="relative w-full sm:max-w-md">
 | 
			
		||||
				<Search className="absolute left-2.5 top-2.5 h-4 w-4 text-muted-foreground" />
 | 
			
		||||
				<Input
 | 
			
		||||
					type="search"
 | 
			
		||||
					placeholder="Search icons by name, aliases, or categories..."
 | 
			
		||||
					className="w-full pl-8"
 | 
			
		||||
					value={searchQuery}
 | 
			
		||||
					onChange={(e) => handleSearch(e.target.value)}
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
 | 
			
		||||
			{filteredIcons.length === 0 ? (
 | 
			
		||||
				<div className="flex flex-col gap-8 py-12 max-w-2xl mx-auto">
 | 
			
		||||
					<div className="text-center">
 | 
			
		||||
						<h2 className="text-5xl font-semibold">We don't have this one...yet!</h2>
 | 
			
		||||
					</div>
 | 
			
		||||
					<IconSubmissionContent />
 | 
			
		||||
				</div>
 | 
			
		||||
			) : (
 | 
			
		||||
				<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-6 xl:grid-cols-8 gap-4 mt-8">
 | 
			
		||||
					{filteredIcons.map(({ name, data }) => (
 | 
			
		||||
						<Link
 | 
			
		||||
							prefetch={false}
 | 
			
		||||
							key={name}
 | 
			
		||||
							href={`/icons/${name}`}
 | 
			
		||||
							className="group flex flex-col items-center p-3 sm:p-4 rounded-lg border border-border hover:border-primary hover:bg-accent transition-colors"
 | 
			
		||||
						>
 | 
			
		||||
							<div className="relative h-12 w-12 sm:h-16 sm:w-16 mb-2">
 | 
			
		||||
								<Image
 | 
			
		||||
									src={`${BASE_URL}/${data.base}/${name}.${data.base}`}
 | 
			
		||||
									alt={`${name} icon`}
 | 
			
		||||
									fill
 | 
			
		||||
									className="object-contain p-1 group-hover:scale-110 transition-transform"
 | 
			
		||||
								/>
 | 
			
		||||
							</div>
 | 
			
		||||
							<span className="text-xs sm:text-sm text-center truncate w-full capitalize">{name.replace(/-/g, " ")}</span>
 | 
			
		||||
						</Link>
 | 
			
		||||
					))}
 | 
			
		||||
				</div>
 | 
			
		||||
			)}
 | 
			
		||||
		</>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										54
									
								
								web/src/app/icons/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								web/src/app/icons/page.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,54 @@
 | 
			
		||||
import { BASE_URL } from "@/constants"
 | 
			
		||||
import { getIconsArray } from "@/lib/api"
 | 
			
		||||
import type { Metadata } from "next"
 | 
			
		||||
import { IconSearch } from "./components/icon-search"
 | 
			
		||||
 | 
			
		||||
export const metadata: Metadata = {
 | 
			
		||||
	title: "Browse icons | Dashboard Icons",
 | 
			
		||||
	description: "Search and browse through our collection of beautiful dashboard icons for your applications",
 | 
			
		||||
	keywords: ["dashboard icons", "browse icons", "icon search", "free icons", "open source icons"],
 | 
			
		||||
	openGraph: {
 | 
			
		||||
		title: "Browse Dashboard Icons Collection",
 | 
			
		||||
		description: "Search and browse through our collection of beautiful dashboard icons for your applications",
 | 
			
		||||
		type: "website",
 | 
			
		||||
		url: `${BASE_URL}/icons`,
 | 
			
		||||
		images: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/og-image-browse.png",
 | 
			
		||||
				width: 1200,
 | 
			
		||||
				height: 630,
 | 
			
		||||
				alt: "Browse Dashboard Icons",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
	},
 | 
			
		||||
	twitter: {
 | 
			
		||||
		card: "summary_large_image",
 | 
			
		||||
		title: "Browse Dashboard Icons Collection",
 | 
			
		||||
		description: "Search and browse through our collection of beautiful dashboard icons for your applications",
 | 
			
		||||
		images: ["/og-image-browse.png"],
 | 
			
		||||
	},
 | 
			
		||||
	alternates: {
 | 
			
		||||
		canonical: `${BASE_URL}/icons`,
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const dynamic = "force-static"
 | 
			
		||||
 | 
			
		||||
export default async function IconsPage() {
 | 
			
		||||
	const icons = await getIconsArray()
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="py-8">
 | 
			
		||||
			<div className="space-y-4 mb-8 mx-auto max-w-[80vw]">
 | 
			
		||||
				<div className="flex flex-col sm:flex-row sm:items-center justify-between gap-4">
 | 
			
		||||
					<div>
 | 
			
		||||
						<h1 className="text-3xl font-bold">Browse icons</h1>
 | 
			
		||||
						<p className="text-muted-foreground">Search through our collection of {icons.length} beautiful icons.</p>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
 | 
			
		||||
				<IconSearch icons={icons} />
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										168
									
								
								web/src/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								web/src/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,168 @@
 | 
			
		||||
import { PostHogProvider } from "@/components/PostHogProvider"
 | 
			
		||||
import { Header } from "@/components/header"
 | 
			
		||||
import { LicenseNotice } from "@/components/license-notice"
 | 
			
		||||
import type { Metadata, Viewport } from "next"
 | 
			
		||||
import { Inter } from "next/font/google"
 | 
			
		||||
import { Toaster } from "sonner"
 | 
			
		||||
import "./globals.css"
 | 
			
		||||
import { ThemeProvider } from "./theme-provider"
 | 
			
		||||
 | 
			
		||||
const inter = Inter({
 | 
			
		||||
	variable: "--font-inter",
 | 
			
		||||
	subsets: ["latin"],
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
export const viewport: Viewport = {
 | 
			
		||||
	width: "device-width",
 | 
			
		||||
	initialScale: 1,
 | 
			
		||||
	themeColor: "#ffffff",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const metadata: Metadata = {
 | 
			
		||||
	metadataBase: new URL("https://dashboardicons.com"),
 | 
			
		||||
	title: "Dashboard Icons",
 | 
			
		||||
	description: "Curated icons for your dashboard",
 | 
			
		||||
	keywords: ["dashboard", "icons", "open source", "free icons", "dashboard design"],
 | 
			
		||||
	robots: {
 | 
			
		||||
		index: true,
 | 
			
		||||
		follow: true,
 | 
			
		||||
		"max-image-preview": "large",
 | 
			
		||||
		"max-snippet": -1,
 | 
			
		||||
		"max-video-preview": -1,
 | 
			
		||||
		googleBot: "index, follow",
 | 
			
		||||
	},
 | 
			
		||||
	openGraph: {
 | 
			
		||||
		siteName: "Dashboard Icons",
 | 
			
		||||
		type: "website",
 | 
			
		||||
		locale: "en_US",
 | 
			
		||||
		title: "Dashboard Icons",
 | 
			
		||||
		description: "Curated icons for your dashboard",
 | 
			
		||||
		url: "https://dashboardicons.com",
 | 
			
		||||
		images: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/og-image.png",
 | 
			
		||||
				width: 1200,
 | 
			
		||||
				height: 630,
 | 
			
		||||
				alt: "Dashboard Icons",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
	},
 | 
			
		||||
	twitter: {
 | 
			
		||||
		card: "summary_large_image",
 | 
			
		||||
		site: "@homarr_app",
 | 
			
		||||
		creator: "@homarr_app",
 | 
			
		||||
		title: "Dashboard Icons",
 | 
			
		||||
		description: "Curated icons for your dashboard",
 | 
			
		||||
		images: ["/og-image.png"],
 | 
			
		||||
	},
 | 
			
		||||
	applicationName: "Dashboard Icons",
 | 
			
		||||
	appleWebApp: {
 | 
			
		||||
		title: "Dashboard Icons",
 | 
			
		||||
		statusBarStyle: "default",
 | 
			
		||||
		capable: true,
 | 
			
		||||
	},
 | 
			
		||||
	alternates: {
 | 
			
		||||
		types: {
 | 
			
		||||
			"application/rss+xml": "https://dashboardicons.com/rss.xml",
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
	icons: {
 | 
			
		||||
		icon: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/favicon.ico",
 | 
			
		||||
				type: "image/x-icon",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/favicon-16x16.png",
 | 
			
		||||
				sizes: "16x16",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/favicon-32x32.png",
 | 
			
		||||
				sizes: "32x32",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/favicon-96x96.png",
 | 
			
		||||
				sizes: "96x96",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/android-chrome-192x192.png",
 | 
			
		||||
				sizes: "192x192",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
		shortcut: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/favicon.ico",
 | 
			
		||||
				type: "image/x-icon",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
		apple: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-57x57.png",
 | 
			
		||||
				sizes: "57x57",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-60x60.png",
 | 
			
		||||
				sizes: "60x60",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-72x72.png",
 | 
			
		||||
				sizes: "72x72",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-76x76.png",
 | 
			
		||||
				sizes: "76x76",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-114x114.png",
 | 
			
		||||
				sizes: "114x114",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-120x120.png",
 | 
			
		||||
				sizes: "120x120",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-144x144.png",
 | 
			
		||||
				sizes: "144x144",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-152x152.png",
 | 
			
		||||
				sizes: "152x152",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
			{
 | 
			
		||||
				url: "/apple-icon-180x180.png",
 | 
			
		||||
				sizes: "180x180",
 | 
			
		||||
				type: "image/png",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default function RootLayout({ children }: Readonly<{ children: React.ReactNode }>) {
 | 
			
		||||
	return (
 | 
			
		||||
		<html lang="en" suppressHydrationWarning>
 | 
			
		||||
			<body className={`${inter.variable} antialiased bg-background`}>
 | 
			
		||||
				<PostHogProvider>
 | 
			
		||||
					<ThemeProvider attribute="class" defaultTheme="system" enableSystem disableTransitionOnChange>
 | 
			
		||||
						<Header />
 | 
			
		||||
						<main>{children}</main>
 | 
			
		||||
						<Toaster />
 | 
			
		||||
						<LicenseNotice />
 | 
			
		||||
					</ThemeProvider>
 | 
			
		||||
				</PostHogProvider>
 | 
			
		||||
			</body>
 | 
			
		||||
		</html>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										30
									
								
								web/src/app/not-found.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								web/src/app/not-found.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,30 @@
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { AlertTriangle, ArrowLeft } from "lucide-react"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
 | 
			
		||||
export default function NotFound({
 | 
			
		||||
	error,
 | 
			
		||||
}: {
 | 
			
		||||
	error: Error & { digest?: string }
 | 
			
		||||
}) {
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="py-16 flex items-center justify-center">
 | 
			
		||||
			<div className="text-center space-y-6 max-w-md">
 | 
			
		||||
				<div className="mx-auto w-16 h-16 bg-red-100 dark:bg-red-900/20 rounded-full flex items-center justify-center text-red-600 dark:text-red-400">
 | 
			
		||||
					<AlertTriangle className="w-8 h-8" />
 | 
			
		||||
				</div>
 | 
			
		||||
				<h1 className="text-2xl font-bold">Icon not found</h1>
 | 
			
		||||
				<p className="text-muted-foreground">The icon you are looking for could not be found or there was an error loading it.</p>
 | 
			
		||||
				<p className="text-muted-foreground">If you believe this is an error, please contact the maintainers of the repository.</p>
 | 
			
		||||
				<div className="flex flex-col sm:flex-row gap-4 justify-center pt-4">
 | 
			
		||||
					<Button asChild>
 | 
			
		||||
						<Link href="/icons">
 | 
			
		||||
							<ArrowLeft className="mr-2 h-4 w-4" />
 | 
			
		||||
							Back to all icons
 | 
			
		||||
						</Link>
 | 
			
		||||
					</Button>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										43
									
								
								web/src/app/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								web/src/app/page.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,43 @@
 | 
			
		||||
import { HeroSection } from "@/components/hero"
 | 
			
		||||
import { BASE_URL } from "@/constants"
 | 
			
		||||
import { getTotalIcons } from "@/lib/api"
 | 
			
		||||
import type { Metadata } from "next"
 | 
			
		||||
 | 
			
		||||
export const metadata: Metadata = {
 | 
			
		||||
	title: "Dashboard Icons - Beautiful icons for your dashboard",
 | 
			
		||||
	description: "Free, open-source icons for your dashboard. Choose from hundreds of high-quality icons for your web applications.",
 | 
			
		||||
	keywords: ["self hosted", "dashboard icons", "free icons", "open source icons", "web dashboard", "application icons"],
 | 
			
		||||
	openGraph: {
 | 
			
		||||
		title: "Dashboard Icons - Your definitive source for dashboard icons",
 | 
			
		||||
		description: "Free, open-source icons for your dashboard. Choose from thousands of high-quality icons.",
 | 
			
		||||
		type: "website",
 | 
			
		||||
		url: BASE_URL,
 | 
			
		||||
		images: [
 | 
			
		||||
			{
 | 
			
		||||
				url: "/og-image.png",
 | 
			
		||||
				width: 1200,
 | 
			
		||||
				height: 630,
 | 
			
		||||
				alt: "Dashboard Icons",
 | 
			
		||||
			},
 | 
			
		||||
		],
 | 
			
		||||
	},
 | 
			
		||||
	twitter: {
 | 
			
		||||
		title: "Dashboard Icons - Your definitive source for dashboard icons",
 | 
			
		||||
		description: "Free, open-source icons for your dashboard. Choose from thousands of high-quality icons.",
 | 
			
		||||
		card: "summary_large_image",
 | 
			
		||||
		images: ["/og-image.png"],
 | 
			
		||||
	},
 | 
			
		||||
	alternates: {
 | 
			
		||||
		canonical: BASE_URL,
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default async function Home() {
 | 
			
		||||
	const { totalIcons } = await getTotalIcons()
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="flex flex-col min-h-screen">
 | 
			
		||||
			<HeroSection totalIcons={totalIcons} />
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										36
									
								
								web/src/app/sitemap.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								web/src/app/sitemap.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
import { BASE_URL, WEB_URL } from "@/constants"
 | 
			
		||||
import { getAllIcons } from "@/lib/api"
 | 
			
		||||
import type { MetadataRoute } from "next"
 | 
			
		||||
 | 
			
		||||
export const dynamic = "force-static"
 | 
			
		||||
 | 
			
		||||
export default async function sitemap(): Promise<MetadataRoute.Sitemap> {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	return [
 | 
			
		||||
		{
 | 
			
		||||
			url: WEB_URL,
 | 
			
		||||
			lastModified: new Date(),
 | 
			
		||||
			changeFrequency: "yearly",
 | 
			
		||||
			priority: 1,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			url: `${WEB_URL}/icons`,
 | 
			
		||||
			lastModified: new Date(),
 | 
			
		||||
			changeFrequency: "daily",
 | 
			
		||||
			priority: 1,
 | 
			
		||||
			images: [`${WEB_URL}/icons/icon.png`],
 | 
			
		||||
		},
 | 
			
		||||
		...Object.keys(iconsData).map((iconName) => ({
 | 
			
		||||
			url: `${WEB_URL}/icons/${iconName}`,
 | 
			
		||||
			lastModified: iconsData[iconName].update.timestamp,
 | 
			
		||||
			changeFrequency: "yearly" as const,
 | 
			
		||||
			priority: 0.8,
 | 
			
		||||
			images: [
 | 
			
		||||
				`${BASE_URL}/png/${iconName}.png`,
 | 
			
		||||
				// SVG is conditional if it exists
 | 
			
		||||
				iconsData[iconName].base === "svg" ? `${BASE_URL}/svg/${iconName}.svg` : null,
 | 
			
		||||
				`${BASE_URL}/webp/${iconName}.webp`,
 | 
			
		||||
			].filter(Boolean) as string[],
 | 
			
		||||
		})),
 | 
			
		||||
	]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										8
									
								
								web/src/app/theme-provider.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								web/src/app/theme-provider.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { ThemeProvider as NextThemesProvider } from "next-themes"
 | 
			
		||||
import type * as React from "react"
 | 
			
		||||
 | 
			
		||||
export function ThemeProvider({ children, ...props }: React.ComponentProps<typeof NextThemesProvider>) {
 | 
			
		||||
	return <NextThemesProvider {...props}>{children}</NextThemesProvider>
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										57
									
								
								web/src/components/PostHogProvider.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								web/src/components/PostHogProvider.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,57 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { usePathname, useSearchParams } from "next/navigation"
 | 
			
		||||
import posthog from "posthog-js"
 | 
			
		||||
import { PostHogProvider as PHProvider, usePostHog } from "posthog-js/react"
 | 
			
		||||
import { Suspense, useEffect } from "react"
 | 
			
		||||
 | 
			
		||||
export function PostHogProvider({ children }: { children: React.ReactNode }) {
 | 
			
		||||
	useEffect(() => {
 | 
			
		||||
		if (process.env.NODE_ENV === "development" || process.env.DISABLE_POSTHOG === "true") return
 | 
			
		||||
		// biome-ignore lint/style/noNonNullAssertion: The NEXT_PUBLIC_POSTHOG_KEY environment variable is guaranteed to be set in production.
 | 
			
		||||
		posthog.init(process.env.NEXT_PUBLIC_POSTHOG_KEY!, {
 | 
			
		||||
			ui_host: "https://eu.posthog.com",
 | 
			
		||||
			api_host: process.env.NEXT_PUBLIC_POSTHOG_HOST || "https://eu.i.posthog.com",
 | 
			
		||||
			capture_pageview: false, // We capture pageviews manually
 | 
			
		||||
			capture_pageleave: true, // Enable pageleave capture
 | 
			
		||||
			loaded(posthogInstance) {
 | 
			
		||||
				// @ts-expect-error
 | 
			
		||||
				window.posthog = posthogInstance
 | 
			
		||||
			},
 | 
			
		||||
		})
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<PHProvider client={posthog}>
 | 
			
		||||
			<SuspendedPostHogPageView />
 | 
			
		||||
			{children}
 | 
			
		||||
		</PHProvider>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PostHogPageView() {
 | 
			
		||||
	const pathname = usePathname()
 | 
			
		||||
	const searchParams = useSearchParams()
 | 
			
		||||
	const posthogClient = usePostHog()
 | 
			
		||||
 | 
			
		||||
	useEffect(() => {
 | 
			
		||||
		if (pathname && posthogClient) {
 | 
			
		||||
			let url = window.origin + pathname
 | 
			
		||||
			const search = searchParams.toString()
 | 
			
		||||
			if (search) {
 | 
			
		||||
				url += `?${search}`
 | 
			
		||||
			}
 | 
			
		||||
			posthogClient.capture("$pageview", { $current_url: url })
 | 
			
		||||
		}
 | 
			
		||||
	}, [pathname, searchParams, posthogClient])
 | 
			
		||||
 | 
			
		||||
	return null
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SuspendedPostHogPageView() {
 | 
			
		||||
	return (
 | 
			
		||||
		<Suspense fallback={null}>
 | 
			
		||||
			<PostHogPageView />
 | 
			
		||||
		</Suspense>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										87
									
								
								web/src/components/carbon.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								web/src/components/carbon.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,87 @@
 | 
			
		||||
import React from "react"
 | 
			
		||||
 | 
			
		||||
export function Carbon() {
 | 
			
		||||
	// biome-ignore lint/style/noNonNullAssertion: <explanation>
 | 
			
		||||
	const ref = React.useRef<HTMLDivElement>(null!)
 | 
			
		||||
	if (process.env.NODE_ENV === "development") {
 | 
			
		||||
		return null
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	React.useEffect(() => {
 | 
			
		||||
		const serve = "CW7IP27L"
 | 
			
		||||
		const placement = "homarrdev"
 | 
			
		||||
		ref.current.innerHTML = ""
 | 
			
		||||
		const s = document.createElement("script")
 | 
			
		||||
		s.id = "_carbonads_js"
 | 
			
		||||
		s.src = `//cdn.carbonads.com/carbon.js?serve=${serve}&placement=${placement}`
 | 
			
		||||
		ref.current.appendChild(s)
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<>
 | 
			
		||||
			<style>
 | 
			
		||||
				{`
 | 
			
		||||
					#carbonads_1 { display: none; }
 | 
			
		||||
					#carbonads * { margin: initial; padding: initial; }
 | 
			
		||||
					#carbonads {
 | 
			
		||||
						font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto,
 | 
			
		||||
							Oxygen-Sans, Ubuntu, Cantarell, 'Helvetica Neue', Helvetica, Arial,
 | 
			
		||||
							sans-serif;
 | 
			
		||||
						display: flex;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads a {
 | 
			
		||||
						text-decoration: none;
 | 
			
		||||
						color: inherit;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads span {
 | 
			
		||||
						position: relative;
 | 
			
		||||
						display: block;
 | 
			
		||||
						overflow: hidden;
 | 
			
		||||
						width: 100%;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads .carbon-wrap {
 | 
			
		||||
						display: flex;
 | 
			
		||||
						flex-direction: column;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads .carbon-img {
 | 
			
		||||
						display: block;
 | 
			
		||||
						margin: 0;
 | 
			
		||||
						line-height: 1;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads .carbon-img img {
 | 
			
		||||
						display: block;
 | 
			
		||||
						height: 100%;
 | 
			
		||||
						max-width: 100% !important;
 | 
			
		||||
						width: 100%;
 | 
			
		||||
						border-radius: 4px;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads .carbon-text {
 | 
			
		||||
						font-size: 11px;
 | 
			
		||||
						padding: 10px;
 | 
			
		||||
						margin-bottom: 16px;
 | 
			
		||||
						line-height: 1.5;
 | 
			
		||||
						text-align: left;
 | 
			
		||||
					}
 | 
			
		||||
					#carbonads .carbon-poweredby {
 | 
			
		||||
						display: block;
 | 
			
		||||
						padding: 6px 8px;
 | 
			
		||||
						text-align: center;
 | 
			
		||||
						text-transform: uppercase;
 | 
			
		||||
						letter-spacing: 0.5px;
 | 
			
		||||
						font-weight: 600;
 | 
			
		||||
						font-size: 8px;
 | 
			
		||||
						line-height: 1;
 | 
			
		||||
						border-top-left-radius: 3px;
 | 
			
		||||
						position: absolute;
 | 
			
		||||
						bottom: 0;
 | 
			
		||||
						right: 0;
 | 
			
		||||
						background: rgba(128, 128, 128, 0.1);
 | 
			
		||||
					}
 | 
			
		||||
				`}
 | 
			
		||||
			</style>
 | 
			
		||||
			<div className="bg-background shadow-xl flex flex-col m-4 space-y-2 rounded-l-lg">
 | 
			
		||||
				<div ref={ref} className="carbon-outer" />
 | 
			
		||||
			</div>
 | 
			
		||||
		</>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										90
									
								
								web/src/components/command-menu.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								web/src/components/command-menu.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,90 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { useRouter } from "next/navigation"
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { CommandDialog, CommandEmpty, CommandInput, CommandItem, CommandList } from "@/components/ui/command"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
 | 
			
		||||
interface CommandMenuProps {
 | 
			
		||||
	icons: string[]
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function CommandMenu({ icons }: CommandMenuProps) {
 | 
			
		||||
	const router = useRouter()
 | 
			
		||||
	const [open, setOpen] = React.useState(false)
 | 
			
		||||
	const [mounted, setMounted] = React.useState(false)
 | 
			
		||||
	const [inputValue, setInputValue] = React.useState("")
 | 
			
		||||
	const getFilteredIcons = React.useCallback(() => {
 | 
			
		||||
		const query = inputValue.toLowerCase().trim()
 | 
			
		||||
		if (!query) return icons.slice(0, 75)
 | 
			
		||||
		return icons.filter((icon) => {
 | 
			
		||||
			const iconName = icon.toLowerCase()
 | 
			
		||||
			if (iconName.includes(query)) return true
 | 
			
		||||
			const parts = query.split(/\s+/)
 | 
			
		||||
			let lastIndex = -1
 | 
			
		||||
			return parts.every((part) => {
 | 
			
		||||
				const index = iconName.indexOf(part, lastIndex + 1)
 | 
			
		||||
				if (index === -1) return false
 | 
			
		||||
				lastIndex = index
 | 
			
		||||
				return true
 | 
			
		||||
			})
 | 
			
		||||
		})
 | 
			
		||||
	}, [icons, inputValue])
 | 
			
		||||
 | 
			
		||||
	const filteredIcons = getFilteredIcons()
 | 
			
		||||
 | 
			
		||||
	React.useEffect(() => {
 | 
			
		||||
		setMounted(true)
 | 
			
		||||
	}, [])
 | 
			
		||||
	React.useEffect(() => {
 | 
			
		||||
		const down = (e: KeyboardEvent) => {
 | 
			
		||||
			if (e.key === "k" && (e.metaKey || e.ctrlKey)) {
 | 
			
		||||
				e.preventDefault()
 | 
			
		||||
				setOpen((open) => !open)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		document.addEventListener("keydown", down)
 | 
			
		||||
		return () => document.removeEventListener("keydown", down)
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	const handleInputChange = React.useCallback((value: string) => {
 | 
			
		||||
		setInputValue(value)
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	const handleSelectIcon = React.useCallback(
 | 
			
		||||
		(iconName: string) => {
 | 
			
		||||
			router.push(`/icons/${iconName}`)
 | 
			
		||||
			setOpen(false)
 | 
			
		||||
		},
 | 
			
		||||
		[router],
 | 
			
		||||
	)
 | 
			
		||||
	if (!mounted) return null
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<>
 | 
			
		||||
			<p className="text-sm text-muted-foreground">
 | 
			
		||||
				Press{" "}
 | 
			
		||||
				<kbd className="pointer-events-none inline-flex h-5 select-none items-center gap-1 rounded border bg-muted px-1.5 font-mono text-[10px] font-medium text-muted-foreground opacity-100">
 | 
			
		||||
					<span className="text-xs">⌘</span>K
 | 
			
		||||
				</kbd>{" "}
 | 
			
		||||
				to search
 | 
			
		||||
			</p>
 | 
			
		||||
			<CommandDialog open={open} onOpenChange={setOpen}>
 | 
			
		||||
				<CommandInput placeholder="Type to search icons..." value={inputValue} onValueChange={handleInputChange} />
 | 
			
		||||
				<CommandList className="max-h-[300px]">
 | 
			
		||||
					{filteredIcons.length === 0 && <CommandEmpty>No results found. Try a different search term.</CommandEmpty>}
 | 
			
		||||
					{filteredIcons.map((icon) => (
 | 
			
		||||
						<CommandItem key={icon} onSelect={() => handleSelectIcon(icon)}>
 | 
			
		||||
							<Link prefetch={filteredIcons.length < 3} href={`/icons/${icon}`} className="flex items-center gap-2">
 | 
			
		||||
								<div className="w-2 h-2 bg-primary-foreground" />
 | 
			
		||||
								<span className="capitalize">{icon.replace(/-/g, " ")}</span>
 | 
			
		||||
							</Link>
 | 
			
		||||
						</CommandItem>
 | 
			
		||||
					))}
 | 
			
		||||
				</CommandList>
 | 
			
		||||
			</CommandDialog>
 | 
			
		||||
		</>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										23
									
								
								web/src/components/grid-background.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								web/src/components/grid-background.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
interface GridBackgroundProps {
 | 
			
		||||
	className?: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function GridBackground({ className }: GridBackgroundProps) {
 | 
			
		||||
	return (
 | 
			
		||||
		<div className={cn("absolute inset-0 overflow-hidden", className)}>
 | 
			
		||||
			{/* Grid pattern */}
 | 
			
		||||
			<div
 | 
			
		||||
				className={cn(
 | 
			
		||||
					"absolute inset-0",
 | 
			
		||||
					"[background-size:40px_40px]",
 | 
			
		||||
					"[background-image:linear-gradient(to_right,rgba(99,102,241,0.1)_1px,transparent_1px),linear-gradient(to_bottom,rgba(99,102,241,0.1)_1px,transparent_1px)]",
 | 
			
		||||
					"dark:[background-image:linear-gradient(to_right,rgba(99,102,241,0.1)_1px,transparent_1px),linear-gradient(to_bottom,rgba(99,102,241,0.1)_1px,transparent_1px)]",
 | 
			
		||||
				)}
 | 
			
		||||
			/>
 | 
			
		||||
 | 
			
		||||
			<div className="pointer-events-none absolute inset-0 flex items-center justify-center bg-slate-900 [mask-image:radial-gradient(ellipse_at_center,transparent_20%,black)] dark:bg-slate-900" />
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										28
									
								
								web/src/components/header-nav.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								web/src/components/header-nav.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { usePathname } from "next/navigation"
 | 
			
		||||
 | 
			
		||||
export function HeaderNav() {
 | 
			
		||||
	const pathname = usePathname()
 | 
			
		||||
	const isIconsActive = pathname === "/icons" || pathname.startsWith("/icons/")
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<nav className="flex items-center gap-2 md:gap-6">
 | 
			
		||||
			<Link
 | 
			
		||||
				href="/"
 | 
			
		||||
				className={cn("text-sm font-medium transition-colors hover:text-primary", pathname === "/" && "text-primary font-semibold")}
 | 
			
		||||
			>
 | 
			
		||||
				Home
 | 
			
		||||
			</Link>
 | 
			
		||||
			<Link
 | 
			
		||||
				prefetch
 | 
			
		||||
				href="/icons"
 | 
			
		||||
				className={cn("text-sm font-medium transition-colors hover:text-primary", isIconsActive && "text-primary font-semibold")}
 | 
			
		||||
			>
 | 
			
		||||
				Icons
 | 
			
		||||
			</Link>
 | 
			
		||||
		</nav>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										33
									
								
								web/src/components/header.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								web/src/components/header.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
import { IconSubmissionForm } from "@/components/icon-submission-form"
 | 
			
		||||
import { ThemeSwitcher } from "@/components/theme-switcher"
 | 
			
		||||
import { REPO_PATH } from "@/constants"
 | 
			
		||||
import { getAllIcons } from "@/lib/api"
 | 
			
		||||
import { Github } from "lucide-react"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { CommandMenu } from "./command-menu"
 | 
			
		||||
import { HeaderNav } from "./header-nav"
 | 
			
		||||
 | 
			
		||||
const icons = await getAllIcons()
 | 
			
		||||
 | 
			
		||||
export async function Header() {
 | 
			
		||||
	return (
 | 
			
		||||
		<header className="border-b">
 | 
			
		||||
			<div className="px-4 md:px-12 flex items-center justify-between h-16">
 | 
			
		||||
				<div className="flex items-center gap-2 md:gap-6">
 | 
			
		||||
					<Link href="/" className="text-lg md:text-xl font-bold">
 | 
			
		||||
						Dashboard Icons
 | 
			
		||||
					</Link>
 | 
			
		||||
					<HeaderNav />
 | 
			
		||||
				</div>
 | 
			
		||||
				<div className="flex items-center gap-2 md:gap-4">
 | 
			
		||||
					<CommandMenu icons={Object.keys(icons)} />
 | 
			
		||||
					<IconSubmissionForm />
 | 
			
		||||
					<Link href={REPO_PATH} target="_blank" className="text-sm font-medium transition-colors hover:text-primary">
 | 
			
		||||
						<Github className="h-5 w-5" />
 | 
			
		||||
					</Link>
 | 
			
		||||
					<ThemeSwitcher />
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</header>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										228
									
								
								web/src/components/hero.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								web/src/components/hero.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,228 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { Card } from "@/components/ui/card"
 | 
			
		||||
import { Input } from "@/components/ui/input"
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { motion } from "framer-motion"
 | 
			
		||||
import { Circle, Github, Search } from "lucide-react"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { useState } from "react"
 | 
			
		||||
 | 
			
		||||
interface IconCardProps {
 | 
			
		||||
	name: string
 | 
			
		||||
	imageUrl: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function IconCard({ name, imageUrl }: IconCardProps) {
 | 
			
		||||
	return (
 | 
			
		||||
		<Card className="p-4 hover:shadow-md transition-shadow duration-300 flex flex-col items-center gap-2 cursor-pointer group">
 | 
			
		||||
			<div className="w-16 h-16 flex items-center justify-center">
 | 
			
		||||
				<img src={imageUrl} alt={name} className="max-w-full max-h-full" />
 | 
			
		||||
			</div>
 | 
			
		||||
			<p className="text-sm text-center text-muted-foreground group-hover:text-foreground transition-colors">{name}</p>
 | 
			
		||||
		</Card>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ElegantShape({
 | 
			
		||||
	className,
 | 
			
		||||
	delay = 0,
 | 
			
		||||
	width = 400,
 | 
			
		||||
	height = 100,
 | 
			
		||||
	rotate = 0,
 | 
			
		||||
	gradient = "from-background/[0.1]",
 | 
			
		||||
}: {
 | 
			
		||||
	className?: string
 | 
			
		||||
	delay?: number
 | 
			
		||||
	width?: number
 | 
			
		||||
	height?: number
 | 
			
		||||
	rotate?: number
 | 
			
		||||
	gradient?: string
 | 
			
		||||
}) {
 | 
			
		||||
	return (
 | 
			
		||||
		<motion.div
 | 
			
		||||
			initial={{
 | 
			
		||||
				opacity: 0,
 | 
			
		||||
				y: -150,
 | 
			
		||||
				rotate: rotate - 15,
 | 
			
		||||
			}}
 | 
			
		||||
			animate={{
 | 
			
		||||
				opacity: 1,
 | 
			
		||||
				y: 0,
 | 
			
		||||
				rotate: rotate,
 | 
			
		||||
			}}
 | 
			
		||||
			transition={{
 | 
			
		||||
				duration: 2.4,
 | 
			
		||||
				delay,
 | 
			
		||||
				ease: [0.23, 0.86, 0.39, 0.96],
 | 
			
		||||
				opacity: { duration: 1.2 },
 | 
			
		||||
			}}
 | 
			
		||||
			className={cn("absolute", className)}
 | 
			
		||||
		>
 | 
			
		||||
			<motion.div
 | 
			
		||||
				animate={{
 | 
			
		||||
					y: [0, 15, 0],
 | 
			
		||||
				}}
 | 
			
		||||
				transition={{
 | 
			
		||||
					duration: 12,
 | 
			
		||||
					repeat: Number.POSITIVE_INFINITY,
 | 
			
		||||
					ease: "easeInOut",
 | 
			
		||||
				}}
 | 
			
		||||
				style={{
 | 
			
		||||
					width,
 | 
			
		||||
					height,
 | 
			
		||||
				}}
 | 
			
		||||
				className="relative"
 | 
			
		||||
			>
 | 
			
		||||
				<div
 | 
			
		||||
					className={cn(
 | 
			
		||||
						"absolute inset-0 rounded-full",
 | 
			
		||||
						"bg-gradient-to-r to-transparent",
 | 
			
		||||
						gradient,
 | 
			
		||||
						"backdrop-blur-[2px] border-2 border-white/[0.15]",
 | 
			
		||||
						"shadow-[0_8px_32px_0_rgba(255,255,255,0.1)]",
 | 
			
		||||
						"after:absolute after:inset-0 after:rounded-full",
 | 
			
		||||
						"after:bg-[radial-gradient(circle_at_50%_50%,rgba(255,255,255,0.2),transparent_70%)]",
 | 
			
		||||
					)}
 | 
			
		||||
				/>
 | 
			
		||||
			</motion.div>
 | 
			
		||||
		</motion.div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function HeroSection({ totalIcons }: { totalIcons: number }) {
 | 
			
		||||
	const [searchQuery, setSearchQuery] = useState("")
 | 
			
		||||
 | 
			
		||||
	const fadeUpVariants = {
 | 
			
		||||
		hidden: { opacity: 0, y: 30 },
 | 
			
		||||
		visible: (i: number) => ({
 | 
			
		||||
			opacity: 1,
 | 
			
		||||
			y: 0,
 | 
			
		||||
			transition: {
 | 
			
		||||
				duration: 1,
 | 
			
		||||
				delay: 0.5 + i * 0.2,
 | 
			
		||||
				ease: [0.25, 0.4, 0.25, 1],
 | 
			
		||||
			},
 | 
			
		||||
		}),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="relative pt-40 w-full flex items-center justify-center overflow-hidden bg-background">
 | 
			
		||||
			<div className="absolute inset-0 bg-gradient-to-br from-indigo-500/[0.05] via-transparent to-rose-500/[0.05] blur-3xl" />
 | 
			
		||||
 | 
			
		||||
			<div className="absolute inset-0 overflow-hidden">
 | 
			
		||||
				<ElegantShape
 | 
			
		||||
					delay={0.3}
 | 
			
		||||
					width={600}
 | 
			
		||||
					height={140}
 | 
			
		||||
					rotate={12}
 | 
			
		||||
					gradient="from-indigo-500/[0.15]"
 | 
			
		||||
					className="left-[-10%] md:left-[-5%] top-[15%] md:top-[20%]"
 | 
			
		||||
				/>
 | 
			
		||||
 | 
			
		||||
				<ElegantShape
 | 
			
		||||
					delay={0.5}
 | 
			
		||||
					width={500}
 | 
			
		||||
					height={120}
 | 
			
		||||
					rotate={-15}
 | 
			
		||||
					gradient="from-rose-500/[0.15]"
 | 
			
		||||
					className="right-[-5%] md:right-[0%] top-[70%] md:top-[75%]"
 | 
			
		||||
				/>
 | 
			
		||||
 | 
			
		||||
				<ElegantShape
 | 
			
		||||
					delay={0.4}
 | 
			
		||||
					width={300}
 | 
			
		||||
					height={80}
 | 
			
		||||
					rotate={-8}
 | 
			
		||||
					gradient="from-violet-500/[0.15]"
 | 
			
		||||
					className="left-[5%] md:left-[10%] bottom-[5%] md:bottom-[10%]"
 | 
			
		||||
				/>
 | 
			
		||||
 | 
			
		||||
				<ElegantShape
 | 
			
		||||
					delay={0.6}
 | 
			
		||||
					width={200}
 | 
			
		||||
					height={60}
 | 
			
		||||
					rotate={20}
 | 
			
		||||
					gradient="from-amber-500/[0.15]"
 | 
			
		||||
					className="right-[15%] md:right-[20%] top-[10%] md:top-[15%]"
 | 
			
		||||
				/>
 | 
			
		||||
 | 
			
		||||
				<ElegantShape
 | 
			
		||||
					delay={0.7}
 | 
			
		||||
					width={150}
 | 
			
		||||
					height={40}
 | 
			
		||||
					rotate={-25}
 | 
			
		||||
					gradient="from-cyan-500/[0.15]"
 | 
			
		||||
					className="left-[20%] md:left-[25%] top-[5%] md:top-[10%]"
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
 | 
			
		||||
			<div className="relative z-10 container mx-auto px-4 md:px-6">
 | 
			
		||||
				<div className="max-w-4xl mx-auto text-center flex flex-col gap-4">
 | 
			
		||||
					<Link prefetch href="https://github.com/homarr-labs" target="_blank" rel="noopener noreferrer" className="mx-auto">
 | 
			
		||||
						<motion.div variants={fadeUpVariants} custom={0} initial="hidden" animate="visible">
 | 
			
		||||
							<Card className="p-2 flex flex-row items-center gap-2 hover:scale-105 transition-all duration-300">
 | 
			
		||||
								<Circle className="h-2 w-2 fill-rose-500/80" />
 | 
			
		||||
								<span className="text-sm text-foreground/60 tracking-wide">by homarr-labs</span>
 | 
			
		||||
							</Card>
 | 
			
		||||
						</motion.div>
 | 
			
		||||
					</Link>
 | 
			
		||||
 | 
			
		||||
					<motion.div custom={1} variants={fadeUpVariants} initial="hidden" animate="visible">
 | 
			
		||||
						<h1 className="text-4xl sm:text-6xl md:text-7xl font-bold mb-6 md:mb-8 tracking-tight">
 | 
			
		||||
							<span className="bg-clip-text text-transparent bg-gradient-to-b from-foreground to-foreground/80">
 | 
			
		||||
								Your definitive source for
 | 
			
		||||
							</span>
 | 
			
		||||
							<br />
 | 
			
		||||
							<span className={cn("bg-clip-text text-transparent bg-gradient-to-r from-indigo-300 via-foreground/90 to-rose-300")}>
 | 
			
		||||
								dashboard icons.
 | 
			
		||||
							</span>
 | 
			
		||||
						</h1>
 | 
			
		||||
					</motion.div>
 | 
			
		||||
 | 
			
		||||
					<motion.div custom={2} variants={fadeUpVariants} initial="hidden" animate="visible">
 | 
			
		||||
						<p className="text-base sm:text-lg md:text-xl text-muted-foreground mb-8 leading-relaxed font-light tracking-wide max-w-2xl mx-auto px-4">
 | 
			
		||||
							A collection of {totalIcons} beautiful, clean and consistent icons for your dashboard, application, or website.
 | 
			
		||||
						</p>
 | 
			
		||||
					</motion.div>
 | 
			
		||||
 | 
			
		||||
					<motion.div
 | 
			
		||||
						custom={3}
 | 
			
		||||
						variants={fadeUpVariants}
 | 
			
		||||
						initial="hidden"
 | 
			
		||||
						animate="visible"
 | 
			
		||||
						className="flex flex-col items-center gap-6 mb-12"
 | 
			
		||||
					>
 | 
			
		||||
						<form action="/icons" method="GET" className="relative w-full max-w-md">
 | 
			
		||||
							<Input
 | 
			
		||||
								name="q"
 | 
			
		||||
								type="search"
 | 
			
		||||
								placeholder={`Search ${totalIcons} icons...`}
 | 
			
		||||
								className="pl-10 h-12 rounded-lg"
 | 
			
		||||
								value={searchQuery}
 | 
			
		||||
								onChange={(e) => setSearchQuery(e.target.value)}
 | 
			
		||||
							/>
 | 
			
		||||
							<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 h-5 w-5 text-muted-foreground" />
 | 
			
		||||
						</form>
 | 
			
		||||
						<div className="flex gap-4">
 | 
			
		||||
							<Button variant="default" className="rounded-lg" size="lg" asChild>
 | 
			
		||||
								<Link href="/icons" className="flex items-center">
 | 
			
		||||
									Browse all icons
 | 
			
		||||
								</Link>
 | 
			
		||||
							</Button>
 | 
			
		||||
							<Button variant="outline" size="lg" className="gap-2" asChild>
 | 
			
		||||
								<Link href="https://github.com/homarr-labs/dashboard-icons" target="_blank" rel="noopener noreferrer">
 | 
			
		||||
									GitHub
 | 
			
		||||
									<Github className="h-4 w-4" />
 | 
			
		||||
								</Link>
 | 
			
		||||
							</Button>
 | 
			
		||||
						</div>
 | 
			
		||||
					</motion.div>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
 | 
			
		||||
			<div className="absolute inset-0 bg-gradient-to-t from-background via-transparent to-background/80 pointer-events-none" />
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										341
									
								
								web/src/components/icon-details.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										341
									
								
								web/src/components/icon-details.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,341 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar"
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
 | 
			
		||||
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "@/components/ui/tooltip"
 | 
			
		||||
import { BASE_URL, REPO_PATH } from "@/constants"
 | 
			
		||||
import type { AuthorData, Icon } from "@/types/icons"
 | 
			
		||||
import { motion } from "framer-motion"
 | 
			
		||||
import { Check, Copy, Download, Github } from "lucide-react"
 | 
			
		||||
import Image from "next/image"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { useState } from "react"
 | 
			
		||||
import { toast } from "sonner"
 | 
			
		||||
import { Carbon } from "./carbon"
 | 
			
		||||
 | 
			
		||||
export type IconDetailsProps = {
 | 
			
		||||
	icon: string
 | 
			
		||||
	iconData: Icon
 | 
			
		||||
	authorData: AuthorData
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function IconDetails({ icon, iconData, authorData }: IconDetailsProps) {
 | 
			
		||||
	const authorName = authorData.name || authorData.login || ""
 | 
			
		||||
	const iconColorVariants = iconData.colors
 | 
			
		||||
	const formattedDate = new Date(iconData.update.timestamp).toLocaleDateString("en-GB", {
 | 
			
		||||
		day: "numeric",
 | 
			
		||||
		month: "long",
 | 
			
		||||
		year: "numeric",
 | 
			
		||||
	})
 | 
			
		||||
	const getAvailableFormats = () => {
 | 
			
		||||
		switch (iconData.base) {
 | 
			
		||||
			case "svg":
 | 
			
		||||
				return ["svg", "png", "webp"]
 | 
			
		||||
			case "png":
 | 
			
		||||
				return ["png", "webp"]
 | 
			
		||||
			default:
 | 
			
		||||
				return [iconData.base]
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const availableFormats = getAvailableFormats()
 | 
			
		||||
	const [copiedVariants, setCopiedVariants] = useState<Record<string, boolean>>({})
 | 
			
		||||
 | 
			
		||||
	const handleCopy = (url: string, variantKey: string) => {
 | 
			
		||||
		navigator.clipboard.writeText(url)
 | 
			
		||||
		setCopiedVariants((prev) => ({
 | 
			
		||||
			...prev,
 | 
			
		||||
			[variantKey]: true,
 | 
			
		||||
		}))
 | 
			
		||||
		setTimeout(() => {
 | 
			
		||||
			setCopiedVariants((prev) => ({
 | 
			
		||||
				...prev,
 | 
			
		||||
				[variantKey]: false,
 | 
			
		||||
			}))
 | 
			
		||||
		}, 2000)
 | 
			
		||||
 | 
			
		||||
		toast.success("URL copied", {
 | 
			
		||||
			description: "The icon URL has been copied to your clipboard",
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const renderVariant = (format: string, iconName: string, theme?: "light" | "dark") => {
 | 
			
		||||
		const variantName = theme && iconColorVariants?.[theme] ? iconColorVariants[theme] : iconName
 | 
			
		||||
		const url = `${BASE_URL}/${format}/${variantName}.${format}`
 | 
			
		||||
		const githubUrl = `${REPO_PATH}/tree/main/${format}/${iconName}.${format}`
 | 
			
		||||
		const variantKey = `${format}-${theme || "default"}`
 | 
			
		||||
		const isCopied = copiedVariants[variantKey] || false
 | 
			
		||||
 | 
			
		||||
		return (
 | 
			
		||||
			<TooltipProvider key={variantKey}>
 | 
			
		||||
				<div className="flex flex-col items-center bg-card rounded-lg p-4 border shadow-sm hover:shadow-md transition-all">
 | 
			
		||||
					<Tooltip>
 | 
			
		||||
						<TooltipTrigger asChild>
 | 
			
		||||
							<motion.div
 | 
			
		||||
								className="relative w-28 h-28 mb-3 cursor-pointer rounded-md overflow-hidden group"
 | 
			
		||||
								whileHover={{ scale: 1.05 }}
 | 
			
		||||
								whileTap={{ scale: 0.95 }}
 | 
			
		||||
								onClick={() => handleCopy(url, variantKey)}
 | 
			
		||||
							>
 | 
			
		||||
								<div className="absolute inset-0 border-2 border-transparent group-hover:border-primary/20 rounded-md z-10 transition-colors" />
 | 
			
		||||
 | 
			
		||||
								<motion.div
 | 
			
		||||
									className="absolute inset-0 bg-primary/10 flex items-center justify-center z-20 rounded-md"
 | 
			
		||||
									initial={{ opacity: 0 }}
 | 
			
		||||
									animate={{ opacity: isCopied ? 1 : 0 }}
 | 
			
		||||
									transition={{ duration: 0.2 }}
 | 
			
		||||
								>
 | 
			
		||||
									<motion.div
 | 
			
		||||
										initial={{ scale: 0.5, opacity: 0 }}
 | 
			
		||||
										animate={{ scale: isCopied ? 1 : 0.5, opacity: isCopied ? 1 : 0 }}
 | 
			
		||||
										transition={{ type: "spring", stiffness: 300, damping: 20 }}
 | 
			
		||||
									>
 | 
			
		||||
										<Check className="w-8 h-8 text-primary" />
 | 
			
		||||
									</motion.div>
 | 
			
		||||
								</motion.div>
 | 
			
		||||
 | 
			
		||||
								<Image
 | 
			
		||||
									src={url}
 | 
			
		||||
									alt={`${iconName} in ${format} format${theme ? ` (${theme} theme)` : ""}`}
 | 
			
		||||
									fill
 | 
			
		||||
									className="object-contain p-2"
 | 
			
		||||
								/>
 | 
			
		||||
							</motion.div>
 | 
			
		||||
						</TooltipTrigger>
 | 
			
		||||
						<TooltipContent>
 | 
			
		||||
							<p>Click to copy URL to clipboard</p>
 | 
			
		||||
						</TooltipContent>
 | 
			
		||||
					</Tooltip>
 | 
			
		||||
 | 
			
		||||
					<p className="text-sm font-medium">{format.toUpperCase()}</p>
 | 
			
		||||
 | 
			
		||||
					<div className="flex gap-2 mt-3 w-full justify-center">
 | 
			
		||||
						<Tooltip>
 | 
			
		||||
							<TooltipTrigger asChild>
 | 
			
		||||
								<Button variant="outline" size="icon" className="h-8 w-8" asChild>
 | 
			
		||||
									<a href={url} download={`${iconName}.${format}`}>
 | 
			
		||||
										<Download className="w-4 h-4" />
 | 
			
		||||
									</a>
 | 
			
		||||
								</Button>
 | 
			
		||||
							</TooltipTrigger>
 | 
			
		||||
							<TooltipContent>
 | 
			
		||||
								<p>Download icon</p>
 | 
			
		||||
							</TooltipContent>
 | 
			
		||||
						</Tooltip>
 | 
			
		||||
 | 
			
		||||
						<Tooltip>
 | 
			
		||||
							<TooltipTrigger asChild>
 | 
			
		||||
								<Button
 | 
			
		||||
									variant="outline"
 | 
			
		||||
									size="icon"
 | 
			
		||||
									className="h-8 w-8 cursor-pointer"
 | 
			
		||||
									onClick={() => handleCopy(url, `btn-${variantKey}`)}
 | 
			
		||||
								>
 | 
			
		||||
									{copiedVariants[`btn-${variantKey}`] ? <Check className="w-4 h-4 text-green-500" /> : <Copy className="w-4 h-4" />}
 | 
			
		||||
								</Button>
 | 
			
		||||
							</TooltipTrigger>
 | 
			
		||||
							<TooltipContent>
 | 
			
		||||
								<p>Copy URL to clipboard</p>
 | 
			
		||||
							</TooltipContent>
 | 
			
		||||
						</Tooltip>
 | 
			
		||||
 | 
			
		||||
						<Tooltip>
 | 
			
		||||
							<TooltipTrigger asChild>
 | 
			
		||||
								<Button variant="outline" size="icon" className="h-8 w-8" asChild>
 | 
			
		||||
									<Link href={githubUrl} target="_blank" rel="noopener noreferrer">
 | 
			
		||||
										<Github className="w-4 h-4" />
 | 
			
		||||
									</Link>
 | 
			
		||||
								</Button>
 | 
			
		||||
							</TooltipTrigger>
 | 
			
		||||
							<TooltipContent>
 | 
			
		||||
								<p>View on GitHub</p>
 | 
			
		||||
							</TooltipContent>
 | 
			
		||||
						</Tooltip>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
			</TooltipProvider>
 | 
			
		||||
		)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="container mx-auto px-4 py-8">
 | 
			
		||||
			<div className="grid grid-cols-1 lg:grid-cols-4 gap-6">
 | 
			
		||||
				{/* Left Column: Icon Info and Author */}
 | 
			
		||||
				<div className="lg:col-span-1">
 | 
			
		||||
					<Card className="h-full">
 | 
			
		||||
						<CardHeader className="pb-4">
 | 
			
		||||
							<div className="flex flex-col items-center">
 | 
			
		||||
								<div className="relative w-32 h-32 bg-background rounded-xl overflow-hidden border flex items-center justify-center p-3 mb-4">
 | 
			
		||||
									<Image
 | 
			
		||||
										src={`${BASE_URL}/${iconData.base}/${icon}.${iconData.base}`}
 | 
			
		||||
										width={96}
 | 
			
		||||
										height={96}
 | 
			
		||||
										alt={icon}
 | 
			
		||||
										className="w-full h-full object-contain"
 | 
			
		||||
									/>
 | 
			
		||||
								</div>
 | 
			
		||||
								<CardTitle className="text-2xl font-bold capitalize text-center mb-2">{icon}</CardTitle>
 | 
			
		||||
							</div>
 | 
			
		||||
						</CardHeader>
 | 
			
		||||
						<CardContent>
 | 
			
		||||
							<div className="space-y-6">
 | 
			
		||||
								<div className="space-y-3">
 | 
			
		||||
									<div className="space-y-2">
 | 
			
		||||
										<div className="flex items-center gap-2">
 | 
			
		||||
											<p className="text-sm">
 | 
			
		||||
												<span className="font-medium">Updated on:</span> {formattedDate}
 | 
			
		||||
											</p>
 | 
			
		||||
										</div>
 | 
			
		||||
										<div className="flex items-center gap-2">
 | 
			
		||||
											<div className="flex items-center gap-2">
 | 
			
		||||
												<p className="text-sm font-medium">By:</p>
 | 
			
		||||
												<Avatar className="h-5 w-5 border">
 | 
			
		||||
													<AvatarImage src={authorData.avatar_url} alt={authorName} />
 | 
			
		||||
													<AvatarFallback>{authorName ? authorName.slice(0, 2).toUpperCase() : "??"}</AvatarFallback>
 | 
			
		||||
												</Avatar>
 | 
			
		||||
												<Link
 | 
			
		||||
													href={authorData.html_url}
 | 
			
		||||
													target="_blank"
 | 
			
		||||
													rel="noopener noreferrer"
 | 
			
		||||
													className="text-primary hover:underline text-sm"
 | 
			
		||||
												>
 | 
			
		||||
													{authorName}
 | 
			
		||||
												</Link>
 | 
			
		||||
											</div>
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								</div>
 | 
			
		||||
 | 
			
		||||
								{iconData.categories && iconData.categories.length > 0 && (
 | 
			
		||||
									<div className="space-y-3">
 | 
			
		||||
										<h3 className="text-sm font-semibold text-muted-foreground">Categories</h3>
 | 
			
		||||
										<div className="flex flex-wrap gap-2">
 | 
			
		||||
											{iconData.categories.map((category) => (
 | 
			
		||||
												<span key={category} className="inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold">
 | 
			
		||||
													{category}
 | 
			
		||||
												</span>
 | 
			
		||||
											))}
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								)}
 | 
			
		||||
 | 
			
		||||
								{iconData.aliases && iconData.aliases.length > 0 && (
 | 
			
		||||
									<div className="space-y-3">
 | 
			
		||||
										<h3 className="text-sm font-semibold text-muted-foreground">Aliases</h3>
 | 
			
		||||
										<div className="flex flex-wrap gap-1">
 | 
			
		||||
											{iconData.aliases.map((alias) => (
 | 
			
		||||
												<span key={alias} className="inline-flex items-center rounded-full bg-muted px-2.5 py-0.5 text-xs">
 | 
			
		||||
													{alias}
 | 
			
		||||
												</span>
 | 
			
		||||
											))}
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								)}
 | 
			
		||||
							</div>
 | 
			
		||||
						</CardContent>
 | 
			
		||||
					</Card>
 | 
			
		||||
				</div>
 | 
			
		||||
 | 
			
		||||
				{/* Middle Column: Icon variants */}
 | 
			
		||||
				<div className="lg:col-span-2">
 | 
			
		||||
					<Card className="h-full">
 | 
			
		||||
						<CardHeader>
 | 
			
		||||
							<CardTitle>Icon variants</CardTitle>
 | 
			
		||||
							<CardDescription>Click on any icon to copy its URL to your clipboard</CardDescription>
 | 
			
		||||
						</CardHeader>
 | 
			
		||||
						<CardContent>
 | 
			
		||||
							{!iconData.colors ? (
 | 
			
		||||
								<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
 | 
			
		||||
									{availableFormats.map((format) => renderVariant(format, icon))}
 | 
			
		||||
								</div>
 | 
			
		||||
							) : (
 | 
			
		||||
								<div className="space-y-10">
 | 
			
		||||
									<div>
 | 
			
		||||
										<h3 className="text-lg font-semibold mb-4 flex items-center gap-2">
 | 
			
		||||
											<span className="inline-block w-3 h-3 rounded-full bg-primary" />
 | 
			
		||||
											Light theme
 | 
			
		||||
										</h3>
 | 
			
		||||
										<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
 | 
			
		||||
											{availableFormats.map((format) => renderVariant(format, icon, "light"))}
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
									<div>
 | 
			
		||||
										<h3 className="text-lg font-semibold mb-4 flex items-center gap-2">
 | 
			
		||||
											<span className="inline-block w-3 h-3 rounded-full bg-primary" />
 | 
			
		||||
											Dark theme
 | 
			
		||||
										</h3>
 | 
			
		||||
										<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
 | 
			
		||||
											{availableFormats.map((format) => renderVariant(format, icon, "dark"))}
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								</div>
 | 
			
		||||
							)}
 | 
			
		||||
						</CardContent>
 | 
			
		||||
					</Card>
 | 
			
		||||
				</div>
 | 
			
		||||
 | 
			
		||||
				{/* Right Column: Technical details */}
 | 
			
		||||
				<div className="lg:col-span-1">
 | 
			
		||||
					<Card className="h-full">
 | 
			
		||||
						<CardHeader>
 | 
			
		||||
							<CardTitle>Technical details</CardTitle>
 | 
			
		||||
						</CardHeader>
 | 
			
		||||
						<CardContent>
 | 
			
		||||
							<div className="space-y-6">
 | 
			
		||||
								<div className="space-y-3">
 | 
			
		||||
									<h3 className="text-sm font-semibold text-muted-foreground">Base format</h3>
 | 
			
		||||
									<div className="flex items-center gap-2">
 | 
			
		||||
										<span className="w-3 h-3 rounded-full bg-primary/80" />
 | 
			
		||||
										<div className="px-3 py-1.5 bg-muted rounded-md text-sm font-medium">{iconData.base.toUpperCase()}</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								</div>
 | 
			
		||||
 | 
			
		||||
								<div className="space-y-3">
 | 
			
		||||
									<h3 className="text-sm font-semibold text-muted-foreground">Available formats</h3>
 | 
			
		||||
									<div className="flex flex-wrap gap-2">
 | 
			
		||||
										{availableFormats.map((format) => (
 | 
			
		||||
											<div key={format} className="px-3 py-1.5 bg-muted rounded-md text-xs font-medium">
 | 
			
		||||
												{format.toUpperCase()}
 | 
			
		||||
											</div>
 | 
			
		||||
										))}
 | 
			
		||||
									</div>
 | 
			
		||||
								</div>
 | 
			
		||||
 | 
			
		||||
								{iconData.colors && (
 | 
			
		||||
									<div className="space-y-3">
 | 
			
		||||
										<h3 className="text-sm font-semibold text-muted-foreground">Color variants</h3>
 | 
			
		||||
										<div className="space-y-2">
 | 
			
		||||
											{Object.entries(iconData.colors).map(([theme, variant]) => (
 | 
			
		||||
												<div key={theme} className="flex items-center gap-2">
 | 
			
		||||
													<span className="w-3 h-3 rounded-full bg-primary/80" />
 | 
			
		||||
													<span className="capitalize font-medium text-sm">{theme}:</span>
 | 
			
		||||
													<code className="bg-muted px-2 py-0.5 rounded text-xs">{variant}</code>
 | 
			
		||||
												</div>
 | 
			
		||||
											))}
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								)}
 | 
			
		||||
 | 
			
		||||
								<div className="space-y-3">
 | 
			
		||||
									<h3 className="text-sm font-semibold text-muted-foreground">Source</h3>
 | 
			
		||||
									<Button variant="outline" className="w-full" asChild>
 | 
			
		||||
										<Link
 | 
			
		||||
											href={`${REPO_PATH}/tree/main/${iconData.base}/${icon}.${iconData.base}`}
 | 
			
		||||
											target="_blank"
 | 
			
		||||
											rel="noopener noreferrer"
 | 
			
		||||
										>
 | 
			
		||||
											<Github className="w-4 h-4 mr-2" />
 | 
			
		||||
											View on GitHub
 | 
			
		||||
										</Link>
 | 
			
		||||
									</Button>
 | 
			
		||||
								</div>
 | 
			
		||||
							</div>
 | 
			
		||||
						</CardContent>
 | 
			
		||||
						<Carbon />
 | 
			
		||||
					</Card>
 | 
			
		||||
				</div>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										87
									
								
								web/src/components/icon-submission-form.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								web/src/components/icon-submission-form.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,87 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogTrigger } from "@/components/ui/dialog"
 | 
			
		||||
import { REPO_PATH } from "@/constants"
 | 
			
		||||
import { DialogDescription } from "@radix-ui/react-dialog"
 | 
			
		||||
import { ExternalLink, PlusCircle } from "lucide-react"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { useState } from "react"
 | 
			
		||||
 | 
			
		||||
export const ISSUE_TEMPLATES = [
 | 
			
		||||
	{
 | 
			
		||||
		id: "add_monochrome_icon",
 | 
			
		||||
		name: "Add light & dark icon",
 | 
			
		||||
		description: "Use this template to add a new icon to the project. Monochrome icons need both light and dark versions.",
 | 
			
		||||
		url: `${REPO_PATH}/issues/new?template=add_monochrome_icon.yml`,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		id: "add_normal_icon",
 | 
			
		||||
		name: "Add normal icon",
 | 
			
		||||
		description: "Use this template to add a new icon to the project. Normal icons work for both light and dark themes.",
 | 
			
		||||
		url: `${REPO_PATH}/issues/new?template=add_normal_icon.yml`,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		id: "update_monochrome_icon",
 | 
			
		||||
		name: "Update light & dark icon",
 | 
			
		||||
		description: "Use this template to update an existing icon. Monochrome icons need both light and dark versions.",
 | 
			
		||||
		url: `${REPO_PATH}/issues/new?template=update_monochrome_icon.yml`,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		id: "update_normal_icon",
 | 
			
		||||
		name: "Update normal icon",
 | 
			
		||||
		description: "Use this template to update an existing icon. Normal icons work for both light and dark themes.",
 | 
			
		||||
		url: `${REPO_PATH}/issues/new?template=update_normal_icon.yml`,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		id: "blank_issue",
 | 
			
		||||
		name: "Something else",
 | 
			
		||||
		description: "You'd like to do something else? Use this template to create a new issue.",
 | 
			
		||||
		url: `${REPO_PATH}/issues/new?template=BLANK_ISSUE`,
 | 
			
		||||
	},
 | 
			
		||||
]
 | 
			
		||||
export function IconSubmissionContent({ onClose }: { onClose?: () => void }) {
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="flex flex-col gap-4">
 | 
			
		||||
			<div className="flex flex-col gap-2">
 | 
			
		||||
				{ISSUE_TEMPLATES.map((template) => (
 | 
			
		||||
					<Link key={template.id} href={template.url} className="w-full" target="_blank" rel="noopener noreferrer">
 | 
			
		||||
						<Button
 | 
			
		||||
							key={template.id}
 | 
			
		||||
							variant="outline"
 | 
			
		||||
							className="w-full flex flex-col items-start gap-1 h-auto p-4 text-left cursor-pointer"
 | 
			
		||||
						>
 | 
			
		||||
							<div className="flex w-full items-center justify-between">
 | 
			
		||||
								<span className="font-medium">{template.name}</span>
 | 
			
		||||
								<ExternalLink className="h-4 w-4 text-muted-foreground" />
 | 
			
		||||
							</div>
 | 
			
		||||
							<span className="text-xs text-muted-foreground">{template.description}</span>
 | 
			
		||||
						</Button>
 | 
			
		||||
					</Link>
 | 
			
		||||
				))}
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
export function IconSubmissionForm() {
 | 
			
		||||
	const [open, setOpen] = useState(false)
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<Dialog open={open} onOpenChange={setOpen}>
 | 
			
		||||
			<DialogTrigger asChild>
 | 
			
		||||
				<Button variant="outline" className="hidden md:inline-flex">
 | 
			
		||||
					<PlusCircle className="h-4 w-4" /> Suggest new icon
 | 
			
		||||
				</Button>
 | 
			
		||||
			</DialogTrigger>
 | 
			
		||||
			<DialogContent className="md:max-w-4xl backdrop-blur-2xl">
 | 
			
		||||
				<DialogHeader>
 | 
			
		||||
					<DialogTitle>Suggest a new icon</DialogTitle>
 | 
			
		||||
					<DialogDescription>You can suggest a new icon by creating an issue on GitHub using one of the templates below.</DialogDescription>
 | 
			
		||||
				</DialogHeader>
 | 
			
		||||
				<div className="mt-4">
 | 
			
		||||
					<IconSubmissionContent onClose={() => setOpen(false)} />
 | 
			
		||||
				</div>
 | 
			
		||||
			</DialogContent>
 | 
			
		||||
		</Dialog>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										75
									
								
								web/src/components/license-notice.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								web/src/components/license-notice.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,75 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { REPO_PATH } from "@/constants"
 | 
			
		||||
import { AnimatePresence, motion } from "framer-motion"
 | 
			
		||||
import { X } from "lucide-react"
 | 
			
		||||
import Link from "next/link"
 | 
			
		||||
import { useEffect, useState } from "react"
 | 
			
		||||
 | 
			
		||||
const LOCAL_STORAGE_KEY = "licenseNoticeDismissed"
 | 
			
		||||
 | 
			
		||||
export function LicenseNotice() {
 | 
			
		||||
	const [isVisible, setIsVisible] = useState(false)
 | 
			
		||||
 | 
			
		||||
	useEffect(() => {
 | 
			
		||||
		// Check local storage only on the client side
 | 
			
		||||
		const dismissed = localStorage.getItem(LOCAL_STORAGE_KEY)
 | 
			
		||||
		if (!dismissed) {
 | 
			
		||||
			setIsVisible(true)
 | 
			
		||||
		}
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	const handleDismiss = () => {
 | 
			
		||||
		localStorage.setItem(LOCAL_STORAGE_KEY, "true")
 | 
			
		||||
		setIsVisible(false)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<AnimatePresence>
 | 
			
		||||
			{isVisible && (
 | 
			
		||||
				<motion.div
 | 
			
		||||
					initial={{ opacity: 0, y: 20 }}
 | 
			
		||||
					animate={{ opacity: 1, y: 0 }}
 | 
			
		||||
					exit={{ opacity: 0, y: 20 }}
 | 
			
		||||
					transition={{ duration: 0.3 }}
 | 
			
		||||
					className="fixed bottom-4 right-4 z-50 max-w-sm rounded-lg border bg-card p-4 text-card-foreground shadow-lg"
 | 
			
		||||
				>
 | 
			
		||||
					<div className="flex items-start justify-between">
 | 
			
		||||
						<div className="text-xs text-muted-foreground space-y-1">
 | 
			
		||||
							<p>
 | 
			
		||||
								Unless otherwise indicated, all images and assets are the property of their respective owners and used for identification
 | 
			
		||||
								purposes only.
 | 
			
		||||
							</p>
 | 
			
		||||
							<p>
 | 
			
		||||
								Read the{" "}
 | 
			
		||||
								<Link
 | 
			
		||||
									href={`${REPO_PATH}/blob/main/LICENSE`}
 | 
			
		||||
									className="underline hover:text-foreground"
 | 
			
		||||
									target="_blank"
 | 
			
		||||
									rel="noopener noreferrer"
 | 
			
		||||
								>
 | 
			
		||||
									LICENSE
 | 
			
		||||
								</Link>{" "}
 | 
			
		||||
								or{" "}
 | 
			
		||||
								<a href="mailto:homarr-labs@proton.me" className="underline hover:text-foreground">
 | 
			
		||||
									contact us
 | 
			
		||||
								</a>
 | 
			
		||||
								.
 | 
			
		||||
							</p>
 | 
			
		||||
						</div>
 | 
			
		||||
						<Button
 | 
			
		||||
							variant="ghost"
 | 
			
		||||
							size="sm"
 | 
			
		||||
							className="-mr-2 -mt-2 h-6 w-6 p-0"
 | 
			
		||||
							onClick={handleDismiss}
 | 
			
		||||
							aria-label="Dismiss license notice"
 | 
			
		||||
						>
 | 
			
		||||
							<X className="h-4 w-4" />
 | 
			
		||||
						</Button>
 | 
			
		||||
					</div>
 | 
			
		||||
				</motion.div>
 | 
			
		||||
			)}
 | 
			
		||||
		</AnimatePresence>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										28
									
								
								web/src/components/theme-switcher.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								web/src/components/theme-switcher.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Moon, Sun } from "lucide-react"
 | 
			
		||||
import { useTheme } from "next-themes"
 | 
			
		||||
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger } from "@/components/ui/dropdown-menu"
 | 
			
		||||
 | 
			
		||||
export function ThemeSwitcher() {
 | 
			
		||||
	const { setTheme } = useTheme()
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<DropdownMenu>
 | 
			
		||||
			<DropdownMenuTrigger asChild>
 | 
			
		||||
				<Button className="hover:text-primary" variant="ghost" size="icon">
 | 
			
		||||
					<Sun className="h-[1.2rem] w-[1.2rem] rotate-0 scale-100 transition-all dark:-rotate-90 dark:scale-0" />
 | 
			
		||||
					<Moon className=" absolute h-[1.2rem] w-[1.2rem] rotate-90 scale-0 transition-all dark:rotate-0 dark:scale-100" />
 | 
			
		||||
					<span className="sr-only">Toggle theme</span>
 | 
			
		||||
				</Button>
 | 
			
		||||
			</DropdownMenuTrigger>
 | 
			
		||||
			<DropdownMenuContent align="end">
 | 
			
		||||
				<DropdownMenuItem onClick={() => setTheme("light")}>Light</DropdownMenuItem>
 | 
			
		||||
				<DropdownMenuItem onClick={() => setTheme("dark")}>Dark</DropdownMenuItem>
 | 
			
		||||
				<DropdownMenuItem onClick={() => setTheme("system")}>System</DropdownMenuItem>
 | 
			
		||||
			</DropdownMenuContent>
 | 
			
		||||
		</DropdownMenu>
 | 
			
		||||
	)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										66
									
								
								web/src/components/ui/accordion.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								web/src/components/ui/accordion.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,66 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as AccordionPrimitive from "@radix-ui/react-accordion"
 | 
			
		||||
import { ChevronDownIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Accordion({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AccordionPrimitive.Root>) {
 | 
			
		||||
  return <AccordionPrimitive.Root data-slot="accordion" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AccordionItem({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AccordionPrimitive.Item>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AccordionPrimitive.Item
 | 
			
		||||
      data-slot="accordion-item"
 | 
			
		||||
      className={cn("border-b last:border-b-0", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AccordionTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AccordionPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AccordionPrimitive.Header className="flex">
 | 
			
		||||
      <AccordionPrimitive.Trigger
 | 
			
		||||
        data-slot="accordion-trigger"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "focus-visible:border-ring focus-visible:ring-ring/50 flex flex-1 items-start justify-between gap-4 rounded-md py-4 text-left text-sm font-medium transition-all outline-none hover:underline focus-visible:ring-[3px] disabled:pointer-events-none disabled:opacity-50 [&[data-state=open]>svg]:rotate-180",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
        <ChevronDownIcon className="text-muted-foreground pointer-events-none size-4 shrink-0 translate-y-0.5 transition-transform duration-200" />
 | 
			
		||||
      </AccordionPrimitive.Trigger>
 | 
			
		||||
    </AccordionPrimitive.Header>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AccordionContent({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AccordionPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AccordionPrimitive.Content
 | 
			
		||||
      data-slot="accordion-content"
 | 
			
		||||
      className="data-[state=closed]:animate-accordion-up data-[state=open]:animate-accordion-down overflow-hidden text-sm"
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <div className={cn("pt-0 pb-4", className)}>{children}</div>
 | 
			
		||||
    </AccordionPrimitive.Content>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Accordion, AccordionItem, AccordionTrigger, AccordionContent }
 | 
			
		||||
							
								
								
									
										157
									
								
								web/src/components/ui/alert-dialog.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								web/src/components/ui/alert-dialog.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,157 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as AlertDialogPrimitive from "@radix-ui/react-alert-dialog"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { buttonVariants } from "@/components/ui/button"
 | 
			
		||||
 | 
			
		||||
function AlertDialog({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Root>) {
 | 
			
		||||
  return <AlertDialogPrimitive.Root data-slot="alert-dialog" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Trigger data-slot="alert-dialog-trigger" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Portal>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Portal data-slot="alert-dialog-portal" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogOverlay({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Overlay>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Overlay
 | 
			
		||||
      data-slot="alert-dialog-overlay"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPortal>
 | 
			
		||||
      <AlertDialogOverlay />
 | 
			
		||||
      <AlertDialogPrimitive.Content
 | 
			
		||||
        data-slot="alert-dialog-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </AlertDialogPortal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogHeader({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="alert-dialog-header"
 | 
			
		||||
      className={cn("flex flex-col gap-2 text-center sm:text-left", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogFooter({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="alert-dialog-footer"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogTitle({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Title>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Title
 | 
			
		||||
      data-slot="alert-dialog-title"
 | 
			
		||||
      className={cn("text-lg font-semibold", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogDescription({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Description>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Description
 | 
			
		||||
      data-slot="alert-dialog-description"
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogAction({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Action>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Action
 | 
			
		||||
      className={cn(buttonVariants(), className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDialogCancel({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AlertDialogPrimitive.Cancel>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AlertDialogPrimitive.Cancel
 | 
			
		||||
      className={cn(buttonVariants({ variant: "outline" }), className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  AlertDialog,
 | 
			
		||||
  AlertDialogPortal,
 | 
			
		||||
  AlertDialogOverlay,
 | 
			
		||||
  AlertDialogTrigger,
 | 
			
		||||
  AlertDialogContent,
 | 
			
		||||
  AlertDialogHeader,
 | 
			
		||||
  AlertDialogFooter,
 | 
			
		||||
  AlertDialogTitle,
 | 
			
		||||
  AlertDialogDescription,
 | 
			
		||||
  AlertDialogAction,
 | 
			
		||||
  AlertDialogCancel,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										66
									
								
								web/src/components/ui/alert.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								web/src/components/ui/alert.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,66 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { cva, type VariantProps } from "class-variance-authority"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
const alertVariants = cva(
 | 
			
		||||
  "relative w-full rounded-lg border px-4 py-3 text-sm grid has-[>svg]:grid-cols-[calc(var(--spacing)*4)_1fr] grid-cols-[0_1fr] has-[>svg]:gap-x-3 gap-y-0.5 items-start [&>svg]:size-4 [&>svg]:translate-y-0.5 [&>svg]:text-current",
 | 
			
		||||
  {
 | 
			
		||||
    variants: {
 | 
			
		||||
      variant: {
 | 
			
		||||
        default: "bg-card text-card-foreground",
 | 
			
		||||
        destructive:
 | 
			
		||||
          "text-destructive bg-card [&>svg]:text-current *:data-[slot=alert-description]:text-destructive/90",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    defaultVariants: {
 | 
			
		||||
      variant: "default",
 | 
			
		||||
    },
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function Alert({
 | 
			
		||||
  className,
 | 
			
		||||
  variant,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & VariantProps<typeof alertVariants>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="alert"
 | 
			
		||||
      role="alert"
 | 
			
		||||
      className={cn(alertVariants({ variant }), className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertTitle({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="alert-title"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "col-start-2 line-clamp-1 min-h-4 font-medium tracking-tight",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AlertDescription({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="alert-description"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground col-start-2 grid justify-items-start gap-1 text-sm [&_p]:leading-relaxed",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Alert, AlertTitle, AlertDescription }
 | 
			
		||||
							
								
								
									
										11
									
								
								web/src/components/ui/aspect-ratio.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								web/src/components/ui/aspect-ratio.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,11 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as AspectRatioPrimitive from "@radix-ui/react-aspect-ratio"
 | 
			
		||||
 | 
			
		||||
function AspectRatio({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AspectRatioPrimitive.Root>) {
 | 
			
		||||
  return <AspectRatioPrimitive.Root data-slot="aspect-ratio" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { AspectRatio }
 | 
			
		||||
							
								
								
									
										53
									
								
								web/src/components/ui/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								web/src/components/ui/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,53 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as AvatarPrimitive from "@radix-ui/react-avatar"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Avatar({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AvatarPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AvatarPrimitive.Root
 | 
			
		||||
      data-slot="avatar"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "relative flex size-8 shrink-0 overflow-hidden rounded-full",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AvatarImage({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AvatarPrimitive.Image>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AvatarPrimitive.Image
 | 
			
		||||
      data-slot="avatar-image"
 | 
			
		||||
      className={cn("aspect-square size-full", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function AvatarFallback({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof AvatarPrimitive.Fallback>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <AvatarPrimitive.Fallback
 | 
			
		||||
      data-slot="avatar-fallback"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-muted flex size-full items-center justify-center rounded-full",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Avatar, AvatarImage, AvatarFallback }
 | 
			
		||||
							
								
								
									
										46
									
								
								web/src/components/ui/badge.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								web/src/components/ui/badge.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,46 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { Slot } from "@radix-ui/react-slot"
 | 
			
		||||
import { cva, type VariantProps } from "class-variance-authority"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
const badgeVariants = cva(
 | 
			
		||||
  "inline-flex items-center justify-center rounded-md border px-2 py-0.5 text-xs font-medium w-fit whitespace-nowrap shrink-0 [&>svg]:size-3 gap-1 [&>svg]:pointer-events-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive transition-[color,box-shadow] overflow-hidden",
 | 
			
		||||
  {
 | 
			
		||||
    variants: {
 | 
			
		||||
      variant: {
 | 
			
		||||
        default:
 | 
			
		||||
          "border-transparent bg-primary text-primary-foreground [a&]:hover:bg-primary/90",
 | 
			
		||||
        secondary:
 | 
			
		||||
          "border-transparent bg-secondary text-secondary-foreground [a&]:hover:bg-secondary/90",
 | 
			
		||||
        destructive:
 | 
			
		||||
          "border-transparent bg-destructive text-white [a&]:hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
 | 
			
		||||
        outline:
 | 
			
		||||
          "text-foreground [a&]:hover:bg-accent [a&]:hover:text-accent-foreground",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    defaultVariants: {
 | 
			
		||||
      variant: "default",
 | 
			
		||||
    },
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function Badge({
 | 
			
		||||
  className,
 | 
			
		||||
  variant,
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span"> &
 | 
			
		||||
  VariantProps<typeof badgeVariants> & { asChild?: boolean }) {
 | 
			
		||||
  const Comp = asChild ? Slot : "span"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="badge"
 | 
			
		||||
      className={cn(badgeVariants({ variant }), className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Badge, badgeVariants }
 | 
			
		||||
							
								
								
									
										109
									
								
								web/src/components/ui/breadcrumb.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								web/src/components/ui/breadcrumb.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,109 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { Slot } from "@radix-ui/react-slot"
 | 
			
		||||
import { ChevronRight, MoreHorizontal } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Breadcrumb({ ...props }: React.ComponentProps<"nav">) {
 | 
			
		||||
  return <nav aria-label="breadcrumb" data-slot="breadcrumb" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbList({ className, ...props }: React.ComponentProps<"ol">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ol
 | 
			
		||||
      data-slot="breadcrumb-list"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground flex flex-wrap items-center gap-1.5 text-sm break-words sm:gap-2.5",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbItem({ className, ...props }: React.ComponentProps<"li">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <li
 | 
			
		||||
      data-slot="breadcrumb-item"
 | 
			
		||||
      className={cn("inline-flex items-center gap-1.5", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbLink({
 | 
			
		||||
  asChild,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"a"> & {
 | 
			
		||||
  asChild?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  const Comp = asChild ? Slot : "a"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="breadcrumb-link"
 | 
			
		||||
      className={cn("hover:text-foreground transition-colors", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbPage({ className, ...props }: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="breadcrumb-page"
 | 
			
		||||
      role="link"
 | 
			
		||||
      aria-disabled="true"
 | 
			
		||||
      aria-current="page"
 | 
			
		||||
      className={cn("text-foreground font-normal", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbSeparator({
 | 
			
		||||
  children,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"li">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <li
 | 
			
		||||
      data-slot="breadcrumb-separator"
 | 
			
		||||
      role="presentation"
 | 
			
		||||
      aria-hidden="true"
 | 
			
		||||
      className={cn("[&>svg]:size-3.5", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children ?? <ChevronRight />}
 | 
			
		||||
    </li>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function BreadcrumbEllipsis({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="breadcrumb-ellipsis"
 | 
			
		||||
      role="presentation"
 | 
			
		||||
      aria-hidden="true"
 | 
			
		||||
      className={cn("flex size-9 items-center justify-center", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <MoreHorizontal className="size-4" />
 | 
			
		||||
      <span className="sr-only">More</span>
 | 
			
		||||
    </span>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Breadcrumb,
 | 
			
		||||
  BreadcrumbList,
 | 
			
		||||
  BreadcrumbItem,
 | 
			
		||||
  BreadcrumbLink,
 | 
			
		||||
  BreadcrumbPage,
 | 
			
		||||
  BreadcrumbSeparator,
 | 
			
		||||
  BreadcrumbEllipsis,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										59
									
								
								web/src/components/ui/button.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								web/src/components/ui/button.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,59 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { Slot } from "@radix-ui/react-slot"
 | 
			
		||||
import { cva, type VariantProps } from "class-variance-authority"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
const buttonVariants = cva(
 | 
			
		||||
  "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
 | 
			
		||||
  {
 | 
			
		||||
    variants: {
 | 
			
		||||
      variant: {
 | 
			
		||||
        default:
 | 
			
		||||
          "bg-primary text-primary-foreground shadow-xs hover:bg-primary/90",
 | 
			
		||||
        destructive:
 | 
			
		||||
          "bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
 | 
			
		||||
        outline:
 | 
			
		||||
          "border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50",
 | 
			
		||||
        secondary:
 | 
			
		||||
          "bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80",
 | 
			
		||||
        ghost:
 | 
			
		||||
          "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
 | 
			
		||||
        link: "text-primary underline-offset-4 hover:underline",
 | 
			
		||||
      },
 | 
			
		||||
      size: {
 | 
			
		||||
        default: "h-9 px-4 py-2 has-[>svg]:px-3",
 | 
			
		||||
        sm: "h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5",
 | 
			
		||||
        lg: "h-10 rounded-md px-6 has-[>svg]:px-4",
 | 
			
		||||
        icon: "size-9",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    defaultVariants: {
 | 
			
		||||
      variant: "default",
 | 
			
		||||
      size: "default",
 | 
			
		||||
    },
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function Button({
 | 
			
		||||
  className,
 | 
			
		||||
  variant,
 | 
			
		||||
  size,
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"button"> &
 | 
			
		||||
  VariantProps<typeof buttonVariants> & {
 | 
			
		||||
    asChild?: boolean
 | 
			
		||||
  }) {
 | 
			
		||||
  const Comp = asChild ? Slot : "button"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="button"
 | 
			
		||||
      className={cn(buttonVariants({ variant, size, className }))}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Button, buttonVariants }
 | 
			
		||||
							
								
								
									
										75
									
								
								web/src/components/ui/calendar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								web/src/components/ui/calendar.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,75 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { ChevronLeft, ChevronRight } from "lucide-react"
 | 
			
		||||
import { DayPicker } from "react-day-picker"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { buttonVariants } from "@/components/ui/button"
 | 
			
		||||
 | 
			
		||||
function Calendar({
 | 
			
		||||
  className,
 | 
			
		||||
  classNames,
 | 
			
		||||
  showOutsideDays = true,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DayPicker>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DayPicker
 | 
			
		||||
      showOutsideDays={showOutsideDays}
 | 
			
		||||
      className={cn("p-3", className)}
 | 
			
		||||
      classNames={{
 | 
			
		||||
        months: "flex flex-col sm:flex-row gap-2",
 | 
			
		||||
        month: "flex flex-col gap-4",
 | 
			
		||||
        caption: "flex justify-center pt-1 relative items-center w-full",
 | 
			
		||||
        caption_label: "text-sm font-medium",
 | 
			
		||||
        nav: "flex items-center gap-1",
 | 
			
		||||
        nav_button: cn(
 | 
			
		||||
          buttonVariants({ variant: "outline" }),
 | 
			
		||||
          "size-7 bg-transparent p-0 opacity-50 hover:opacity-100"
 | 
			
		||||
        ),
 | 
			
		||||
        nav_button_previous: "absolute left-1",
 | 
			
		||||
        nav_button_next: "absolute right-1",
 | 
			
		||||
        table: "w-full border-collapse space-x-1",
 | 
			
		||||
        head_row: "flex",
 | 
			
		||||
        head_cell:
 | 
			
		||||
          "text-muted-foreground rounded-md w-8 font-normal text-[0.8rem]",
 | 
			
		||||
        row: "flex w-full mt-2",
 | 
			
		||||
        cell: cn(
 | 
			
		||||
          "relative p-0 text-center text-sm focus-within:relative focus-within:z-20 [&:has([aria-selected])]:bg-accent [&:has([aria-selected].day-range-end)]:rounded-r-md",
 | 
			
		||||
          props.mode === "range"
 | 
			
		||||
            ? "[&:has(>.day-range-end)]:rounded-r-md [&:has(>.day-range-start)]:rounded-l-md first:[&:has([aria-selected])]:rounded-l-md last:[&:has([aria-selected])]:rounded-r-md"
 | 
			
		||||
            : "[&:has([aria-selected])]:rounded-md"
 | 
			
		||||
        ),
 | 
			
		||||
        day: cn(
 | 
			
		||||
          buttonVariants({ variant: "ghost" }),
 | 
			
		||||
          "size-8 p-0 font-normal aria-selected:opacity-100"
 | 
			
		||||
        ),
 | 
			
		||||
        day_range_start:
 | 
			
		||||
          "day-range-start aria-selected:bg-primary aria-selected:text-primary-foreground",
 | 
			
		||||
        day_range_end:
 | 
			
		||||
          "day-range-end aria-selected:bg-primary aria-selected:text-primary-foreground",
 | 
			
		||||
        day_selected:
 | 
			
		||||
          "bg-primary text-primary-foreground hover:bg-primary hover:text-primary-foreground focus:bg-primary focus:text-primary-foreground",
 | 
			
		||||
        day_today: "bg-accent text-accent-foreground",
 | 
			
		||||
        day_outside:
 | 
			
		||||
          "day-outside text-muted-foreground aria-selected:text-muted-foreground",
 | 
			
		||||
        day_disabled: "text-muted-foreground opacity-50",
 | 
			
		||||
        day_range_middle:
 | 
			
		||||
          "aria-selected:bg-accent aria-selected:text-accent-foreground",
 | 
			
		||||
        day_hidden: "invisible",
 | 
			
		||||
        ...classNames,
 | 
			
		||||
      }}
 | 
			
		||||
      components={{
 | 
			
		||||
        IconLeft: ({ className, ...props }) => (
 | 
			
		||||
          <ChevronLeft className={cn("size-4", className)} {...props} />
 | 
			
		||||
        ),
 | 
			
		||||
        IconRight: ({ className, ...props }) => (
 | 
			
		||||
          <ChevronRight className={cn("size-4", className)} {...props} />
 | 
			
		||||
        ),
 | 
			
		||||
      }}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Calendar }
 | 
			
		||||
							
								
								
									
										92
									
								
								web/src/components/ui/card.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								web/src/components/ui/card.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,92 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Card({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-card text-card-foreground flex flex-col gap-6 rounded-xl border py-6 shadow-sm",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardHeader({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-header"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "@container/card-header grid auto-rows-min grid-rows-[auto_auto] items-start gap-1.5 px-6 has-data-[slot=card-action]:grid-cols-[1fr_auto] [.border-b]:pb-6",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardTitle({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-title"
 | 
			
		||||
      className={cn("leading-none font-semibold", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardDescription({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-description"
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardAction({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-action"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "col-start-2 row-span-2 row-start-1 self-start justify-self-end",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardContent({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-content"
 | 
			
		||||
      className={cn("px-6", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CardFooter({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="card-footer"
 | 
			
		||||
      className={cn("flex items-center px-6 [.border-t]:pt-6", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Card,
 | 
			
		||||
  CardHeader,
 | 
			
		||||
  CardFooter,
 | 
			
		||||
  CardTitle,
 | 
			
		||||
  CardAction,
 | 
			
		||||
  CardDescription,
 | 
			
		||||
  CardContent,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										241
									
								
								web/src/components/ui/carousel.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										241
									
								
								web/src/components/ui/carousel.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,241 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import useEmblaCarousel, {
 | 
			
		||||
  type UseEmblaCarouselType,
 | 
			
		||||
} from "embla-carousel-react"
 | 
			
		||||
import { ArrowLeft, ArrowRight } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
 | 
			
		||||
type CarouselApi = UseEmblaCarouselType[1]
 | 
			
		||||
type UseCarouselParameters = Parameters<typeof useEmblaCarousel>
 | 
			
		||||
type CarouselOptions = UseCarouselParameters[0]
 | 
			
		||||
type CarouselPlugin = UseCarouselParameters[1]
 | 
			
		||||
 | 
			
		||||
type CarouselProps = {
 | 
			
		||||
  opts?: CarouselOptions
 | 
			
		||||
  plugins?: CarouselPlugin
 | 
			
		||||
  orientation?: "horizontal" | "vertical"
 | 
			
		||||
  setApi?: (api: CarouselApi) => void
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type CarouselContextProps = {
 | 
			
		||||
  carouselRef: ReturnType<typeof useEmblaCarousel>[0]
 | 
			
		||||
  api: ReturnType<typeof useEmblaCarousel>[1]
 | 
			
		||||
  scrollPrev: () => void
 | 
			
		||||
  scrollNext: () => void
 | 
			
		||||
  canScrollPrev: boolean
 | 
			
		||||
  canScrollNext: boolean
 | 
			
		||||
} & CarouselProps
 | 
			
		||||
 | 
			
		||||
const CarouselContext = React.createContext<CarouselContextProps | null>(null)
 | 
			
		||||
 | 
			
		||||
function useCarousel() {
 | 
			
		||||
  const context = React.useContext(CarouselContext)
 | 
			
		||||
 | 
			
		||||
  if (!context) {
 | 
			
		||||
    throw new Error("useCarousel must be used within a <Carousel />")
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return context
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function Carousel({
 | 
			
		||||
  orientation = "horizontal",
 | 
			
		||||
  opts,
 | 
			
		||||
  setApi,
 | 
			
		||||
  plugins,
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & CarouselProps) {
 | 
			
		||||
  const [carouselRef, api] = useEmblaCarousel(
 | 
			
		||||
    {
 | 
			
		||||
      ...opts,
 | 
			
		||||
      axis: orientation === "horizontal" ? "x" : "y",
 | 
			
		||||
    },
 | 
			
		||||
    plugins
 | 
			
		||||
  )
 | 
			
		||||
  const [canScrollPrev, setCanScrollPrev] = React.useState(false)
 | 
			
		||||
  const [canScrollNext, setCanScrollNext] = React.useState(false)
 | 
			
		||||
 | 
			
		||||
  const onSelect = React.useCallback((api: CarouselApi) => {
 | 
			
		||||
    if (!api) return
 | 
			
		||||
    setCanScrollPrev(api.canScrollPrev())
 | 
			
		||||
    setCanScrollNext(api.canScrollNext())
 | 
			
		||||
  }, [])
 | 
			
		||||
 | 
			
		||||
  const scrollPrev = React.useCallback(() => {
 | 
			
		||||
    api?.scrollPrev()
 | 
			
		||||
  }, [api])
 | 
			
		||||
 | 
			
		||||
  const scrollNext = React.useCallback(() => {
 | 
			
		||||
    api?.scrollNext()
 | 
			
		||||
  }, [api])
 | 
			
		||||
 | 
			
		||||
  const handleKeyDown = React.useCallback(
 | 
			
		||||
    (event: React.KeyboardEvent<HTMLDivElement>) => {
 | 
			
		||||
      if (event.key === "ArrowLeft") {
 | 
			
		||||
        event.preventDefault()
 | 
			
		||||
        scrollPrev()
 | 
			
		||||
      } else if (event.key === "ArrowRight") {
 | 
			
		||||
        event.preventDefault()
 | 
			
		||||
        scrollNext()
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    [scrollPrev, scrollNext]
 | 
			
		||||
  )
 | 
			
		||||
 | 
			
		||||
  React.useEffect(() => {
 | 
			
		||||
    if (!api || !setApi) return
 | 
			
		||||
    setApi(api)
 | 
			
		||||
  }, [api, setApi])
 | 
			
		||||
 | 
			
		||||
  React.useEffect(() => {
 | 
			
		||||
    if (!api) return
 | 
			
		||||
    onSelect(api)
 | 
			
		||||
    api.on("reInit", onSelect)
 | 
			
		||||
    api.on("select", onSelect)
 | 
			
		||||
 | 
			
		||||
    return () => {
 | 
			
		||||
      api?.off("select", onSelect)
 | 
			
		||||
    }
 | 
			
		||||
  }, [api, onSelect])
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <CarouselContext.Provider
 | 
			
		||||
      value={{
 | 
			
		||||
        carouselRef,
 | 
			
		||||
        api: api,
 | 
			
		||||
        opts,
 | 
			
		||||
        orientation:
 | 
			
		||||
          orientation || (opts?.axis === "y" ? "vertical" : "horizontal"),
 | 
			
		||||
        scrollPrev,
 | 
			
		||||
        scrollNext,
 | 
			
		||||
        canScrollPrev,
 | 
			
		||||
        canScrollNext,
 | 
			
		||||
      }}
 | 
			
		||||
    >
 | 
			
		||||
      <div
 | 
			
		||||
        onKeyDownCapture={handleKeyDown}
 | 
			
		||||
        className={cn("relative", className)}
 | 
			
		||||
        role="region"
 | 
			
		||||
        aria-roledescription="carousel"
 | 
			
		||||
        data-slot="carousel"
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
      </div>
 | 
			
		||||
    </CarouselContext.Provider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CarouselContent({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  const { carouselRef, orientation } = useCarousel()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      ref={carouselRef}
 | 
			
		||||
      className="overflow-hidden"
 | 
			
		||||
      data-slot="carousel-content"
 | 
			
		||||
    >
 | 
			
		||||
      <div
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "flex",
 | 
			
		||||
          orientation === "horizontal" ? "-ml-4" : "-mt-4 flex-col",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CarouselItem({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  const { orientation } = useCarousel()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      role="group"
 | 
			
		||||
      aria-roledescription="slide"
 | 
			
		||||
      data-slot="carousel-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "min-w-0 shrink-0 grow-0 basis-full",
 | 
			
		||||
        orientation === "horizontal" ? "pl-4" : "pt-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CarouselPrevious({
 | 
			
		||||
  className,
 | 
			
		||||
  variant = "outline",
 | 
			
		||||
  size = "icon",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Button>) {
 | 
			
		||||
  const { orientation, scrollPrev, canScrollPrev } = useCarousel()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Button
 | 
			
		||||
      data-slot="carousel-previous"
 | 
			
		||||
      variant={variant}
 | 
			
		||||
      size={size}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "absolute size-8 rounded-full",
 | 
			
		||||
        orientation === "horizontal"
 | 
			
		||||
          ? "top-1/2 -left-12 -translate-y-1/2"
 | 
			
		||||
          : "-top-12 left-1/2 -translate-x-1/2 rotate-90",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      disabled={!canScrollPrev}
 | 
			
		||||
      onClick={scrollPrev}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ArrowLeft />
 | 
			
		||||
      <span className="sr-only">Previous slide</span>
 | 
			
		||||
    </Button>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CarouselNext({
 | 
			
		||||
  className,
 | 
			
		||||
  variant = "outline",
 | 
			
		||||
  size = "icon",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Button>) {
 | 
			
		||||
  const { orientation, scrollNext, canScrollNext } = useCarousel()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Button
 | 
			
		||||
      data-slot="carousel-next"
 | 
			
		||||
      variant={variant}
 | 
			
		||||
      size={size}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "absolute size-8 rounded-full",
 | 
			
		||||
        orientation === "horizontal"
 | 
			
		||||
          ? "top-1/2 -right-12 -translate-y-1/2"
 | 
			
		||||
          : "-bottom-12 left-1/2 -translate-x-1/2 rotate-90",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      disabled={!canScrollNext}
 | 
			
		||||
      onClick={scrollNext}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ArrowRight />
 | 
			
		||||
      <span className="sr-only">Next slide</span>
 | 
			
		||||
    </Button>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  type CarouselApi,
 | 
			
		||||
  Carousel,
 | 
			
		||||
  CarouselContent,
 | 
			
		||||
  CarouselItem,
 | 
			
		||||
  CarouselPrevious,
 | 
			
		||||
  CarouselNext,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										348
									
								
								web/src/components/ui/chart.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										348
									
								
								web/src/components/ui/chart.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,348 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as RechartsPrimitive from "recharts"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
const THEMES = { light: "", dark: ".dark" } as const
 | 
			
		||||
 | 
			
		||||
export type ChartConfig = {
 | 
			
		||||
  [k in string]: {
 | 
			
		||||
    label?: React.ReactNode
 | 
			
		||||
    icon?: React.ComponentType
 | 
			
		||||
  } & (
 | 
			
		||||
    | { color?: string; theme?: never }
 | 
			
		||||
    | { color?: never; theme: Record<keyof typeof THEMES, string> }
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type ChartContextProps = {
 | 
			
		||||
  config: ChartConfig
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const ChartContext = React.createContext<ChartContextProps | null>(null)
 | 
			
		||||
 | 
			
		||||
function useChart() {
 | 
			
		||||
  const context = React.useContext(ChartContext)
 | 
			
		||||
 | 
			
		||||
  if (!context) {
 | 
			
		||||
    throw new Error("useChart must be used within a <ChartContainer />")
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return context
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ChartContainer({
 | 
			
		||||
  id,
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  config,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & {
 | 
			
		||||
  config: ChartConfig
 | 
			
		||||
  children: React.ComponentProps<
 | 
			
		||||
    typeof RechartsPrimitive.ResponsiveContainer
 | 
			
		||||
  >["children"]
 | 
			
		||||
}) {
 | 
			
		||||
  const uniqueId = React.useId()
 | 
			
		||||
  const chartId = `chart-${id || uniqueId.replace(/:/g, "")}`
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <ChartContext.Provider value={{ config }}>
 | 
			
		||||
      <div
 | 
			
		||||
        data-slot="chart"
 | 
			
		||||
        data-chart={chartId}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "[&_.recharts-cartesian-axis-tick_text]:fill-muted-foreground [&_.recharts-cartesian-grid_line[stroke='#ccc']]:stroke-border/50 [&_.recharts-curve.recharts-tooltip-cursor]:stroke-border [&_.recharts-polar-grid_[stroke='#ccc']]:stroke-border [&_.recharts-radial-bar-background-sector]:fill-muted [&_.recharts-rectangle.recharts-tooltip-cursor]:fill-muted [&_.recharts-reference-line_[stroke='#ccc']]:stroke-border flex aspect-video justify-center text-xs [&_.recharts-dot[stroke='#fff']]:stroke-transparent [&_.recharts-layer]:outline-hidden [&_.recharts-sector]:outline-hidden [&_.recharts-sector[stroke='#fff']]:stroke-transparent [&_.recharts-surface]:outline-hidden",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        <ChartStyle id={chartId} config={config} />
 | 
			
		||||
        <RechartsPrimitive.ResponsiveContainer>
 | 
			
		||||
          {children}
 | 
			
		||||
        </RechartsPrimitive.ResponsiveContainer>
 | 
			
		||||
      </div>
 | 
			
		||||
    </ChartContext.Provider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const ChartStyle = ({ id, config }: { id: string; config: ChartConfig }) => {
 | 
			
		||||
  const colorConfig = Object.entries(config).filter(
 | 
			
		||||
    ([, config]) => config.theme || config.color
 | 
			
		||||
  )
 | 
			
		||||
 | 
			
		||||
  if (!colorConfig.length) {
 | 
			
		||||
    return null
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <style
 | 
			
		||||
      dangerouslySetInnerHTML={{
 | 
			
		||||
        __html: Object.entries(THEMES)
 | 
			
		||||
          .map(
 | 
			
		||||
            ([theme, prefix]) => `
 | 
			
		||||
${prefix} [data-chart=${id}] {
 | 
			
		||||
${colorConfig
 | 
			
		||||
  .map(([key, itemConfig]) => {
 | 
			
		||||
    const color =
 | 
			
		||||
      itemConfig.theme?.[theme as keyof typeof itemConfig.theme] ||
 | 
			
		||||
      itemConfig.color
 | 
			
		||||
    return color ? `  --color-${key}: ${color};` : null
 | 
			
		||||
  })
 | 
			
		||||
  .join("\n")}
 | 
			
		||||
}
 | 
			
		||||
`
 | 
			
		||||
          )
 | 
			
		||||
          .join("\n"),
 | 
			
		||||
      }}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const ChartTooltip = RechartsPrimitive.Tooltip
 | 
			
		||||
 | 
			
		||||
function ChartTooltipContent({
 | 
			
		||||
  active,
 | 
			
		||||
  payload,
 | 
			
		||||
  className,
 | 
			
		||||
  indicator = "dot",
 | 
			
		||||
  hideLabel = false,
 | 
			
		||||
  hideIndicator = false,
 | 
			
		||||
  label,
 | 
			
		||||
  labelFormatter,
 | 
			
		||||
  labelClassName,
 | 
			
		||||
  formatter,
 | 
			
		||||
  color,
 | 
			
		||||
  nameKey,
 | 
			
		||||
  labelKey,
 | 
			
		||||
}: React.ComponentProps<typeof RechartsPrimitive.Tooltip> &
 | 
			
		||||
  React.ComponentProps<"div"> & {
 | 
			
		||||
    hideLabel?: boolean
 | 
			
		||||
    hideIndicator?: boolean
 | 
			
		||||
    indicator?: "line" | "dot" | "dashed"
 | 
			
		||||
    nameKey?: string
 | 
			
		||||
    labelKey?: string
 | 
			
		||||
  }) {
 | 
			
		||||
  const { config } = useChart()
 | 
			
		||||
 | 
			
		||||
  const tooltipLabel = React.useMemo(() => {
 | 
			
		||||
    if (hideLabel || !payload?.length) {
 | 
			
		||||
      return null
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const [item] = payload
 | 
			
		||||
    const key = `${labelKey || item?.dataKey || item?.name || "value"}`
 | 
			
		||||
    const itemConfig = getPayloadConfigFromPayload(config, item, key)
 | 
			
		||||
    const value =
 | 
			
		||||
      !labelKey && typeof label === "string"
 | 
			
		||||
        ? config[label as keyof typeof config]?.label || label
 | 
			
		||||
        : itemConfig?.label
 | 
			
		||||
 | 
			
		||||
    if (labelFormatter) {
 | 
			
		||||
      return (
 | 
			
		||||
        <div className={cn("font-medium", labelClassName)}>
 | 
			
		||||
          {labelFormatter(value, payload)}
 | 
			
		||||
        </div>
 | 
			
		||||
      )
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (!value) {
 | 
			
		||||
      return null
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return <div className={cn("font-medium", labelClassName)}>{value}</div>
 | 
			
		||||
  }, [
 | 
			
		||||
    label,
 | 
			
		||||
    labelFormatter,
 | 
			
		||||
    payload,
 | 
			
		||||
    hideLabel,
 | 
			
		||||
    labelClassName,
 | 
			
		||||
    config,
 | 
			
		||||
    labelKey,
 | 
			
		||||
  ])
 | 
			
		||||
 | 
			
		||||
  if (!active || !payload?.length) {
 | 
			
		||||
    return null
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const nestLabel = payload.length === 1 && indicator !== "dot"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "border-border/50 bg-background grid min-w-[8rem] items-start gap-1.5 rounded-lg border px-2.5 py-1.5 text-xs shadow-xl",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
    >
 | 
			
		||||
      {!nestLabel ? tooltipLabel : null}
 | 
			
		||||
      <div className="grid gap-1.5">
 | 
			
		||||
        {payload.map((item, index) => {
 | 
			
		||||
          const key = `${nameKey || item.name || item.dataKey || "value"}`
 | 
			
		||||
          const itemConfig = getPayloadConfigFromPayload(config, item, key)
 | 
			
		||||
          const indicatorColor = color || item.payload.fill || item.color
 | 
			
		||||
 | 
			
		||||
          return (
 | 
			
		||||
            <div
 | 
			
		||||
              key={item.dataKey}
 | 
			
		||||
              className={cn(
 | 
			
		||||
                "[&>svg]:text-muted-foreground flex w-full flex-wrap items-stretch gap-2 [&>svg]:h-2.5 [&>svg]:w-2.5",
 | 
			
		||||
                indicator === "dot" && "items-center"
 | 
			
		||||
              )}
 | 
			
		||||
            >
 | 
			
		||||
              {formatter && item?.value !== undefined && item.name ? (
 | 
			
		||||
                formatter(item.value, item.name, item, index, item.payload)
 | 
			
		||||
              ) : (
 | 
			
		||||
                <>
 | 
			
		||||
                  {itemConfig?.icon ? (
 | 
			
		||||
                    <itemConfig.icon />
 | 
			
		||||
                  ) : (
 | 
			
		||||
                    !hideIndicator && (
 | 
			
		||||
                      <div
 | 
			
		||||
                        className={cn(
 | 
			
		||||
                          "shrink-0 rounded-[2px] border-(--color-border) bg-(--color-bg)",
 | 
			
		||||
                          {
 | 
			
		||||
                            "h-2.5 w-2.5": indicator === "dot",
 | 
			
		||||
                            "w-1": indicator === "line",
 | 
			
		||||
                            "w-0 border-[1.5px] border-dashed bg-transparent":
 | 
			
		||||
                              indicator === "dashed",
 | 
			
		||||
                            "my-0.5": nestLabel && indicator === "dashed",
 | 
			
		||||
                          }
 | 
			
		||||
                        )}
 | 
			
		||||
                        style={
 | 
			
		||||
                          {
 | 
			
		||||
                            "--color-bg": indicatorColor,
 | 
			
		||||
                            "--color-border": indicatorColor,
 | 
			
		||||
                          } as React.CSSProperties
 | 
			
		||||
                        }
 | 
			
		||||
                      />
 | 
			
		||||
                    )
 | 
			
		||||
                  )}
 | 
			
		||||
                  <div
 | 
			
		||||
                    className={cn(
 | 
			
		||||
                      "flex flex-1 justify-between leading-none",
 | 
			
		||||
                      nestLabel ? "items-end" : "items-center"
 | 
			
		||||
                    )}
 | 
			
		||||
                  >
 | 
			
		||||
                    <div className="grid gap-1.5">
 | 
			
		||||
                      {nestLabel ? tooltipLabel : null}
 | 
			
		||||
                      <span className="text-muted-foreground">
 | 
			
		||||
                        {itemConfig?.label || item.name}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    {item.value && (
 | 
			
		||||
                      <span className="text-foreground font-mono font-medium tabular-nums">
 | 
			
		||||
                        {item.value.toLocaleString()}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    )}
 | 
			
		||||
                  </div>
 | 
			
		||||
                </>
 | 
			
		||||
              )}
 | 
			
		||||
            </div>
 | 
			
		||||
          )
 | 
			
		||||
        })}
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const ChartLegend = RechartsPrimitive.Legend
 | 
			
		||||
 | 
			
		||||
function ChartLegendContent({
 | 
			
		||||
  className,
 | 
			
		||||
  hideIcon = false,
 | 
			
		||||
  payload,
 | 
			
		||||
  verticalAlign = "bottom",
 | 
			
		||||
  nameKey,
 | 
			
		||||
}: React.ComponentProps<"div"> &
 | 
			
		||||
  Pick<RechartsPrimitive.LegendProps, "payload" | "verticalAlign"> & {
 | 
			
		||||
    hideIcon?: boolean
 | 
			
		||||
    nameKey?: string
 | 
			
		||||
  }) {
 | 
			
		||||
  const { config } = useChart()
 | 
			
		||||
 | 
			
		||||
  if (!payload?.length) {
 | 
			
		||||
    return null
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex items-center justify-center gap-4",
 | 
			
		||||
        verticalAlign === "top" ? "pb-3" : "pt-3",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
    >
 | 
			
		||||
      {payload.map((item) => {
 | 
			
		||||
        const key = `${nameKey || item.dataKey || "value"}`
 | 
			
		||||
        const itemConfig = getPayloadConfigFromPayload(config, item, key)
 | 
			
		||||
 | 
			
		||||
        return (
 | 
			
		||||
          <div
 | 
			
		||||
            key={item.value}
 | 
			
		||||
            className={cn(
 | 
			
		||||
              "[&>svg]:text-muted-foreground flex items-center gap-1.5 [&>svg]:h-3 [&>svg]:w-3"
 | 
			
		||||
            )}
 | 
			
		||||
          >
 | 
			
		||||
            {itemConfig?.icon && !hideIcon ? (
 | 
			
		||||
              <itemConfig.icon />
 | 
			
		||||
            ) : (
 | 
			
		||||
              <div
 | 
			
		||||
                className="h-2 w-2 shrink-0 rounded-[2px]"
 | 
			
		||||
                style={{
 | 
			
		||||
                  backgroundColor: item.color,
 | 
			
		||||
                }}
 | 
			
		||||
              />
 | 
			
		||||
            )}
 | 
			
		||||
            {itemConfig?.label}
 | 
			
		||||
          </div>
 | 
			
		||||
        )
 | 
			
		||||
      })}
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
function getPayloadConfigFromPayload(
 | 
			
		||||
  config: ChartConfig,
 | 
			
		||||
  payload: unknown,
 | 
			
		||||
  key: string
 | 
			
		||||
) {
 | 
			
		||||
  if (typeof payload !== "object" || payload === null) {
 | 
			
		||||
    return undefined
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const payloadPayload =
 | 
			
		||||
    "payload" in payload &&
 | 
			
		||||
    typeof payload.payload === "object" &&
 | 
			
		||||
    payload.payload !== null
 | 
			
		||||
      ? payload.payload
 | 
			
		||||
      : undefined
 | 
			
		||||
 | 
			
		||||
  let configLabelKey: string = key
 | 
			
		||||
 | 
			
		||||
  if (
 | 
			
		||||
    key in payload &&
 | 
			
		||||
    typeof payload[key as keyof typeof payload] === "string"
 | 
			
		||||
  ) {
 | 
			
		||||
    configLabelKey = payload[key as keyof typeof payload] as string
 | 
			
		||||
  } else if (
 | 
			
		||||
    payloadPayload &&
 | 
			
		||||
    key in payloadPayload &&
 | 
			
		||||
    typeof payloadPayload[key as keyof typeof payloadPayload] === "string"
 | 
			
		||||
  ) {
 | 
			
		||||
    configLabelKey = payloadPayload[
 | 
			
		||||
      key as keyof typeof payloadPayload
 | 
			
		||||
    ] as string
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return configLabelKey in config
 | 
			
		||||
    ? config[configLabelKey]
 | 
			
		||||
    : config[key as keyof typeof config]
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
	ChartContainer, ChartLegend,
 | 
			
		||||
	ChartLegendContent,
 | 
			
		||||
	ChartStyle, ChartTooltip,
 | 
			
		||||
	ChartTooltipContent
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										32
									
								
								web/src/components/ui/checkbox.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								web/src/components/ui/checkbox.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,32 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as CheckboxPrimitive from "@radix-ui/react-checkbox"
 | 
			
		||||
import { CheckIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Checkbox({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CheckboxPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CheckboxPrimitive.Root
 | 
			
		||||
      data-slot="checkbox"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "peer border-input dark:bg-input/30 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground dark:data-[state=checked]:bg-primary data-[state=checked]:border-primary focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive size-4 shrink-0 rounded-[4px] border shadow-xs transition-shadow outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <CheckboxPrimitive.Indicator
 | 
			
		||||
        data-slot="checkbox-indicator"
 | 
			
		||||
        className="flex items-center justify-center text-current transition-none"
 | 
			
		||||
      >
 | 
			
		||||
        <CheckIcon className="size-3.5" />
 | 
			
		||||
      </CheckboxPrimitive.Indicator>
 | 
			
		||||
    </CheckboxPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Checkbox }
 | 
			
		||||
							
								
								
									
										33
									
								
								web/src/components/ui/collapsible.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								web/src/components/ui/collapsible.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as CollapsiblePrimitive from "@radix-ui/react-collapsible"
 | 
			
		||||
 | 
			
		||||
function Collapsible({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CollapsiblePrimitive.Root>) {
 | 
			
		||||
  return <CollapsiblePrimitive.Root data-slot="collapsible" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CollapsibleTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CollapsiblePrimitive.CollapsibleTrigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CollapsiblePrimitive.CollapsibleTrigger
 | 
			
		||||
      data-slot="collapsible-trigger"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CollapsibleContent({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CollapsiblePrimitive.CollapsibleContent>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CollapsiblePrimitive.CollapsibleContent
 | 
			
		||||
      data-slot="collapsible-content"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Collapsible, CollapsibleTrigger, CollapsibleContent }
 | 
			
		||||
							
								
								
									
										177
									
								
								web/src/components/ui/command.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										177
									
								
								web/src/components/ui/command.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,177 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { Command as CommandPrimitive } from "cmdk"
 | 
			
		||||
import { SearchIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import {
 | 
			
		||||
  Dialog,
 | 
			
		||||
  DialogContent,
 | 
			
		||||
  DialogDescription,
 | 
			
		||||
  DialogHeader,
 | 
			
		||||
  DialogTitle,
 | 
			
		||||
} from "@/components/ui/dialog"
 | 
			
		||||
 | 
			
		||||
function Command({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive
 | 
			
		||||
      data-slot="command"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-popover text-popover-foreground flex h-full w-full flex-col overflow-hidden rounded-md",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandDialog({
 | 
			
		||||
  title = "Command Palette",
 | 
			
		||||
  description = "Search for a command to run...",
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Dialog> & {
 | 
			
		||||
  title?: string
 | 
			
		||||
  description?: string
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Dialog {...props}>
 | 
			
		||||
      <DialogHeader className="sr-only">
 | 
			
		||||
        <DialogTitle>{title}</DialogTitle>
 | 
			
		||||
        <DialogDescription>{description}</DialogDescription>
 | 
			
		||||
      </DialogHeader>
 | 
			
		||||
      <DialogContent className="overflow-hidden p-0">
 | 
			
		||||
        <Command className="[&_[cmdk-group-heading]]:text-muted-foreground **:data-[slot=command-input-wrapper]:h-12 [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group]]:px-2 [&_[cmdk-group]:not([hidden])_~[cmdk-group]]:pt-0 [&_[cmdk-input-wrapper]_svg]:h-5 [&_[cmdk-input-wrapper]_svg]:w-5 [&_[cmdk-input]]:h-12 [&_[cmdk-item]]:px-2 [&_[cmdk-item]]:py-3 [&_[cmdk-item]_svg]:h-5 [&_[cmdk-item]_svg]:w-5">
 | 
			
		||||
          {children}
 | 
			
		||||
        </Command>
 | 
			
		||||
      </DialogContent>
 | 
			
		||||
    </Dialog>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandInput({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.Input>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="command-input-wrapper"
 | 
			
		||||
      className="flex h-9 items-center gap-2 border-b px-3"
 | 
			
		||||
    >
 | 
			
		||||
      <SearchIcon className="size-4 shrink-0 opacity-50" />
 | 
			
		||||
      <CommandPrimitive.Input
 | 
			
		||||
        data-slot="command-input"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "placeholder:text-muted-foreground flex h-10 w-full rounded-md bg-transparent py-3 text-sm outline-hidden disabled:cursor-not-allowed disabled:opacity-50",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandList({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.List>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive.List
 | 
			
		||||
      data-slot="command-list"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "max-h-[300px] scroll-py-1 overflow-x-hidden overflow-y-auto",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandEmpty({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.Empty>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive.Empty
 | 
			
		||||
      data-slot="command-empty"
 | 
			
		||||
      className="py-6 text-center text-sm"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandGroup({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.Group>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive.Group
 | 
			
		||||
      data-slot="command-group"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-foreground [&_[cmdk-group-heading]]:text-muted-foreground overflow-hidden p-1 [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive.Separator
 | 
			
		||||
      data-slot="command-separator"
 | 
			
		||||
      className={cn("bg-border -mx-1 h-px", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandItem({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof CommandPrimitive.Item>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <CommandPrimitive.Item
 | 
			
		||||
      data-slot="command-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[selected=true]:bg-accent data-[selected=true]:text-accent-foreground [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled=true]:pointer-events-none data-[disabled=true]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function CommandShortcut({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="command-shortcut"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground ml-auto text-xs tracking-widest",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Command,
 | 
			
		||||
  CommandDialog,
 | 
			
		||||
  CommandInput,
 | 
			
		||||
  CommandList,
 | 
			
		||||
  CommandEmpty,
 | 
			
		||||
  CommandGroup,
 | 
			
		||||
  CommandItem,
 | 
			
		||||
  CommandShortcut,
 | 
			
		||||
  CommandSeparator,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										252
									
								
								web/src/components/ui/context-menu.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								web/src/components/ui/context-menu.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,252 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as ContextMenuPrimitive from "@radix-ui/react-context-menu"
 | 
			
		||||
import { CheckIcon, ChevronRightIcon, CircleIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function ContextMenu({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Root>) {
 | 
			
		||||
  return <ContextMenuPrimitive.Root data-slot="context-menu" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Trigger data-slot="context-menu-trigger" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Group>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Group data-slot="context-menu-group" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Portal>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Portal data-slot="context-menu-portal" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuSub({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Sub>) {
 | 
			
		||||
  return <ContextMenuPrimitive.Sub data-slot="context-menu-sub" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuRadioGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.RadioGroup>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.RadioGroup
 | 
			
		||||
      data-slot="context-menu-radio-group"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuSubTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.SubTrigger> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.SubTrigger
 | 
			
		||||
      data-slot="context-menu-sub-trigger"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
      <ChevronRightIcon className="ml-auto" />
 | 
			
		||||
    </ContextMenuPrimitive.SubTrigger>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuSubContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.SubContent>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.SubContent
 | 
			
		||||
      data-slot="context-menu-sub-content"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[8rem] origin-(--radix-context-menu-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-lg",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Portal>
 | 
			
		||||
      <ContextMenuPrimitive.Content
 | 
			
		||||
        data-slot="context-menu-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-context-menu-content-available-height) min-w-[8rem] origin-(--radix-context-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </ContextMenuPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuItem({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  variant = "default",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Item> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
  variant?: "default" | "destructive"
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Item
 | 
			
		||||
      data-slot="context-menu-item"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      data-variant={variant}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 dark:data-[variant=destructive]:focus:bg-destructive/20 data-[variant=destructive]:focus:text-destructive data-[variant=destructive]:*:[svg]:!text-destructive [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuCheckboxItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  checked,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.CheckboxItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.CheckboxItem
 | 
			
		||||
      data-slot="context-menu-checkbox-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      checked={checked}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <ContextMenuPrimitive.ItemIndicator>
 | 
			
		||||
          <CheckIcon className="size-4" />
 | 
			
		||||
        </ContextMenuPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </ContextMenuPrimitive.CheckboxItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuRadioItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.RadioItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.RadioItem
 | 
			
		||||
      data-slot="context-menu-radio-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <ContextMenuPrimitive.ItemIndicator>
 | 
			
		||||
          <CircleIcon className="size-2 fill-current" />
 | 
			
		||||
        </ContextMenuPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </ContextMenuPrimitive.RadioItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Label> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Label
 | 
			
		||||
      data-slot="context-menu-label"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-foreground px-2 py-1.5 text-sm font-medium data-[inset]:pl-8",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ContextMenuPrimitive.Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ContextMenuPrimitive.Separator
 | 
			
		||||
      data-slot="context-menu-separator"
 | 
			
		||||
      className={cn("bg-border -mx-1 my-1 h-px", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ContextMenuShortcut({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="context-menu-shortcut"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground ml-auto text-xs tracking-widest",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  ContextMenu,
 | 
			
		||||
  ContextMenuTrigger,
 | 
			
		||||
  ContextMenuContent,
 | 
			
		||||
  ContextMenuItem,
 | 
			
		||||
  ContextMenuCheckboxItem,
 | 
			
		||||
  ContextMenuRadioItem,
 | 
			
		||||
  ContextMenuLabel,
 | 
			
		||||
  ContextMenuSeparator,
 | 
			
		||||
  ContextMenuShortcut,
 | 
			
		||||
  ContextMenuGroup,
 | 
			
		||||
  ContextMenuPortal,
 | 
			
		||||
  ContextMenuSub,
 | 
			
		||||
  ContextMenuSubContent,
 | 
			
		||||
  ContextMenuSubTrigger,
 | 
			
		||||
  ContextMenuRadioGroup,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										135
									
								
								web/src/components/ui/dialog.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								web/src/components/ui/dialog.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,135 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as DialogPrimitive from "@radix-ui/react-dialog"
 | 
			
		||||
import { XIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Dialog({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Root>) {
 | 
			
		||||
  return <DialogPrimitive.Root data-slot="dialog" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Trigger>) {
 | 
			
		||||
  return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Portal>) {
 | 
			
		||||
  return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogClose({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Close>) {
 | 
			
		||||
  return <DialogPrimitive.Close data-slot="dialog-close" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogOverlay({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Overlay>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DialogPrimitive.Overlay
 | 
			
		||||
      data-slot="dialog-overlay"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "backdrop-blur-xs data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogContent({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DialogPortal data-slot="dialog-portal">
 | 
			
		||||
      <DialogOverlay />
 | 
			
		||||
      <DialogPrimitive.Content
 | 
			
		||||
        data-slot="dialog-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
        <DialogPrimitive.Close className="ring-offset-background focus:ring-ring data-[state=open]:bg-accent data-[state=open]:text-muted-foreground absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4">
 | 
			
		||||
          <XIcon />
 | 
			
		||||
          <span className="sr-only">Close</span>
 | 
			
		||||
        </DialogPrimitive.Close>
 | 
			
		||||
      </DialogPrimitive.Content>
 | 
			
		||||
    </DialogPortal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogHeader({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="dialog-header"
 | 
			
		||||
      className={cn("flex flex-col gap-2 text-center sm:text-left", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogFooter({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="dialog-footer"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogTitle({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Title>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DialogPrimitive.Title
 | 
			
		||||
      data-slot="dialog-title"
 | 
			
		||||
      className={cn("text-lg leading-none font-semibold", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DialogDescription({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DialogPrimitive.Description>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DialogPrimitive.Description
 | 
			
		||||
      data-slot="dialog-description"
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Dialog,
 | 
			
		||||
  DialogClose,
 | 
			
		||||
  DialogContent,
 | 
			
		||||
  DialogDescription,
 | 
			
		||||
  DialogFooter,
 | 
			
		||||
  DialogHeader,
 | 
			
		||||
  DialogOverlay,
 | 
			
		||||
  DialogPortal,
 | 
			
		||||
  DialogTitle,
 | 
			
		||||
  DialogTrigger,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										132
									
								
								web/src/components/ui/drawer.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								web/src/components/ui/drawer.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,132 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { Drawer as DrawerPrimitive } from "vaul"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Drawer({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Root>) {
 | 
			
		||||
  return <DrawerPrimitive.Root data-slot="drawer" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Trigger>) {
 | 
			
		||||
  return <DrawerPrimitive.Trigger data-slot="drawer-trigger" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Portal>) {
 | 
			
		||||
  return <DrawerPrimitive.Portal data-slot="drawer-portal" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerClose({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Close>) {
 | 
			
		||||
  return <DrawerPrimitive.Close data-slot="drawer-close" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerOverlay({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Overlay>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DrawerPrimitive.Overlay
 | 
			
		||||
      data-slot="drawer-overlay"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerContent({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DrawerPortal data-slot="drawer-portal">
 | 
			
		||||
      <DrawerOverlay />
 | 
			
		||||
      <DrawerPrimitive.Content
 | 
			
		||||
        data-slot="drawer-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "group/drawer-content bg-background fixed z-50 flex h-auto flex-col",
 | 
			
		||||
          "data-[vaul-drawer-direction=top]:inset-x-0 data-[vaul-drawer-direction=top]:top-0 data-[vaul-drawer-direction=top]:mb-24 data-[vaul-drawer-direction=top]:max-h-[80vh] data-[vaul-drawer-direction=top]:rounded-b-lg data-[vaul-drawer-direction=top]:border-b",
 | 
			
		||||
          "data-[vaul-drawer-direction=bottom]:inset-x-0 data-[vaul-drawer-direction=bottom]:bottom-0 data-[vaul-drawer-direction=bottom]:mt-24 data-[vaul-drawer-direction=bottom]:max-h-[80vh] data-[vaul-drawer-direction=bottom]:rounded-t-lg data-[vaul-drawer-direction=bottom]:border-t",
 | 
			
		||||
          "data-[vaul-drawer-direction=right]:inset-y-0 data-[vaul-drawer-direction=right]:right-0 data-[vaul-drawer-direction=right]:w-3/4 data-[vaul-drawer-direction=right]:border-l data-[vaul-drawer-direction=right]:sm:max-w-sm",
 | 
			
		||||
          "data-[vaul-drawer-direction=left]:inset-y-0 data-[vaul-drawer-direction=left]:left-0 data-[vaul-drawer-direction=left]:w-3/4 data-[vaul-drawer-direction=left]:border-r data-[vaul-drawer-direction=left]:sm:max-w-sm",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        <div className="bg-muted mx-auto mt-4 hidden h-2 w-[100px] shrink-0 rounded-full group-data-[vaul-drawer-direction=bottom]/drawer-content:block" />
 | 
			
		||||
        {children}
 | 
			
		||||
      </DrawerPrimitive.Content>
 | 
			
		||||
    </DrawerPortal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerHeader({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="drawer-header"
 | 
			
		||||
      className={cn("flex flex-col gap-1.5 p-4", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerFooter({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="drawer-footer"
 | 
			
		||||
      className={cn("mt-auto flex flex-col gap-2 p-4", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerTitle({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Title>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DrawerPrimitive.Title
 | 
			
		||||
      data-slot="drawer-title"
 | 
			
		||||
      className={cn("text-foreground font-semibold", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DrawerDescription({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DrawerPrimitive.Description>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DrawerPrimitive.Description
 | 
			
		||||
      data-slot="drawer-description"
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Drawer,
 | 
			
		||||
  DrawerPortal,
 | 
			
		||||
  DrawerOverlay,
 | 
			
		||||
  DrawerTrigger,
 | 
			
		||||
  DrawerClose,
 | 
			
		||||
  DrawerContent,
 | 
			
		||||
  DrawerHeader,
 | 
			
		||||
  DrawerFooter,
 | 
			
		||||
  DrawerTitle,
 | 
			
		||||
  DrawerDescription,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										257
									
								
								web/src/components/ui/dropdown-menu.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										257
									
								
								web/src/components/ui/dropdown-menu.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,257 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as DropdownMenuPrimitive from "@radix-ui/react-dropdown-menu"
 | 
			
		||||
import { CheckIcon, ChevronRightIcon, CircleIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function DropdownMenu({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Root>) {
 | 
			
		||||
  return <DropdownMenuPrimitive.Root data-slot="dropdown-menu" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Portal>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Portal data-slot="dropdown-menu-portal" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Trigger
 | 
			
		||||
      data-slot="dropdown-menu-trigger"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuContent({
 | 
			
		||||
  className,
 | 
			
		||||
  sideOffset = 4,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Portal>
 | 
			
		||||
      <DropdownMenuPrimitive.Content
 | 
			
		||||
        data-slot="dropdown-menu-content"
 | 
			
		||||
        sideOffset={sideOffset}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-dropdown-menu-content-available-height) min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </DropdownMenuPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Group>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Group data-slot="dropdown-menu-group" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuItem({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  variant = "default",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Item> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
  variant?: "default" | "destructive"
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Item
 | 
			
		||||
      data-slot="dropdown-menu-item"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      data-variant={variant}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 dark:data-[variant=destructive]:focus:bg-destructive/20 data-[variant=destructive]:focus:text-destructive data-[variant=destructive]:*:[svg]:!text-destructive [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuCheckboxItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  checked,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.CheckboxItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.CheckboxItem
 | 
			
		||||
      data-slot="dropdown-menu-checkbox-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      checked={checked}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <DropdownMenuPrimitive.ItemIndicator>
 | 
			
		||||
          <CheckIcon className="size-4" />
 | 
			
		||||
        </DropdownMenuPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </DropdownMenuPrimitive.CheckboxItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuRadioGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.RadioGroup>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.RadioGroup
 | 
			
		||||
      data-slot="dropdown-menu-radio-group"
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuRadioItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.RadioItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.RadioItem
 | 
			
		||||
      data-slot="dropdown-menu-radio-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <DropdownMenuPrimitive.ItemIndicator>
 | 
			
		||||
          <CircleIcon className="size-2 fill-current" />
 | 
			
		||||
        </DropdownMenuPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </DropdownMenuPrimitive.RadioItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Label> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Label
 | 
			
		||||
      data-slot="dropdown-menu-label"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "px-2 py-1.5 text-sm font-medium data-[inset]:pl-8",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.Separator
 | 
			
		||||
      data-slot="dropdown-menu-separator"
 | 
			
		||||
      className={cn("bg-border -mx-1 my-1 h-px", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuShortcut({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="dropdown-menu-shortcut"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground ml-auto text-xs tracking-widest",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuSub({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.Sub>) {
 | 
			
		||||
  return <DropdownMenuPrimitive.Sub data-slot="dropdown-menu-sub" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuSubTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.SubTrigger> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.SubTrigger
 | 
			
		||||
      data-slot="dropdown-menu-sub-trigger"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[inset]:pl-8",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
      <ChevronRightIcon className="ml-auto size-4" />
 | 
			
		||||
    </DropdownMenuPrimitive.SubTrigger>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function DropdownMenuSubContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof DropdownMenuPrimitive.SubContent>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <DropdownMenuPrimitive.SubContent
 | 
			
		||||
      data-slot="dropdown-menu-sub-content"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-lg",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  DropdownMenu,
 | 
			
		||||
  DropdownMenuPortal,
 | 
			
		||||
  DropdownMenuTrigger,
 | 
			
		||||
  DropdownMenuContent,
 | 
			
		||||
  DropdownMenuGroup,
 | 
			
		||||
  DropdownMenuLabel,
 | 
			
		||||
  DropdownMenuItem,
 | 
			
		||||
  DropdownMenuCheckboxItem,
 | 
			
		||||
  DropdownMenuRadioGroup,
 | 
			
		||||
  DropdownMenuRadioItem,
 | 
			
		||||
  DropdownMenuSeparator,
 | 
			
		||||
  DropdownMenuShortcut,
 | 
			
		||||
  DropdownMenuSub,
 | 
			
		||||
  DropdownMenuSubTrigger,
 | 
			
		||||
  DropdownMenuSubContent,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										167
									
								
								web/src/components/ui/form.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								web/src/components/ui/form.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,167 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as LabelPrimitive from "@radix-ui/react-label"
 | 
			
		||||
import { Slot } from "@radix-ui/react-slot"
 | 
			
		||||
import {
 | 
			
		||||
  Controller,
 | 
			
		||||
  FormProvider,
 | 
			
		||||
  useFormContext,
 | 
			
		||||
  useFormState,
 | 
			
		||||
  type ControllerProps,
 | 
			
		||||
  type FieldPath,
 | 
			
		||||
  type FieldValues,
 | 
			
		||||
} from "react-hook-form"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { Label } from "@/components/ui/label"
 | 
			
		||||
 | 
			
		||||
const Form = FormProvider
 | 
			
		||||
 | 
			
		||||
type FormFieldContextValue<
 | 
			
		||||
  TFieldValues extends FieldValues = FieldValues,
 | 
			
		||||
  TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
 | 
			
		||||
> = {
 | 
			
		||||
  name: TName
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const FormFieldContext = React.createContext<FormFieldContextValue>(
 | 
			
		||||
  {} as FormFieldContextValue
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const FormField = <
 | 
			
		||||
  TFieldValues extends FieldValues = FieldValues,
 | 
			
		||||
  TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues>,
 | 
			
		||||
>({
 | 
			
		||||
  ...props
 | 
			
		||||
}: ControllerProps<TFieldValues, TName>) => {
 | 
			
		||||
  return (
 | 
			
		||||
    <FormFieldContext.Provider value={{ name: props.name }}>
 | 
			
		||||
      <Controller {...props} />
 | 
			
		||||
    </FormFieldContext.Provider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const useFormField = () => {
 | 
			
		||||
  const fieldContext = React.useContext(FormFieldContext)
 | 
			
		||||
  const itemContext = React.useContext(FormItemContext)
 | 
			
		||||
  const { getFieldState } = useFormContext()
 | 
			
		||||
  const formState = useFormState({ name: fieldContext.name })
 | 
			
		||||
  const fieldState = getFieldState(fieldContext.name, formState)
 | 
			
		||||
 | 
			
		||||
  if (!fieldContext) {
 | 
			
		||||
    throw new Error("useFormField should be used within <FormField>")
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const { id } = itemContext
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    id,
 | 
			
		||||
    name: fieldContext.name,
 | 
			
		||||
    formItemId: `${id}-form-item`,
 | 
			
		||||
    formDescriptionId: `${id}-form-item-description`,
 | 
			
		||||
    formMessageId: `${id}-form-item-message`,
 | 
			
		||||
    ...fieldState,
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type FormItemContextValue = {
 | 
			
		||||
  id: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const FormItemContext = React.createContext<FormItemContextValue>(
 | 
			
		||||
  {} as FormItemContextValue
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function FormItem({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  const id = React.useId()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <FormItemContext.Provider value={{ id }}>
 | 
			
		||||
      <div
 | 
			
		||||
        data-slot="form-item"
 | 
			
		||||
        className={cn("grid gap-2", className)}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </FormItemContext.Provider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function FormLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof LabelPrimitive.Root>) {
 | 
			
		||||
  const { error, formItemId } = useFormField()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Label
 | 
			
		||||
      data-slot="form-label"
 | 
			
		||||
      data-error={!!error}
 | 
			
		||||
      className={cn("data-[error=true]:text-destructive", className)}
 | 
			
		||||
      htmlFor={formItemId}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function FormControl({ ...props }: React.ComponentProps<typeof Slot>) {
 | 
			
		||||
  const { error, formItemId, formDescriptionId, formMessageId } = useFormField()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Slot
 | 
			
		||||
      data-slot="form-control"
 | 
			
		||||
      id={formItemId}
 | 
			
		||||
      aria-describedby={
 | 
			
		||||
        !error
 | 
			
		||||
          ? `${formDescriptionId}`
 | 
			
		||||
          : `${formDescriptionId} ${formMessageId}`
 | 
			
		||||
      }
 | 
			
		||||
      aria-invalid={!!error}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function FormDescription({ className, ...props }: React.ComponentProps<"p">) {
 | 
			
		||||
  const { formDescriptionId } = useFormField()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <p
 | 
			
		||||
      data-slot="form-description"
 | 
			
		||||
      id={formDescriptionId}
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function FormMessage({ className, ...props }: React.ComponentProps<"p">) {
 | 
			
		||||
  const { error, formMessageId } = useFormField()
 | 
			
		||||
  const body = error ? String(error?.message ?? "") : props.children
 | 
			
		||||
 | 
			
		||||
  if (!body) {
 | 
			
		||||
    return null
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <p
 | 
			
		||||
      data-slot="form-message"
 | 
			
		||||
      id={formMessageId}
 | 
			
		||||
      className={cn("text-destructive text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {body}
 | 
			
		||||
    </p>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  useFormField,
 | 
			
		||||
  Form,
 | 
			
		||||
  FormItem,
 | 
			
		||||
  FormLabel,
 | 
			
		||||
  FormControl,
 | 
			
		||||
  FormDescription,
 | 
			
		||||
  FormMessage,
 | 
			
		||||
  FormField,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										44
									
								
								web/src/components/ui/hover-card.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								web/src/components/ui/hover-card.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,44 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as HoverCardPrimitive from "@radix-ui/react-hover-card"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function HoverCard({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof HoverCardPrimitive.Root>) {
 | 
			
		||||
  return <HoverCardPrimitive.Root data-slot="hover-card" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function HoverCardTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof HoverCardPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <HoverCardPrimitive.Trigger data-slot="hover-card-trigger" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function HoverCardContent({
 | 
			
		||||
  className,
 | 
			
		||||
  align = "center",
 | 
			
		||||
  sideOffset = 4,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof HoverCardPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <HoverCardPrimitive.Portal data-slot="hover-card-portal">
 | 
			
		||||
      <HoverCardPrimitive.Content
 | 
			
		||||
        data-slot="hover-card-content"
 | 
			
		||||
        align={align}
 | 
			
		||||
        sideOffset={sideOffset}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-64 origin-(--radix-hover-card-content-transform-origin) rounded-md border p-4 shadow-md outline-hidden",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </HoverCardPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { HoverCard, HoverCardTrigger, HoverCardContent }
 | 
			
		||||
							
								
								
									
										77
									
								
								web/src/components/ui/input-otp.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								web/src/components/ui/input-otp.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,77 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { OTPInput, OTPInputContext } from "input-otp"
 | 
			
		||||
import { MinusIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function InputOTP({
 | 
			
		||||
  className,
 | 
			
		||||
  containerClassName,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof OTPInput> & {
 | 
			
		||||
  containerClassName?: string
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <OTPInput
 | 
			
		||||
      data-slot="input-otp"
 | 
			
		||||
      containerClassName={cn(
 | 
			
		||||
        "flex items-center gap-2 has-disabled:opacity-50",
 | 
			
		||||
        containerClassName
 | 
			
		||||
      )}
 | 
			
		||||
      className={cn("disabled:cursor-not-allowed", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function InputOTPGroup({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="input-otp-group"
 | 
			
		||||
      className={cn("flex items-center", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function InputOTPSlot({
 | 
			
		||||
  index,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & {
 | 
			
		||||
  index: number
 | 
			
		||||
}) {
 | 
			
		||||
  const inputOTPContext = React.useContext(OTPInputContext)
 | 
			
		||||
  const { char, hasFakeCaret, isActive } = inputOTPContext?.slots[index] ?? {}
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="input-otp-slot"
 | 
			
		||||
      data-active={isActive}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[active=true]:border-ring data-[active=true]:ring-ring/50 data-[active=true]:aria-invalid:ring-destructive/20 dark:data-[active=true]:aria-invalid:ring-destructive/40 aria-invalid:border-destructive data-[active=true]:aria-invalid:border-destructive dark:bg-input/30 border-input relative flex h-9 w-9 items-center justify-center border-y border-r text-sm shadow-xs transition-all outline-none first:rounded-l-md first:border-l last:rounded-r-md data-[active=true]:z-10 data-[active=true]:ring-[3px]",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {char}
 | 
			
		||||
      {hasFakeCaret && (
 | 
			
		||||
        <div className="pointer-events-none absolute inset-0 flex items-center justify-center">
 | 
			
		||||
          <div className="animate-caret-blink bg-foreground h-4 w-px duration-1000" />
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function InputOTPSeparator({ ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div data-slot="input-otp-separator" role="separator" {...props}>
 | 
			
		||||
      <MinusIcon />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { InputOTP, InputOTPGroup, InputOTPSlot, InputOTPSeparator }
 | 
			
		||||
							
								
								
									
										21
									
								
								web/src/components/ui/input.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								web/src/components/ui/input.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Input({ className, type, ...props }: React.ComponentProps<"input">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <input
 | 
			
		||||
      type={type}
 | 
			
		||||
      data-slot="input"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 border-input flex h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
 | 
			
		||||
        "focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]",
 | 
			
		||||
        "aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Input }
 | 
			
		||||
							
								
								
									
										24
									
								
								web/src/components/ui/label.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								web/src/components/ui/label.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as LabelPrimitive from "@radix-ui/react-label"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Label({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof LabelPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <LabelPrimitive.Root
 | 
			
		||||
      data-slot="label"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex items-center gap-2 text-sm leading-none font-medium select-none group-data-[disabled=true]:pointer-events-none group-data-[disabled=true]:opacity-50 peer-disabled:cursor-not-allowed peer-disabled:opacity-50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Label }
 | 
			
		||||
							
								
								
									
										276
									
								
								web/src/components/ui/menubar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										276
									
								
								web/src/components/ui/menubar.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,276 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as MenubarPrimitive from "@radix-ui/react-menubar"
 | 
			
		||||
import { CheckIcon, ChevronRightIcon, CircleIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Menubar({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.Root
 | 
			
		||||
      data-slot="menubar"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-background flex h-9 items-center gap-1 rounded-md border p-1 shadow-xs",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarMenu({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Menu>) {
 | 
			
		||||
  return <MenubarPrimitive.Menu data-slot="menubar-menu" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Group>) {
 | 
			
		||||
  return <MenubarPrimitive.Group data-slot="menubar-group" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Portal>) {
 | 
			
		||||
  return <MenubarPrimitive.Portal data-slot="menubar-portal" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarRadioGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.RadioGroup>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.RadioGroup data-slot="menubar-radio-group" {...props} />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.Trigger
 | 
			
		||||
      data-slot="menubar-trigger"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex items-center rounded-sm px-2 py-1 text-sm font-medium outline-hidden select-none",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarContent({
 | 
			
		||||
  className,
 | 
			
		||||
  align = "start",
 | 
			
		||||
  alignOffset = -4,
 | 
			
		||||
  sideOffset = 8,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPortal>
 | 
			
		||||
      <MenubarPrimitive.Content
 | 
			
		||||
        data-slot="menubar-content"
 | 
			
		||||
        align={align}
 | 
			
		||||
        alignOffset={alignOffset}
 | 
			
		||||
        sideOffset={sideOffset}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[12rem] origin-(--radix-menubar-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-md",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </MenubarPortal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarItem({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  variant = "default",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Item> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
  variant?: "default" | "destructive"
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.Item
 | 
			
		||||
      data-slot="menubar-item"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      data-variant={variant}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 dark:data-[variant=destructive]:focus:bg-destructive/20 data-[variant=destructive]:focus:text-destructive data-[variant=destructive]:*:[svg]:!text-destructive [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarCheckboxItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  checked,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.CheckboxItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.CheckboxItem
 | 
			
		||||
      data-slot="menubar-checkbox-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-xs py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      checked={checked}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <MenubarPrimitive.ItemIndicator>
 | 
			
		||||
          <CheckIcon className="size-4" />
 | 
			
		||||
        </MenubarPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </MenubarPrimitive.CheckboxItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarRadioItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.RadioItem>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.RadioItem
 | 
			
		||||
      data-slot="menubar-radio-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-xs py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <MenubarPrimitive.ItemIndicator>
 | 
			
		||||
          <CircleIcon className="size-2 fill-current" />
 | 
			
		||||
        </MenubarPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      {children}
 | 
			
		||||
    </MenubarPrimitive.RadioItem>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Label> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.Label
 | 
			
		||||
      data-slot="menubar-label"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "px-2 py-1.5 text-sm font-medium data-[inset]:pl-8",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.Separator
 | 
			
		||||
      data-slot="menubar-separator"
 | 
			
		||||
      className={cn("bg-border -mx-1 my-1 h-px", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarShortcut({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      data-slot="menubar-shortcut"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-muted-foreground ml-auto text-xs tracking-widest",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarSub({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.Sub>) {
 | 
			
		||||
  return <MenubarPrimitive.Sub data-slot="menubar-sub" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarSubTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  inset,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.SubTrigger> & {
 | 
			
		||||
  inset?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.SubTrigger
 | 
			
		||||
      data-slot="menubar-sub-trigger"
 | 
			
		||||
      data-inset={inset}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-none select-none data-[inset]:pl-8",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
      <ChevronRightIcon className="ml-auto h-4 w-4" />
 | 
			
		||||
    </MenubarPrimitive.SubTrigger>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function MenubarSubContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof MenubarPrimitive.SubContent>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <MenubarPrimitive.SubContent
 | 
			
		||||
      data-slot="menubar-sub-content"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[8rem] origin-(--radix-menubar-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-lg",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Menubar,
 | 
			
		||||
  MenubarPortal,
 | 
			
		||||
  MenubarMenu,
 | 
			
		||||
  MenubarTrigger,
 | 
			
		||||
  MenubarContent,
 | 
			
		||||
  MenubarGroup,
 | 
			
		||||
  MenubarSeparator,
 | 
			
		||||
  MenubarLabel,
 | 
			
		||||
  MenubarItem,
 | 
			
		||||
  MenubarShortcut,
 | 
			
		||||
  MenubarCheckboxItem,
 | 
			
		||||
  MenubarRadioGroup,
 | 
			
		||||
  MenubarRadioItem,
 | 
			
		||||
  MenubarSub,
 | 
			
		||||
  MenubarSubTrigger,
 | 
			
		||||
  MenubarSubContent,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										168
									
								
								web/src/components/ui/navigation-menu.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								web/src/components/ui/navigation-menu.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,168 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as NavigationMenuPrimitive from "@radix-ui/react-navigation-menu"
 | 
			
		||||
import { cva } from "class-variance-authority"
 | 
			
		||||
import { ChevronDownIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function NavigationMenu({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  viewport = true,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Root> & {
 | 
			
		||||
  viewport?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Root
 | 
			
		||||
      data-slot="navigation-menu"
 | 
			
		||||
      data-viewport={viewport}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "group/navigation-menu relative flex max-w-max flex-1 items-center justify-center",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
      {viewport && <NavigationMenuViewport />}
 | 
			
		||||
    </NavigationMenuPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuList({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.List>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.List
 | 
			
		||||
      data-slot="navigation-menu-list"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "group flex flex-1 list-none items-center justify-center gap-1",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuItem({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Item>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Item
 | 
			
		||||
      data-slot="navigation-menu-item"
 | 
			
		||||
      className={cn("relative", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const navigationMenuTriggerStyle = cva(
 | 
			
		||||
  "group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground disabled:pointer-events-none disabled:opacity-50 data-[state=open]:hover:bg-accent data-[state=open]:text-accent-foreground data-[state=open]:focus:bg-accent data-[state=open]:bg-accent/50 focus-visible:ring-ring/50 outline-none transition-[color,box-shadow] focus-visible:ring-[3px] focus-visible:outline-1"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function NavigationMenuTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Trigger
 | 
			
		||||
      data-slot="navigation-menu-trigger"
 | 
			
		||||
      className={cn(navigationMenuTriggerStyle(), "group", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}{" "}
 | 
			
		||||
      <ChevronDownIcon
 | 
			
		||||
        className="relative top-[1px] ml-1 size-3 transition duration-300 group-data-[state=open]:rotate-180"
 | 
			
		||||
        aria-hidden="true"
 | 
			
		||||
      />
 | 
			
		||||
    </NavigationMenuPrimitive.Trigger>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Content
 | 
			
		||||
      data-slot="navigation-menu-content"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[motion^=from-]:animate-in data-[motion^=to-]:animate-out data-[motion^=from-]:fade-in data-[motion^=to-]:fade-out data-[motion=from-end]:slide-in-from-right-52 data-[motion=from-start]:slide-in-from-left-52 data-[motion=to-end]:slide-out-to-right-52 data-[motion=to-start]:slide-out-to-left-52 top-0 left-0 w-full p-2 pr-2.5 md:absolute md:w-auto",
 | 
			
		||||
        "group-data-[viewport=false]/navigation-menu:bg-popover group-data-[viewport=false]/navigation-menu:text-popover-foreground group-data-[viewport=false]/navigation-menu:data-[state=open]:animate-in group-data-[viewport=false]/navigation-menu:data-[state=closed]:animate-out group-data-[viewport=false]/navigation-menu:data-[state=closed]:zoom-out-95 group-data-[viewport=false]/navigation-menu:data-[state=open]:zoom-in-95 group-data-[viewport=false]/navigation-menu:data-[state=open]:fade-in-0 group-data-[viewport=false]/navigation-menu:data-[state=closed]:fade-out-0 group-data-[viewport=false]/navigation-menu:top-full group-data-[viewport=false]/navigation-menu:mt-1.5 group-data-[viewport=false]/navigation-menu:overflow-hidden group-data-[viewport=false]/navigation-menu:rounded-md group-data-[viewport=false]/navigation-menu:border group-data-[viewport=false]/navigation-menu:shadow group-data-[viewport=false]/navigation-menu:duration-200 **:data-[slot=navigation-menu-link]:focus:ring-0 **:data-[slot=navigation-menu-link]:focus:outline-none",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuViewport({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Viewport>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "absolute top-full left-0 isolate z-50 flex justify-center"
 | 
			
		||||
      )}
 | 
			
		||||
    >
 | 
			
		||||
      <NavigationMenuPrimitive.Viewport
 | 
			
		||||
        data-slot="navigation-menu-viewport"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "origin-top-center bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-90 relative mt-1.5 h-[var(--radix-navigation-menu-viewport-height)] w-full overflow-hidden rounded-md border shadow md:w-[var(--radix-navigation-menu-viewport-width)]",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuLink({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Link>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Link
 | 
			
		||||
      data-slot="navigation-menu-link"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[active=true]:focus:bg-accent data-[active=true]:hover:bg-accent data-[active=true]:bg-accent/50 data-[active=true]:text-accent-foreground hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground focus-visible:ring-ring/50 [&_svg:not([class*='text-'])]:text-muted-foreground flex flex-col gap-1 rounded-sm p-2 text-sm transition-all outline-none focus-visible:ring-[3px] focus-visible:outline-1 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function NavigationMenuIndicator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof NavigationMenuPrimitive.Indicator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <NavigationMenuPrimitive.Indicator
 | 
			
		||||
      data-slot="navigation-menu-indicator"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[state=visible]:animate-in data-[state=hidden]:animate-out data-[state=hidden]:fade-out data-[state=visible]:fade-in top-full z-[1] flex h-1.5 items-end justify-center overflow-hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <div className="bg-border relative top-[60%] h-2 w-2 rotate-45 rounded-tl-sm shadow-md" />
 | 
			
		||||
    </NavigationMenuPrimitive.Indicator>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  NavigationMenu,
 | 
			
		||||
  NavigationMenuList,
 | 
			
		||||
  NavigationMenuItem,
 | 
			
		||||
  NavigationMenuContent,
 | 
			
		||||
  NavigationMenuTrigger,
 | 
			
		||||
  NavigationMenuLink,
 | 
			
		||||
  NavigationMenuIndicator,
 | 
			
		||||
  NavigationMenuViewport,
 | 
			
		||||
  navigationMenuTriggerStyle,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										127
									
								
								web/src/components/ui/pagination.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								web/src/components/ui/pagination.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,127 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import {
 | 
			
		||||
  ChevronLeftIcon,
 | 
			
		||||
  ChevronRightIcon,
 | 
			
		||||
  MoreHorizontalIcon,
 | 
			
		||||
} from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { Button, buttonVariants } from "@/components/ui/button"
 | 
			
		||||
 | 
			
		||||
function Pagination({ className, ...props }: React.ComponentProps<"nav">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <nav
 | 
			
		||||
      role="navigation"
 | 
			
		||||
      aria-label="pagination"
 | 
			
		||||
      data-slot="pagination"
 | 
			
		||||
      className={cn("mx-auto flex w-full justify-center", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PaginationContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"ul">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ul
 | 
			
		||||
      data-slot="pagination-content"
 | 
			
		||||
      className={cn("flex flex-row items-center gap-1", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PaginationItem({ ...props }: React.ComponentProps<"li">) {
 | 
			
		||||
  return <li data-slot="pagination-item" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type PaginationLinkProps = {
 | 
			
		||||
  isActive?: boolean
 | 
			
		||||
} & Pick<React.ComponentProps<typeof Button>, "size"> &
 | 
			
		||||
  React.ComponentProps<"a">
 | 
			
		||||
 | 
			
		||||
function PaginationLink({
 | 
			
		||||
  className,
 | 
			
		||||
  isActive,
 | 
			
		||||
  size = "icon",
 | 
			
		||||
  ...props
 | 
			
		||||
}: PaginationLinkProps) {
 | 
			
		||||
  return (
 | 
			
		||||
    <a
 | 
			
		||||
      aria-current={isActive ? "page" : undefined}
 | 
			
		||||
      data-slot="pagination-link"
 | 
			
		||||
      data-active={isActive}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        buttonVariants({
 | 
			
		||||
          variant: isActive ? "outline" : "ghost",
 | 
			
		||||
          size,
 | 
			
		||||
        }),
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PaginationPrevious({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PaginationLink>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <PaginationLink
 | 
			
		||||
      aria-label="Go to previous page"
 | 
			
		||||
      size="default"
 | 
			
		||||
      className={cn("gap-1 px-2.5 sm:pl-2.5", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ChevronLeftIcon />
 | 
			
		||||
      <span className="hidden sm:block">Previous</span>
 | 
			
		||||
    </PaginationLink>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PaginationNext({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PaginationLink>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <PaginationLink
 | 
			
		||||
      aria-label="Go to next page"
 | 
			
		||||
      size="default"
 | 
			
		||||
      className={cn("gap-1 px-2.5 sm:pr-2.5", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="hidden sm:block">Next</span>
 | 
			
		||||
      <ChevronRightIcon />
 | 
			
		||||
    </PaginationLink>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PaginationEllipsis({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"span">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <span
 | 
			
		||||
      aria-hidden
 | 
			
		||||
      data-slot="pagination-ellipsis"
 | 
			
		||||
      className={cn("flex size-9 items-center justify-center", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <MoreHorizontalIcon className="size-4" />
 | 
			
		||||
      <span className="sr-only">More pages</span>
 | 
			
		||||
    </span>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Pagination,
 | 
			
		||||
  PaginationContent,
 | 
			
		||||
  PaginationLink,
 | 
			
		||||
  PaginationItem,
 | 
			
		||||
  PaginationPrevious,
 | 
			
		||||
  PaginationNext,
 | 
			
		||||
  PaginationEllipsis,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										48
									
								
								web/src/components/ui/popover.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								web/src/components/ui/popover.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,48 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as PopoverPrimitive from "@radix-ui/react-popover"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Popover({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PopoverPrimitive.Root>) {
 | 
			
		||||
  return <PopoverPrimitive.Root data-slot="popover" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PopoverTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PopoverPrimitive.Trigger>) {
 | 
			
		||||
  return <PopoverPrimitive.Trigger data-slot="popover-trigger" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PopoverContent({
 | 
			
		||||
  className,
 | 
			
		||||
  align = "center",
 | 
			
		||||
  sideOffset = 4,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PopoverPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <PopoverPrimitive.Portal>
 | 
			
		||||
      <PopoverPrimitive.Content
 | 
			
		||||
        data-slot="popover-content"
 | 
			
		||||
        align={align}
 | 
			
		||||
        sideOffset={sideOffset}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-72 origin-(--radix-popover-content-transform-origin) rounded-md border p-4 shadow-md outline-hidden",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </PopoverPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function PopoverAnchor({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof PopoverPrimitive.Anchor>) {
 | 
			
		||||
  return <PopoverPrimitive.Anchor data-slot="popover-anchor" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Popover, PopoverTrigger, PopoverContent, PopoverAnchor }
 | 
			
		||||
							
								
								
									
										31
									
								
								web/src/components/ui/progress.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								web/src/components/ui/progress.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as ProgressPrimitive from "@radix-ui/react-progress"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Progress({
 | 
			
		||||
  className,
 | 
			
		||||
  value,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ProgressPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ProgressPrimitive.Root
 | 
			
		||||
      data-slot="progress"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-primary/20 relative h-2 w-full overflow-hidden rounded-full",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ProgressPrimitive.Indicator
 | 
			
		||||
        data-slot="progress-indicator"
 | 
			
		||||
        className="bg-primary h-full w-full flex-1 transition-all"
 | 
			
		||||
        style={{ transform: `translateX(-${100 - (value || 0)}%)` }}
 | 
			
		||||
      />
 | 
			
		||||
    </ProgressPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Progress }
 | 
			
		||||
							
								
								
									
										45
									
								
								web/src/components/ui/radio-group.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								web/src/components/ui/radio-group.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,45 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as RadioGroupPrimitive from "@radix-ui/react-radio-group"
 | 
			
		||||
import { CircleIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function RadioGroup({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof RadioGroupPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <RadioGroupPrimitive.Root
 | 
			
		||||
      data-slot="radio-group"
 | 
			
		||||
      className={cn("grid gap-3", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function RadioGroupItem({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof RadioGroupPrimitive.Item>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <RadioGroupPrimitive.Item
 | 
			
		||||
      data-slot="radio-group-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "border-input text-primary focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive dark:bg-input/30 aspect-square size-4 shrink-0 rounded-full border shadow-xs transition-[color,box-shadow] outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <RadioGroupPrimitive.Indicator
 | 
			
		||||
        data-slot="radio-group-indicator"
 | 
			
		||||
        className="relative flex items-center justify-center"
 | 
			
		||||
      >
 | 
			
		||||
        <CircleIcon className="fill-primary absolute top-1/2 left-1/2 size-2 -translate-x-1/2 -translate-y-1/2" />
 | 
			
		||||
      </RadioGroupPrimitive.Indicator>
 | 
			
		||||
    </RadioGroupPrimitive.Item>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { RadioGroup, RadioGroupItem }
 | 
			
		||||
							
								
								
									
										56
									
								
								web/src/components/ui/resizable.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								web/src/components/ui/resizable.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,56 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import { GripVerticalIcon } from "lucide-react"
 | 
			
		||||
import * as ResizablePrimitive from "react-resizable-panels"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function ResizablePanelGroup({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ResizablePrimitive.PanelGroup>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ResizablePrimitive.PanelGroup
 | 
			
		||||
      data-slot="resizable-panel-group"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex h-full w-full data-[panel-group-direction=vertical]:flex-col",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ResizablePanel({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ResizablePrimitive.Panel>) {
 | 
			
		||||
  return <ResizablePrimitive.Panel data-slot="resizable-panel" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ResizableHandle({
 | 
			
		||||
  withHandle,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ResizablePrimitive.PanelResizeHandle> & {
 | 
			
		||||
  withHandle?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ResizablePrimitive.PanelResizeHandle
 | 
			
		||||
      data-slot="resizable-handle"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-border focus-visible:ring-ring relative flex w-px items-center justify-center after:absolute after:inset-y-0 after:left-1/2 after:w-1 after:-translate-x-1/2 focus-visible:ring-1 focus-visible:ring-offset-1 focus-visible:outline-hidden data-[panel-group-direction=vertical]:h-px data-[panel-group-direction=vertical]:w-full data-[panel-group-direction=vertical]:after:left-0 data-[panel-group-direction=vertical]:after:h-1 data-[panel-group-direction=vertical]:after:w-full data-[panel-group-direction=vertical]:after:-translate-y-1/2 data-[panel-group-direction=vertical]:after:translate-x-0 [&[data-panel-group-direction=vertical]>div]:rotate-90",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {withHandle && (
 | 
			
		||||
        <div className="bg-border z-10 flex h-4 w-3 items-center justify-center rounded-xs border">
 | 
			
		||||
          <GripVerticalIcon className="size-2.5" />
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
    </ResizablePrimitive.PanelResizeHandle>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { ResizablePanelGroup, ResizablePanel, ResizableHandle }
 | 
			
		||||
							
								
								
									
										58
									
								
								web/src/components/ui/scroll-area.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								web/src/components/ui/scroll-area.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,58 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as ScrollAreaPrimitive from "@radix-ui/react-scroll-area"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function ScrollArea({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ScrollAreaPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ScrollAreaPrimitive.Root
 | 
			
		||||
      data-slot="scroll-area"
 | 
			
		||||
      className={cn("relative", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ScrollAreaPrimitive.Viewport
 | 
			
		||||
        data-slot="scroll-area-viewport"
 | 
			
		||||
        className="focus-visible:ring-ring/50 size-full rounded-[inherit] transition-[color,box-shadow] outline-none focus-visible:ring-[3px] focus-visible:outline-1"
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
      </ScrollAreaPrimitive.Viewport>
 | 
			
		||||
      <ScrollBar />
 | 
			
		||||
      <ScrollAreaPrimitive.Corner />
 | 
			
		||||
    </ScrollAreaPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ScrollBar({
 | 
			
		||||
  className,
 | 
			
		||||
  orientation = "vertical",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ScrollAreaPrimitive.ScrollAreaScrollbar>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ScrollAreaPrimitive.ScrollAreaScrollbar
 | 
			
		||||
      data-slot="scroll-area-scrollbar"
 | 
			
		||||
      orientation={orientation}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex touch-none p-px transition-colors select-none",
 | 
			
		||||
        orientation === "vertical" &&
 | 
			
		||||
          "h-full w-2.5 border-l border-l-transparent",
 | 
			
		||||
        orientation === "horizontal" &&
 | 
			
		||||
          "h-2.5 flex-col border-t border-t-transparent",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ScrollAreaPrimitive.ScrollAreaThumb
 | 
			
		||||
        data-slot="scroll-area-thumb"
 | 
			
		||||
        className="bg-border relative flex-1 rounded-full"
 | 
			
		||||
      />
 | 
			
		||||
    </ScrollAreaPrimitive.ScrollAreaScrollbar>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { ScrollArea, ScrollBar }
 | 
			
		||||
							
								
								
									
										185
									
								
								web/src/components/ui/select.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										185
									
								
								web/src/components/ui/select.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,185 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as SelectPrimitive from "@radix-ui/react-select"
 | 
			
		||||
import { CheckIcon, ChevronDownIcon, ChevronUpIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Select({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Root>) {
 | 
			
		||||
  return <SelectPrimitive.Root data-slot="select" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectGroup({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Group>) {
 | 
			
		||||
  return <SelectPrimitive.Group data-slot="select-group" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectValue({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Value>) {
 | 
			
		||||
  return <SelectPrimitive.Value data-slot="select-value" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  size = "default",
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Trigger> & {
 | 
			
		||||
  size?: "sm" | "default"
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.Trigger
 | 
			
		||||
      data-slot="select-trigger"
 | 
			
		||||
      data-size={size}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "border-input data-[placeholder]:text-muted-foreground [&_svg:not([class*='text-'])]:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive dark:bg-input/30 dark:hover:bg-input/50 flex w-fit items-center justify-between gap-2 rounded-md border bg-transparent px-3 py-2 text-sm whitespace-nowrap shadow-xs transition-[color,box-shadow] outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50 data-[size=default]:h-9 data-[size=sm]:h-8 *:data-[slot=select-value]:line-clamp-1 *:data-[slot=select-value]:flex *:data-[slot=select-value]:items-center *:data-[slot=select-value]:gap-2 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
      <SelectPrimitive.Icon asChild>
 | 
			
		||||
        <ChevronDownIcon className="size-4 opacity-50" />
 | 
			
		||||
      </SelectPrimitive.Icon>
 | 
			
		||||
    </SelectPrimitive.Trigger>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectContent({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  position = "popper",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.Portal>
 | 
			
		||||
      <SelectPrimitive.Content
 | 
			
		||||
        data-slot="select-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 relative z-50 max-h-(--radix-select-content-available-height) min-w-[8rem] origin-(--radix-select-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border shadow-md",
 | 
			
		||||
          position === "popper" &&
 | 
			
		||||
            "data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        position={position}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        <SelectScrollUpButton />
 | 
			
		||||
        <SelectPrimitive.Viewport
 | 
			
		||||
          className={cn(
 | 
			
		||||
            "p-1",
 | 
			
		||||
            position === "popper" &&
 | 
			
		||||
              "h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)] scroll-my-1"
 | 
			
		||||
          )}
 | 
			
		||||
        >
 | 
			
		||||
          {children}
 | 
			
		||||
        </SelectPrimitive.Viewport>
 | 
			
		||||
        <SelectScrollDownButton />
 | 
			
		||||
      </SelectPrimitive.Content>
 | 
			
		||||
    </SelectPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Label>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.Label
 | 
			
		||||
      data-slot="select-label"
 | 
			
		||||
      className={cn("text-muted-foreground px-2 py-1.5 text-xs", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Item>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.Item
 | 
			
		||||
      data-slot="select-item"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "focus:bg-accent focus:text-accent-foreground [&_svg:not([class*='text-'])]:text-muted-foreground relative flex w-full cursor-default items-center gap-2 rounded-sm py-1.5 pr-8 pl-2 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 *:[span]:last:flex *:[span]:last:items-center *:[span]:last:gap-2",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <span className="absolute right-2 flex size-3.5 items-center justify-center">
 | 
			
		||||
        <SelectPrimitive.ItemIndicator>
 | 
			
		||||
          <CheckIcon className="size-4" />
 | 
			
		||||
        </SelectPrimitive.ItemIndicator>
 | 
			
		||||
      </span>
 | 
			
		||||
      <SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
 | 
			
		||||
    </SelectPrimitive.Item>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.Separator
 | 
			
		||||
      data-slot="select-separator"
 | 
			
		||||
      className={cn("bg-border pointer-events-none -mx-1 my-1 h-px", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectScrollUpButton({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.ScrollUpButton>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.ScrollUpButton
 | 
			
		||||
      data-slot="select-scroll-up-button"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex cursor-default items-center justify-center py-1",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ChevronUpIcon className="size-4" />
 | 
			
		||||
    </SelectPrimitive.ScrollUpButton>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SelectScrollDownButton({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SelectPrimitive.ScrollDownButton>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SelectPrimitive.ScrollDownButton
 | 
			
		||||
      data-slot="select-scroll-down-button"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex cursor-default items-center justify-center py-1",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ChevronDownIcon className="size-4" />
 | 
			
		||||
    </SelectPrimitive.ScrollDownButton>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Select,
 | 
			
		||||
  SelectContent,
 | 
			
		||||
  SelectGroup,
 | 
			
		||||
  SelectItem,
 | 
			
		||||
  SelectLabel,
 | 
			
		||||
  SelectScrollDownButton,
 | 
			
		||||
  SelectScrollUpButton,
 | 
			
		||||
  SelectSeparator,
 | 
			
		||||
  SelectTrigger,
 | 
			
		||||
  SelectValue,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										28
									
								
								web/src/components/ui/separator.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								web/src/components/ui/separator.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as SeparatorPrimitive from "@radix-ui/react-separator"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Separator({
 | 
			
		||||
  className,
 | 
			
		||||
  orientation = "horizontal",
 | 
			
		||||
  decorative = true,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SeparatorPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SeparatorPrimitive.Root
 | 
			
		||||
      data-slot="separator-root"
 | 
			
		||||
      decorative={decorative}
 | 
			
		||||
      orientation={orientation}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-border shrink-0 data-[orientation=horizontal]:h-px data-[orientation=horizontal]:w-full data-[orientation=vertical]:h-full data-[orientation=vertical]:w-px",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Separator }
 | 
			
		||||
							
								
								
									
										139
									
								
								web/src/components/ui/sheet.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								web/src/components/ui/sheet.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,139 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as SheetPrimitive from "@radix-ui/react-dialog"
 | 
			
		||||
import { XIcon } from "lucide-react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Sheet({ ...props }: React.ComponentProps<typeof SheetPrimitive.Root>) {
 | 
			
		||||
  return <SheetPrimitive.Root data-slot="sheet" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Trigger>) {
 | 
			
		||||
  return <SheetPrimitive.Trigger data-slot="sheet-trigger" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetClose({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Close>) {
 | 
			
		||||
  return <SheetPrimitive.Close data-slot="sheet-close" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetPortal({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Portal>) {
 | 
			
		||||
  return <SheetPrimitive.Portal data-slot="sheet-portal" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetOverlay({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Overlay>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SheetPrimitive.Overlay
 | 
			
		||||
      data-slot="sheet-overlay"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetContent({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  side = "right",
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Content> & {
 | 
			
		||||
  side?: "top" | "right" | "bottom" | "left"
 | 
			
		||||
}) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SheetPortal>
 | 
			
		||||
      <SheetOverlay />
 | 
			
		||||
      <SheetPrimitive.Content
 | 
			
		||||
        data-slot="sheet-content"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out fixed z-50 flex flex-col gap-4 shadow-lg transition ease-in-out data-[state=closed]:duration-300 data-[state=open]:duration-500",
 | 
			
		||||
          side === "right" &&
 | 
			
		||||
            "data-[state=closed]:slide-out-to-right data-[state=open]:slide-in-from-right inset-y-0 right-0 h-full w-3/4 border-l sm:max-w-sm",
 | 
			
		||||
          side === "left" &&
 | 
			
		||||
            "data-[state=closed]:slide-out-to-left data-[state=open]:slide-in-from-left inset-y-0 left-0 h-full w-3/4 border-r sm:max-w-sm",
 | 
			
		||||
          side === "top" &&
 | 
			
		||||
            "data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top inset-x-0 top-0 h-auto border-b",
 | 
			
		||||
          side === "bottom" &&
 | 
			
		||||
            "data-[state=closed]:slide-out-to-bottom data-[state=open]:slide-in-from-bottom inset-x-0 bottom-0 h-auto border-t",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
        <SheetPrimitive.Close className="ring-offset-background focus:ring-ring data-[state=open]:bg-secondary absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none">
 | 
			
		||||
          <XIcon className="size-4" />
 | 
			
		||||
          <span className="sr-only">Close</span>
 | 
			
		||||
        </SheetPrimitive.Close>
 | 
			
		||||
      </SheetPrimitive.Content>
 | 
			
		||||
    </SheetPortal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetHeader({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sheet-header"
 | 
			
		||||
      className={cn("flex flex-col gap-1.5 p-4", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetFooter({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sheet-footer"
 | 
			
		||||
      className={cn("mt-auto flex flex-col gap-2 p-4", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetTitle({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Title>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SheetPrimitive.Title
 | 
			
		||||
      data-slot="sheet-title"
 | 
			
		||||
      className={cn("text-foreground font-semibold", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SheetDescription({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SheetPrimitive.Description>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SheetPrimitive.Description
 | 
			
		||||
      data-slot="sheet-description"
 | 
			
		||||
      className={cn("text-muted-foreground text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Sheet,
 | 
			
		||||
  SheetTrigger,
 | 
			
		||||
  SheetClose,
 | 
			
		||||
  SheetContent,
 | 
			
		||||
  SheetHeader,
 | 
			
		||||
  SheetFooter,
 | 
			
		||||
  SheetTitle,
 | 
			
		||||
  SheetDescription,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										711
									
								
								web/src/components/ui/sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										711
									
								
								web/src/components/ui/sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,711 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { Slot } from "@radix-ui/react-slot"
 | 
			
		||||
import { VariantProps, cva } from "class-variance-authority"
 | 
			
		||||
import { PanelLeftIcon } from "lucide-react"
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { Button } from "@/components/ui/button"
 | 
			
		||||
import { Input } from "@/components/ui/input"
 | 
			
		||||
import { Separator } from "@/components/ui/separator"
 | 
			
		||||
import {
 | 
			
		||||
	Sheet,
 | 
			
		||||
	SheetContent,
 | 
			
		||||
	SheetDescription,
 | 
			
		||||
	SheetHeader,
 | 
			
		||||
	SheetTitle,
 | 
			
		||||
} from "@/components/ui/sheet"
 | 
			
		||||
import { Skeleton } from "@/components/ui/skeleton"
 | 
			
		||||
import {
 | 
			
		||||
	Tooltip,
 | 
			
		||||
	TooltipContent,
 | 
			
		||||
	TooltipProvider,
 | 
			
		||||
	TooltipTrigger,
 | 
			
		||||
} from "@/components/ui/tooltip"
 | 
			
		||||
import { useIsMobile } from "@/hooks/use-mobile"
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
const SIDEBAR_COOKIE_NAME = "sidebar_state"
 | 
			
		||||
const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7
 | 
			
		||||
const SIDEBAR_WIDTH = "16rem"
 | 
			
		||||
const SIDEBAR_WIDTH_MOBILE = "18rem"
 | 
			
		||||
const SIDEBAR_WIDTH_ICON = "3rem"
 | 
			
		||||
const SIDEBAR_KEYBOARD_SHORTCUT = "b"
 | 
			
		||||
 | 
			
		||||
type SidebarContextProps = {
 | 
			
		||||
  state: "expanded" | "collapsed"
 | 
			
		||||
  open: boolean
 | 
			
		||||
  setOpen: (open: boolean) => void
 | 
			
		||||
  openMobile: boolean
 | 
			
		||||
  setOpenMobile: (open: boolean) => void
 | 
			
		||||
  isMobile: boolean
 | 
			
		||||
  toggleSidebar: () => void
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const SidebarContext = React.createContext<SidebarContextProps | null>(null)
 | 
			
		||||
 | 
			
		||||
function useSidebar() {
 | 
			
		||||
  const context = React.useContext(SidebarContext)
 | 
			
		||||
  if (!context) {
 | 
			
		||||
    throw new Error("useSidebar must be used within a SidebarProvider.")
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return context
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarProvider({
 | 
			
		||||
  defaultOpen = true,
 | 
			
		||||
  open: openProp,
 | 
			
		||||
  onOpenChange: setOpenProp,
 | 
			
		||||
  className,
 | 
			
		||||
  style,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & {
 | 
			
		||||
  defaultOpen?: boolean
 | 
			
		||||
  open?: boolean
 | 
			
		||||
  onOpenChange?: (open: boolean) => void
 | 
			
		||||
}) {
 | 
			
		||||
  const isMobile = useIsMobile()
 | 
			
		||||
  const [openMobile, setOpenMobile] = React.useState(false)
 | 
			
		||||
  const [_open, _setOpen] = React.useState(defaultOpen)
 | 
			
		||||
  const open = openProp ?? _open
 | 
			
		||||
  const setOpen = React.useCallback(
 | 
			
		||||
    (value: boolean | ((value: boolean) => boolean)) => {
 | 
			
		||||
      const openState = typeof value === "function" ? value(open) : value
 | 
			
		||||
      if (setOpenProp) {
 | 
			
		||||
        setOpenProp(openState)
 | 
			
		||||
      } else {
 | 
			
		||||
        _setOpen(openState)
 | 
			
		||||
      }
 | 
			
		||||
      document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`
 | 
			
		||||
    },
 | 
			
		||||
    [setOpenProp, open]
 | 
			
		||||
  )
 | 
			
		||||
  const toggleSidebar = React.useCallback(() => {
 | 
			
		||||
    return isMobile ? setOpenMobile((open) => !open) : setOpen((open) => !open)
 | 
			
		||||
  }, [isMobile, setOpen, setOpenMobile])
 | 
			
		||||
  React.useEffect(() => {
 | 
			
		||||
    const handleKeyDown = (event: KeyboardEvent) => {
 | 
			
		||||
      if (
 | 
			
		||||
        event.key === SIDEBAR_KEYBOARD_SHORTCUT &&
 | 
			
		||||
        (event.metaKey || event.ctrlKey)
 | 
			
		||||
      ) {
 | 
			
		||||
        event.preventDefault()
 | 
			
		||||
        toggleSidebar()
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    window.addEventListener("keydown", handleKeyDown)
 | 
			
		||||
    return () => window.removeEventListener("keydown", handleKeyDown)
 | 
			
		||||
  }, [toggleSidebar])
 | 
			
		||||
  const state = open ? "expanded" : "collapsed"
 | 
			
		||||
 | 
			
		||||
  const contextValue = React.useMemo<SidebarContextProps>(
 | 
			
		||||
    () => ({
 | 
			
		||||
      state,
 | 
			
		||||
      open,
 | 
			
		||||
      setOpen,
 | 
			
		||||
      isMobile,
 | 
			
		||||
      openMobile,
 | 
			
		||||
      setOpenMobile,
 | 
			
		||||
      toggleSidebar,
 | 
			
		||||
    }),
 | 
			
		||||
    [state, open, setOpen, isMobile, openMobile, setOpenMobile, toggleSidebar]
 | 
			
		||||
  )
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SidebarContext.Provider value={contextValue}>
 | 
			
		||||
      <TooltipProvider delayDuration={0}>
 | 
			
		||||
        <div
 | 
			
		||||
          data-slot="sidebar-wrapper"
 | 
			
		||||
          style={
 | 
			
		||||
            {
 | 
			
		||||
              "--sidebar-width": SIDEBAR_WIDTH,
 | 
			
		||||
              "--sidebar-width-icon": SIDEBAR_WIDTH_ICON,
 | 
			
		||||
              ...style,
 | 
			
		||||
            } as React.CSSProperties
 | 
			
		||||
          }
 | 
			
		||||
          className={cn(
 | 
			
		||||
            "group/sidebar-wrapper has-data-[variant=inset]:bg-sidebar flex min-h-svh w-full",
 | 
			
		||||
            className
 | 
			
		||||
          )}
 | 
			
		||||
          {...props}
 | 
			
		||||
        >
 | 
			
		||||
          {children}
 | 
			
		||||
        </div>
 | 
			
		||||
      </TooltipProvider>
 | 
			
		||||
    </SidebarContext.Provider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function Sidebar({
 | 
			
		||||
  side = "left",
 | 
			
		||||
  variant = "sidebar",
 | 
			
		||||
  collapsible = "offcanvas",
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & {
 | 
			
		||||
  side?: "left" | "right"
 | 
			
		||||
  variant?: "sidebar" | "floating" | "inset"
 | 
			
		||||
  collapsible?: "offcanvas" | "icon" | "none"
 | 
			
		||||
}) {
 | 
			
		||||
  const { isMobile, state, openMobile, setOpenMobile } = useSidebar()
 | 
			
		||||
 | 
			
		||||
  if (collapsible === "none") {
 | 
			
		||||
    return (
 | 
			
		||||
      <div
 | 
			
		||||
        data-slot="sidebar"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
      </div>
 | 
			
		||||
    )
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (isMobile) {
 | 
			
		||||
    return (
 | 
			
		||||
      <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}>
 | 
			
		||||
        <SheetContent
 | 
			
		||||
          data-sidebar="sidebar"
 | 
			
		||||
          data-slot="sidebar"
 | 
			
		||||
          data-mobile="true"
 | 
			
		||||
          className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
 | 
			
		||||
          style={
 | 
			
		||||
            {
 | 
			
		||||
              "--sidebar-width": SIDEBAR_WIDTH_MOBILE,
 | 
			
		||||
            } as React.CSSProperties
 | 
			
		||||
          }
 | 
			
		||||
          side={side}
 | 
			
		||||
        >
 | 
			
		||||
          <SheetHeader className="sr-only">
 | 
			
		||||
            <SheetTitle>Sidebar</SheetTitle>
 | 
			
		||||
            <SheetDescription>Displays the mobile sidebar.</SheetDescription>
 | 
			
		||||
          </SheetHeader>
 | 
			
		||||
          <div className="flex h-full w-full flex-col">{children}</div>
 | 
			
		||||
        </SheetContent>
 | 
			
		||||
      </Sheet>
 | 
			
		||||
    )
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className="group peer text-sidebar-foreground hidden md:block"
 | 
			
		||||
      data-state={state}
 | 
			
		||||
      data-collapsible={state === "collapsed" ? collapsible : ""}
 | 
			
		||||
      data-variant={variant}
 | 
			
		||||
      data-side={side}
 | 
			
		||||
      data-slot="sidebar"
 | 
			
		||||
    >
 | 
			
		||||
      {/* This is what handles the sidebar gap on desktop */}
 | 
			
		||||
      <div
 | 
			
		||||
        data-slot="sidebar-gap"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "relative w-(--sidebar-width) bg-transparent transition-[width] duration-200 ease-linear",
 | 
			
		||||
          "group-data-[collapsible=offcanvas]:w-0",
 | 
			
		||||
          "group-data-[side=right]:rotate-180",
 | 
			
		||||
          variant === "floating" || variant === "inset"
 | 
			
		||||
            ? "group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]"
 | 
			
		||||
            : "group-data-[collapsible=icon]:w-(--sidebar-width-icon)"
 | 
			
		||||
        )}
 | 
			
		||||
      />
 | 
			
		||||
      <div
 | 
			
		||||
        data-slot="sidebar-container"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "fixed inset-y-0 z-10 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:flex",
 | 
			
		||||
          side === "left"
 | 
			
		||||
            ? "left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]"
 | 
			
		||||
            : "right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]",
 | 
			
		||||
          variant === "floating" || variant === "inset"
 | 
			
		||||
            ? "p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]"
 | 
			
		||||
            : "group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        <div
 | 
			
		||||
          data-sidebar="sidebar"
 | 
			
		||||
          data-slot="sidebar-inner"
 | 
			
		||||
          className="bg-sidebar group-data-[variant=floating]:border-sidebar-border flex h-full w-full flex-col group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:shadow-sm"
 | 
			
		||||
        >
 | 
			
		||||
          {children}
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  onClick,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Button>) {
 | 
			
		||||
  const { toggleSidebar } = useSidebar()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Button
 | 
			
		||||
      data-sidebar="trigger"
 | 
			
		||||
      data-slot="sidebar-trigger"
 | 
			
		||||
      variant="ghost"
 | 
			
		||||
      size="icon"
 | 
			
		||||
      className={cn("size-7", className)}
 | 
			
		||||
      onClick={(event) => {
 | 
			
		||||
        onClick?.(event)
 | 
			
		||||
        toggleSidebar()
 | 
			
		||||
      }}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <PanelLeftIcon />
 | 
			
		||||
      <span className="sr-only">Toggle Sidebar</span>
 | 
			
		||||
    </Button>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarRail({ className, ...props }: React.ComponentProps<"button">) {
 | 
			
		||||
  const { toggleSidebar } = useSidebar()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <button
 | 
			
		||||
      data-sidebar="rail"
 | 
			
		||||
      data-slot="sidebar-rail"
 | 
			
		||||
      aria-label="Toggle Sidebar"
 | 
			
		||||
      tabIndex={-1}
 | 
			
		||||
      onClick={toggleSidebar}
 | 
			
		||||
      title="Toggle Sidebar"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "hover:after:bg-sidebar-border absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear group-data-[side=left]:-right-4 group-data-[side=right]:left-0 after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] sm:flex",
 | 
			
		||||
        "in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize",
 | 
			
		||||
        "[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize",
 | 
			
		||||
        "hover:group-data-[collapsible=offcanvas]:bg-sidebar group-data-[collapsible=offcanvas]:translate-x-0 group-data-[collapsible=offcanvas]:after:left-full",
 | 
			
		||||
        "[[data-side=left][data-collapsible=offcanvas]_&]:-right-2",
 | 
			
		||||
        "[[data-side=right][data-collapsible=offcanvas]_&]:-left-2",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarInset({ className, ...props }: React.ComponentProps<"main">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <main
 | 
			
		||||
      data-slot="sidebar-inset"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-background relative flex w-full flex-1 flex-col",
 | 
			
		||||
        "md:peer-data-[variant=inset]:m-2 md:peer-data-[variant=inset]:ml-0 md:peer-data-[variant=inset]:rounded-xl md:peer-data-[variant=inset]:shadow-sm md:peer-data-[variant=inset]:peer-data-[state=collapsed]:ml-2",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarInput({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Input>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Input
 | 
			
		||||
      data-slot="sidebar-input"
 | 
			
		||||
      data-sidebar="input"
 | 
			
		||||
      className={cn("bg-background h-8 w-full shadow-none", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarHeader({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-header"
 | 
			
		||||
      data-sidebar="header"
 | 
			
		||||
      className={cn("flex flex-col gap-2 p-2", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarFooter({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-footer"
 | 
			
		||||
      data-sidebar="footer"
 | 
			
		||||
      className={cn("flex flex-col gap-2 p-2", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarSeparator({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof Separator>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <Separator
 | 
			
		||||
      data-slot="sidebar-separator"
 | 
			
		||||
      data-sidebar="separator"
 | 
			
		||||
      className={cn("bg-sidebar-border mx-2 w-auto", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarContent({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-content"
 | 
			
		||||
      data-sidebar="content"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarGroup({ className, ...props }: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-group"
 | 
			
		||||
      data-sidebar="group"
 | 
			
		||||
      className={cn("relative flex w-full min-w-0 flex-col p-2", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarGroupLabel({
 | 
			
		||||
  className,
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & { asChild?: boolean }) {
 | 
			
		||||
  const Comp = asChild ? Slot : "div"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="sidebar-group-label"
 | 
			
		||||
      data-sidebar="group-label"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-sidebar-foreground/70 ring-sidebar-ring flex h-8 shrink-0 items-center rounded-md px-2 text-xs font-medium outline-hidden transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0",
 | 
			
		||||
        "group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarGroupAction({
 | 
			
		||||
  className,
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"button"> & { asChild?: boolean }) {
 | 
			
		||||
  const Comp = asChild ? Slot : "button"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="sidebar-group-action"
 | 
			
		||||
      data-sidebar="group-action"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0",
 | 
			
		||||
        "after:absolute after:-inset-2 md:after:hidden",
 | 
			
		||||
        "group-data-[collapsible=icon]:hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarGroupContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-group-content"
 | 
			
		||||
      data-sidebar="group-content"
 | 
			
		||||
      className={cn("w-full text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenu({ className, ...props }: React.ComponentProps<"ul">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ul
 | 
			
		||||
      data-slot="sidebar-menu"
 | 
			
		||||
      data-sidebar="menu"
 | 
			
		||||
      className={cn("flex w-full min-w-0 flex-col gap-1", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuItem({ className, ...props }: React.ComponentProps<"li">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <li
 | 
			
		||||
      data-slot="sidebar-menu-item"
 | 
			
		||||
      data-sidebar="menu-item"
 | 
			
		||||
      className={cn("group/menu-item relative", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const sidebarMenuButtonVariants = cva(
 | 
			
		||||
  "peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left text-sm outline-hidden ring-sidebar-ring transition-[width,height,padding] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 active:bg-sidebar-accent active:text-sidebar-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-data-[sidebar=menu-action]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 data-[active=true]:bg-sidebar-accent data-[active=true]:font-medium data-[active=true]:text-sidebar-accent-foreground data-[state=open]:hover:bg-sidebar-accent data-[state=open]:hover:text-sidebar-accent-foreground group-data-[collapsible=icon]:size-8! group-data-[collapsible=icon]:p-2! [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0",
 | 
			
		||||
  {
 | 
			
		||||
    variants: {
 | 
			
		||||
      variant: {
 | 
			
		||||
        default: "hover:bg-sidebar-accent hover:text-sidebar-accent-foreground",
 | 
			
		||||
        outline:
 | 
			
		||||
          "bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]",
 | 
			
		||||
      },
 | 
			
		||||
      size: {
 | 
			
		||||
        default: "h-8 text-sm",
 | 
			
		||||
        sm: "h-7 text-xs",
 | 
			
		||||
        lg: "h-12 text-sm group-data-[collapsible=icon]:p-0!",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    defaultVariants: {
 | 
			
		||||
      variant: "default",
 | 
			
		||||
      size: "default",
 | 
			
		||||
    },
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function SidebarMenuButton({
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  isActive = false,
 | 
			
		||||
  variant = "default",
 | 
			
		||||
  size = "default",
 | 
			
		||||
  tooltip,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"button"> & {
 | 
			
		||||
  asChild?: boolean
 | 
			
		||||
  isActive?: boolean
 | 
			
		||||
  tooltip?: string | React.ComponentProps<typeof TooltipContent>
 | 
			
		||||
} & VariantProps<typeof sidebarMenuButtonVariants>) {
 | 
			
		||||
  const Comp = asChild ? Slot : "button"
 | 
			
		||||
  const { isMobile, state } = useSidebar()
 | 
			
		||||
 | 
			
		||||
  const button = (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="sidebar-menu-button"
 | 
			
		||||
      data-sidebar="menu-button"
 | 
			
		||||
      data-size={size}
 | 
			
		||||
      data-active={isActive}
 | 
			
		||||
      className={cn(sidebarMenuButtonVariants({ variant, size }), className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
 | 
			
		||||
  if (!tooltip) {
 | 
			
		||||
    return button
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (typeof tooltip === "string") {
 | 
			
		||||
    tooltip = {
 | 
			
		||||
      children: tooltip,
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Tooltip>
 | 
			
		||||
      <TooltipTrigger asChild>{button}</TooltipTrigger>
 | 
			
		||||
      <TooltipContent
 | 
			
		||||
        side="right"
 | 
			
		||||
        align="center"
 | 
			
		||||
        hidden={state !== "collapsed" || isMobile}
 | 
			
		||||
        {...tooltip}
 | 
			
		||||
      />
 | 
			
		||||
    </Tooltip>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuAction({
 | 
			
		||||
  className,
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  showOnHover = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"button"> & {
 | 
			
		||||
  asChild?: boolean
 | 
			
		||||
  showOnHover?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  const Comp = asChild ? Slot : "button"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="sidebar-menu-action"
 | 
			
		||||
      data-sidebar="menu-action"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 flex aspect-square w-5 items-center justify-center rounded-md p-0 outline-hidden transition-transform focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0",
 | 
			
		||||
        "after:absolute after:-inset-2 md:after:hidden",
 | 
			
		||||
        "peer-data-[size=sm]/menu-button:top-1",
 | 
			
		||||
        "peer-data-[size=default]/menu-button:top-1.5",
 | 
			
		||||
        "peer-data-[size=lg]/menu-button:top-2.5",
 | 
			
		||||
        "group-data-[collapsible=icon]:hidden",
 | 
			
		||||
        showOnHover &&
 | 
			
		||||
          "peer-data-[active=true]/menu-button:text-sidebar-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-[state=open]:opacity-100 md:opacity-0",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuBadge({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-menu-badge"
 | 
			
		||||
      data-sidebar="menu-badge"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-sidebar-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums select-none",
 | 
			
		||||
        "peer-hover/menu-button:text-sidebar-accent-foreground peer-data-[active=true]/menu-button:text-sidebar-accent-foreground",
 | 
			
		||||
        "peer-data-[size=sm]/menu-button:top-1",
 | 
			
		||||
        "peer-data-[size=default]/menu-button:top-1.5",
 | 
			
		||||
        "peer-data-[size=lg]/menu-button:top-2.5",
 | 
			
		||||
        "group-data-[collapsible=icon]:hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuSkeleton({
 | 
			
		||||
  className,
 | 
			
		||||
  showIcon = false,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"div"> & {
 | 
			
		||||
  showIcon?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  const width = React.useMemo(() => {
 | 
			
		||||
    return `${Math.floor(Math.random() * 40) + 50}%`
 | 
			
		||||
  }, [])
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="sidebar-menu-skeleton"
 | 
			
		||||
      data-sidebar="menu-skeleton"
 | 
			
		||||
      className={cn("flex h-8 items-center gap-2 rounded-md px-2", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {showIcon && (
 | 
			
		||||
        <Skeleton
 | 
			
		||||
          className="size-4 rounded-md"
 | 
			
		||||
          data-sidebar="menu-skeleton-icon"
 | 
			
		||||
        />
 | 
			
		||||
      )}
 | 
			
		||||
      <Skeleton
 | 
			
		||||
        className="h-4 max-w-(--skeleton-width) flex-1"
 | 
			
		||||
        data-sidebar="menu-skeleton-text"
 | 
			
		||||
        style={
 | 
			
		||||
          {
 | 
			
		||||
            "--skeleton-width": width,
 | 
			
		||||
          } as React.CSSProperties
 | 
			
		||||
        }
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuSub({ className, ...props }: React.ComponentProps<"ul">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ul
 | 
			
		||||
      data-slot="sidebar-menu-sub"
 | 
			
		||||
      data-sidebar="menu-sub"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "border-sidebar-border mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-l px-2.5 py-0.5",
 | 
			
		||||
        "group-data-[collapsible=icon]:hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuSubItem({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"li">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <li
 | 
			
		||||
      data-slot="sidebar-menu-sub-item"
 | 
			
		||||
      data-sidebar="menu-sub-item"
 | 
			
		||||
      className={cn("group/menu-sub-item relative", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function SidebarMenuSubButton({
 | 
			
		||||
  asChild = false,
 | 
			
		||||
  size = "md",
 | 
			
		||||
  isActive = false,
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"a"> & {
 | 
			
		||||
  asChild?: boolean
 | 
			
		||||
  size?: "sm" | "md"
 | 
			
		||||
  isActive?: boolean
 | 
			
		||||
}) {
 | 
			
		||||
  const Comp = asChild ? Slot : "a"
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Comp
 | 
			
		||||
      data-slot="sidebar-menu-sub-button"
 | 
			
		||||
      data-sidebar="menu-sub-button"
 | 
			
		||||
      data-size={size}
 | 
			
		||||
      data-active={isActive}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground [&>svg]:text-sidebar-accent-foreground flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 outline-hidden focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0",
 | 
			
		||||
        "data-[active=true]:bg-sidebar-accent data-[active=true]:text-sidebar-accent-foreground",
 | 
			
		||||
        size === "sm" && "text-xs",
 | 
			
		||||
        size === "md" && "text-sm",
 | 
			
		||||
        "group-data-[collapsible=icon]:hidden",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
	Sidebar,
 | 
			
		||||
	SidebarContent,
 | 
			
		||||
	SidebarFooter,
 | 
			
		||||
	SidebarGroup,
 | 
			
		||||
	SidebarGroupAction,
 | 
			
		||||
	SidebarGroupContent,
 | 
			
		||||
	SidebarGroupLabel,
 | 
			
		||||
	SidebarHeader,
 | 
			
		||||
	SidebarInput,
 | 
			
		||||
	SidebarInset,
 | 
			
		||||
	SidebarMenu,
 | 
			
		||||
	SidebarMenuAction,
 | 
			
		||||
	SidebarMenuBadge,
 | 
			
		||||
	SidebarMenuButton,
 | 
			
		||||
	SidebarMenuItem,
 | 
			
		||||
	SidebarMenuSkeleton,
 | 
			
		||||
	SidebarMenuSub,
 | 
			
		||||
	SidebarMenuSubButton,
 | 
			
		||||
	SidebarMenuSubItem,
 | 
			
		||||
	SidebarProvider,
 | 
			
		||||
	SidebarRail,
 | 
			
		||||
	SidebarSeparator,
 | 
			
		||||
	SidebarTrigger,
 | 
			
		||||
	useSidebar
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										15
									
								
								web/src/components/ui/skeleton.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								web/src/components/ui/skeleton.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Skeleton({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.HTMLAttributes<HTMLDivElement>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className={cn("animate-pulse rounded-md bg-primary/10", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Skeleton }
 | 
			
		||||
							
								
								
									
										63
									
								
								web/src/components/ui/slider.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								web/src/components/ui/slider.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,63 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as SliderPrimitive from "@radix-ui/react-slider"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Slider({
 | 
			
		||||
  className,
 | 
			
		||||
  defaultValue,
 | 
			
		||||
  value,
 | 
			
		||||
  min = 0,
 | 
			
		||||
  max = 100,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SliderPrimitive.Root>) {
 | 
			
		||||
  const _values = React.useMemo(
 | 
			
		||||
    () =>
 | 
			
		||||
      Array.isArray(value)
 | 
			
		||||
        ? value
 | 
			
		||||
        : Array.isArray(defaultValue)
 | 
			
		||||
          ? defaultValue
 | 
			
		||||
          : [min, max],
 | 
			
		||||
    [value, defaultValue, min, max]
 | 
			
		||||
  )
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <SliderPrimitive.Root
 | 
			
		||||
      data-slot="slider"
 | 
			
		||||
      defaultValue={defaultValue}
 | 
			
		||||
      value={value}
 | 
			
		||||
      min={min}
 | 
			
		||||
      max={max}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "relative flex w-full touch-none items-center select-none data-[disabled]:opacity-50 data-[orientation=vertical]:h-full data-[orientation=vertical]:min-h-44 data-[orientation=vertical]:w-auto data-[orientation=vertical]:flex-col",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <SliderPrimitive.Track
 | 
			
		||||
        data-slot="slider-track"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-muted relative grow overflow-hidden rounded-full data-[orientation=horizontal]:h-1.5 data-[orientation=horizontal]:w-full data-[orientation=vertical]:h-full data-[orientation=vertical]:w-1.5"
 | 
			
		||||
        )}
 | 
			
		||||
      >
 | 
			
		||||
        <SliderPrimitive.Range
 | 
			
		||||
          data-slot="slider-range"
 | 
			
		||||
          className={cn(
 | 
			
		||||
            "bg-primary absolute data-[orientation=horizontal]:h-full data-[orientation=vertical]:w-full"
 | 
			
		||||
          )}
 | 
			
		||||
        />
 | 
			
		||||
      </SliderPrimitive.Track>
 | 
			
		||||
      {Array.from({ length: _values.length }, (_, index) => (
 | 
			
		||||
        <SliderPrimitive.Thumb
 | 
			
		||||
          data-slot="slider-thumb"
 | 
			
		||||
          key={index}
 | 
			
		||||
          className="border-primary bg-background ring-ring/50 block size-4 shrink-0 rounded-full border shadow-sm transition-[color,box-shadow] hover:ring-4 focus-visible:ring-4 focus-visible:outline-hidden disabled:pointer-events-none disabled:opacity-50"
 | 
			
		||||
        />
 | 
			
		||||
      ))}
 | 
			
		||||
    </SliderPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Slider }
 | 
			
		||||
							
								
								
									
										25
									
								
								web/src/components/ui/sonner.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								web/src/components/ui/sonner.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,25 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import { useTheme } from "next-themes"
 | 
			
		||||
import { Toaster as Sonner, ToasterProps } from "sonner"
 | 
			
		||||
 | 
			
		||||
const Toaster = ({ ...props }: ToasterProps) => {
 | 
			
		||||
  const { theme = "system" } = useTheme()
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Sonner
 | 
			
		||||
      theme={theme as ToasterProps["theme"]}
 | 
			
		||||
      className="toaster group"
 | 
			
		||||
      style={
 | 
			
		||||
        {
 | 
			
		||||
          "--normal-bg": "var(--popover)",
 | 
			
		||||
          "--normal-text": "var(--popover-foreground)",
 | 
			
		||||
          "--normal-border": "var(--border)",
 | 
			
		||||
        } as React.CSSProperties
 | 
			
		||||
      }
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Toaster }
 | 
			
		||||
							
								
								
									
										31
									
								
								web/src/components/ui/switch.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								web/src/components/ui/switch.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as SwitchPrimitive from "@radix-ui/react-switch"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Switch({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof SwitchPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <SwitchPrimitive.Root
 | 
			
		||||
      data-slot="switch"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "peer data-[state=checked]:bg-primary data-[state=unchecked]:bg-input focus-visible:border-ring focus-visible:ring-ring/50 dark:data-[state=unchecked]:bg-input/80 inline-flex h-[1.15rem] w-8 shrink-0 items-center rounded-full border border-transparent shadow-xs transition-all outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <SwitchPrimitive.Thumb
 | 
			
		||||
        data-slot="switch-thumb"
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-background dark:data-[state=unchecked]:bg-foreground dark:data-[state=checked]:bg-primary-foreground pointer-events-none block size-4 rounded-full ring-0 transition-transform data-[state=checked]:translate-x-[calc(100%-2px)] data-[state=unchecked]:translate-x-0"
 | 
			
		||||
        )}
 | 
			
		||||
      />
 | 
			
		||||
    </SwitchPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Switch }
 | 
			
		||||
							
								
								
									
										116
									
								
								web/src/components/ui/table.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								web/src/components/ui/table.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,116 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Table({ className, ...props }: React.ComponentProps<"table">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-slot="table-container"
 | 
			
		||||
      className="relative w-full overflow-x-auto"
 | 
			
		||||
    >
 | 
			
		||||
      <table
 | 
			
		||||
        data-slot="table"
 | 
			
		||||
        className={cn("w-full caption-bottom text-sm", className)}
 | 
			
		||||
        {...props}
 | 
			
		||||
      />
 | 
			
		||||
    </div>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableHeader({ className, ...props }: React.ComponentProps<"thead">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <thead
 | 
			
		||||
      data-slot="table-header"
 | 
			
		||||
      className={cn("[&_tr]:border-b", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableBody({ className, ...props }: React.ComponentProps<"tbody">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <tbody
 | 
			
		||||
      data-slot="table-body"
 | 
			
		||||
      className={cn("[&_tr:last-child]:border-0", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableFooter({ className, ...props }: React.ComponentProps<"tfoot">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <tfoot
 | 
			
		||||
      data-slot="table-footer"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-muted/50 border-t font-medium [&>tr]:last:border-b-0",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableRow({ className, ...props }: React.ComponentProps<"tr">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <tr
 | 
			
		||||
      data-slot="table-row"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "hover:bg-muted/50 data-[state=selected]:bg-muted border-b transition-colors",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableHead({ className, ...props }: React.ComponentProps<"th">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <th
 | 
			
		||||
      data-slot="table-head"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "text-foreground h-10 px-2 text-left align-middle font-medium whitespace-nowrap [&:has([role=checkbox])]:pr-0 [&>[role=checkbox]]:translate-y-[2px]",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableCell({ className, ...props }: React.ComponentProps<"td">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <td
 | 
			
		||||
      data-slot="table-cell"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "p-2 align-middle whitespace-nowrap [&:has([role=checkbox])]:pr-0 [&>[role=checkbox]]:translate-y-[2px]",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TableCaption({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<"caption">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <caption
 | 
			
		||||
      data-slot="table-caption"
 | 
			
		||||
      className={cn("text-muted-foreground mt-4 text-sm", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Table,
 | 
			
		||||
  TableHeader,
 | 
			
		||||
  TableBody,
 | 
			
		||||
  TableFooter,
 | 
			
		||||
  TableHead,
 | 
			
		||||
  TableRow,
 | 
			
		||||
  TableCell,
 | 
			
		||||
  TableCaption,
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										66
									
								
								web/src/components/ui/tabs.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								web/src/components/ui/tabs.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,66 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as TabsPrimitive from "@radix-ui/react-tabs"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Tabs({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TabsPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TabsPrimitive.Root
 | 
			
		||||
      data-slot="tabs"
 | 
			
		||||
      className={cn("flex flex-col gap-2", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TabsList({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TabsPrimitive.List>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TabsPrimitive.List
 | 
			
		||||
      data-slot="tabs-list"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "bg-muted text-muted-foreground inline-flex h-9 w-fit items-center justify-center rounded-lg p-[3px]",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TabsTrigger({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TabsPrimitive.Trigger>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TabsPrimitive.Trigger
 | 
			
		||||
      data-slot="tabs-trigger"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "data-[state=active]:bg-background dark:data-[state=active]:text-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:outline-ring dark:data-[state=active]:border-input dark:data-[state=active]:bg-input/30 text-foreground dark:text-muted-foreground inline-flex h-[calc(100%-1px)] flex-1 items-center justify-center gap-1.5 rounded-md border border-transparent px-2 py-1 text-sm font-medium whitespace-nowrap transition-[color,box-shadow] focus-visible:ring-[3px] focus-visible:outline-1 disabled:pointer-events-none disabled:opacity-50 data-[state=active]:shadow-sm [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TabsContent({
 | 
			
		||||
  className,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TabsPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TabsPrimitive.Content
 | 
			
		||||
      data-slot="tabs-content"
 | 
			
		||||
      className={cn("flex-1 outline-none", className)}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Tabs, TabsList, TabsTrigger, TabsContent }
 | 
			
		||||
							
								
								
									
										18
									
								
								web/src/components/ui/textarea.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								web/src/components/ui/textarea.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,18 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function Textarea({ className, ...props }: React.ComponentProps<"textarea">) {
 | 
			
		||||
  return (
 | 
			
		||||
    <textarea
 | 
			
		||||
      data-slot="textarea"
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "border-input placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive dark:bg-input/30 flex field-sizing-content min-h-16 w-full rounded-md border bg-transparent px-3 py-2 text-base shadow-xs transition-[color,box-shadow] outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Textarea }
 | 
			
		||||
							
								
								
									
										73
									
								
								web/src/components/ui/toggle-group.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								web/src/components/ui/toggle-group.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,73 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as ToggleGroupPrimitive from "@radix-ui/react-toggle-group"
 | 
			
		||||
import { type VariantProps } from "class-variance-authority"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
import { toggleVariants } from "@/components/ui/toggle"
 | 
			
		||||
 | 
			
		||||
const ToggleGroupContext = React.createContext<
 | 
			
		||||
  VariantProps<typeof toggleVariants>
 | 
			
		||||
>({
 | 
			
		||||
  size: "default",
 | 
			
		||||
  variant: "default",
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
function ToggleGroup({
 | 
			
		||||
  className,
 | 
			
		||||
  variant,
 | 
			
		||||
  size,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ToggleGroupPrimitive.Root> &
 | 
			
		||||
  VariantProps<typeof toggleVariants>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <ToggleGroupPrimitive.Root
 | 
			
		||||
      data-slot="toggle-group"
 | 
			
		||||
      data-variant={variant}
 | 
			
		||||
      data-size={size}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        "group/toggle-group flex w-fit items-center rounded-md data-[variant=outline]:shadow-xs",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      <ToggleGroupContext.Provider value={{ variant, size }}>
 | 
			
		||||
        {children}
 | 
			
		||||
      </ToggleGroupContext.Provider>
 | 
			
		||||
    </ToggleGroupPrimitive.Root>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function ToggleGroupItem({
 | 
			
		||||
  className,
 | 
			
		||||
  children,
 | 
			
		||||
  variant,
 | 
			
		||||
  size,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof ToggleGroupPrimitive.Item> &
 | 
			
		||||
  VariantProps<typeof toggleVariants>) {
 | 
			
		||||
  const context = React.useContext(ToggleGroupContext)
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <ToggleGroupPrimitive.Item
 | 
			
		||||
      data-slot="toggle-group-item"
 | 
			
		||||
      data-variant={context.variant || variant}
 | 
			
		||||
      data-size={context.size || size}
 | 
			
		||||
      className={cn(
 | 
			
		||||
        toggleVariants({
 | 
			
		||||
          variant: context.variant || variant,
 | 
			
		||||
          size: context.size || size,
 | 
			
		||||
        }),
 | 
			
		||||
        "min-w-0 flex-1 shrink-0 rounded-none shadow-none first:rounded-l-md last:rounded-r-md focus:z-10 focus-visible:z-10 data-[variant=outline]:border-l-0 data-[variant=outline]:first:border-l",
 | 
			
		||||
        className
 | 
			
		||||
      )}
 | 
			
		||||
      {...props}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
    </ToggleGroupPrimitive.Item>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { ToggleGroup, ToggleGroupItem }
 | 
			
		||||
							
								
								
									
										47
									
								
								web/src/components/ui/toggle.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								web/src/components/ui/toggle.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,47 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as TogglePrimitive from "@radix-ui/react-toggle"
 | 
			
		||||
import { cva, type VariantProps } from "class-variance-authority"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
const toggleVariants = cva(
 | 
			
		||||
  "inline-flex items-center justify-center gap-2 rounded-md text-sm font-medium hover:bg-muted hover:text-muted-foreground disabled:pointer-events-none disabled:opacity-50 data-[state=on]:bg-accent data-[state=on]:text-accent-foreground [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 [&_svg]:shrink-0 focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] outline-none transition-[color,box-shadow] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive whitespace-nowrap",
 | 
			
		||||
  {
 | 
			
		||||
    variants: {
 | 
			
		||||
      variant: {
 | 
			
		||||
        default: "bg-transparent",
 | 
			
		||||
        outline:
 | 
			
		||||
          "border border-input bg-transparent shadow-xs hover:bg-accent hover:text-accent-foreground",
 | 
			
		||||
      },
 | 
			
		||||
      size: {
 | 
			
		||||
        default: "h-9 px-2 min-w-9",
 | 
			
		||||
        sm: "h-8 px-1.5 min-w-8",
 | 
			
		||||
        lg: "h-10 px-2.5 min-w-10",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    defaultVariants: {
 | 
			
		||||
      variant: "default",
 | 
			
		||||
      size: "default",
 | 
			
		||||
    },
 | 
			
		||||
  }
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
function Toggle({
 | 
			
		||||
  className,
 | 
			
		||||
  variant,
 | 
			
		||||
  size,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TogglePrimitive.Root> &
 | 
			
		||||
  VariantProps<typeof toggleVariants>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TogglePrimitive.Root
 | 
			
		||||
      data-slot="toggle"
 | 
			
		||||
      className={cn(toggleVariants({ variant, size, className }))}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Toggle, toggleVariants }
 | 
			
		||||
							
								
								
									
										61
									
								
								web/src/components/ui/tooltip.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								web/src/components/ui/tooltip.tsx
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,61 @@
 | 
			
		||||
"use client"
 | 
			
		||||
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
import * as TooltipPrimitive from "@radix-ui/react-tooltip"
 | 
			
		||||
 | 
			
		||||
import { cn } from "@/lib/utils"
 | 
			
		||||
 | 
			
		||||
function TooltipProvider({
 | 
			
		||||
  delayDuration = 0,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TooltipPrimitive.Provider>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TooltipPrimitive.Provider
 | 
			
		||||
      data-slot="tooltip-provider"
 | 
			
		||||
      delayDuration={delayDuration}
 | 
			
		||||
      {...props}
 | 
			
		||||
    />
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function Tooltip({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TooltipPrimitive.Root>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TooltipProvider>
 | 
			
		||||
      <TooltipPrimitive.Root data-slot="tooltip" {...props} />
 | 
			
		||||
    </TooltipProvider>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TooltipTrigger({
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TooltipPrimitive.Trigger>) {
 | 
			
		||||
  return <TooltipPrimitive.Trigger data-slot="tooltip-trigger" {...props} />
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function TooltipContent({
 | 
			
		||||
  className,
 | 
			
		||||
  sideOffset = 0,
 | 
			
		||||
  children,
 | 
			
		||||
  ...props
 | 
			
		||||
}: React.ComponentProps<typeof TooltipPrimitive.Content>) {
 | 
			
		||||
  return (
 | 
			
		||||
    <TooltipPrimitive.Portal>
 | 
			
		||||
      <TooltipPrimitive.Content
 | 
			
		||||
        data-slot="tooltip-content"
 | 
			
		||||
        sideOffset={sideOffset}
 | 
			
		||||
        className={cn(
 | 
			
		||||
          "bg-primary text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-fit origin-(--radix-tooltip-content-transform-origin) rounded-md px-3 py-1.5 text-xs text-balance",
 | 
			
		||||
          className
 | 
			
		||||
        )}
 | 
			
		||||
        {...props}
 | 
			
		||||
      >
 | 
			
		||||
        {children}
 | 
			
		||||
        <TooltipPrimitive.Arrow className="bg-primary fill-primary z-50 size-2.5 translate-y-[calc(-50%_-_2px)] rotate-45 rounded-[2px]" />
 | 
			
		||||
      </TooltipPrimitive.Content>
 | 
			
		||||
    </TooltipPrimitive.Portal>
 | 
			
		||||
  )
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { Tooltip, TooltipTrigger, TooltipContent, TooltipProvider }
 | 
			
		||||
							
								
								
									
										4
									
								
								web/src/constants.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								web/src/constants.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
export const BASE_URL = "https://cdn.jsdelivr.net/gh/homarr-labs/dashboard-icons"
 | 
			
		||||
export const REPO_PATH = "https://github.com/homarr-labs/dashboard-icons"
 | 
			
		||||
export const METADATA_URL = "https://raw.githubusercontent.com/homarr-labs/dashboard-icons/refs/heads/main/metadata.json"
 | 
			
		||||
export const WEB_URL = "https://icons.homarr.dev"
 | 
			
		||||
							
								
								
									
										19
									
								
								web/src/hooks/use-mobile.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								web/src/hooks/use-mobile.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
import * as React from "react"
 | 
			
		||||
 | 
			
		||||
const MOBILE_BREAKPOINT = 768
 | 
			
		||||
 | 
			
		||||
export function useIsMobile() {
 | 
			
		||||
	const [isMobile, setIsMobile] = React.useState<boolean | undefined>(undefined)
 | 
			
		||||
 | 
			
		||||
	React.useEffect(() => {
 | 
			
		||||
		const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`)
 | 
			
		||||
		const onChange = () => {
 | 
			
		||||
			setIsMobile(window.innerWidth < MOBILE_BREAKPOINT)
 | 
			
		||||
		}
 | 
			
		||||
		mql.addEventListener("change", onChange)
 | 
			
		||||
		setIsMobile(window.innerWidth < MOBILE_BREAKPOINT)
 | 
			
		||||
		return () => mql.removeEventListener("change", onChange)
 | 
			
		||||
	}, [])
 | 
			
		||||
 | 
			
		||||
	return !!isMobile
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										74
									
								
								web/src/lib/api.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								web/src/lib/api.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,74 @@
 | 
			
		||||
import { METADATA_URL } from "@/constants"
 | 
			
		||||
import type { IconFile, IconWithName } from "@/types/icons"
 | 
			
		||||
/**
 | 
			
		||||
 * Fetches all icon data from the metadata.json file
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
export async function getAllIcons(): Promise<IconFile> {
 | 
			
		||||
	const file = await fetch(METADATA_URL)
 | 
			
		||||
	return (await file.json()) as IconFile
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Gets a list of all icon names.
 | 
			
		||||
 */
 | 
			
		||||
export const getIconNames = async (): Promise<string[]> => {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	return Object.keys(iconsData)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Converts icon data to an array format for easier rendering
 | 
			
		||||
 */
 | 
			
		||||
export async function getIconsArray(): Promise<IconWithName[]> {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
 | 
			
		||||
	return Object.entries(iconsData)
 | 
			
		||||
		.map(([name, data]) => ({
 | 
			
		||||
			name,
 | 
			
		||||
			data,
 | 
			
		||||
		}))
 | 
			
		||||
		.sort((a, b) => a.name.localeCompare(b.name))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Fetches data for a specific icon
 | 
			
		||||
 */
 | 
			
		||||
export async function getIconData(iconName: string): Promise<IconWithName | null> {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
	const iconData = iconsData[iconName]
 | 
			
		||||
 | 
			
		||||
	if (!iconData) {
 | 
			
		||||
		return null
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return {
 | 
			
		||||
		name: iconName,
 | 
			
		||||
		data: iconData,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Fetches author data from GitHub API
 | 
			
		||||
 */
 | 
			
		||||
export async function getAuthorData(authorId: number) {
 | 
			
		||||
	const response = await fetch(`https://api.github.com/user/${authorId}`, {
 | 
			
		||||
		headers: {
 | 
			
		||||
			Authorization: `Bearer ${process.env.GITHUB_TOKEN}`,
 | 
			
		||||
			"Cache-Control": "public, max-age=86400",
 | 
			
		||||
		},
 | 
			
		||||
		next: { revalidate: 86400 }, // Revalidate cache once a day
 | 
			
		||||
	})
 | 
			
		||||
	return response.json()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Fetches featured icons for the homepage
 | 
			
		||||
 */
 | 
			
		||||
export async function getTotalIcons() {
 | 
			
		||||
	const iconsData = await getAllIcons()
 | 
			
		||||
 | 
			
		||||
	return {
 | 
			
		||||
		totalIcons: Object.keys(iconsData).length,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										6
									
								
								web/src/lib/utils.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								web/src/lib/utils.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,6 @@
 | 
			
		||||
import { type ClassValue, clsx } from "clsx"
 | 
			
		||||
import { twMerge } from "tailwind-merge"
 | 
			
		||||
 | 
			
		||||
export function cn(...inputs: ClassValue[]) {
 | 
			
		||||
	return twMerge(clsx(inputs))
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										44
									
								
								web/src/types/icons.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								web/src/types/icons.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,44 @@
 | 
			
		||||
export type IconAuthor = {
 | 
			
		||||
	id: number
 | 
			
		||||
	name?: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type IconUpdate = {
 | 
			
		||||
	timestamp: string
 | 
			
		||||
	author: IconAuthor
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type IconColors = {
 | 
			
		||||
	dark?: string
 | 
			
		||||
	light?: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type Icon = {
 | 
			
		||||
	base: string | "svg" | "png" | "webp"
 | 
			
		||||
	aliases: string[]
 | 
			
		||||
	categories: string[]
 | 
			
		||||
	update: IconUpdate
 | 
			
		||||
	colors?: IconColors
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type IconFile = {
 | 
			
		||||
	[key: string]: Icon
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type IconWithName = {
 | 
			
		||||
	name: string
 | 
			
		||||
	data: Icon
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type IconSearchProps = {
 | 
			
		||||
	icons: IconWithName[]
 | 
			
		||||
	initialQuery?: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type AuthorData = {
 | 
			
		||||
	id: number
 | 
			
		||||
	name?: string
 | 
			
		||||
	login: string
 | 
			
		||||
	avatar_url: string
 | 
			
		||||
	html_url: string
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								web/src/types/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								web/src/types/index.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
export * from "./icons"
 | 
			
		||||
		Reference in New Issue
	
	Block a user