diff --git a/.mcp.json b/.mcp.json index 453b55c..16fc2a6 100644 --- a/.mcp.json +++ b/.mcp.json @@ -66,10 +66,7 @@ }, "shadcn": { "command": "npx", - "args": [ - "shadcn@latest", - "mcp" - ] + "args": ["shadcn@latest", "mcp"] } } } diff --git a/apps/landing/components.json b/apps/landing/components.json new file mode 100644 index 0000000..edcaef2 --- /dev/null +++ b/apps/landing/components.json @@ -0,0 +1,22 @@ +{ + "$schema": "https://ui.shadcn.com/schema.json", + "style": "new-york", + "rsc": true, + "tsx": true, + "tailwind": { + "config": "", + "css": "src/app/globals.css", + "baseColor": "neutral", + "cssVariables": true, + "prefix": "" + }, + "iconLibrary": "lucide", + "aliases": { + "components": "@/components", + "utils": "@/lib/utils", + "ui": "@/components/ui", + "lib": "@/lib", + "hooks": "@/hooks" + }, + "registries": {} +} diff --git a/apps/landing/package.json b/apps/landing/package.json index 6b93aea..6c16392 100644 --- a/apps/landing/package.json +++ b/apps/landing/package.json @@ -11,10 +11,19 @@ }, "dependencies": { "@banatie/database": "workspace:*", + "@radix-ui/react-checkbox": "^1.3.3", + "@radix-ui/react-collapsible": "^1.1.12", + "@radix-ui/react-label": "^2.1.8", + "@radix-ui/react-radio-group": "^1.3.8", + "@radix-ui/react-select": "^2.2.6", + "@radix-ui/react-slot": "^1.2.4", + "class-variance-authority": "^0.7.1", + "clsx": "^2.1.1", "lucide-react": "^0.400.0", "next": "15.5.4", "react": "19.1.0", - "react-dom": "19.1.0" + "react-dom": "19.1.0", + "tailwind-merge": "^3.4.0" }, "devDependencies": { "@tailwindcss/postcss": "^4", @@ -22,6 +31,7 @@ "@types/react": "^19", "@types/react-dom": "^19", "tailwindcss": "^4", + "tw-animate-css": "^1.4.0", "typescript": "^5" } } diff --git a/apps/landing/src/app/globals.css b/apps/landing/src/app/globals.css index bafa22b..c65a01d 100644 --- a/apps/landing/src/app/globals.css +++ b/apps/landing/src/app/globals.css @@ -1,20 +1,49 @@ @import 'tailwindcss'; +@import "tw-animate-css"; -:root { - --background: #0f172a; - --foreground: #f8fafc; -} +@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); } body { - background: var(--background); - color: var(--foreground); font-family: 'Inter', -apple-system, @@ -99,3 +128,81 @@ pre::-webkit-scrollbar-thumb { pre::-webkit-scrollbar-thumb:hover { background: rgb(148, 163, 184); /* slate-400 */ } + +/* Dark theme as default (no .dark class needed) */ +:root { + --radius: 0.625rem; + --background: oklch(0.145 0 0); + --foreground: oklch(0.985 0 0); + --card: oklch(0.205 0 0); + --card-foreground: oklch(0.985 0 0); + --popover: oklch(0.205 0 0); + --popover-foreground: oklch(0.985 0 0); + --primary: oklch(0.922 0 0); + --primary-foreground: oklch(0.205 0 0); + --secondary: oklch(0.269 0 0); + --secondary-foreground: oklch(0.985 0 0); + --muted: oklch(0.269 0 0); + --muted-foreground: oklch(0.708 0 0); + --accent: oklch(0.269 0 0); + --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.556 0 0); + --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.205 0 0); + --sidebar-foreground: oklch(0.985 0 0); + --sidebar-primary: oklch(0.488 0.243 264.376); + --sidebar-primary-foreground: oklch(0.985 0 0); + --sidebar-accent: oklch(0.269 0 0); + --sidebar-accent-foreground: oklch(0.985 0 0); + --sidebar-border: oklch(1 0 0 / 10%); + --sidebar-ring: oklch(0.556 0 0); +} + +/* Light theme (optional, requires .light class) */ +.light { + --background: oklch(1 0 0); + --foreground: oklch(0.145 0 0); + --card: oklch(1 0 0); + --card-foreground: oklch(0.145 0 0); + --popover: oklch(1 0 0); + --popover-foreground: oklch(0.145 0 0); + --primary: oklch(0.205 0 0); + --primary-foreground: oklch(0.985 0 0); + --secondary: oklch(0.97 0 0); + --secondary-foreground: oklch(0.205 0 0); + --muted: oklch(0.97 0 0); + --muted-foreground: oklch(0.556 0 0); + --accent: oklch(0.97 0 0); + --accent-foreground: oklch(0.205 0 0); + --destructive: oklch(0.577 0.245 27.325); + --border: oklch(0.922 0 0); + --input: oklch(0.922 0 0); + --ring: oklch(0.708 0 0); + --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.145 0 0); + --sidebar-primary: oklch(0.205 0 0); + --sidebar-primary-foreground: oklch(0.985 0 0); + --sidebar-accent: oklch(0.97 0 0); + --sidebar-accent-foreground: oklch(0.205 0 0); + --sidebar-border: oklch(0.922 0 0); + --sidebar-ring: oklch(0.708 0 0); +} + +@layer base { + * { + @apply border-border outline-ring/50; + } + /* Body background handled by page-specific gradients, not Shadcn variables */ +} diff --git a/apps/landing/src/components/lab/FilterPlaceholder.tsx b/apps/landing/src/components/lab/FilterPlaceholder.tsx index 2ffc47d..e8b3060 100644 --- a/apps/landing/src/components/lab/FilterPlaceholder.tsx +++ b/apps/landing/src/components/lab/FilterPlaceholder.tsx @@ -5,17 +5,20 @@ * * Checkbox/radio button group for sidebar filters. * Supports both single-select (radio) and multi-select (checkbox) modes. - * Uses slate palette for form controls (matches form styling). + * Uses Shadcn UI components with custom slate styling. * * Features: - * - Radio buttons for single selection - * - Checkboxes for multiple selection + * - Radio buttons for single selection (Shadcn RadioGroup) + * - Checkboxes for multiple selection (Shadcn Checkbox) * - Option counts (e.g., "All (127)") * - Accessible keyboard navigation * - Focus indicators */ import { useState } from 'react'; +import { Checkbox } from '@/components/ui/checkbox'; +import { RadioGroup, RadioGroupItem } from '@/components/ui/radio-group'; +import { Label } from '@/components/ui/label'; type FilterOption = { id: string; @@ -36,37 +39,75 @@ export const FilterPlaceholder = ({ }: FilterPlaceholderProps) => { const [selected, setSelected] = useState(multiSelect ? [] : [options[0]?.id || '']); - const handleSelect = (optionId: string) => { - if (multiSelect) { - setSelected((prev) => - prev.includes(optionId) ? prev.filter((id) => id !== optionId) : [...prev, optionId] - ); - } else { - setSelected([optionId]); - } + const handleCheckboxChange = (optionId: string, checked: boolean) => { + setSelected((prev) => + checked ? [...prev, optionId] : prev.filter((id) => id !== optionId) + ); + }; + + const handleRadioChange = (value: string) => { + setSelected([value]); }; const isSelected = (optionId: string) => selected.includes(optionId); + if (multiSelect) { + // Checkbox mode for multi-select + return ( +
+ {options.map((option) => { + const checked = isSelected(option.id); + const inputId = `${groupId}-${option.id}`; + + return ( + + ); + })} +
+ ); + } + + // Radio mode for single-select return ( -
+ {options.map((option) => { const checked = isSelected(option.id); const inputId = `${groupId}-${option.id}`; return ( - + ); })} -
+ ); }; diff --git a/apps/landing/src/components/lab/GenerateFormPlaceholder.tsx b/apps/landing/src/components/lab/GenerateFormPlaceholder.tsx index d14e550..e35bb3d 100644 --- a/apps/landing/src/components/lab/GenerateFormPlaceholder.tsx +++ b/apps/landing/src/components/lab/GenerateFormPlaceholder.tsx @@ -5,18 +5,19 @@ * * Main content for lab generation interface. * Clean, work-focused design with compact layout. + * Uses Shadcn UI components with custom slate styling. * * Features: * - Compact info banner (dismissible) - * - Prompt textarea with keyboard shortcuts - * - Options grid (aspect ratio, style, advanced) + * - Prompt textarea with keyboard shortcuts (Shadcn Textarea) + * - Options grid with Shadcn Select dropdowns * - Results area with empty state * - Lucide icons throughout * * Color scheme: Forms use slate palette (original style) */ -import { useState, useRef, KeyboardEvent } from 'react'; +import { useState, KeyboardEvent } from 'react'; import { Info, X, @@ -30,6 +31,27 @@ import { Image, Layers, } from 'lucide-react'; +import { Button } from '@/components/ui/button'; +import { Textarea } from '@/components/ui/textarea'; +import { Input } from '@/components/ui/input'; +import { Label } from '@/components/ui/label'; +import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'; +import { + Select, + SelectContent, + SelectItem, + SelectTrigger, + SelectValue, +} from '@/components/ui/select'; + +// Shared styles for consistent theming +const selectTriggerStyles = "w-full bg-slate-800 border-slate-700 text-white focus:ring-purple-500 focus:ring-offset-0 hover:bg-slate-700/50"; +const selectContentStyles = "bg-slate-800 border-slate-700"; +const selectItemStyles = "text-white focus:bg-slate-700 focus:text-white"; +const inputStyles = "bg-slate-800 border-slate-700 text-white placeholder:text-gray-500 focus-visible:ring-purple-500 focus-visible:ring-offset-0"; +const textareaStyles = "bg-slate-800 border-slate-700 text-white placeholder:text-gray-500 focus-visible:ring-purple-500 focus-visible:ring-offset-0 resize-none"; +const labelStyles = "text-xs font-medium text-gray-400"; +const cardStyles = "p-4 bg-slate-900/80 backdrop-blur-sm border-slate-700 rounded-xl gap-0 py-4"; export const GenerateFormPlaceholder = () => { const [prompt, setPrompt] = useState(''); @@ -37,7 +59,6 @@ export const GenerateFormPlaceholder = () => { const [template, setTemplate] = useState('photorealistic'); const [generating, setGenerating] = useState(false); const [showBanner, setShowBanner] = useState(true); - const textareaRef = useRef(null); const handleGenerate = () => { if (!prompt.trim()) return; @@ -65,142 +86,130 @@ export const GenerateFormPlaceholder = () => { Experimental features enabled. API calls may be rate-limited.

- + )} {/* Generation Form Card */} -
- {/* Prompt Textarea */} -
- -