- LoadingState: bg-kodo-slate → bg-muted for skeleton variant - PlayerLoading: fullScreen overlay bg-black/50 → bg-background/80 backdrop-blur-sm - Header: bg-white/5 → bg-muted/30, border-white/* → border-border, focus:ring-ring - Sidebar: overlay bg-black/60 → bg-background/80, hover:text-white → hover:text-foreground - Navbar: text-white → text-foreground, ring-white/5 → ring-border Co-authored-by: Cursor <cursoragent@cursor.com>
277 lines
12 KiB
TypeScript
277 lines
12 KiB
TypeScript
import React from 'react';
|
|
import { useNavigate, useLocation, Link } from 'react-router-dom';
|
|
import {
|
|
Home, Users, Disc, Radio, Settings, LogOut, ShoppingBag,
|
|
GraduationCap, BarChart2, Shield, Box, MessageSquare, Cloud,
|
|
Layers, Cpu, Heart, ListMusic, CreditCard, DollarSign, Terminal,
|
|
ChevronLeft, ChevronRight,
|
|
} from 'lucide-react';
|
|
import { NavItem } from '../../types';
|
|
import { useAuthStore } from '@/features/auth/store/authStore';
|
|
import { useUIStore } from '@/stores/ui';
|
|
import { cn } from '@/lib/utils';
|
|
import { Button } from '@/components/ui/button';
|
|
import { Tooltip } from '@/components/ui/tooltip';
|
|
|
|
interface SidebarProps {
|
|
currentView?: string;
|
|
onNavigate?: (viewId: string) => void;
|
|
onLogout?: () => void;
|
|
}
|
|
|
|
const navItems: { section: string; items: NavItem[] }[] = [
|
|
{
|
|
section: 'My Studio',
|
|
items: [
|
|
{ id: 'dashboard', label: 'Command Center', icon: <Home className="w-4 h-4" /> },
|
|
{ id: 'studio', label: 'Cloud Files', icon: <Cloud className="w-4 h-4" /> },
|
|
{ id: 'tracks', label: 'Projects', icon: <Layers className="w-4 h-4" /> },
|
|
{ id: 'gear', label: 'Gear Locker', icon: <Box className="w-4 h-4" /> },
|
|
{ id: 'analytics', label: 'Performance', icon: <BarChart2 className="w-4 h-4" /> },
|
|
],
|
|
},
|
|
{
|
|
section: 'Veza Network',
|
|
items: [
|
|
{ id: 'social', label: 'Community Feed', icon: <Users className="w-4 h-4" /> },
|
|
{ id: 'marketplace', label: 'Marketplace', icon: <ShoppingBag className="w-4 h-4" /> },
|
|
{ id: 'live', label: 'Live Sessions', icon: <Radio className="w-4 h-4" />, badge: 3 },
|
|
{ id: 'chat', label: 'Channels', icon: <MessageSquare className="w-4 h-4" />, badge: 12 },
|
|
{ id: 'education', label: 'Academy', icon: <GraduationCap className="w-4 h-4" /> },
|
|
],
|
|
},
|
|
{
|
|
section: 'Commerce',
|
|
items: [
|
|
{ id: 'sell', label: 'Seller Dashboard', icon: <DollarSign className="w-4 h-4" /> },
|
|
{ id: 'wishlist', label: 'Wishlist', icon: <Heart className="w-4 h-4" /> },
|
|
{ id: 'purchases', label: 'Purchases', icon: <CreditCard className="w-4 h-4" /> },
|
|
],
|
|
},
|
|
{
|
|
section: 'Library',
|
|
items: [
|
|
{ id: 'playlists', label: 'Playlists', icon: <ListMusic className="w-4 h-4" /> },
|
|
{ id: 'queue', label: 'Play Queue', icon: <Disc className="w-4 h-4" /> },
|
|
],
|
|
},
|
|
{
|
|
section: 'System',
|
|
items: [
|
|
{ id: 'developer', label: 'Developer API', icon: <Terminal className="w-4 h-4" /> },
|
|
{ id: 'admin', label: 'Admin Panel', icon: <Shield className="w-4 h-4" /> },
|
|
],
|
|
},
|
|
];
|
|
|
|
const routeMap: Record<string, string> = {
|
|
dashboard: '/dashboard', studio: '/library', tracks: '/library', gear: '/gear',
|
|
analytics: '/analytics', social: '/social', marketplace: '/marketplace', live: '/live',
|
|
chat: '/chat', education: '/education', sell: '/sell', wishlist: '/wishlist',
|
|
purchases: '/purchases', playlists: '/playlists', queue: '/queue', developer: '/developer',
|
|
admin: '/admin', settings: '/settings',
|
|
};
|
|
|
|
export const Sidebar: React.FC<SidebarProps> = ({ currentView, onNavigate, onLogout }) => {
|
|
const navigate = useNavigate();
|
|
const location = useLocation();
|
|
const { logout } = useAuthStore();
|
|
const { sidebarOpen, setSidebarOpen } = useUIStore();
|
|
|
|
const handleMobileNav = () => {
|
|
if (window.innerWidth < 1024) setSidebarOpen(false);
|
|
};
|
|
|
|
const activeView = currentView || Object.keys(routeMap).find((key) => routeMap[key] === location.pathname) || 'dashboard';
|
|
|
|
const handleLogout = () => {
|
|
logout();
|
|
navigate('/login');
|
|
onLogout?.();
|
|
};
|
|
|
|
return (
|
|
<>
|
|
{sidebarOpen && (
|
|
<div
|
|
className="fixed inset-0 bg-background/80 backdrop-blur-sm lg:hidden z-sidebar-overlay"
|
|
onClick={() => setSidebarOpen(false)}
|
|
aria-hidden="true"
|
|
/>
|
|
)}
|
|
|
|
<aside
|
|
data-testid="app-sidebar"
|
|
className={cn(
|
|
'fixed left-sidebar bottom-sidebar top-sidebar rounded-xl flex flex-col transition-shell z-sidebar overflow-hidden bg-[var(--sidebar)] backdrop-blur-md',
|
|
sidebarOpen ? 'w-sidebar-expanded translate-x-0 opacity-100' : '-translate-x-full lg:translate-x-0 lg:opacity-100 lg:w-sidebar-collapsed'
|
|
)}
|
|
>
|
|
{/* Header — minimal Spotify-style */}
|
|
<div className="px-4 py-4 flex items-center gap-3 relative">
|
|
<div className="w-8 h-8 rounded-lg bg-sidebar-accent flex items-center justify-center flex-shrink-0">
|
|
<Cpu className="w-4 h-4 text-muted-foreground" />
|
|
</div>
|
|
|
|
<div className={cn("transition-shell overflow-hidden min-w-0", sidebarOpen ? "opacity-100" : "w-0 opacity-0")}>
|
|
<h2 className="text-sm font-semibold text-foreground truncate">
|
|
System Hub
|
|
</h2>
|
|
<div className="flex items-center gap-1.5 mt-0.5">
|
|
<span className="w-1.5 h-1.5 rounded-full bg-primary shrink-0 animate-pulse" />
|
|
<span className="text-xs text-muted-foreground truncate">
|
|
Online
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<Button
|
|
variant="ghost"
|
|
size="icon"
|
|
onClick={() => setSidebarOpen(!sidebarOpen)}
|
|
className={cn(
|
|
"ml-auto text-muted-foreground hover:text-foreground hidden lg:flex hover:bg-sidebar-accent",
|
|
!sidebarOpen && "absolute left-1/2 -translate-x-1/2 top-1/2 -translate-y-1/2"
|
|
)}
|
|
>
|
|
{sidebarOpen ? <ChevronLeft className="w-4 h-4" /> : <ChevronRight className="w-4 h-4" />}
|
|
</Button>
|
|
</div>
|
|
|
|
{/* Nav — Discord/Spotify: pill indicator, micro-animations, section dividers */}
|
|
<div className="flex-1 overflow-y-auto custom-scrollbar px-3 py-2">
|
|
{navItems.map((group, idx) => (
|
|
<div key={idx}>
|
|
{/* Section divider between groups */}
|
|
{idx > 0 && (
|
|
<div className={cn(
|
|
"h-px bg-border/50 mx-3 my-1.5 transition-opacity duration-[var(--duration-normal)]",
|
|
!sidebarOpen && "mx-1"
|
|
)} />
|
|
)}
|
|
|
|
<h3 className={cn(
|
|
"text-xs font-medium text-muted-foreground mb-2 px-3 transition-all duration-[var(--duration-normal)] uppercase tracking-wider",
|
|
!sidebarOpen && "opacity-0 h-0 overflow-hidden mb-0 px-0"
|
|
)}>
|
|
{group.section}
|
|
</h3>
|
|
|
|
<div className="space-y-0.5">
|
|
{group.items.map((item) => {
|
|
const route = routeMap[item.id] || '/dashboard';
|
|
const isActive = activeView === item.id;
|
|
|
|
return (
|
|
<Tooltip key={item.id} content={item.label} position="right" disabled={sidebarOpen}>
|
|
<Link
|
|
to={route}
|
|
onClick={() => {
|
|
handleMobileNav();
|
|
onNavigate?.(item.id);
|
|
}}
|
|
className={cn(
|
|
'w-full flex items-center px-3 py-2 rounded-lg text-sm font-medium transition-all duration-[var(--duration-fast)] group relative',
|
|
'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-primary/50 focus-visible:ring-offset-2 focus-visible:ring-offset-background',
|
|
isActive
|
|
? 'bg-primary/10 text-primary'
|
|
: 'text-muted-foreground hover:text-foreground hover:bg-sidebar-accent active:bg-sidebar-accent/80',
|
|
!sidebarOpen && "justify-center px-0"
|
|
)}
|
|
>
|
|
{/* Discord-style active pill indicator */}
|
|
{isActive && (
|
|
<span className="absolute left-0 top-1/2 -translate-y-1/2 w-1 h-5 rounded-full bg-primary transition-all duration-[var(--duration-normal)]" />
|
|
)}
|
|
|
|
<div className={cn("flex items-center gap-3 relative z-10 min-w-0", !sidebarOpen && "justify-center")}>
|
|
<span className={cn(
|
|
'shrink-0 transition-all duration-[var(--duration-fast)]',
|
|
'group-hover:scale-110',
|
|
isActive ? 'text-primary' : 'text-muted-foreground group-hover:text-foreground'
|
|
)}>
|
|
{item.icon}
|
|
</span>
|
|
|
|
<span className={cn(
|
|
"transition-all duration-[var(--duration-normal)] whitespace-nowrap truncate",
|
|
sidebarOpen ? "opacity-100" : "w-0 opacity-0 overflow-hidden"
|
|
)}>
|
|
{item.label}
|
|
</span>
|
|
</div>
|
|
|
|
{/* Badge — expanded: colored pill */}
|
|
{item.badge != null && sidebarOpen && (
|
|
<span className="ml-auto flex h-5 min-w-5 items-center justify-center rounded-full bg-primary/15 text-primary text-xs font-semibold tabular-nums shrink-0">
|
|
{item.badge}
|
|
</span>
|
|
)}
|
|
|
|
{/* Badge — collapsed: pinging dot */}
|
|
{item.badge != null && !sidebarOpen && (
|
|
<span className="absolute top-1.5 right-1.5 flex h-2 w-2">
|
|
<span className="animate-ping absolute inline-flex h-full w-full rounded-full bg-primary opacity-75" />
|
|
<span className="relative inline-flex rounded-full h-2 w-2 bg-primary" />
|
|
</span>
|
|
)}
|
|
</Link>
|
|
</Tooltip>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
|
|
{/* Footer */}
|
|
<div className="p-2 border-t border-sidebar-border space-y-0.5">
|
|
<Tooltip content="Settings" position="right" disabled={sidebarOpen}>
|
|
<Link
|
|
to="/settings"
|
|
onClick={() => { handleMobileNav(); onNavigate?.('settings'); }}
|
|
className={cn(
|
|
'flex items-center gap-3 px-3 py-2 rounded-lg text-sm font-medium transition-all duration-[var(--duration-fast)] group relative',
|
|
'focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-primary/50 focus-visible:ring-offset-2 focus-visible:ring-offset-background',
|
|
activeView === 'settings'
|
|
? 'bg-primary/10 text-primary'
|
|
: 'text-muted-foreground hover:text-foreground hover:bg-sidebar-accent active:bg-sidebar-accent/80',
|
|
!sidebarOpen && "justify-center px-0"
|
|
)}
|
|
>
|
|
{activeView === 'settings' && (
|
|
<span className="absolute left-0 top-1/2 -translate-y-1/2 w-1 h-5 rounded-full bg-primary transition-all duration-[var(--duration-normal)]" />
|
|
)}
|
|
<Settings className={cn(
|
|
"w-4 h-4 shrink-0 transition-all duration-[var(--duration-fast)]",
|
|
"group-hover:scale-110",
|
|
activeView === 'settings' ? 'text-primary' : 'text-muted-foreground group-hover:text-foreground'
|
|
)} />
|
|
<span className={cn(
|
|
"truncate transition-all duration-[var(--duration-normal)]",
|
|
sidebarOpen ? "opacity-100" : "w-0 opacity-0 overflow-hidden"
|
|
)}>Settings</span>
|
|
</Link>
|
|
</Tooltip>
|
|
|
|
<Tooltip content="Sign Out" position="right" disabled={sidebarOpen}>
|
|
<Button
|
|
variant="ghost"
|
|
onClick={handleLogout}
|
|
className={cn(
|
|
"w-full text-muted-foreground hover:text-destructive hover:bg-destructive/10 gap-3 justify-start rounded-lg group",
|
|
!sidebarOpen && "justify-center px-0"
|
|
)}
|
|
>
|
|
<LogOut className="w-4 h-4 shrink-0 transition-transform duration-[var(--duration-fast)] group-hover:scale-110" />
|
|
<span className={cn(
|
|
"whitespace-nowrap transition-all duration-[var(--duration-normal)]",
|
|
sidebarOpen ? "opacity-100" : "w-0 opacity-0 overflow-hidden"
|
|
)}>Sign Out</span>
|
|
</Button>
|
|
</Tooltip>
|
|
</div>
|
|
</aside>
|
|
</>
|
|
);
|
|
};
|