feat(app): add error/loading boundaries and update dashboard routes

- Add error.tsx and loading.tsx boundaries for admin, parent, student, teacher routes

- Add dashboard-error-fallback and dashboard-loading-skeleton components

- Add student/learning page, parent/leave routes, teacher textbook components

- Update existing app routes across auth, dashboard, and API endpoints

- Update proxy middleware and next-auth type declarations
This commit is contained in:
SpecialX
2026-06-23 17:38:28 +08:00
parent c4d3433cc9
commit 1a9377222c
90 changed files with 1690 additions and 741 deletions

View File

@@ -5,6 +5,7 @@ import { requirePermission } from "@/shared/lib/auth-guard"
import { Permissions } from "@/shared/types/permissions"
import { getClassMasterySummary } from "@/modules/diagnostic/data-access"
import { ClassDiagnosticView } from "@/modules/diagnostic/components/class-diagnostic-view"
import { WidgetBoundary } from "@/modules/grades/components/widget-boundary"
export const dynamic = "force-dynamic"
@@ -41,7 +42,9 @@ export default async function ClassDiagnosticPage({
Class-level knowledge point mastery overview and student attention list.
</p>
</div>
<ClassDiagnosticView summary={summary} />
<WidgetBoundary title="班级学情诊断" skeletonHeight={400}>
<ClassDiagnosticView summary={summary} />
</WidgetBoundary>
</div>
)
}

View File

@@ -39,16 +39,19 @@ export default async function TeacherDiagnosticPage({
const reportType = getParam(sp, "reportType")
const status = getParam(sp, "status")
const reports = await getDiagnosticReports({
reportType: reportType && reportType !== "all" ? parseReportType(reportType) : undefined,
status: status && status !== "all" ? parseReportStatus(status) : undefined,
})
const reports = await getDiagnosticReports(
{
reportType: reportType && reportType !== "all" ? parseReportType(reportType) : undefined,
status: status && status !== "all" ? parseReportStatus(status) : undefined,
},
ctx.dataScope,
)
// 学生角色仅查看自己的报告;其他角色查看全部
const visibleReports =
ctx.dataScope.type === "class_members"
? reports.filter((r) => r.studentId === ctx.userId)
: reports
? reports.reports.filter((r) => r.studentId === ctx.userId)
: reports.reports
return (
<div className="h-full flex-1 flex-col space-y-8 p-8 md:flex">

View File

@@ -8,7 +8,9 @@ import {
getKnowledgePointStats,
} from "@/modules/diagnostic/data-access"
import { getDiagnosticReports } from "@/modules/diagnostic/data-access-reports"
import { getStudentActiveClassId } from "@/modules/classes/data-access"
import { StudentDiagnosticView } from "@/modules/diagnostic/components/student-diagnostic-view"
import { WidgetBoundary } from "@/modules/grades/components/widget-boundary"
import type { MasteryRadarPoint } from "@/modules/diagnostic/types"
export const dynamic = "force-dynamic"
@@ -29,11 +31,26 @@ export default async function StudentDiagnosticPage({
notFound()
}
const [summary, reports, classStats] = await Promise.all([
// v4-P1-2: class_taught scope 校验师生关系
// 教师只能查看自己所教班级的学生诊断
if (ctx.dataScope.type === "class_taught") {
const studentClassId = await getStudentActiveClassId(studentId)
if (!studentClassId || !ctx.dataScope.classIds.includes(studentClassId)) {
notFound()
}
}
// 先查询学生所属班级,再用 classId 调用 getKnowledgePointStats
// 否则无参调用会导致 studentIds=[] 直接返回空数组,班级平均对比功能失效
const studentClassId = await getStudentActiveClassId(studentId)
const [summary, reportsResult, classStats] = await Promise.all([
getStudentMasterySummary(studentId),
getDiagnosticReports({ studentId }),
getKnowledgePointStats(),
// v4-P1-3: 教师视角可查看所有状态报告(含草稿),便于审核
getDiagnosticReports({ studentId }, ctx.dataScope),
studentClassId ? getKnowledgePointStats(studentClassId) : Promise.resolve([]),
])
const reports = reportsResult.reports
// 班级平均掌握度(用于雷达图对比)
let classAverageMastery: MasteryRadarPoint[] | undefined
@@ -56,11 +73,14 @@ export default async function StudentDiagnosticPage({
Knowledge point mastery analysis and diagnostic reports.
</p>
</div>
<StudentDiagnosticView
summary={summary}
reports={reports}
classAverageMastery={classAverageMastery}
/>
<WidgetBoundary title="学生学情诊断" skeletonHeight={400}>
<StudentDiagnosticView
summary={summary}
reports={reports}
classAverageMastery={classAverageMastery}
practiceHrefBase="/teacher/questions"
/>
</WidgetBoundary>
</div>
)
}