feat: started working on competition runner page, updated colors, placed mock elems in separated file

This commit is contained in:
rngsurrounded
2025-03-01 18:19:56 +09:00
parent 049c31e394
commit 36f5ed8a63
7 changed files with 200 additions and 69 deletions
@@ -3,26 +3,9 @@ import { useParams, useNavigate } from "react-router-dom";
import Navbar from "@/modules/Navbar";
import { Button } from "@/components/ui/button";
import { ArrowLeft } from "lucide-react";
import { Competition, Status } from "@/shared/types/types";
import { Competition } from "@/shared/types/types";
import { mockCompetitions, mockTasks } from "@/shared/mocks/mocks";
const mockCompetitions: Competition[] = [
{
id: '1',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: true,
status: Status.InProgress,
description: 'Проверка глубоких знаний и навыков в анализе данных. Будет несколько творческих заданий со свободным ответом. Задания выполняются индивидуально, вес тура в итоговом результате – 0,5. Этап пройдет онлайн в заданное время, с применением системы прокторинга. На работу дается 240 минут.'
},
{
id: '2',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: false,
status: Status.NotParticipating,
description: 'Индивидуальный этап олимпиады DANO 2025 – это уникальная возможность для студентов продемонстрировать свои навыки анализа данных и решения сложных задач. Участники будут работать с реальными наборами данных и применять современные методы машинного обучения и статистического анализа.'
},
];
const CompetitionPreview = () => {
const { id } = useParams<{ id: string }>();
@@ -52,7 +35,16 @@ const CompetitionPreview = () => {
};
const handleContinue = () => {
console.log("Continue to competition:", competition?.id);
if (competition?.id) {
const competitionTasks = mockTasks[competition.id];
if (competitionTasks && competitionTasks.length > 0) {
const firstTaskId = competitionTasks[0].id;
navigate(`/competition/${competition.id}/tasks/${firstTaskId}`);
} else {
navigate(`/competition/${competition.id}/tasks`);
}
}
};
return (
@@ -0,0 +1,98 @@
import { useState } from "react";
import { useParams } from "react-router-dom";
import Navbar from "@/modules/Navbar";
import { Task, TaskStatus } from "@/shared/types/types";
const sampleTasks: Task[] = [
{ id: "1", number: "1.1", status: "uncleared" },
{ id: "2", number: "1.2", status: "checking" },
{ id: "3", number: "1.3", status: "correct" },
{ id: "4", number: "2.1", status: "partial" },
{ id: "5", number: "2.2", status: "wrong" },
{ id: "6", number: "2.3", status: "uncleared" },
{ id: "7", number: "3.1", status: "checking" },
{ id: "8", number: "3.2", status: "correct" },
];
const CompetitionRunnerPage = () => {
const { id } = useParams<{ id: string }>();
const [competitionTitle, setCompetitionTitle] = useState("Олимпиада DANO 2025. Индивидуальный этап");
const [tasks, setTasks] = useState<Task[]>(sampleTasks);
const [selectedTaskId, setSelectedTaskId] = useState<string | null>(null);
const getTaskBgColor = (status: TaskStatus): string => {
switch (status) {
case "uncleared": return "bg-[var(--color-task-uncleared)]";
case "checking": return "bg-[var(--color-task-checking)]";
case "correct": return "bg-[var(--color-task-correct)]";
case "partial": return "bg-[var(--color-task-partial)]";
case "wrong": return "bg-[var(--color-task-wrong)]";
}
};
const getTaskTextColor = (status: TaskStatus): string => {
switch (status) {
case "uncleared": return "text-gray-600";
case "checking": return "text-gray-800";
case "correct": return "text-green-800";
case "partial": return "text-green-700";
case "wrong": return "text-red-800";
}
};
const handleTaskClick = (taskId: string) => {
setSelectedTaskId(taskId);
};
return (
<>
<Navbar />
<div className="sticky top-16 z-10 bg-white border-b border-gray-200 shadow-sm">
<div className="container mx-auto px-4">
<div className="py-4">
<h1 className="text-xl font-semibold font-hse-sans">{competitionTitle}</h1>
</div>
<div className="flex items-center gap-3 pb-4 overflow-x-auto scrollbar-thin scrollbar-thumb-gray-300">
{tasks.map((task) => (
<div
key={task.id}
className={`${getTaskBgColor(task.status)} ${getTaskTextColor(task.status)}
rounded-lg px-4 py-2 font-medium text-sm font-hse-sans cursor-pointer
transition-transform hover:scale-105 flex-shrink-0
${selectedTaskId === task.id ? 'ring-2 ring-black' : ''}`}
onClick={() => handleTaskClick(task.id)}
>
{task.number}
</div>
))}
</div>
</div>
</div>
<div className="container mx-auto px-4 py-8">
<div className="bg-white rounded-lg p-6 shadow-sm">
{selectedTaskId ? (
<div className="font-hse-sans">
<h2 className="text-lg font-medium mb-4">
Задание {tasks.find(t => t.id === selectedTaskId)?.number}
</h2>
<p className="text-gray-700">
Содержание задания будет отображаться здесь.
</p>
</div>
) : (
<p className="font-hse-sans text-gray-500">
Выберите задание для просмотра
</p>
)}
</div>
</div>
</>
);
};
export default CompetitionRunnerPage;
@@ -5,52 +5,7 @@ import { Alert, AlertDescription } from "@/components/ui/alert";
import { AlertCircle } from "lucide-react";
import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs";
import Navbar from '@/modules/Navbar';
const mockCompetitions: Competition[] = [
{
id: '1',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: true,
status: Status.InProgress
},
{
id: '2',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: false,
status: Status.NotParticipating
},
{
id: '3',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: false,
status: Status.InProgress
},
{
id: '4',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: true,
status: Status.Completed
},
{
id: '5',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: false,
status: Status.Completed
},
{
id: '6',
name: 'Олимпиада DANO 2025. Индивидуальный этап',
imageUrl: '/DANO.png',
isOlympics: true,
status: Status.NotParticipating
}
];
import { mockCompetitions } from '@/shared/mocks/mocks';
const CompetitionsPage = () => {
const [competitions, setCompetitions] = useState<Competition[]>([]);