This repository has been archived on 2026-02-15. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
breakpilot-pwa/studio-v2/components/Layout.tsx
Benjamin Admin bfdaf63ba9 fix: Restore all files lost during destructive rebase
A previous `git pull --rebase origin main` dropped 177 local commits,
losing 3400+ files across admin-v2, backend, studio-v2, website,
klausur-service, and many other services. The partial restore attempt
(660295e2) only recovered some files.

This commit restores all missing files from pre-rebase ref 98933f5e
while preserving post-rebase additions (night-scheduler, night-mode UI,
NightModeWidget dashboard integration).

Restored features include:
- AI Module Sidebar (FAB), OCR Labeling, OCR Compare
- GPU Dashboard, RAG Pipeline, Magic Help
- Klausur-Korrektur (8 files), Abitur-Archiv (5+ files)
- Companion, Zeugnisse-Crawler, Screen Flow
- Full backend, studio-v2, website, klausur-service
- All compliance SDKs, agent-core, voice-service
- CI/CD configs, documentation, scripts

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-09 09:51:32 +01:00

194 lines
7.0 KiB
TypeScript

'use client'
import { useState, ReactNode } from 'react'
import Link from 'next/link'
import { usePathname } from 'next/navigation'
// ============================================
// TYPES
// ============================================
interface NavItem {
id: string
name: string
href: string
icon: ReactNode
description?: string
}
// ============================================
// NAVIGATION - Hier werden alle Tabs definiert
// ============================================
const navigation: NavItem[] = [
{
id: 'magic-help',
name: 'Magic Help',
href: '/magic-help',
icon: (
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 3v4M3 5h4M6 17v4m-2-2h4m5-16l2.286 6.857L21 12l-5.714 2.143L13 21l-2.286-6.857L5 12l5.714-2.143L13 3z" />
</svg>
),
description: 'Handschrift-OCR & Klausur-Korrektur',
},
{
id: 'meet',
name: 'Meet',
href: '/meet',
icon: (
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 10l4.553-2.276A1 1 0 0121 8.618v6.764a1 1 0 01-1.447.894L15 14M5 18h8a2 2 0 002-2V8a2 2 0 00-2-2H5a2 2 0 00-2 2v8a2 2 0 002 2z" />
</svg>
),
description: 'Videokonferenzen & Meetings',
},
]
// ============================================
// LAYOUT COMPONENT
// ============================================
interface LayoutProps {
children: ReactNode
}
export default function Layout({ children }: LayoutProps) {
const pathname = usePathname()
const [sidebarCollapsed, setSidebarCollapsed] = useState(false)
const isActive = (href: string) => {
if (href === '/') return pathname === '/'
return pathname.startsWith(href)
}
return (
<div className="min-h-screen flex flex-col">
{/* ============================================
HEADER
============================================ */}
<header className="h-14 bg-white border-b border-slate-200 flex items-center justify-between px-4 fixed top-0 left-0 right-0 z-30">
{/* Logo */}
<div className="flex items-center gap-3">
<div className="w-8 h-8 bg-primary-500 rounded-lg flex items-center justify-center text-white font-bold text-sm">
BP
</div>
<div>
<div className="font-bold text-primary-500 text-lg leading-none">BreakPilot</div>
<div className="text-xs text-slate-400">Studio v2</div>
</div>
</div>
{/* Spacer - hier kommen später weitere Header-Elemente */}
<div className="flex-1" />
{/* Platzhalter für spätere Features (Login, Theme, Language) */}
<div className="flex items-center gap-2 text-sm text-slate-400">
<span className="px-3 py-1 bg-slate-100 rounded-full">Preview Build</span>
</div>
</header>
<div className="flex flex-1 pt-14">
{/* ============================================
SIDEBAR
============================================ */}
<aside
className={`${
sidebarCollapsed ? 'w-16' : 'w-64'
} bg-slate-900 text-white flex flex-col transition-all duration-200 fixed left-0 top-14 bottom-0 z-20`}
>
{/* Collapse Button */}
<div className="p-2 border-b border-slate-700">
<button
onClick={() => setSidebarCollapsed(!sidebarCollapsed)}
className="w-full p-2 rounded-lg hover:bg-slate-800 transition-colors flex items-center justify-center"
title={sidebarCollapsed ? 'Sidebar erweitern' : 'Sidebar einklappen'}
>
<svg
className={`w-5 h-5 transition-transform ${sidebarCollapsed ? 'rotate-180' : ''}`}
fill="none"
stroke="currentColor"
viewBox="0 0 24 24"
>
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M11 19l-7-7 7-7m8 14l-7-7 7-7" />
</svg>
</button>
</div>
{/* Navigation */}
<nav className="flex-1 py-4 overflow-y-auto">
<ul className="space-y-1 px-2">
{navigation.map((item) => (
<li key={item.id}>
<Link
href={item.href}
className={`flex items-center gap-3 px-3 py-2.5 rounded-lg transition-colors ${
isActive(item.href)
? 'bg-primary-500 text-white'
: 'text-slate-300 hover:bg-slate-800 hover:text-white'
}`}
title={sidebarCollapsed ? item.name : undefined}
>
<span className="flex-shrink-0">{item.icon}</span>
{!sidebarCollapsed && (
<div className="flex-1 min-w-0">
<span className="block truncate">{item.name}</span>
{item.description && (
<span className="block text-xs text-slate-400 truncate">{item.description}</span>
)}
</div>
)}
</Link>
</li>
))}
</ul>
{/* Hinweis wenn keine weiteren Tabs */}
{navigation.length === 1 && !sidebarCollapsed && (
<div className="mx-4 mt-6 p-3 bg-slate-800 rounded-lg text-xs text-slate-400">
<p className="font-medium text-slate-300 mb-1">Schritt für Schritt</p>
<p>Weitere Module werden nach und nach hinzugefügt.</p>
</div>
)}
</nav>
{/* Sidebar Footer */}
<div className="p-4 border-t border-slate-700">
{!sidebarCollapsed && (
<div className="text-xs text-slate-500">
<p>Studio v2 - Build #1</p>
<p className="mt-1">Port 3001</p>
</div>
)}
</div>
</aside>
{/* ============================================
MAIN CONTENT
============================================ */}
<main
className={`flex-1 transition-all duration-200 ${
sidebarCollapsed ? 'ml-16' : 'ml-64'
}`}
>
<div className="p-6 min-h-[calc(100vh-3.5rem-3rem)]">
{children}
</div>
{/* ============================================
FOOTER
============================================ */}
<footer className="h-12 bg-white border-t border-slate-200 flex items-center justify-between px-6 text-sm text-slate-500">
<div>BreakPilot Studio v2</div>
<div className="flex items-center gap-4">
<span>Port 3001</span>
<span className="text-slate-300">|</span>
<span>Backend: Port 8000</span>
</div>
</footer>
</main>
</div>
</div>
)
}