Fertigstellen-Button + drawing_created Status-Flow

- DrawIt: Button "Fertigstellen" unter Objektliste setzt Picture-Status auf drawing_created
- Bild verschwindet danach aus der Annotieren-Ansicht
- GenerateIt: lädt jetzt Directus-Bilder mit status=drawing_created
- GenerateIt: zeigt Bild-Vorschau + Directus-Objekte
- app.py: PATCH-Endpunkt für Pictures + Status-Parameter im GET

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-25 21:49:13 +02:00
parent e18d9a5796
commit a42fadef09
4 changed files with 111 additions and 65 deletions

View File

@@ -3,7 +3,9 @@ import ObjectsList from '../components/ObjectsList'
import DetailsPanel from '../components/DetailsPanel'
import SentencesList from '../components/SentencesList'
import Topbar from '../components/Topbar'
import { getImages, getObjects, generateDetails, generateSentence, getSentences } from '../api'
import { getDirectusPictures, directusAssetUrl, type DirectusPicture, getDirectusObjects, generateDetails, generateSentence, getSentences } from '../api'
import { useAuth } from '../context/AuthContext'
import type { DirectusObject, Selection } from '../types'
import type { ObjectMeta, Sentence } from '../types'
const ChevronLeftIcon = () => (
@@ -30,54 +32,58 @@ const ChatIcon = () => (
</svg>
)
function directusObjToMeta(obj: DirectusObject, index: number): ObjectMeta {
const first = obj.selections?.[0]
return {
id: obj.id,
image_file: '',
title_de: '',
position_de: '',
action_de: '',
condition_de: '',
hierarchy: 1,
parent_id: obj.parent ?? undefined,
user_notes: obj.user_notes ?? '',
selections: obj.selections ?? [],
} as unknown as ObjectMeta
}
export default function GenerateIt() {
const [imageList, setImageList] = useState<string[]>([])
const { token } = useAuth()
const [pictureList, setPictureList] = useState<DirectusPicture[]>([])
const [currentIndex, setCurrentIndex] = useState(-1)
const [directusObjects, setDirectusObjects] = useState<DirectusObject[]>([])
const [objects, setObjects] = useState<ObjectMeta[]>([])
const [selectedObj, setSelectedObj] = useState<ObjectMeta | null>(null)
const [sentences, setSentences] = useState<Sentence[]>([])
const [isGeneratingDetails, setIsGeneratingDetails] = useState(false)
const [isGeneratingSentence, setIsGeneratingSentence] = useState(false)
const currentFilename =
currentIndex >= 0 && currentIndex < imageList.length ? imageList[currentIndex] : null
const currentPicture: DirectusPicture | null =
currentIndex >= 0 && currentIndex < pictureList.length ? pictureList[currentIndex] : null
useEffect(() => {
getImages('generate')
.then(imgs => {
setImageList(imgs)
setCurrentIndex(imgs.length - 1)
})
if (!token) return
getDirectusPictures(token, 'drawing_created')
.then(pics => { setPictureList(pics); setCurrentIndex(pics.length > 0 ? 0 : -1) })
.catch(console.error)
}, [])
}, [token])
useEffect(() => {
if (!currentFilename) {
setObjects([])
setSelectedObj(null)
setSentences([])
return
}
getObjects(currentFilename)
if (!currentPicture || !token) { setDirectusObjects([]); setObjects([]); setSelectedObj(null); setSentences([]); return }
getDirectusObjects(currentPicture.id, token)
.then(objs => {
setObjects(objs)
if (objs.length > 0) {
setSelectedObj(objs[0])
loadSentences(objs[0].id)
} else {
setSelectedObj(null)
setSentences([])
}
setDirectusObjects(objs)
const metas = objs.map((o, i) => directusObjToMeta(o, i))
setObjects(metas)
if (metas.length > 0) { setSelectedObj(metas[0]); loadSentences(metas[0].id) }
else { setSelectedObj(null); setSentences([]) }
})
.catch(console.error)
}, [currentFilename])
}, [currentPicture?.id, token])
const loadSentences = async (objId: string) => {
try {
setSentences(await getSentences(objId))
} catch (e) {
console.error(e)
}
try { setSentences(await getSentences(objId)) } catch (e) { console.error(e) }
}
const handleGenerateDetails = async () => {
@@ -114,48 +120,32 @@ export default function GenerateIt() {
const imageNav = (
<div style={{ display: 'flex', alignItems: 'center', gap: 8 }}>
<div className="image-nav">
<button
className="btn-icon"
onClick={() => setCurrentIndex(i => i - 1)}
disabled={currentIndex <= 0}
>
<button className="btn-icon" onClick={() => setCurrentIndex(i => i - 1)} disabled={currentIndex <= 0}>
<ChevronLeftIcon />
</button>
<span className="image-counter">
{imageList.length > 0 ? (
{pictureList.length > 0 ? (
<>
<span className="image-counter-num">{currentIndex + 1}</span>
<span className="image-counter-sep">/</span>
<span className="image-counter-total">{imageList.length}</span>
<span className="image-counter-total">{pictureList.length}</span>
</>
) : (
<span className="image-counter-empty">Keine Bilder</span>
)}
</span>
<button
className="btn-icon"
onClick={() => setCurrentIndex(i => i + 1)}
disabled={currentIndex >= imageList.length - 1}
>
<button className="btn-icon" onClick={() => setCurrentIndex(i => i + 1)} disabled={currentIndex >= pictureList.length - 1}>
<ChevronRightIcon />
</button>
</div>
<div style={{ width: 1, height: 24, background: 'var(--border)' }} />
<button
className="btn-ghost btn-sm"
onClick={handleGenerateDetails}
disabled={isGeneratingDetails || !selectedObj}
>
<button className="btn-ghost btn-sm" onClick={handleGenerateDetails} disabled={isGeneratingDetails || !selectedObj}>
<SparkleIcon />
{isGeneratingDetails ? 'Generiere…' : 'KI-Details'}
</button>
<button
className="btn-ghost btn-sm"
onClick={handleGenerateSentence}
disabled={isGeneratingSentence || !selectedObj}
>
<button className="btn-ghost btn-sm" onClick={handleGenerateSentence} disabled={isGeneratingSentence || !selectedObj}>
<ChatIcon />
{isGeneratingSentence ? 'Generiere…' : 'KI-Sentence'}
</button>
@@ -170,16 +160,16 @@ export default function GenerateIt() {
{/* Left: Objects */}
<aside className="sidebar">
<div className="sidebar-panel" style={{ flex: 1 }}>
<h3 className="sidebar-heading">Objekte</h3>
<h3 className="sidebar-heading">
Objekte
{objects.length > 0 && <span className="badge">{objects.length}</span>}
</h3>
<ObjectsList
objects={objects}
selectedObjectId={selectedObj?.id ?? null}
onSelect={id => {
const obj = objects.find(o => o.id === id)
if (obj) {
setSelectedObj(obj)
loadSentences(obj.id)
}
if (obj) { setSelectedObj(obj); loadSentences(obj.id) }
}}
onObjectsChange={setObjects}
isGeneratePage={true}
@@ -189,9 +179,16 @@ export default function GenerateIt() {
</div>
</aside>
{/* Center: Details */}
<main className="canvas-area" style={{ alignItems: 'flex-start', justifyContent: 'flex-start' }}>
<div style={{ width: '100%', maxWidth: 520, display: 'flex', flexDirection: 'column', gap: 1 }}>
{/* Center: Image + Details */}
<main className="canvas-area" style={{ alignItems: 'flex-start', justifyContent: 'flex-start', flexDirection: 'column', gap: 12, padding: 16 }}>
{currentPicture && token && (
<img
src={directusAssetUrl(currentPicture.media, token)}
alt="Bild"
style={{ maxWidth: '100%', maxHeight: 320, borderRadius: 8, objectFit: 'contain', border: '1px solid var(--border)' }}
/>
)}
<div style={{ width: '100%', maxWidth: 520 }}>
<div className="sidebar-panel" style={{ background: 'var(--surface)', borderRadius: 'var(--r-lg)', border: '1px solid var(--border)' }}>
<DetailsPanel obj={selectedObj} objects={objects} sentences={sentences} />
</div>