P0-1: 10 个页面补充 requirePermission 权限校验 P0-2: diagnostic/data-access-reports.ts 移除直查 users 表,改用 getUserNamesByIds P0-3: 新增 grade/grades/diagnostic 三组 i18n 翻译文件(zh-CN/en) P0-4: 新增 /management/grade 重定向页面 P1-2: 抽取 toNumber/normalize/buildScopeClassFilter 到 lib/grade-utils.ts P1-3: 为 12 个 Action 新增 Zod safeParse 校验(schema.ts +12 查询 schema) P1-4: 修复 as 断言违规,改用类型守卫函数 P2-2: 移除 diagnostic 组件中 Tailwind 任意值 同步更新架构图文档 004 和 005
67 lines
2.2 KiB
TypeScript
67 lines
2.2 KiB
TypeScript
import type { JSX } from "react"
|
|
import { notFound } from "next/navigation"
|
|
import { Stethoscope } from "lucide-react"
|
|
import { requirePermission } from "@/shared/lib/auth-guard"
|
|
import { Permissions } from "@/shared/types/permissions"
|
|
import {
|
|
getStudentMasterySummary,
|
|
getKnowledgePointStats,
|
|
} from "@/modules/diagnostic/data-access"
|
|
import { getDiagnosticReports } from "@/modules/diagnostic/data-access-reports"
|
|
import { StudentDiagnosticView } from "@/modules/diagnostic/components/student-diagnostic-view"
|
|
import type { MasteryRadarPoint } from "@/modules/diagnostic/types"
|
|
|
|
export const dynamic = "force-dynamic"
|
|
|
|
export default async function StudentDiagnosticPage({
|
|
params,
|
|
}: {
|
|
params: Promise<{ studentId: string }>
|
|
}): Promise<JSX.Element> {
|
|
const { studentId } = await params
|
|
const ctx = await requirePermission(Permissions.DIAGNOSTIC_READ)
|
|
|
|
// DataScope 二次校验:学生只能看自己,家长只能看子女
|
|
if (ctx.dataScope.type === "class_members" && ctx.userId !== studentId) {
|
|
notFound()
|
|
}
|
|
if (ctx.dataScope.type === "children" && !ctx.dataScope.childrenIds.includes(studentId)) {
|
|
notFound()
|
|
}
|
|
|
|
const [summary, reports, classStats] = await Promise.all([
|
|
getStudentMasterySummary(studentId),
|
|
getDiagnosticReports({ studentId }),
|
|
getKnowledgePointStats(),
|
|
])
|
|
|
|
// 班级平均掌握度(用于雷达图对比)
|
|
let classAverageMastery: MasteryRadarPoint[] | undefined
|
|
if (summary) {
|
|
classAverageMastery = classStats.map((k) => ({
|
|
knowledgePoint: k.knowledgePointName,
|
|
student: 0,
|
|
classAverage: k.averageMastery,
|
|
}))
|
|
}
|
|
|
|
return (
|
|
<div className="h-full flex-1 flex-col space-y-8 p-8 md:flex">
|
|
<div>
|
|
<h1 className="flex items-center gap-2 text-2xl font-bold tracking-tight">
|
|
<Stethoscope className="h-6 w-6" aria-hidden="true" />
|
|
Student Diagnostic
|
|
</h1>
|
|
<p className="text-muted-foreground">
|
|
Knowledge point mastery analysis and diagnostic reports.
|
|
</p>
|
|
</div>
|
|
<StudentDiagnosticView
|
|
summary={summary}
|
|
reports={reports}
|
|
classAverageMastery={classAverageMastery}
|
|
/>
|
|
</div>
|
|
)
|
|
}
|