Whistleblower (1220 -> 349 LOC) split into 6 colocated components:
TabNavigation, StatCard, FilterBar, ReportCard, WhistleblowerCreateModal,
CaseDetailPanel. All under the 300 LOC soft target.
Drive-by fix: the earlier fc6a330 split of compliance-scope-types.ts
dropped several helper exports that downstream consumers still import
(lib/sdk/index.ts, compliance-scope-engine.ts, obligations page,
compliance-scope page, constraint-enforcer, drafting-engine validate).
Restored them in the appropriate domain modules:
- core-levels.ts: maxDepthLevel, getDepthLevelNumeric, depthLevelFromNumeric
- state.ts: createEmptyScopeState
- decisions.ts: createEmptyScopeDecision + ApplicableRegulation,
RegulationObligation, RegulationAssessmentResult, SupervisoryAuthorityInfo
Verification: next build clean (142 pages generated), /sdk/whistleblower
still builds at ~11.5 kB.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
224 lines
8.5 KiB
TypeScript
224 lines
8.5 KiB
TypeScript
'use client'
|
|
|
|
import React, { useState } from 'react'
|
|
|
|
export function WhistleblowerCreateModal({
|
|
onClose,
|
|
onSuccess
|
|
}: {
|
|
onClose: () => void
|
|
onSuccess: () => void
|
|
}) {
|
|
const [title, setTitle] = useState('')
|
|
const [description, setDescription] = useState('')
|
|
const [category, setCategory] = useState<string>('corruption')
|
|
const [priority, setPriority] = useState<string>('normal')
|
|
const [isAnonymous, setIsAnonymous] = useState(true)
|
|
const [reporterName, setReporterName] = useState('')
|
|
const [reporterEmail, setReporterEmail] = useState('')
|
|
const [isSaving, setIsSaving] = useState(false)
|
|
const [error, setError] = useState<string | null>(null)
|
|
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
e.preventDefault()
|
|
if (!title.trim() || !description.trim()) return
|
|
|
|
setIsSaving(true)
|
|
setError(null)
|
|
try {
|
|
const body: Record<string, unknown> = {
|
|
title: title.trim(),
|
|
description: description.trim(),
|
|
category,
|
|
priority,
|
|
isAnonymous,
|
|
status: 'new'
|
|
}
|
|
if (!isAnonymous) {
|
|
body.reporterName = reporterName.trim()
|
|
body.reporterEmail = reporterEmail.trim()
|
|
}
|
|
|
|
const res = await fetch('/api/sdk/v1/whistleblower/reports', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(body)
|
|
})
|
|
if (!res.ok) {
|
|
const data = await res.json().catch(() => ({}))
|
|
throw new Error(data?.detail || data?.message || `Fehler ${res.status}`)
|
|
}
|
|
onSuccess()
|
|
} catch (err: unknown) {
|
|
setError(err instanceof Error ? err.message : 'Unbekannter Fehler')
|
|
} finally {
|
|
setIsSaving(false)
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="fixed inset-0 z-50 flex items-center justify-center">
|
|
{/* Backdrop */}
|
|
<div
|
|
className="absolute inset-0 bg-black/50"
|
|
onClick={onClose}
|
|
/>
|
|
|
|
{/* Modal */}
|
|
<div className="relative bg-white rounded-2xl shadow-xl w-full max-w-lg mx-4 max-h-[90vh] overflow-y-auto">
|
|
<div className="p-6">
|
|
<div className="flex items-center justify-between mb-6">
|
|
<h2 className="text-xl font-semibold text-gray-900">Neue Meldung erfassen</h2>
|
|
<button
|
|
onClick={onClose}
|
|
className="p-2 text-gray-400 hover:text-gray-600 hover:bg-gray-100 rounded-lg 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>
|
|
|
|
{error && (
|
|
<div className="mb-4 p-3 bg-red-50 border border-red-200 rounded-lg text-sm text-red-700">
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<form onSubmit={handleSubmit} className="space-y-4">
|
|
{/* Title */}
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
Titel <span className="text-red-500">*</span>
|
|
</label>
|
|
<input
|
|
type="text"
|
|
value={title}
|
|
onChange={(e) => setTitle(e.target.value)}
|
|
required
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm"
|
|
placeholder="Kurze Beschreibung des Vorfalls"
|
|
/>
|
|
</div>
|
|
|
|
{/* Description */}
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
Beschreibung <span className="text-red-500">*</span>
|
|
</label>
|
|
<textarea
|
|
value={description}
|
|
onChange={(e) => setDescription(e.target.value)}
|
|
required
|
|
rows={4}
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm resize-none"
|
|
placeholder="Detaillierte Beschreibung des Vorfalls..."
|
|
/>
|
|
</div>
|
|
|
|
{/* Category */}
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
Kategorie
|
|
</label>
|
|
<select
|
|
value={category}
|
|
onChange={(e) => setCategory(e.target.value)}
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm"
|
|
>
|
|
<option value="corruption">Korruption</option>
|
|
<option value="fraud">Betrug</option>
|
|
<option value="data_protection">Datenschutz</option>
|
|
<option value="discrimination">Diskriminierung</option>
|
|
<option value="environment">Umwelt</option>
|
|
<option value="competition">Wettbewerb</option>
|
|
<option value="product_safety">Produktsicherheit</option>
|
|
<option value="tax_evasion">Steuerhinterziehung</option>
|
|
<option value="other">Sonstiges</option>
|
|
</select>
|
|
</div>
|
|
|
|
{/* Priority */}
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
Prioritaet
|
|
</label>
|
|
<select
|
|
value={priority}
|
|
onChange={(e) => setPriority(e.target.value)}
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm"
|
|
>
|
|
<option value="normal">Normal</option>
|
|
<option value="high">Hoch</option>
|
|
<option value="critical">Kritisch</option>
|
|
</select>
|
|
</div>
|
|
|
|
{/* Anonymous */}
|
|
<div className="flex items-center gap-3">
|
|
<input
|
|
type="checkbox"
|
|
id="isAnonymous"
|
|
checked={isAnonymous}
|
|
onChange={(e) => setIsAnonymous(e.target.checked)}
|
|
className="w-4 h-4 text-purple-600 border-gray-300 rounded focus:ring-purple-500"
|
|
/>
|
|
<label htmlFor="isAnonymous" className="text-sm font-medium text-gray-700">
|
|
Anonyme Einreichung
|
|
</label>
|
|
</div>
|
|
|
|
{/* Reporter fields (only if not anonymous) */}
|
|
{!isAnonymous && (
|
|
<>
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
Name des Hinweisgebers
|
|
</label>
|
|
<input
|
|
type="text"
|
|
value={reporterName}
|
|
onChange={(e) => setReporterName(e.target.value)}
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm"
|
|
placeholder="Vor- und Nachname"
|
|
/>
|
|
</div>
|
|
<div>
|
|
<label className="block text-sm font-medium text-gray-700 mb-1">
|
|
E-Mail des Hinweisgebers
|
|
</label>
|
|
<input
|
|
type="email"
|
|
value={reporterEmail}
|
|
onChange={(e) => setReporterEmail(e.target.value)}
|
|
className="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 focus:border-purple-500 text-sm"
|
|
placeholder="email@beispiel.de"
|
|
/>
|
|
</div>
|
|
</>
|
|
)}
|
|
|
|
{/* Buttons */}
|
|
<div className="flex items-center justify-end gap-3 pt-2">
|
|
<button
|
|
type="button"
|
|
onClick={onClose}
|
|
className="px-4 py-2 text-sm text-gray-600 hover:bg-gray-100 rounded-lg transition-colors"
|
|
>
|
|
Abbrechen
|
|
</button>
|
|
<button
|
|
type="submit"
|
|
disabled={isSaving || !title.trim() || !description.trim()}
|
|
className="px-4 py-2 text-sm bg-purple-600 text-white rounded-lg hover:bg-purple-700 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
|
>
|
|
{isSaving ? 'Wird eingereicht...' : 'Einreichen'}
|
|
</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|