Files
breakpilot-lehrer/studio-v2/app/korrektur/[klausurId]/_components/UploadModal.tsx
Benjamin Admin 451365a312 [split-required] Split remaining 500-680 LOC files (final batch)
website (17 pages + 3 components):
- multiplayer/wizard, middleware/wizard+test-wizard, communication
- builds/wizard, staff-search, voice, sbom/wizard
- foerderantrag, mail/tasks, tools/communication, sbom
- compliance/evidence, uni-crawler, brandbook (already done)
- CollectionsTab, IngestionTab, RiskHeatmap

backend-lehrer (5 files):
- letters_api (641 → 2), certificates_api (636 → 2)
- alerts_agent/db/models (636 → 3)
- llm_gateway/communication_service (614 → 2)
- game/database already done in prior batch

klausur-service (2 files):
- hybrid_vocab_extractor (664 → 2)
- klausur-service/frontend: api.ts (620 → 3), EHUploadWizard (591 → 2)

voice-service (3 files):
- bqas/rag_judge (618 → 3), runner (529 → 2)
- enhanced_task_orchestrator (519 → 2)

studio-v2 (6 files):
- korrektur/[klausurId] (578 → 4), fairness (569 → 2)
- AlertsWizard (552 → 2), OnboardingWizard (513 → 2)
- korrektur/api.ts (506 → 3), geo-lernwelt (501 → 2)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-25 08:56:45 +02:00

153 lines
6.1 KiB
TypeScript

'use client'
import { useState, useRef } from 'react'
import { GlassCard } from './GlassCard'
interface UploadModalProps {
isOpen: boolean
onClose: () => void
onUpload: (files: File[], anonymIds: string[]) => void
isUploading: boolean
}
export function UploadModal({ isOpen, onClose, onUpload, isUploading }: UploadModalProps) {
const [files, setFiles] = useState<File[]>([])
const [anonymIds, setAnonymIds] = useState<string[]>([])
const fileInputRef = useRef<HTMLInputElement>(null)
if (!isOpen) return null
const handleFileSelect = (selectedFiles: FileList | null) => {
if (!selectedFiles) return
const newFiles = Array.from(selectedFiles)
setFiles((prev) => [...prev, ...newFiles])
setAnonymIds((prev) => [
...prev,
...newFiles.map((_, i) => `Arbeit-${prev.length + i + 1}`),
])
}
const handleDrop = (e: React.DragEvent) => {
e.preventDefault()
handleFileSelect(e.dataTransfer.files)
}
const removeFile = (index: number) => {
setFiles((prev) => prev.filter((_, i) => i !== index))
setAnonymIds((prev) => prev.filter((_, i) => i !== index))
}
const updateAnonymId = (index: number, value: string) => {
setAnonymIds((prev) => {
const updated = [...prev]
updated[index] = value
return updated
})
}
const handleSubmit = () => {
if (files.length > 0) {
onUpload(files, anonymIds)
}
}
return (
<div className="fixed inset-0 z-50 flex items-center justify-center p-4">
<div className="absolute inset-0 bg-black/50 backdrop-blur-sm" onClick={onClose} />
<GlassCard className="relative w-full max-w-2xl max-h-[80vh] overflow-hidden" size="lg" delay={0}>
<div className="flex items-center justify-between mb-6">
<h2 className="text-xl font-bold text-white">Arbeiten hochladen</h2>
<button
onClick={onClose}
className="p-2 rounded-lg hover:bg-white/10 text-white/60 transition-colors"
>
<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>
{/* Drop Zone */}
<div
className="border-2 border-dashed border-white/20 rounded-2xl p-8 text-center cursor-pointer transition-all hover:border-purple-400/50 hover:bg-white/5 mb-6"
onDrop={handleDrop}
onDragOver={(e) => e.preventDefault()}
onClick={() => fileInputRef.current?.click()}
>
<input
ref={fileInputRef}
type="file"
accept="image/*,.pdf"
multiple
onChange={(e) => handleFileSelect(e.target.files)}
className="hidden"
/>
<svg className="w-12 h-12 mx-auto mb-3 text-white/30" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M7 16a4 4 0 01-.88-7.903A5 5 0 1115.9 6L16 6a5 5 0 011 9.9M15 13l-3-3m0 0l-3 3m3-3v12" />
</svg>
<p className="text-white font-medium">Dateien hierher ziehen</p>
<p className="text-white/50 text-sm mt-1">oder klicken zum Auswaehlen</p>
</div>
{/* File List */}
{files.length > 0 && (
<div className="max-h-64 overflow-y-auto space-y-2 mb-6">
{files.map((file, index) => (
<div key={index} className="flex items-center gap-3 p-3 rounded-xl bg-white/5">
<span className="text-lg">{file.type.startsWith('image/') ? '\uD83D\uDDBC\uFE0F' : '\uD83D\uDCC4'}</span>
<div className="flex-1 min-w-0">
<p className="text-white text-sm truncate">{file.name}</p>
<input
type="text"
value={anonymIds[index] || ''}
onChange={(e) => updateAnonymId(index, e.target.value)}
placeholder="Anonym-ID"
className="mt-1 w-full px-2 py-1 rounded bg-white/10 border border-white/10 text-white text-sm placeholder-white/40 focus:outline-none focus:border-purple-500"
/>
</div>
<button
onClick={() => removeFile(index)}
className="p-2 rounded-lg hover:bg-red-500/20 text-red-400 transition-colors"
>
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
</svg>
</button>
</div>
))}
</div>
)}
{/* Actions */}
<div className="flex gap-3">
<button
onClick={onClose}
className="flex-1 px-4 py-3 rounded-xl bg-white/10 text-white hover:bg-white/20 transition-colors"
>
Abbrechen
</button>
<button
onClick={handleSubmit}
disabled={isUploading || files.length === 0}
className="flex-1 px-4 py-3 rounded-xl bg-gradient-to-r from-purple-500 to-pink-500 text-white font-semibold hover:shadow-lg hover:shadow-purple-500/30 transition-all disabled:opacity-50 flex items-center justify-center gap-2"
>
{isUploading ? (
<>
<div className="w-5 h-5 border-2 border-white border-t-transparent rounded-full animate-spin" />
Hochladen...
</>
) : (
<>
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-8l-4-4m0 0L8 8m4-4v12" />
</svg>
{files.length} Dateien hochladen
</>
)}
</button>
</div>
</GlassCard>
</div>
)
}