feat(admin): sidebar shell + routing

This commit is contained in:
Bas van Rossem
2026-06-17 19:03:35 +02:00
parent 77659edf8e
commit 286e2d29db
6 changed files with 346 additions and 6 deletions

View File

@@ -1,7 +1,21 @@
import { render, screen } from '@testing-library/react'; import { render, screen, within } from '@testing-library/react';
import { describe, expect, it } from 'vitest'; import userEvent from '@testing-library/user-event';
import { afterEach, describe, expect, it, vi } from 'vitest';
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'; import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
import App from './App'; import App from './App';
import { clearToken, getToken, setToken } from './lib/auth-storage';
// Stub the network layer so the authed shell renders without real requests.
// /api/me resolves an admin so the gate stays authed and the header shows the email.
vi.mock('./lib/api', async () => {
const actual = await vi.importActual<typeof import('./lib/api')>('./lib/api');
return {
...actual,
apiFetch: vi.fn().mockResolvedValue({
user: { id: 'u1', email: 'admin@solelog.local', name: 'Admin', role: 'admin' },
}),
};
});
function renderApp() { function renderApp() {
const queryClient = new QueryClient({ defaultOptions: { queries: { retry: false } } }); const queryClient = new QueryClient({ defaultOptions: { queries: { retry: false } } });
@@ -13,8 +27,30 @@ function renderApp() {
} }
describe('App', () => { describe('App', () => {
it('renders the admin app name', () => { afterEach(() => {
clearToken();
});
it('shows the login screen when there is no token', () => {
clearToken();
renderApp(); renderApp();
expect(screen.getByText('SoleLog Admin')).toBeInTheDocument(); expect(screen.getByText('E-mailadres')).toBeInTheDocument();
expect(screen.getByRole('button', { name: 'Inloggen' })).toBeInTheDocument();
});
it('shows the sidebar nav when a token is present', () => {
setToken('tok');
renderApp();
const nav = within(screen.getByRole('navigation'));
expect(nav.getByText('Live')).toBeInTheDocument();
expect(nav.getByText('Handelingen')).toBeInTheDocument();
});
it('clears the token when logout is clicked', async () => {
setToken('tok');
renderApp();
expect(getToken()).toBe('tok');
await userEvent.click(screen.getByRole('button', { name: 'Uitloggen' }));
expect(getToken()).toBeNull();
}); });
}); });

View File

@@ -1,9 +1,24 @@
import { BrowserRouter, Route, Routes } from 'react-router-dom';
import { AuthProvider, useAuth } from './auth/AuthContext'; import { AuthProvider, useAuth } from './auth/AuthContext';
import Login from './screens/Login'; import Login from './screens/Login';
import Sidebar from './components/Sidebar';
import Live from './screens/Live';
import Activities from './screens/Activities';
function AuthedShell() { function AuthedShell() {
// Placeholder shell — replaced by the sidebar + routing in Task 4. return (
return <div>SoleLog Admin</div>; <BrowserRouter>
<div className="admin-shell">
<Sidebar />
<main className="admin-main">
<Routes>
<Route path="/" element={<Live />} />
<Route path="/handelingen" element={<Activities />} />
</Routes>
</main>
</div>
</BrowserRouter>
);
} }
function Gate() { function Gate() {

View File

@@ -0,0 +1,52 @@
import { NavLink } from 'react-router-dom';
import { useAuth } from '../auth/AuthContext';
import { useMe } from '../api/me';
const navItems = [
{ to: '/', label: 'Live' },
{ to: '/handelingen', label: 'Handelingen' },
] as const;
// Sections planned for Phase 3b — shown muted/disabled as a hint of what's coming.
const soonItems = ['Rapporten', 'Gebruikers', 'Handmatig'] as const;
export default function Sidebar() {
const { signOut } = useAuth();
const meQuery = useMe();
const email = meQuery.data?.user.email;
return (
<aside className="sidebar">
<div className="sidebar-brand">SoleLog Admin</div>
<nav className="sidebar-nav">
{navItems.map((item) => (
<NavLink
key={item.to}
to={item.to}
end={item.to === '/'}
className={({ isActive }) => (isActive ? 'nav-link nav-link-active' : 'nav-link')}
>
{item.label}
</NavLink>
))}
<div className="nav-soon">
<span className="nav-soon-label">Binnenkort</span>
{soonItems.map((label) => (
<span key={label} className="nav-disabled" aria-disabled="true">
{label}
</span>
))}
</div>
</nav>
<div className="topbar">
{email && <span className="topbar-email">{email}</span>}
<button type="button" className="btn-logout" aria-label="Uitloggen" onClick={signOut}>
Uitloggen
</button>
</div>
</aside>
);
}

View File

@@ -0,0 +1,4 @@
// Placeholder — replaced by the activity-management view in Task 6.
export default function Activities() {
return <div className="screen">Handelingen</div>;
}

View File

@@ -0,0 +1,4 @@
// Placeholder — replaced by the live active-work view in Task 5.
export default function Live() {
return <div className="screen">Live</div>;
}

View File

@@ -0,0 +1,229 @@
:root {
--primary: #2563eb;
--primary-light: #eff6ff;
--text: #111827;
--text-muted: #6b7280;
--border: #e5e7eb;
--danger: #dc2626;
--amber: #d97706;
}
* {
box-sizing: border-box;
}
body {
margin: 0;
font-family:
'Inter',
system-ui,
-apple-system,
sans-serif;
color: var(--text);
background: #f9fafb;
}
/* ---- Admin shell layout (desktop) ---- */
.admin-shell {
display: grid;
grid-template-columns: 220px 1fr;
min-height: 100vh;
}
.admin-main {
padding: 32px;
overflow-y: auto;
}
.screen {
max-width: 960px;
margin: 0 auto;
}
.screen-title {
font-size: 28px;
font-weight: 600;
margin: 0 0 8px;
}
.screen-subtitle {
margin: 0 0 24px;
color: var(--text-muted);
font-size: 15px;
}
/* ---- Sidebar ---- */
.sidebar {
display: flex;
flex-direction: column;
gap: 24px;
padding: 24px 16px;
background: #ffffff;
border-right: 1px solid var(--border);
}
.sidebar-brand {
font-size: 18px;
font-weight: 700;
color: var(--primary);
padding: 0 8px;
}
.sidebar-nav {
display: flex;
flex-direction: column;
gap: 4px;
}
.nav-link {
display: block;
padding: 10px 12px;
font-size: 15px;
font-weight: 500;
text-decoration: none;
color: var(--text-muted);
border-radius: 10px;
}
.nav-link:hover {
background: #f3f4f6;
}
.nav-link-active {
color: var(--primary);
background: var(--primary-light);
}
.nav-soon {
display: flex;
flex-direction: column;
gap: 2px;
margin-top: 16px;
padding-top: 16px;
border-top: 1px solid var(--border);
}
.nav-soon-label {
font-size: 11px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.04em;
color: var(--text-muted);
padding: 0 12px 4px;
}
.nav-disabled {
display: block;
padding: 10px 12px;
font-size: 15px;
font-weight: 500;
color: var(--border);
cursor: not-allowed;
}
/* ---- Topbar (signed-in identity + logout) ---- */
.topbar {
margin-top: auto;
display: flex;
flex-direction: column;
gap: 12px;
padding-top: 16px;
border-top: 1px solid var(--border);
}
.topbar-email {
font-size: 13px;
color: var(--text-muted);
padding: 0 8px;
word-break: break-all;
}
.btn-logout {
width: 100%;
padding: 10px;
font-size: 14px;
font-weight: 600;
color: var(--danger);
background: #ffffff;
border: 1px solid var(--danger);
border-radius: 12px;
cursor: pointer;
}
/* ---- Login screen ---- */
.login {
max-width: 420px;
margin: 0 auto;
padding: 48px 24px;
display: flex;
flex-direction: column;
gap: 24px;
min-height: 100vh;
justify-content: center;
}
.login-title {
font-size: 36px;
font-weight: 600;
text-align: center;
color: var(--primary);
margin: 0;
}
.login-form {
display: flex;
flex-direction: column;
gap: 16px;
}
.field {
display: flex;
flex-direction: column;
gap: 6px;
}
.field-label {
font-size: 14px;
font-weight: 500;
color: var(--text-muted);
}
.field-input {
width: 100%;
padding: 14px 16px;
font-size: 16px;
border: 1px solid var(--border);
border-radius: 12px;
background: #ffffff;
color: var(--text);
}
.field-input:focus {
outline: none;
border-color: var(--primary);
}
.btn-primary {
width: 100%;
padding: 16px;
font-size: 16px;
font-weight: 600;
color: #ffffff;
background: var(--primary);
border: none;
border-radius: 16px;
cursor: pointer;
}
.btn-primary:disabled {
background: var(--border);
color: var(--text-muted);
cursor: not-allowed;
}
.login-error {
color: var(--danger);
font-size: 14px;
margin: 0;
text-align: center;
}