mirror of
https://gitlab.com/megazordpobeda/DataRush.git
synced 2026-05-23 01:37:11 +00:00
feat: started working on competition runner page, updated colors, placed mock elems in separated file
This commit is contained in:
@@ -1,13 +1,16 @@
|
|||||||
import { Routes, Route } from "react-router";
|
import { Routes, Route } from "react-router";
|
||||||
import "./styles/globals.css";
|
import "./styles/globals.css";
|
||||||
import CompetitionsPage from "./pages/CompetitionsPage";
|
import CompetitionsPage from "./pages/CompetitionsPage";
|
||||||
import CompetitionPreview from "./pages/CompetitionPreview";
|
import CompetitionPreviewPage from "./pages/CompetitionPreviewPage";
|
||||||
|
import CompetitionRunnerPage from "./pages/CompetitionRunnerPage";
|
||||||
|
|
||||||
|
|
||||||
const App = () => {
|
const App = () => {
|
||||||
return (
|
return (
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path="/" element={<CompetitionsPage/>} />
|
<Route path="/" element={<CompetitionsPage/>} />
|
||||||
<Route path="/competition/:id" element={<CompetitionPreview />} />
|
<Route path="/competition/:id" element={<CompetitionPreviewPage />} />
|
||||||
|
<Route path="/competition/:id/tasks/:taskId" element={<CompetitionRunnerPage />} />
|
||||||
</Routes>
|
</Routes>
|
||||||
|
|
||||||
);
|
);
|
||||||
|
|||||||
+12
-20
@@ -3,26 +3,9 @@ import { useParams, useNavigate } from "react-router-dom";
|
|||||||
import Navbar from "@/modules/Navbar";
|
import Navbar from "@/modules/Navbar";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { ArrowLeft } from "lucide-react";
|
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 CompetitionPreview = () => {
|
||||||
const { id } = useParams<{ id: string }>();
|
const { id } = useParams<{ id: string }>();
|
||||||
@@ -52,7 +35,16 @@ const CompetitionPreview = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleContinue = () => {
|
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 (
|
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 { AlertCircle } from "lucide-react";
|
||||||
import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
||||||
import Navbar from '@/modules/Navbar';
|
import Navbar from '@/modules/Navbar';
|
||||||
|
import { mockCompetitions } from '@/shared/mocks/mocks';
|
||||||
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
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
|
|
||||||
const CompetitionsPage = () => {
|
const CompetitionsPage = () => {
|
||||||
const [competitions, setCompetitions] = useState<Competition[]>([]);
|
const [competitions, setCompetitions] = useState<Competition[]>([]);
|
||||||
|
|||||||
@@ -0,0 +1,62 @@
|
|||||||
|
import { Competition, Status } from "../types/types";
|
||||||
|
|
||||||
|
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 – это уникальная возможность для студентов продемонстрировать свои навыки анализа данных и решения сложных задач. Участники будут работать с реальными наборами данных и применять современные методы машинного обучения и статистического анализа.'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
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
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
const mockTasks = {
|
||||||
|
'1': [
|
||||||
|
{ id: "1.1", number: "1.1", status: "uncleared" },
|
||||||
|
{ id: "1.2", number: "1.2", status: "checking" },
|
||||||
|
{ id: "1.3", number: "1.3", status: "correct" },
|
||||||
|
],
|
||||||
|
'2': [
|
||||||
|
{ id: "2.1", number: "1.1", status: "uncleared" },
|
||||||
|
{ id: "2.2", number: "1.2", status: "uncleared" },
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
export { mockCompetitions, mockTasks }
|
||||||
@@ -13,5 +13,13 @@ interface Competition {
|
|||||||
description?: string;
|
description?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TaskStatus = "uncleared" | "checking" | "correct" | "partial" | "wrong";
|
||||||
|
|
||||||
|
interface Task {
|
||||||
|
id: string;
|
||||||
|
number: string;
|
||||||
|
status: TaskStatus;
|
||||||
|
}
|
||||||
|
|
||||||
export {Status}
|
export {Status}
|
||||||
export type {Competition}
|
export type {Competition, TaskStatus, Task}
|
||||||
@@ -76,6 +76,12 @@
|
|||||||
--sidebar-accent-foreground: oklch(0.985 0 0);
|
--sidebar-accent-foreground: oklch(0.985 0 0);
|
||||||
--sidebar-border: oklch(0.269 0 0);
|
--sidebar-border: oklch(0.269 0 0);
|
||||||
--sidebar-ring: oklch(0.439 0 0);
|
--sidebar-ring: oklch(0.439 0 0);
|
||||||
|
|
||||||
|
--task-uncleared: oklch(0.955 0 0);
|
||||||
|
--task-checking: oklch(0.899 0.1763 97.07);
|
||||||
|
--task-correct: oklch(0.962 0.0561 158.62);
|
||||||
|
--task-partial: oklch(0.971 0.0616 131.35);
|
||||||
|
--task-wrong: oklch(0.906 0.0484 18.08);
|
||||||
}
|
}
|
||||||
|
|
||||||
@theme inline {
|
@theme inline {
|
||||||
@@ -115,6 +121,13 @@
|
|||||||
--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
|
--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
|
||||||
--color-sidebar-border: var(--sidebar-border);
|
--color-sidebar-border: var(--sidebar-border);
|
||||||
--color-sidebar-ring: var(--sidebar-ring);
|
--color-sidebar-ring: var(--sidebar-ring);
|
||||||
|
|
||||||
|
--color-task-uncleared: var(--task-uncleared);
|
||||||
|
--color-task-checking: var(--task-checking);
|
||||||
|
--color-task-correct: var(--task-correct);
|
||||||
|
--color-task-partial: var(--task-partial);
|
||||||
|
--color-task-wrong: var(--task-wrong);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
|
|||||||
Reference in New Issue
Block a user