'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'; import CustomSelect from './CustomSelect'; 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 snapTo15(value: string): string { if (!value) return value; const d = new Date(value); if (isNaN(d.getTime())) return value; const rem = d.getMinutes() % 15; if (rem !== 0) { d.setMinutes(d.getMinutes() + (15 - rem)); d.setSeconds(0); } const pad = (n: number) => String(n).padStart(2, '0'); return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`; } function nowLocalDatetime(): string { const now = new Date(); const pad = (n: number) => String(n).padStart(2, '0'); const raw = `${now.getFullYear()}-${pad(now.getMonth() + 1)}-${pad(now.getDate())}T${pad(now.getHours())}:${pad(now.getMinutes())}`; return snapTo15(raw); } const NO_OBJEKTE_ARTEN: ArtFuehrung[] = ['BEO-Sitzung', 'Technischer Dienst']; const SONNE_ART: ArtFuehrung = 'Sonnenführung'; 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); const showObjekte = !NO_OBJEKTE_ARTEN.includes(artFuehrung); const showBesucher = !NO_OBJEKTE_ARTEN.includes(artFuehrung); const isSonne = artFuehrung === SONNE_ART; 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]); // Objekte-Vorauswahl je nach Art der Führung useEffect(() => { if (artFuehrung === SONNE_ART) { setObjekte([{ ID: null, Name: 'Sonne' }]); } else if (NO_OBJEKTE_ARTEN.includes(artFuehrung)) { setObjekte([]); } }, [artFuehrung]); 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: showObjekte ? 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); } } const inputCls = 'w-full px-3 py-2 border-2 border-gray-400 rounded-lg bg-white text-base focus:border-blue-500 focus:outline-none'; const labelCls = 'block text-sm font-medium text-gray-700 mb-0.5'; return (
{/* Art der Führung — volle Breite */}
({ value: a, label: a }))} onChange={(v) => setArtFuehrung(v as ArtFuehrung)} />
{/* Beginn / Ende / Besucher — eine Zeile */}
setBeginn(snapTo15(e.target.value))} required step={900} className="w-full px-2 py-2 border-2 border-gray-400 rounded-lg bg-white text-sm focus:border-blue-500 focus:outline-none" />
setEnde(snapTo15(e.target.value))} required step={900} className="w-full px-2 py-2 border-2 border-gray-400 rounded-lg bg-white text-sm focus:border-blue-500 focus:outline-none" />
{showBesucher && (
setBesucher(parseInt(e.target.value) || 0)} min={0} max={9999} className="w-20 px-2 py-2 border-2 border-gray-400 rounded-lg bg-white text-base focus:border-blue-500 focus:outline-none" />
)}
{/* BEOs */}
{/* Objekte — abhängig von der Art der Führung */} {showObjekte && (
{isSonne ? (
Sonne (bei Sonnenführung fest vorgegeben)
) : ( )}
)} {/* Bemerkungen */}