P1-3: empty-state 默认按钮 variant 改为 outline 并新增 variant prop;button.tsx 导出 ButtonProps;统一 5 个详情页返回路径为 ghost+ArrowLeft+文字标签;course-plan-detail raw a 改为 Link。P2-1: formatLongDate 默认 locale 改为 zh-CN,weekday 改为 short;返回按钮文案中文化;course-plan-detail 全量中文化;grades/analytics 标题中文化。验证:tsc 0 错误,lint 0 错误,架构图 004/005 已同步。
100 lines
4.7 KiB
TypeScript
100 lines
4.7 KiB
TypeScript
import type { JSX } from "react"
|
|
import Link from "next/link"
|
|
import { notFound } from "next/navigation"
|
|
import { getHomeworkAssignmentAnalytics } from "@/modules/homework/data-access"
|
|
import { HomeworkAssignmentExamContentCard } from "@/modules/homework/components/homework-assignment-exam-content-card"
|
|
import { HomeworkAssignmentQuestionErrorOverviewCard } from "@/modules/homework/components/homework-assignment-question-error-overview-card"
|
|
import { Badge } from "@/shared/components/ui/badge"
|
|
import { Button } from "@/shared/components/ui/button"
|
|
import { formatDate } from "@/shared/lib/utils"
|
|
import { ArrowLeft, Users, Calendar, BarChart3, CheckCircle2 } from "lucide-react"
|
|
|
|
export const dynamic = "force-dynamic"
|
|
|
|
export default async function HomeworkAssignmentDetailPage({ params }: { params: Promise<{ id: string }> }): Promise<JSX.Element> {
|
|
const { id } = await params
|
|
const analytics = await getHomeworkAssignmentAnalytics(id)
|
|
|
|
if (!analytics) return notFound()
|
|
|
|
const { assignment, questions, gradedSampleCount } = analytics
|
|
|
|
return (
|
|
<div className="flex flex-col min-h-full">
|
|
{/* Header */}
|
|
<div className="border-b bg-background px-8 py-5">
|
|
<div className="flex flex-col gap-4 md:flex-row md:items-start md:justify-between">
|
|
<div className="min-w-0 flex flex-col gap-2">
|
|
<Button asChild variant="ghost" size="sm" className="w-fit">
|
|
<Link href="/teacher/homework/assignments">
|
|
<ArrowLeft className="mr-2 h-4 w-4" aria-hidden="true" />
|
|
返回作业列表
|
|
</Link>
|
|
</Button>
|
|
<div className="flex items-center gap-3">
|
|
<h1 className="text-2xl font-bold tracking-tight text-foreground line-clamp-2">{assignment.title}</h1>
|
|
<Badge variant={assignment.status === "published" ? "default" : "secondary"} className="capitalize">
|
|
{assignment.status}
|
|
</Badge>
|
|
</div>
|
|
<p className="text-muted-foreground text-sm max-w-2xl">{assignment.description || "No description provided."}</p>
|
|
</div>
|
|
|
|
<div className="flex items-center gap-3 mt-2 md:mt-0">
|
|
<Button asChild variant="outline" className="shadow-sm">
|
|
<Link href={`/teacher/homework/assignments/${assignment.id}/submissions`}>
|
|
<Users className="h-4 w-4 mr-2" aria-hidden="true" />
|
|
View Submissions
|
|
</Link>
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Quick Stats Row */}
|
|
<div className="flex flex-wrap gap-x-8 gap-y-2 mt-6 text-sm">
|
|
<div className="flex items-center gap-2 text-muted-foreground">
|
|
<Calendar className="h-4 w-4" aria-hidden="true" />
|
|
<span>Due: <span className="font-medium text-foreground tabular-nums">{assignment.dueAt ? formatDate(assignment.dueAt) : "No due date"}</span></span>
|
|
</div>
|
|
<div className="flex items-center gap-2 text-muted-foreground">
|
|
<Users className="h-4 w-4" aria-hidden="true" />
|
|
<span>Targets: <span className="font-medium text-foreground tabular-nums">{assignment.targetCount}</span></span>
|
|
</div>
|
|
<div className="flex items-center gap-2 text-muted-foreground">
|
|
<CheckCircle2 className="h-4 w-4" aria-hidden="true" />
|
|
<span>Submissions: <span className="font-medium text-foreground tabular-nums">{assignment.submissionCount}</span></span>
|
|
</div>
|
|
<div className="flex items-center gap-2 text-muted-foreground">
|
|
<BarChart3 className="h-4 w-4" aria-hidden="true" />
|
|
<span>Graded: <span className="font-medium text-foreground tabular-nums">{gradedSampleCount}</span></span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex-1 p-8 space-y-8 bg-muted/5">
|
|
{/* Analytics Section */}
|
|
<section className="space-y-4">
|
|
<div className="flex items-center justify-between">
|
|
<h2 className="text-lg font-semibold tracking-tight">Performance Analytics</h2>
|
|
</div>
|
|
<div className="grid gap-6 md:grid-cols-1">
|
|
<HomeworkAssignmentQuestionErrorOverviewCard questions={questions} gradedSampleCount={gradedSampleCount} />
|
|
</div>
|
|
</section>
|
|
|
|
{/* Content Section */}
|
|
<section className="space-y-4">
|
|
<div className="flex items-center justify-between">
|
|
<h2 className="text-lg font-semibold tracking-tight">Assignment Content</h2>
|
|
</div>
|
|
<HomeworkAssignmentExamContentCard
|
|
structure={assignment.structure}
|
|
questions={questions}
|
|
gradedSampleCount={gradedSampleCount}
|
|
/>
|
|
</section>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|