StudentManagement.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <?php
  2. namespace App\Filament\Pages;
  3. use App\Filament\Traits\HasUserRole;
  4. use App\Filament\Widgets\StudentStatsWidget;
  5. use App\Models\Student;
  6. use App\Models\Teacher;
  7. use BackedEnum;
  8. use Filament\Actions\Action;
  9. use Filament\Actions\BulkAction;
  10. use Filament\Actions\BulkActionGroup;
  11. use Filament\Actions\CreateAction;
  12. use Filament\Tables;
  13. use Filament\Tables\Table;
  14. use Filament\Pages\Page;
  15. use Illuminate\Support\Facades\DB;
  16. use Filament\Tables\Concerns\InteractsWithTable;
  17. use Filament\Tables\Contracts\HasTable;
  18. use UnitEnum;
  19. class StudentManagement extends Page implements HasTable
  20. {
  21. use HasUserRole, InteractsWithTable;
  22. protected static string|BackedEnum|null $navigationIcon = 'heroicon-o-academic-cap';
  23. protected static ?string $navigationLabel = '师生管理';
  24. protected static string|UnitEnum|null $navigationGroup = '管理';
  25. protected static ?int $navigationSort = 16;
  26. protected string $view = 'filament.pages.student-management';
  27. public ?string $selectedTeacherId = null;
  28. public ?string $selectedTeacherName = null;
  29. public function mount(): void
  30. {
  31. $this->initializeUserRole();
  32. }
  33. public function getTitle(): string
  34. {
  35. return '师生管理';
  36. }
  37. public function getBreadcrumb(): string
  38. {
  39. return '师生管理';
  40. }
  41. public function filterByTeacher(?string $teacherId): void
  42. {
  43. $this->selectedTeacherId = $teacherId;
  44. if (! $teacherId) {
  45. $this->selectedTeacherName = null;
  46. return;
  47. }
  48. $teacher = Teacher::with('user')->find($teacherId);
  49. $this->selectedTeacherName = $teacher?->user?->full_name
  50. ?? $teacher?->name
  51. ?? '未知老师';
  52. }
  53. public function resetTeacherFilter(): void
  54. {
  55. $this->selectedTeacherId = null;
  56. $this->selectedTeacherName = null;
  57. }
  58. public function table(Table $table): Table
  59. {
  60. $currentUser = auth()->user();
  61. return $table
  62. ->query(
  63. Student::query()
  64. ->with(['teacher.user', 'user'])
  65. ->when($this->selectedTeacherId, fn ($query) => $query->where('teacher_id', $this->selectedTeacherId))
  66. ->when($currentUser?->isTeacher() ?? false, function ($query) use ($currentUser) {
  67. // 如果是老师登录,只显示该老师的学生
  68. $teacherId = $this->getCurrentTeacherId();
  69. if ($teacherId) {
  70. $query->where('teacher_id', $teacherId);
  71. }
  72. })
  73. )
  74. ->columns([
  75. Tables\Columns\TextColumn::make('student_id')
  76. ->label('学生ID')
  77. ->searchable()
  78. ->copyable()
  79. ->toggleable(isToggledHiddenByDefault: true),
  80. Tables\Columns\TextColumn::make('name')
  81. ->label('姓名')
  82. ->searchable()
  83. ->sortable()
  84. ->weight('bold'),
  85. Tables\Columns\TextColumn::make('grade')
  86. ->label('年级')
  87. ->sortable()
  88. ->badge()
  89. ->color(fn(string $state): string => match($state) {
  90. '一年级' => 'gray',
  91. '二年级' => 'gray',
  92. '三年级' => 'blue',
  93. '四年级' => 'blue',
  94. '五年级' => 'green',
  95. '六年级' => 'green',
  96. default => 'primary',
  97. }),
  98. Tables\Columns\TextColumn::make('class_name')
  99. ->label('班级')
  100. ->sortable(),
  101. Tables\Columns\TextColumn::make('teacher.user.full_name')
  102. ->label('指导老师')
  103. ->sortable()
  104. ->url(fn($record): string =>
  105. optional($record->teacher?->user)->email
  106. ? "mailto:{$record->teacher?->user?->email}"
  107. : ''
  108. )
  109. ->openUrlInNewTab()
  110. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时不显示老师列
  111. Tables\Columns\TextColumn::make('user.email')
  112. ->label('邮箱')
  113. ->copyable()
  114. ->toggleable(),
  115. Tables\Columns\TextColumn::make('user.login_count')
  116. ->label('登录次数')
  117. ->sortable()
  118. ->alignCenter()
  119. ->badge()
  120. ->color(fn(?int $state): string =>
  121. ($state ?? 0) === 0 ? 'danger' : (($state ?? 0) < 5 ? 'warning' : 'success')
  122. ),
  123. Tables\Columns\TextColumn::make('user.last_login')
  124. ->label('最后登录')
  125. ->dateTime('Y-m-d H:i')
  126. ->sortable()
  127. ->description(fn($record): string =>
  128. $record->user?->last_login ?
  129. \Carbon\Carbon::parse($record->user->last_login)->diffForHumans() :
  130. '从未登录'
  131. ),
  132. Tables\Columns\TextColumn::make('created_at')
  133. ->label('创建时间')
  134. ->dateTime('Y-m-d H:i')
  135. ->sortable()
  136. ->toggleable(isToggledHiddenByDefault: true),
  137. ])
  138. ->filters([
  139. Tables\Filters\SelectFilter::make('grade')
  140. ->label('年级')
  141. ->options(fn() => Student::query()
  142. ->orderBy('grade')
  143. ->pluck('grade', 'grade')
  144. ->filter()
  145. ->toArray()),
  146. Tables\Filters\SelectFilter::make('class_name')
  147. ->label('班级')
  148. ->options(fn() => Student::query()
  149. ->orderBy('class_name')
  150. ->pluck('class_name', 'class_name')
  151. ->filter()
  152. ->toArray()),
  153. Tables\Filters\SelectFilter::make('teacher_id')
  154. ->label('指导老师')
  155. ->options(fn() => Teacher::query()
  156. ->with('user')
  157. ->get()
  158. ->mapWithKeys(fn (Teacher $teacher) => [
  159. $teacher->teacher_id => $teacher->user->full_name
  160. ?? $teacher->name
  161. ?? "老师 #{$teacher->teacher_id}",
  162. ])
  163. ->toArray())
  164. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时隐藏
  165. Tables\Filters\Filter::make('has_logged_in')
  166. ->label('登录状态')
  167. ->query(fn($query) => $query->whereHas('user', fn ($sub) => $sub->whereNotNull('last_login'))),
  168. ])
  169. ->actions([
  170. Action::make('view')
  171. ->label('查看')
  172. ->icon('heroicon-o-eye')
  173. ->url(fn($record): string => route('filament.admin.resources.students.view', $record))
  174. ->openUrlInNewTab(),
  175. Action::make('edit')
  176. ->label('编辑')
  177. ->icon('heroicon-o-pencil-square')
  178. ->url(fn($record): string => route('filament.admin.resources.students.edit', $record))
  179. ->openUrlInNewTab(),
  180. Action::make('reset_password')
  181. ->label('重置密码')
  182. ->icon('heroicon-o-key')
  183. ->color('warning')
  184. ->action(function ($record) {
  185. $newPassword = 'student123';
  186. $hashedPassword = \Hash::make($newPassword);
  187. DB::table('users')
  188. ->where('user_id', $record->student_id)
  189. ->update(['password_hash' => $hashedPassword]);
  190. \Filament\Notifications\Notification::make()
  191. ->success()
  192. ->title('密码重置成功')
  193. ->body("学生 {$record->name} 的密码已重置为: {$newPassword}")
  194. ->send();
  195. })
  196. ->requiresConfirmation()
  197. ->modalHeading('重置学生密码')
  198. ->modalDescription('确定要重置该学生的密码吗?新密码将是: student123')
  199. ->modalSubmitActionLabel('确认重置'),
  200. ])
  201. ->bulkActions([
  202. BulkActionGroup::make([
  203. BulkAction::make('reset_passwords')
  204. ->label('批量重置密码')
  205. ->icon('heroicon-o-key')
  206. ->color('warning')
  207. ->action(function ($records) {
  208. $newPassword = 'student123';
  209. $hashedPassword = \Hash::make($newPassword);
  210. foreach ($records as $record) {
  211. DB::table('users')
  212. ->where('user_id', $record->student_id)
  213. ->update(['password_hash' => $hashedPassword]);
  214. }
  215. \Filament\Notifications\Notification::make()
  216. ->success()
  217. ->title('批量重置密码成功')
  218. ->body("已为 {$records->count()} 位学生重置密码为: {$newPassword}")
  219. ->send();
  220. })
  221. ->requiresConfirmation()
  222. ->modalHeading('批量重置密码')
  223. ->modalDescription('确定要重置所选学生的密码吗?新密码将是: student123')
  224. ->modalSubmitActionLabel('确认重置'),
  225. ]),
  226. ])
  227. ->searchPlaceholder('搜索学生姓名、ID或邮箱...')
  228. ->emptyStateHeading('暂无学生数据')
  229. ->emptyStateDescription('还没有添加任何学生数据。')
  230. ->emptyStateActions([
  231. CreateAction::make()
  232. ->label('添加新学生')
  233. ->url(route('filament.admin.resources.students.create'))
  234. ->icon('heroicon-o-plus'),
  235. ])
  236. ->paginated([10, 25, 50, 100])
  237. ->poll('60s'); // 每60秒刷新一次
  238. }
  239. public function getHeaderWidgets(): array
  240. {
  241. return [
  242. StudentStatsWidget::class,
  243. ];
  244. }
  245. public function getTeacherOverviewProperty(): array
  246. {
  247. $currentUser = auth()->user();
  248. // 如果是老师,只返回自己的信息
  249. if ($currentUser?->isTeacher() ?? false) {
  250. $teacher = Teacher::query()
  251. ->with([
  252. 'user',
  253. 'students' => fn ($query) => $query
  254. ->select('student_id', 'name', 'grade', 'class_name', 'teacher_id')
  255. ->orderBy('name'),
  256. ])
  257. ->where('teacher_id', $this->getCurrentTeacherId())
  258. ->withCount('students')
  259. ->withMax('students', 'updated_at')
  260. ->first();
  261. if (!$teacher) {
  262. return [];
  263. }
  264. return [[
  265. 'teacher_id' => $teacher->teacher_id,
  266. 'teacher_name' => $teacher->user->full_name ?? $teacher->name ?? '未命名老师',
  267. 'teacher_email' => $teacher->user->email ?? null,
  268. 'students_count' => $teacher->students_count ?? 0,
  269. 'latest_student_activity' => $teacher->students_max_updated_at,
  270. 'students' => $teacher->students
  271. ->sortBy('name')
  272. ->take(4)
  273. ->map(fn (Student $student) => [
  274. 'student_id' => $student->student_id,
  275. 'name' => $student->name,
  276. 'grade' => $student->grade,
  277. 'class_name' => $student->class_name,
  278. ])->values()->toArray(),
  279. ]];
  280. }
  281. // 如果是管理员,返回所有老师信息
  282. $teachers = Teacher::query()
  283. ->with([
  284. 'user',
  285. 'students' => fn ($query) => $query
  286. ->select('student_id', 'name', 'grade', 'class_name', 'teacher_id')
  287. ->orderBy('name'),
  288. ])
  289. ->withCount('students')
  290. ->withMax('students', 'updated_at')
  291. ->orderByDesc('students_count')
  292. ->get();
  293. return $teachers->map(function (Teacher $teacher) {
  294. return [
  295. 'teacher_id' => $teacher->teacher_id,
  296. 'teacher_name' => $teacher->user->full_name ?? $teacher->name ?? '未命名老师',
  297. 'teacher_email' => $teacher->user->email ?? null,
  298. 'students_count' => $teacher->students_count ?? 0,
  299. 'latest_student_activity' => $teacher->students_max_updated_at,
  300. 'students' => $teacher->students
  301. ->sortBy('name')
  302. ->take(4)
  303. ->map(fn (Student $student) => [
  304. 'student_id' => $student->student_id,
  305. 'name' => $student->name,
  306. 'grade' => $student->grade,
  307. 'class_name' => $student->class_name,
  308. ])->values()->toArray(),
  309. ];
  310. })->toArray();
  311. }
  312. }