Agent-completed splits committed after agents hit rate limits before committing their work. All 4 pages now under 500 LOC: - consent-management: 1303 -> 193 LOC (+ 7 _components, _hooks, _data, _types) - control-library: 1210 -> 298 LOC (+ _components, _types) - incidents: 1150 -> 373 LOC (+ _components) - training: 1127 -> 366 LOC (+ _components) Verification: next build clean (142 pages generated). Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
104 lines
5.7 KiB
TypeScript
104 lines
5.7 KiB
TypeScript
'use client'
|
|
|
|
import { useState } from 'react'
|
|
import { updateModule, deleteModule } from '@/lib/sdk/training/api'
|
|
import type { TrainingModule } from '@/lib/sdk/training/types'
|
|
import { REGULATION_LABELS, REGULATION_COLORS } from '@/lib/sdk/training/types'
|
|
|
|
export function ModuleEditDrawer({ module, onClose, onSaved }: { module: TrainingModule; onClose: () => void; onSaved: () => void }) {
|
|
const [title, setTitle] = useState(module.title)
|
|
const [description, setDescription] = useState(module.description || '')
|
|
const [durationMinutes, setDurationMinutes] = useState(module.duration_minutes)
|
|
const [passThreshold, setPassThreshold] = useState(module.pass_threshold)
|
|
const [isActive, setIsActive] = useState(module.is_active)
|
|
const [saving, setSaving] = useState(false)
|
|
const [deleting, setDeleting] = useState(false)
|
|
const [error, setError] = useState<string | null>(null)
|
|
|
|
const handleSave = async () => {
|
|
setSaving(true)
|
|
setError(null)
|
|
try {
|
|
await updateModule(module.id, { title, description, duration_minutes: durationMinutes, pass_threshold: passThreshold, is_active: isActive })
|
|
onSaved()
|
|
} catch (e) {
|
|
setError(e instanceof Error ? e.message : 'Fehler beim Speichern')
|
|
} finally {
|
|
setSaving(false)
|
|
}
|
|
}
|
|
|
|
const handleDelete = async () => {
|
|
if (!window.confirm(`Modul "${module.title}" wirklich loeschen?`)) return
|
|
setDeleting(true)
|
|
try {
|
|
await deleteModule(module.id)
|
|
onSaved()
|
|
} catch (e) {
|
|
setError(e instanceof Error ? e.message : 'Fehler beim Loeschen')
|
|
setDeleting(false)
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="fixed inset-0 bg-black/50 z-50 flex items-start justify-end">
|
|
<div className="h-full w-full max-w-lg bg-white shadow-xl flex flex-col">
|
|
<div className="flex items-center justify-between p-6 border-b border-gray-200">
|
|
<div>
|
|
<div className="flex items-center gap-2">
|
|
<span className={`px-2 py-0.5 rounded text-xs font-medium ${REGULATION_COLORS[module.regulation_area]?.bg || 'bg-gray-100'} ${REGULATION_COLORS[module.regulation_area]?.text || 'text-gray-700'}`}>
|
|
{REGULATION_LABELS[module.regulation_area] || module.regulation_area}
|
|
</span>
|
|
{module.nis2_relevant && <span className="text-xs px-1.5 py-0.5 bg-purple-100 text-purple-700 rounded">NIS2</span>}
|
|
</div>
|
|
<h2 className="text-lg font-semibold text-gray-900 mt-1">{module.module_code}</h2>
|
|
</div>
|
|
<button onClick={onClose} className="text-gray-400 hover:text-gray-600">
|
|
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
<div className="flex-1 overflow-y-auto p-6 space-y-4">
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">Titel</label>
|
|
<input type="text" value={title} onChange={e => setTitle(e.target.value)} className="w-full px-3 py-2 border border-gray-300 rounded-lg text-sm focus:ring-2 focus:ring-blue-500 focus:border-blue-500" />
|
|
</div>
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">Beschreibung</label>
|
|
<textarea value={description} onChange={e => setDescription(e.target.value)} rows={3} className="w-full px-3 py-2 border border-gray-300 rounded-lg text-sm focus:ring-2 focus:ring-blue-500 focus:border-blue-500" />
|
|
</div>
|
|
<div className="grid grid-cols-2 gap-4">
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">Dauer (Minuten)</label>
|
|
<input type="number" min={1} value={durationMinutes} onChange={e => setDurationMinutes(Number(e.target.value))} className="w-full px-3 py-2 border border-gray-300 rounded-lg text-sm focus:ring-2 focus:ring-blue-500 focus:border-blue-500" />
|
|
</div>
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">Bestehensgrenze (%)</label>
|
|
<input type="number" min={0} max={100} value={passThreshold} onChange={e => setPassThreshold(Number(e.target.value))} className="w-full px-3 py-2 border border-gray-300 rounded-lg text-sm focus:ring-2 focus:ring-blue-500 focus:border-blue-500" />
|
|
</div>
|
|
</div>
|
|
<div className="flex items-center justify-between p-3 bg-gray-50 rounded-lg">
|
|
<span className="text-sm font-medium text-gray-700">Modul aktiv</span>
|
|
<button
|
|
onClick={() => setIsActive(!isActive)}
|
|
className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${isActive ? 'bg-blue-600' : 'bg-gray-200'}`}
|
|
>
|
|
<span className={`inline-block h-4 w-4 transform rounded-full bg-white transition-transform ${isActive ? 'translate-x-6' : 'translate-x-1'}`} />
|
|
</button>
|
|
</div>
|
|
{error && <p className="text-sm text-red-600">{error}</p>}
|
|
</div>
|
|
<div className="p-6 border-t border-gray-200 space-y-3">
|
|
<button onClick={handleSave} disabled={saving} className="w-full px-4 py-2 text-sm bg-blue-600 text-white rounded-lg hover:bg-blue-700 disabled:opacity-50 transition-colors">
|
|
{saving ? 'Speichern...' : 'Aenderungen speichern'}
|
|
</button>
|
|
<button onClick={handleDelete} disabled={deleting} className="w-full px-4 py-2 text-sm bg-red-50 text-red-600 border border-red-200 rounded-lg hover:bg-red-100 disabled:opacity-50 transition-colors">
|
|
{deleting ? 'Loeschen...' : 'Modul loeschen'}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|