Compost / frontend /_src /components /sidebar.tsx
abc1181's picture
Fix health endpoint routing order
de886f2
Raw
History Blame Contribute Delete
4.08 kB
'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>
)
}