Mark Duppenthaler
Image examples cleanup
503a577
import React, { useState, useEffect } from 'react'
import API from '../API'
import LoadingSpinner from './LoadingSpinner'
import ImageGallery from './ImageGallery'
import AudioGallery from './AudioGallery'
import VideoGallery from './VideoGallery'
interface ExamplesProps {
fileType: 'image' | 'audio' | 'video'
}
// Move ExamplesData type export to allow import in Galleries.tsx
export type ExamplesData = {
image_url: string
audio_url?: string
video_url?: string
name: string
metadata: {
[key: string]: string | boolean
}
}
const Examples = ({ fileType }: ExamplesProps) => {
const [examples, setExamples] = useState<{
[model: string]: { [attack: string]: ExamplesData[] }
}>({})
const [loading, setLoading] = useState(false)
const [error, setError] = useState<string | null>(null)
const [selectedModel, setSelectedModel] = useState<string | null>(null)
const [selectedAttack, setSelectedAttack] = useState<string | null>(null)
useEffect(() => {
setLoading(true)
setError(null)
API.fetchExamplesByType(fileType)
.then((data) => {
// data is a dictionary from {[model]: {[attack]: {image_url, audio_url, video_url, description}}}
setExamples(data)
// get the first model and attack if available
const models = Object.keys(data)
if (models.length > 0) {
setSelectedModel(models[0])
const attacks = Object.keys(data[models[0]])
if (attacks.length > 0) {
setSelectedAttack(attacks[0]) // Set the first attack of the first model
} else {
setSelectedAttack(null) // No attacks available
}
} else {
setSelectedModel(null)
setSelectedAttack(null) // No models available
}
// Reset loading state
setLoading(false)
})
.catch((err) => {
setError(err.message)
setLoading(false)
})
}, [fileType])
if (loading) {
return <LoadingSpinner />
}
return (
<div className="examples-container">
<div className="selectors-container flex flex-col md:flex-row gap-4 mb-4">
<fieldset className="fieldset">
<legend className="fieldset-legend">Model</legend>
<select
className="select select-bordered w-full"
value={selectedModel || ''}
onChange={(e) => setSelectedModel(e.target.value || null)}
>
{Object.keys(examples).map((model) => (
<option key={model} value={model}>
{model}
</option>
))}
</select>
</fieldset>
{selectedModel && (
<fieldset className="fieldset">
<legend className="fieldset-legend">Attack</legend>
<select
className="select select-bordered w-full"
value={selectedAttack || ''}
onChange={(e) => setSelectedAttack(e.target.value || null)}
>
{Object.keys(examples[selectedModel]).map((attack) => (
<option key={attack} value={attack}>
{attack}
</option>
))}
</select>
</fieldset>
)}
</div>
{error && <p className="error">Error: {error}</p>}
{selectedModel && selectedAttack && fileType === 'image' && (
<ImageGallery
selectedModel={selectedModel}
selectedAttack={selectedAttack}
examples={examples}
/>
)}
{selectedModel && selectedAttack && fileType === 'audio' && (
<AudioGallery
selectedModel={selectedModel}
selectedAttack={selectedAttack}
examples={examples}
/>
)}
{selectedModel && selectedAttack && fileType === 'video' && (
<VideoGallery
selectedModel={selectedModel}
selectedAttack={selectedAttack}
examples={examples}
/>
)}
</div>
)
}
export default Examples