'use client'; import { useEffect, useState } from 'react'; import type { Kuppel, ArtFuehrung, BeoOption, SelectedObjekt, Wetter, LogbuchEintrag } from '@/types/logbuch'; import { ARTEN } from '@/types/logbuch'; import BeoSelector from './BeoSelector'; import ObjektSelector from './ObjektSelector'; interface Props { kuppel: Kuppel; currentUserBeo: BeoOption; editEntry?: LogbuchEintrag | null; onSaved: () => void; } function toLocalDatetimeValue(isoOrDatetime: string): string { if (!isoOrDatetime) return ''; return isoOrDatetime.slice(0, 16); } function nowLocalDatetime(): string { const now = new Date(); const pad = (n: number) => String(n).padStart(2, '0'); return `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}T${pad(now.getHours())}:${pad(now.getMinutes())}`; } export default function LogbuchForm({ kuppel, currentUserBeo, editEntry, onSaved }: Props) { const [artFuehrung, setArtFuehrung] = useState('Reguläre Führung'); const [beginn, setBeginn] = useState(nowLocalDatetime()); const [ende, setEnde] = useState(nowLocalDatetime()); const [besucher, setBesucher] = useState(0); const [beos, setBeos] = useState([currentUserBeo]); const [objekte, setObjekte] = useState([]); const [bemerkungen, setBemerkungen] = useState(''); const [wetter, setWetter] = useState(null); const [saving, setSaving] = useState(false); const [error, setError] = useState(''); const [success, setSuccess] = useState(false); useEffect(() => { fetch('/api/wetter') .then((r) => { if (!r.ok) throw new Error(); return r.json(); }) .then(setWetter) .catch(() => {}); }, []); useEffect(() => { if (editEntry) { setArtFuehrung(editEntry.ArtFuehrung); setBeginn(toLocalDatetimeValue(editEntry.Beginn)); setEnde(toLocalDatetimeValue(editEntry.Ende)); setBesucher(editEntry.Besucher); setBemerkungen(editEntry.Bemerkungen ?? ''); if (editEntry.WetterTemp !== null) { setWetter({ temp: editEntry.WetterTemp ?? 0, feuchte: editEntry.WetterFeuchte ?? 0, druck: editEntry.WetterDruck ?? 0, }); } } else { setArtFuehrung('Reguläre Führung'); setBeginn(nowLocalDatetime()); setEnde(nowLocalDatetime()); setBesucher(0); setBeos([currentUserBeo]); setObjekte([]); setBemerkungen(''); } }, [editEntry, currentUserBeo]); useEffect(() => { if (editEntry && editEntry.BEOs) { fetch('/api/beos') .then((r) => r.json()) .then((all: BeoOption[]) => { const kuerzel = editEntry.BEOs.split(', ').map((k) => k.trim()); setBeos(all.filter((b) => kuerzel.includes(b.Kuerzel))); }) .catch(() => {}); } if (editEntry && editEntry.Objekte) { const names = editEntry.Objekte.split(', ').map((n) => n.trim()); fetch('/api/objekte') .then((r) => r.json()) .then((all: { ID: number; Name: string }[]) => { const result: SelectedObjekt[] = names.map((name) => { const found = all.find((o) => o.Name === name); return { ID: found?.ID ?? null, Name: name }; }); setObjekte(result); }) .catch(() => {}); } }, [editEntry]); async function handleSubmit(e: React.FormEvent) { e.preventDefault(); setSaving(true); setError(''); setSuccess(false); const body = { Kuppel: kuppel, ArtFuehrung: artFuehrung, Beginn: beginn, Ende: ende, Besucher: besucher, beoIds: beos.map((b) => b.ID), objekte, Bemerkungen: bemerkungen, Wetter: wetter, }; const url = editEntry ? `/api/logbuch/${editEntry.ID}` : '/api/logbuch'; const method = editEntry ? 'PUT' : 'POST'; try { const res = await fetch(url, { method, headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(body), }); if (!res.ok) throw new Error(await res.text()); setSuccess(true); if (!editEntry) { setBeginn(nowLocalDatetime()); setEnde(nowLocalDatetime()); setBesucher(0); setBeos([currentUserBeo]); setObjekte([]); setBemerkungen(''); } onSaved(); } catch { setError('Fehler beim Speichern. Bitte erneut versuchen.'); } finally { setSaving(false); } } return (
setBeginn(e.target.value)} required className="w-full px-3 py-2 border-2 border-gray-400 rounded-lg bg-white text-sm focus:border-blue-500 focus:outline-none" />
setEnde(e.target.value)} required className="w-full px-3 py-2 border-2 border-gray-400 rounded-lg bg-white text-sm focus:border-blue-500 focus:outline-none" />
setBesucher(parseInt(e.target.value) || 0)} min={0} max={9999} className="w-32 px-3 py-2 border-2 border-gray-400 rounded-lg bg-white text-sm focus:border-blue-500 focus:outline-none" />