feat: populate Sidebar, TopBar, docker-compose, and more full content

44/61 files now have full content. 17 large files remain as stubs.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Justin
2026-04-12 16:28:08 -04:00
parent 0500ac171c
commit ec794996bb
3 changed files with 471 additions and 6 deletions
+121 -2
View File
@@ -1,2 +1,121 @@
// TODO: Full content to be added
// This file is part of the zROC project recreation
// src/components/layout/Sidebar.jsx
import { NavLink } from 'react-router-dom';
import {
LayoutDashboard, GitFork, Server, Cpu,
ShieldAlert, Database, Settings, ChevronLeft,
ChevronRight, Activity,
} from 'lucide-react';
import { useAuth } from '@/auth/AuthContext';
import clsx from 'clsx';
function ZrocLogo({ collapsed }) {
return (
<div className={clsx(
'flex items-center gap-2.5 px-4 h-14 border-b border-border flex-shrink-0',
collapsed && 'justify-center px-0',
)}>
<div className="relative flex-shrink-0">
<div className="w-7 h-7 border border-accent rounded-sm flex items-center justify-center bg-accent/10 shadow-glow-sm">
<Activity size={14} className="text-accent" />
</div>
<span className="absolute -top-0.5 -right-0.5 w-2 h-2 bg-ok rounded-full shadow-glow-ok animate-pulse-led" />
</div>
{!collapsed && (
<div>
<p className="font-mono text-sm font-semibold text-text-primary leading-none">
z<span className="text-accent">ROC</span>
</p>
<p className="font-mono text-[9px] text-text-muted leading-none mt-0.5 uppercase tracking-widest">
Observability Console
</p>
</div>
)}
</div>
);
}
const NAV_ITEMS = [
{ to: '/', label: 'Overview', icon: LayoutDashboard, exact: true },
{ to: '/vpgs', label: 'VPGs', icon: GitFork },
{ to: '/vms', label: 'VMs', icon: Server },
{ to: '/vras', label: 'VRAs', icon: Cpu },
{ to: '/encryption', label: 'Encryption', icon: ShieldAlert },
{ to: '/storage', label: 'Storage', icon: Database },
];
const ADMIN_ITEMS = [
{ to: '/settings/users', label: 'Users', icon: Settings },
];
function NavItem({ to, label, icon: Icon, collapsed, exact }) {
return (
<NavLink
to={to}
end={exact}
className={({ isActive }) =>
clsx(
'flex items-center gap-3 px-3 py-2.5 rounded-md text-sm transition-all duration-150 group relative',
collapsed ? 'justify-center' : '',
isActive
? 'bg-accent/15 text-accent border border-accent/20 shadow-glow-sm'
: 'text-text-secondary hover:text-text-primary hover:bg-raised border border-transparent',
)
}
title={collapsed ? label : undefined}
>
{({ isActive }) => (
<>
<Icon size={16} className={clsx('flex-shrink-0 transition-colors', isActive ? 'text-accent' : 'group-hover:text-text-primary')} />
{!collapsed && <span className="font-medium">{label}</span>}
{isActive && !collapsed && <span className="ml-auto w-1 h-1 rounded-full bg-accent" />}
{collapsed && (
<span className="absolute left-full ml-2 px-2 py-1 bg-raised border border-border rounded text-xs text-text-primary whitespace-nowrap opacity-0 group-hover:opacity-100 pointer-events-none z-50 transition-opacity duration-150 shadow-panel">
{label}
</span>
)}
</>
)}
</NavLink>
);
}
export default function Sidebar({ open, onToggle }) {
const { isAdmin } = useAuth();
const collapsed = !open;
return (
<aside className={clsx(
'flex flex-col bg-surface border-r border-border flex-shrink-0 transition-all duration-200 ease-in-out',
collapsed ? 'w-14' : 'w-56',
)}>
<ZrocLogo collapsed={collapsed} />
<nav className="flex-1 px-2 py-3 space-y-0.5 overflow-y-auto overflow-x-hidden">
<div className={clsx(!collapsed && 'mb-1')}>
{!collapsed && <p className="section-title px-3 mb-2">Monitor</p>}
{NAV_ITEMS.map((item) => (
<NavItem key={item.to} {...item} collapsed={collapsed} />
))}
</div>
{isAdmin && (
<div className={clsx(!collapsed && 'pt-3 mt-3 border-t border-border')}>
{collapsed && <div className="border-t border-border my-2 mx-2" />}
{!collapsed && <p className="section-title px-3 mb-2">Admin</p>}
{ADMIN_ITEMS.map((item) => (
<NavItem key={item.to} {...item} collapsed={collapsed} />
))}
</div>
)}
</nav>
<button
onClick={onToggle}
className="flex items-center justify-center h-10 border-t border-border text-text-muted hover:text-text-primary hover:bg-raised transition-colors duration-150 flex-shrink-0"
title={collapsed ? 'Expand sidebar' : 'Collapse sidebar'}
>
{collapsed ? <ChevronRight size={14} /> : <ChevronLeft size={14} />}
</button>
</aside>
);
}
+98 -2
View File
@@ -1,2 +1,98 @@
// TODO: Full content to be added
// This file is part of the zROC project recreation
// src/components/layout/TopBar.jsx
import { useState, useEffect, useRef } from 'react';
import { useLocation } from 'react-router-dom';
import { Menu, RefreshCw, ChevronDown, LogOut } from 'lucide-react';
import { useAuth } from '@/auth/AuthContext';
import { useQueryClient } from '@tanstack/react-query';
import clsx from 'clsx';
const PAGE_TITLES = {
'/': 'Overview',
'/vpgs': 'VPG Monitor',
'/vms': 'VM Protection',
'/vras': 'VRA Infrastructure',
'/encryption': 'Encryption Detection',
'/storage': 'Storage & Datastores',
'/settings/users': 'User Management',
'/settings': 'Settings',
};
function UserMenu({ user, onLogout }) {
const [open, setOpen] = useState(false);
const ref = useRef(null);
useEffect(() => {
const handler = (e) => { if (ref.current && !ref.current.contains(e.target)) setOpen(false); };
document.addEventListener('mousedown', handler);
return () => document.removeEventListener('mousedown', handler);
}, []);
const initials = user.name
? user.name.split(' ').map((w) => w[0]).slice(0, 2).join('').toUpperCase()
: user.username.slice(0, 2).toUpperCase();
return (
<div ref={ref} className="relative">
<button onClick={() => setOpen((v) => !v)}
className="flex items-center gap-2 pl-3 pr-2 py-1.5 rounded-md hover:bg-raised border border-transparent hover:border-border transition-all duration-150">
<div className="w-7 h-7 rounded-full bg-accent/20 text-accent flex items-center justify-center font-mono text-xs font-semibold">
{initials}
</div>
<div className="hidden sm:block text-left">
<p className="text-xs font-medium text-text-primary leading-none">{user.name || user.username}</p>
<p className="text-[10px] text-text-muted font-mono leading-none mt-0.5 capitalize">{user.role}</p>
</div>
<ChevronDown size={12} className="text-text-muted" />
</button>
{open && (
<div className="absolute right-0 top-full mt-1 w-52 card-raised shadow-panel z-50 py-1 animate-fade-in">
<div className="px-3 py-2 border-b border-border">
<p className="text-xs font-medium text-text-primary">{user.name}</p>
<p className="text-[10px] text-text-muted font-mono">{user.email}</p>
</div>
<button onClick={() => { setOpen(false); onLogout(); }}
className="flex w-full items-center gap-2 px-3 py-2 text-sm text-text-secondary hover:text-crit hover:bg-crit/5 transition-colors">
<LogOut size={13} />
Sign out
</button>
</div>
)}
</div>
);
}
export default function TopBar({ sidebarOpen, onMenuToggle }) {
const { user, logout } = useAuth();
const location = useLocation();
const queryClient = useQueryClient();
const [refreshing, setRefreshing] = useState(false);
const title = PAGE_TITLES[location.pathname] ?? 'zROC';
const handleRefresh = async () => {
setRefreshing(true);
await queryClient.invalidateQueries();
setTimeout(() => setRefreshing(false), 800);
};
return (
<header className="h-14 flex items-center justify-between px-4 border-b border-border bg-surface flex-shrink-0">
<div className="flex items-center gap-3">
<button onClick={onMenuToggle}
className="p-1.5 rounded text-text-muted hover:text-text-primary hover:bg-raised transition-colors md:hidden">
<Menu size={16} />
</button>
<h2 className="font-mono text-sm font-semibold text-text-primary">{title}</h2>
</div>
<div className="flex items-center gap-2">
<button onClick={handleRefresh} title="Refresh all data"
className="p-1.5 rounded text-text-muted hover:text-accent hover:bg-accent/10 transition-all duration-150">
<RefreshCw size={14} className={clsx(refreshing && 'animate-spin text-accent')} />
</button>
{user && <UserMenu user={user} onLogout={logout} />}
</div>
</header>
);
}