177 lines
5.5 KiB
TypeScript
177 lines
5.5 KiB
TypeScript
'use client';
|
|
|
|
import { useMemo, useState } from "react";
|
|
import CourseCard from "./CourseCard";
|
|
|
|
type CourseStatus = "전체" | "수강 예정" | "수강중" | "수강 완료";
|
|
|
|
type Lesson = {
|
|
id: string;
|
|
title: string;
|
|
durationMin: number;
|
|
progressPct: number;
|
|
isCompleted: boolean;
|
|
};
|
|
|
|
type Course = {
|
|
id: string;
|
|
title: string;
|
|
description: string;
|
|
thumbnail: string;
|
|
status: CourseStatus;
|
|
progressPct: number;
|
|
lessons: Lesson[];
|
|
};
|
|
|
|
const COURSES: Course[] = [
|
|
{
|
|
id: "c1",
|
|
title: "원자로 운전 및 계통",
|
|
description:
|
|
"원자로 운전 원리와 주요 계통의 구조 및 기능을 이해하고, 일련 단계 가이드 기반의 현업에서 배워야 할 핵심을 익힙니다.",
|
|
thumbnail: "/imgs/talk.png",
|
|
status: "수강중",
|
|
progressPct: 80,
|
|
lessons: [
|
|
{ id: "c1l1", title: "1. 원자로 기초 및 핵분열 원리", durationMin: 12, progressPct: 100, isCompleted: true },
|
|
{ id: "c1l2", title: "2. 제어봉 및 원자로 출력 제어", durationMin: 18, progressPct: 100, isCompleted: true },
|
|
{ id: "c1l3", title: "3. 터빈-발전기 계통 및 전력 생산", durationMin: 13, progressPct: 60, isCompleted: false },
|
|
{ id: "c1l4", title: "4. 보조 계통 및 안전 계통", durationMin: 13, progressPct: 0, isCompleted: false },
|
|
{ id: "c1l5", title: "5. 원자로 냉각재 계통 (RCS) 및 열수력", durationMin: 8, progressPct: 0, isCompleted: false },
|
|
{ id: "c1l6", title: "6. 원자로 시동, 운전 및 정지 절차", durationMin: 13, progressPct: 0, isCompleted: false },
|
|
],
|
|
},
|
|
{
|
|
id: "c2",
|
|
title: "확률론",
|
|
description:
|
|
"확률과 통계의 주요 개념을 가이드하며, 일련 단계 실습을 기반으로 문제 해결력을 기릅니다.",
|
|
thumbnail: "/imgs/talk.png",
|
|
status: "수강 완료",
|
|
progressPct: 100,
|
|
lessons: [
|
|
{ id: "c2l1", title: "기초 개념", durationMin: 10, progressPct: 100, isCompleted: true },
|
|
],
|
|
},
|
|
{
|
|
id: "c3",
|
|
title: "부서간 협업",
|
|
description:
|
|
"부서간 협업 절차와 기록 기준을 가이드하며, 협업 중 생길 수 있는 리스크 관리법을 다룹니다.",
|
|
thumbnail: "/imgs/talk.png",
|
|
status: "수강중",
|
|
progressPct: 60,
|
|
lessons: [
|
|
{ id: "c3l1", title: "의사소통 원칙", durationMin: 9, progressPct: 100, isCompleted: true },
|
|
{ id: "c3l2", title: "문서 공유와 승인", durationMin: 14, progressPct: 30, isCompleted: false },
|
|
],
|
|
},
|
|
{
|
|
id: "c4",
|
|
title: "방사선의 이해",
|
|
description:
|
|
"방사선 안전 기준을 가이드하며, 일과 관련된 위험과 보호 장비 선택법을 배웁니다.",
|
|
thumbnail: "/imgs/talk.png",
|
|
status: "수강 예정",
|
|
progressPct: 0,
|
|
lessons: [{ id: "c4l1", title: "기초 이론", durationMin: 12, progressPct: 0, isCompleted: false }],
|
|
},
|
|
];
|
|
|
|
const TABS: CourseStatus[] = ["전체", "수강 예정", "수강중", "수강 완료"];
|
|
|
|
export default function CoursesPage() {
|
|
const [activeTab, setActiveTab] = useState<CourseStatus>("전체");
|
|
|
|
const countsByStatus = useMemo(() => {
|
|
return {
|
|
전체: COURSES.length,
|
|
"수강 예정": COURSES.filter((c) => c.status === "수강 예정").length,
|
|
수강중: COURSES.filter((c) => c.status === "수강중").length,
|
|
"수강 완료": COURSES.filter((c) => c.status === "수강 완료").length,
|
|
};
|
|
}, []);
|
|
|
|
const filtered = useMemo(() => {
|
|
if (activeTab === "전체") return COURSES;
|
|
return COURSES.filter((c) => c.status === activeTab);
|
|
}, [activeTab]);
|
|
|
|
return (
|
|
<main className="flex w-full flex-col">
|
|
<div className="flex h-[100px] items-center px-8">
|
|
<h1 className="text-[24px] font-bold leading-[1.5] text-[#1b2027]">내 강좌실</h1>
|
|
</div>
|
|
|
|
<div className="px-8">
|
|
<div className="border-b border-[#ecf0ff]">
|
|
<ul className="flex gap-6">
|
|
{TABS.map((tab) => {
|
|
const isActive = activeTab === tab;
|
|
return (
|
|
<li key={tab}>
|
|
<button
|
|
type="button"
|
|
onClick={() => setActiveTab(tab)}
|
|
className={[
|
|
"relative -mb-px h-12 px-1 text-[14px] leading-[1.5]",
|
|
isActive ? "font-semibold text-[#1f2b91]" : "font-medium text-[#6c7682]",
|
|
].join(" ")}
|
|
>
|
|
{tab}{" "}
|
|
<span className="ml-1 text-[#8c95a1]">
|
|
{countsByStatus[tab as keyof typeof countsByStatus] ?? 0}
|
|
</span>
|
|
{isActive ? (
|
|
<span className="absolute inset-x-0 -bottom-[1px] h-[2px] bg-[#1f2b91]" />
|
|
) : null}
|
|
</button>
|
|
</li>
|
|
);
|
|
})}
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="px-8 pb-16 pt-6">
|
|
<div className="flex flex-col gap-4">
|
|
{filtered.map((course) => (
|
|
<CourseCard key={course.id} course={course} />
|
|
))}
|
|
</div>
|
|
|
|
{/* pagination */}
|
|
<div className="mt-10 flex items-center justify-center gap-2">
|
|
<button
|
|
type="button"
|
|
className="flex h-8 w-8 items-center justify-center rounded-full text-[#8c95a1] hover:bg-[#f1f3f5]"
|
|
aria-label="이전 페이지"
|
|
>
|
|
‹
|
|
</button>
|
|
{[1, 2, 3].map((p) => (
|
|
<button
|
|
key={p}
|
|
type="button"
|
|
className={[
|
|
"flex h-8 w-8 items-center justify-center rounded-full text-[13px] leading-[1.4]",
|
|
p === 2 ? "bg-[#1f2b91] text-white" : "text-[#4c5561] hover:bg-[#f1f3f5]",
|
|
].join(" ")}
|
|
>
|
|
{p}
|
|
</button>
|
|
))}
|
|
<button
|
|
type="button"
|
|
className="flex h-8 w-8 items-center justify-center rounded-full text-[#8c95a1] hover:bg-[#f1f3f5]"
|
|
aria-label="다음 페이지"
|
|
>
|
|
›
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</main>
|
|
);
|
|
}
|
|
|