Initial commit

This commit is contained in:
Zdeněk Burda
2026-01-09 21:26:40 +01:00
parent e83aec6dca
commit 41e3ce6f25
404 changed files with 61250 additions and 28 deletions

View File

@@ -0,0 +1,261 @@
import { useEffect, useState } from "react";
import axios from "axios";
import { useTranslation } from "react-i18next";
import { useLanguageStore } from "@/stores/languageStore";
import { useContestRefreshStore } from "@/stores/contestRefreshStore";
import {
Table,
TableHeader,
TableColumn,
TableBody,
TableRow,
TableCell,
type Selection,
} from "@heroui/react";
export type Contest = {
id: number;
name: string;
description?: string | null;
bands?: { id: number; name: string }[];
categories?: { id: number; name: string }[];
power_categories?: { id: number; name: string }[];
evaluator?: string | null;
email?: string | null;
email2?: string | null;
is_mcr: boolean;
is_sixhr: boolean;
is_active: boolean;
is_test: boolean;
start_time: string | null; // "HH:MM:SS"
duration: number; // hodiny
logs_deadline_days: number;
};
type PaginatedResponse<T> = {
data: T[];
};
type ContestsTableProps = {
/** Handler vybraného řádku (klik/označení). */
onRowSelect?: (contest: Contest) => void;
/** Handler pro klik na ikonu editace (tužka). */
onEditContest?: (contest: Contest) => void;
/** Show/hide the edit (pencil) actions column. Default: true */
enableEdit?: boolean;
/** Při kliknutí/označení řádku zavolat onSelectContest. Default: false */
selectOnRowClick?: boolean;
/** Filter the list to only active contests (is_active === true). Default: false */
onlyActive?: boolean;
/** Show/hide the is_active column. Default: true */
showActiveColumn?: boolean;
/** Zahrnout testovací závody. Default: false */
showTests?: boolean;
};
export default function ContestsTable({
onRowSelect,
onEditContest,
enableEdit = true,
onlyActive = false,
showActiveColumn = true,
selectOnRowClick = false,
showTests = false,
}: ContestsTableProps) {
const { t } = useTranslation("common");
const locale = useLanguageStore((s) => s.locale);
const refreshKey = useContestRefreshStore((s) => s.refreshKey);
const [items, setItems] = useState<Contest[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const [selectedKeys, setSelectedKeys] = useState<Selection>(new Set([]));
useEffect(() => {
let active = true;
(async () => {
try {
setLoading(true);
setError(null);
const res = await axios.get<PaginatedResponse<Contest> | Contest[]>(
"/api/contests",
{
withCredentials: true,
headers: { Accept: "application/json" },
params: {
lang: locale,
include_tests: showTests ? 1 : 0,
only_active: onlyActive ? 1 : 0,
},
}
);
if (!active) return;
const data = Array.isArray(res.data)
? res.data
: (res.data as PaginatedResponse<Contest>).data;
setItems(data);
} catch {
if (!active) return;
setError(t("unable_to_load_contests") ?? "Nepodařilo se načíst seznam závodů.");
} finally {
if (active) setLoading(false);
}
})();
return () => { active = false; };
}, [locale, t, refreshKey, showTests, onlyActive]);
const handleEdit = onEditContest ?? onRowSelect;
const handleRowSelect = onRowSelect;
const canEdit = Boolean(enableEdit && handleEdit);
const visibleItems = onlyActive ? items.filter((c) => c.is_active) : items;
if (loading) return <div>{t("contests_loading") ?? "Načítám závody…"}</div>;
if (error) return <div className="text-sm text-red-600">{error}</div>;
if (visibleItems.length === 0) {
return (
<div>
{t("contests_empty") ?? "Žádné závody nejsou k dispozici."}
</div>
);
}
const columns = [
{ key: "name", label: t("contest_name") },
{ key: "description", label: t("contest_description") },
{ key: "bands", label: t("bands") ?? "Pásma" },
{ key: "categories", label: t("categories") ?? "Kategorie" },
{ key: "power_categories", label: t("power_categories") ?? "Výkonové kategorie" },
...(showActiveColumn ? [{ key: "is_active", label: t("contest_active") }] : []),
...(canEdit ? [{ key: "actions", label: "" }] : []),
];
const handleSelectionChange = (keys: Selection) => {
setSelectedKeys(keys);
if (!selectOnRowClick || !handleRowSelect) return;
if (keys === "all") return;
const id = Array.from(keys)[0];
if (!id) return;
const selected = visibleItems.find((c) => String(c.id) === String(id));
if (selected) handleRowSelect(selected);
};
return (
<Table
aria-label="Contests table"
selectionMode="single"
selectedKeys={selectedKeys}
onSelectionChange={handleSelectionChange}
>
<TableHeader columns={columns}>
{(column) => (
<TableColumn key={column.key} className="text-left">
{column.label}
</TableColumn>
)}
</TableHeader>
<TableBody items={visibleItems}>
{(item) => (
<TableRow key={item.id}>
{(columnKey) => {
switch (columnKey) {
case "name":
return (
<TableCell>
<span className="font-medium">{item.name}</span>
</TableCell>
);
case "description":
return (
<TableCell>
<span className="text-xs text-foreground-500 line-clamp-2">
{item.description || "—"}
</span>
</TableCell>
);
case "bands": {
const names = item.bands?.map((b) => b.name).join(", ");
return (
<TableCell>
<span className="text-xs text-foreground-500 line-clamp-2">
{names || "—"}
</span>
</TableCell>
);
}
case "categories": {
const names = item.categories?.map((c) => c.name).join(", ");
return (
<TableCell>
<span className="text-xs text-foreground-500 line-clamp-2">
{names || "—"}
</span>
</TableCell>
);
}
case "power_categories": {
const names = item.power_categories?.map((p) => p.name).join(", ");
return (
<TableCell>
<span className="text-xs text-foreground-500 line-clamp-2">
{names || "—"}
</span>
</TableCell>
);
}
case "is_active":
return (
<TableCell>
{item.is_active
? t("yes") ?? "Ano"
: t("no") ?? "Ne"}
</TableCell>
);
case "actions":
return (
<TableCell>
{canEdit && (
<button
type="button"
onClick={() => handleEdit?.(item)}
className="inline-flex items-center p-1 rounded hover:bg-default-100 text-default-500 hover:text-default-700"
aria-label={t("edit_contest") ?? "Edit contest"}
>
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 20 20"
fill="currentColor"
className="h-4 w-4"
>
<path d="M13.586 3.586a2 2 0 0 1 2.828 2.828l-8.25 8.25a2 2 0 0 1-.878.518l-3.122.89a.75.75 0 0 1-.926-.926l.89-3.122a2 2 0 0 1 .518-.878l8.25-8.25Z" />
<path d="M12.475 4.697 4.75 12.422l-.69 2.422 2.422-.69 7.725-7.725-1.732-1.732Z" />
</svg>
</button>
)}
</TableCell>
);
default:
return <TableCell />;
}
}}
</TableRow>
)}
</TableBody>
</Table>
);
}