| 'use client' |
|
|
| import Link from 'next/link' |
| import { usePathname } from 'next/navigation' |
| import { LayoutDashboard, Plug, Wrench, Key, BarChart3, Settings, LogOut, ExternalLink } from 'lucide-react' |
| import { cn } from '@/lib/utils' |
|
|
| const navItems = [ |
| { href: '/', icon: LayoutDashboard, label: 'Dashboard' }, |
| { href: '/apps', icon: Plug, label: 'Apps' }, |
| { href: '/tools', icon: Wrench, label: 'Tools' }, |
| ] |
|
|
| const configItems = [ |
| { href: '/api-keys', icon: Key, label: 'API Keys' }, |
| { href: '/analytics', icon: BarChart3, label: 'Analytics' }, |
| { href: '/settings', icon: Settings, label: 'Settings' }, |
| ] |
|
|
| export function Sidebar({ integrationCount = 0 }: { integrationCount?: number }) { |
| const pathname = usePathname() |
|
|
| return ( |
| <aside className="w-60 bg-bg-secondary border-r border-border flex flex-col shrink-0"> |
| <div className="px-5 py-4 border-b border-border"> |
| <div className="flex items-center gap-3"> |
| <div className="w-8 h-8 bg-accent-cyan rounded-lg flex items-center justify-center font-bold text-bg-primary text-sm">C</div> |
| <span className="text-base font-semibold tracking-tight">Compost</span> |
| </div> |
| </div> |
| |
| <nav className="flex-1 px-3 py-4 space-y-1"> |
| <div className="px-3 mb-2 text-[11px] font-semibold uppercase tracking-wider text-text-dim">Main</div> |
| {navItems.map((item) => ( |
| <Link key={item.href} href={item.href} |
| className={cn( |
| 'flex items-center gap-3 px-3 py-2 rounded-lg text-sm transition-colors', |
| pathname === item.href |
| ? 'bg-accent-cyan/10 text-accent-cyanLight font-medium' |
| : 'text-text-secondary hover:bg-bg-hover hover:text-text-primary' |
| )} |
| > |
| <item.icon className="w-4 h-4" /> |
| {item.label} |
| {item.href === '/apps' && integrationCount > 0 && ( |
| <span className="ml-auto text-[11px] bg-bg-tertiary text-text-muted px-1.5 py-0.5 rounded">{integrationCount}</span> |
| )} |
| </Link> |
| ))} |
| |
| <div className="px-3 mt-6 mb-2 text-[11px] font-semibold uppercase tracking-wider text-text-dim">Configuration</div> |
| {configItems.map((item) => ( |
| <Link key={item.href} href={item.href} |
| className={cn( |
| 'flex items-center gap-3 px-3 py-2 rounded-lg text-sm transition-colors', |
| pathname === item.href |
| ? 'bg-accent-cyan/10 text-accent-cyanLight font-medium' |
| : 'text-text-secondary hover:bg-bg-hover hover:text-text-primary' |
| )} |
| > |
| <item.icon className="w-4 h-4" /> |
| {item.label} |
| </Link> |
| ))} |
| |
| <a href="/api/docs" target="_blank" rel="noopener noreferrer" |
| className="flex items-center gap-3 px-3 py-2 rounded-lg text-sm text-text-secondary hover:bg-bg-hover hover:text-text-primary transition-colors" |
| > |
| <ExternalLink className="w-4 h-4" /> |
| API Docs |
| </a> |
| </nav> |
| |
| <div className="px-3 py-4 border-t border-border"> |
| <div className="flex items-center justify-between"> |
| <div className="flex items-center gap-2 min-w-0"> |
| <div className="w-7 h-7 bg-bg-tertiary rounded-full flex items-center justify-center text-xs font-semibold shrink-0"> |
| {typeof window !== 'undefined' ? (localStorage.getItem('email')?.[0]?.toUpperCase() || 'U') : 'U'} |
| </div> |
| <span className="text-xs text-text-secondary truncate"> |
| {typeof window !== 'undefined' ? localStorage.getItem('email') || 'User' : 'User'} |
| </span> |
| </div> |
| <button onClick={() => { localStorage.removeItem('token'); localStorage.removeItem('email'); window.location.href = '/login' }} |
| className="p-1.5 text-text-muted hover:text-accent-red hover:bg-accent-red/10 rounded-lg transition-colors" |
| > |
| <LogOut className="w-3.5 h-3.5" /> |
| </button> |
| </div> |
| </div> |
| </aside> |
| ) |
| } |