import React, { useState, useEffect, useMemo } from 'react' import { Link } from 'react-router-dom' import api from '../utils/api' import { SKILL_LEVEL_OPTIONS } from '../constants/skillLevels' import SearchableSelect from '../components/SearchableSelect' import MultiSelectCombo from '../components/MultiSelectCombo' const PAGE_SIZE = 100 const LEVEL_FILTER_OPTS = SKILL_LEVEL_OPTIONS.filter((o) => o.level != null) function ExercisesListPage() { const [exercises, setExercises] = useState([]) const [catalogs, setCatalogs] = useState({ focusAreas: [], styleDirections: [], trainingTypes: [], targetGroups: [], skills: [], }) const [catalogsReady, setCatalogsReady] = useState(false) const [loading, setLoading] = useState(true) const [loadingMore, setLoadingMore] = useState(false) const [offset, setOffset] = useState(0) const [hasMore, setHasMore] = useState(false) const [searchInput, setSearchInput] = useState('') const [aiSearchInput, setAiSearchInput] = useState('') const [debouncedSearch, setDebouncedSearch] = useState('') const [debouncedAiSearch, setDebouncedAiSearch] = useState('') const [filters, setFilters] = useState({ focus_area_ids: [], style_direction_ids: [], training_type_ids: [], target_group_ids: [], skill_ids: [], skill_min_level: '', skill_max_level: '', visibility_any: [], status_any: [], }) useEffect(() => { const t = setTimeout(() => setDebouncedSearch(searchInput.trim()), 400) return () => clearTimeout(t) }, [searchInput]) useEffect(() => { const t = setTimeout(() => setDebouncedAiSearch(aiSearchInput.trim()), 400) return () => clearTimeout(t) }, [aiSearchInput]) const focusOptions = useMemo( () => catalogs.focusAreas.map((fa) => ({ id: fa.id, label: `${fa.icon || ''} ${fa.name || ''}`.trim(), })), [catalogs.focusAreas] ) const styleOptions = useMemo( () => catalogs.styleDirections.map((s) => ({ id: s.id, label: s.name || String(s.id) })), [catalogs.styleDirections] ) const trainingTypeOptions = useMemo( () => catalogs.trainingTypes.map((t) => ({ id: t.id, label: t.name || String(t.id) })), [catalogs.trainingTypes] ) const targetGroupOptions = useMemo( () => catalogs.targetGroups.map((g) => ({ id: g.id, label: g.name || String(g.id) })), [catalogs.targetGroups] ) const skillOptions = useMemo( () => catalogs.skills.map((s) => ({ id: s.id, label: s.name || String(s.id) })), [catalogs.skills] ) const levelFilterOptions = useMemo( () => LEVEL_FILTER_OPTS.map((o) => ({ id: o.level, label: o.label })), [] ) const visibilityOptions = useMemo( () => [ { id: 'private', label: 'Privat' }, { id: 'club', label: 'Verein' }, { id: 'official', label: 'Offiziell' }, ], [] ) const statusOptions = useMemo( () => [ { id: 'draft', label: 'Entwurf' }, { id: 'in_review', label: 'In Prüfung' }, { id: 'approved', label: 'Freigegeben' }, { id: 'archived', label: 'Archiviert' }, ], [] ) const queryBase = useMemo(() => { const q = {} const n = (v) => (v === '' || v == null ? undefined : Number(v)) const ids = (arr) => Array.isArray(arr) && arr.length ? arr.map((x) => Number(x)).filter((x) => !Number.isNaN(x)) : undefined const fa = ids(filters.focus_area_ids) if (fa?.length) q.focus_area_ids = fa const sd = ids(filters.style_direction_ids) if (sd?.length) q.style_direction_ids = sd const tt = ids(filters.training_type_ids) if (tt?.length) q.training_type_ids = tt const tg = ids(filters.target_group_ids) if (tg?.length) q.target_group_ids = tg const sk = ids(filters.skill_ids) if (sk?.length) q.skill_ids = sk if (filters.skill_min_level) q.skill_min_level = n(filters.skill_min_level) if (filters.skill_max_level) q.skill_max_level = n(filters.skill_max_level) if (filters.visibility_any?.length) q.visibility_any = [...filters.visibility_any] if (filters.status_any?.length) q.status_any = [...filters.status_any] if (debouncedSearch) q.search = debouncedSearch if (debouncedAiSearch) q.ai_search = debouncedAiSearch return q }, [filters, debouncedSearch, debouncedAiSearch]) useEffect(() => { let cancelled = false ;(async () => { try { const [fa, sd, tt, tg, sk] = await Promise.all([ api.listFocusAreas(), api.listStyleDirections(), api.listTrainingTypes(), api.listTargetGroups(), api.listSkills(), ]) if (!cancelled) { setCatalogs({ focusAreas: fa, styleDirections: sd, trainingTypes: tt, targetGroups: tg, skills: sk, }) setCatalogsReady(true) } } catch (err) { if (!cancelled) { console.error(err) alert('Kataloge konnten nicht geladen werden: ' + err.message) setCatalogsReady(true) } } })() return () => { cancelled = true } }, []) useEffect(() => { if (!catalogsReady) return let cancelled = false const run = async () => { setLoading(true) setOffset(0) try { const batch = await api.listExercises({ ...queryBase, limit: PAGE_SIZE, offset: 0 }) if (cancelled) return setExercises(batch) setHasMore(batch.length === PAGE_SIZE) setOffset(batch.length) } catch (err) { if (!cancelled) { console.error('Failed to load data:', err) alert('Fehler beim Laden: ' + err.message) } } finally { if (!cancelled) setLoading(false) } } run() return () => { cancelled = true } }, [queryBase, catalogsReady]) const loadMore = async () => { if (loadingMore || !hasMore) return setLoadingMore(true) try { const batch = await api.listExercises({ ...queryBase, limit: PAGE_SIZE, offset }) setExercises((prev) => [...prev, ...batch]) setHasMore(batch.length === PAGE_SIZE) setOffset((o) => o + batch.length) } catch (err) { alert('Fehler: ' + err.message) } finally { setLoadingMore(false) } } const handleDelete = async (exercise) => { if (!confirm(`Übung "${exercise.title}" wirklich löschen?`)) return try { await api.deleteExercise(exercise.id) setExercises((prev) => prev.filter((e) => e.id !== exercise.id)) } catch (err) { alert('Fehler beim Löschen: ' + err.message) } } if (!catalogsReady || loading) { return (
Laden...
Filter untereinander werden mit UND kombiniert. Mehrere Einträge in einem Feld mit{' '} ODER (die Übung muss mindestens eine der gewählten Zuordnungen erfüllen).
Standardfilter aus Verein und Trainerrolle folgen später; die KI-Nutzung der zweiten Zeile kommt mit einem eigenen Endpunkt.
Keine Übungen gefunden.
{exercises.length} angezeigt {hasMore ? ' · es gibt weitere Einträge' : ''}
{exercise.summary.length > 160 ? `${exercise.summary.slice(0, 160)}…` : exercise.summary}
)}