StudentResource.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. <?php
  2. namespace App\Filament\Resources;
  3. use App\Filament\Resources\StudentResource\Pages;
  4. use App\Models\Student;
  5. use App\Models\Teacher;
  6. use BackedEnum;
  7. use Filament\Forms\Components\Select;
  8. use Filament\Forms\Components\Textarea;
  9. use Filament\Forms\Components\TextInput;
  10. use Filament\Resources\Resource;
  11. use Filament\Schemas\Components\Section;
  12. use Filament\Schemas\Schema;
  13. use Filament\Tables;
  14. use Filament\Tables\Table;
  15. use Illuminate\Support\Facades\Cache;
  16. class StudentResource extends Resource
  17. {
  18. protected static ?string $model = Student::class;
  19. protected static BackedEnum | string | null $navigationIcon = 'heroicon-o-academic-cap';
  20. protected static bool $shouldRegisterNavigation = false;
  21. public static function form(Schema $schema): Schema
  22. {
  23. $currentUser = auth()->user();
  24. $isTeacher = $currentUser?->isTeacher() ?? false;
  25. return $schema->schema([
  26. TextInput::make('student_id')
  27. ->label('学生ID')
  28. ->disabled()
  29. ->hidden(fn (?Student $record) => blank($record))
  30. ->formatStateUsing(fn (?Student $record): string => $record?->student_id ?? ''),
  31. TextInput::make('name')
  32. ->label('姓名')
  33. ->required()
  34. ->maxLength(128)
  35. ->placeholder('请输入学生姓名')
  36. ->autofocus(),
  37. TextInput::make('grade')
  38. ->label('年级')
  39. ->required()
  40. ->maxLength(32)
  41. ->placeholder('例如:高一、高二'),
  42. TextInput::make('class_name')
  43. ->label('班级')
  44. ->maxLength(64)
  45. ->placeholder('例如:1班、2班'),
  46. Select::make('teacher_id')
  47. ->label('指导老师')
  48. ->options(fn () => self::teacherOptionsForCurrentUser())
  49. ->searchable()
  50. ->required()
  51. ->preload()
  52. ->placeholder('请选择指导老师')
  53. ->hidden(fn () => $isTeacher),
  54. Textarea::make('remark')
  55. ->label('备注')
  56. ->rows(3)
  57. ->placeholder('选填')
  58. ->columnSpanFull(),
  59. ])->columns(2);
  60. }
  61. public static function table(Table $table): Table
  62. {
  63. $currentUser = auth()->user();
  64. return $table
  65. ->query(
  66. Student::query()
  67. ->with(['teacher.user', 'user'])
  68. ->when($currentUser?->isTeacher() ?? false, function ($query) use ($currentUser) {
  69. // 如果是老师登录,只显示该老师的学生
  70. $teacherId = $currentUser->teacher?->teacher_id;
  71. if ($teacherId) {
  72. $query->where('teacher_id', $teacherId);
  73. }
  74. })
  75. )
  76. ->columns([
  77. Tables\Columns\TextColumn::make('student_id')
  78. ->label('学生ID')
  79. ->badge()
  80. ->color('primary')
  81. ->copyable()
  82. ->copyMessage('学生ID已复制')
  83. ->copyMessageDuration(1500)
  84. ->sortable()
  85. ->searchable(),
  86. Tables\Columns\TextColumn::make('name')
  87. ->label('姓名')
  88. ->weight('bold')
  89. ->searchable()
  90. ->sortable(),
  91. Tables\Columns\TextColumn::make('grade')
  92. ->label('年级')
  93. ->badge()
  94. ->color('success')
  95. ->sortable(),
  96. Tables\Columns\TextColumn::make('class_name')
  97. ->label('班级')
  98. ->placeholder('未分配')
  99. ->sortable()
  100. ->formatStateUsing(fn ($state) => $state ?: '未分配'),
  101. Tables\Columns\TextColumn::make('teacher.user.full_name')
  102. ->label('指导老师')
  103. ->sortable()
  104. ->searchable()
  105. ->getStateUsing(fn (Student $record) => $record->teacher?->user?->full_name
  106. ?? $record->teacher?->name
  107. ?? '未分配')
  108. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时不显示老师列
  109. ])
  110. ->filters([
  111. Tables\Filters\SelectFilter::make('grade')
  112. ->label('年级')
  113. ->options(fn () => self::gradeOptions())
  114. ->placeholder('全部年级'),
  115. Tables\Filters\SelectFilter::make('class_name')
  116. ->label('班级')
  117. ->options(fn () => self::classOptions())
  118. ->placeholder('全部班级'),
  119. Tables\Filters\SelectFilter::make('teacher_id')
  120. ->label('指导老师')
  121. ->options(fn () => self::teacherOptionsForCurrentUser())
  122. ->placeholder('全部老师')
  123. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时隐藏老师筛选
  124. ])
  125. ->actions([])
  126. ->bulkActions([])
  127. ->emptyStateHeading('暂无学生记录')
  128. ->emptyStateDescription('开始创建你的第一个学生吧')
  129. ->emptyStateActions([]);
  130. }
  131. public static function getPages(): array
  132. {
  133. return [
  134. 'index' => Pages\ListStudents::route('/'),
  135. 'create' => Pages\CreateStudent::route('/create'),
  136. 'view' => Pages\ViewStudent::route('/{record}'),
  137. 'edit' => Pages\EditStudent::route('/{record}/edit'),
  138. ];
  139. }
  140. protected static function teacherOptions(): array
  141. {
  142. // 使用缓存优化性能,缓存1小时
  143. $options = cache()->remember('teacher_options', 3600, function () {
  144. return self::queryTeacherOptions();
  145. });
  146. // 如果缓存因历史关联错误导致为空,自动重建一次
  147. if (empty($options)) {
  148. $options = self::queryTeacherOptions();
  149. cache()->put('teacher_options', $options, 3600);
  150. }
  151. return $options;
  152. }
  153. protected static function teacherOptionsForCurrentUser(): array
  154. {
  155. $currentRecordId = request()->route('record') ?? request()->input('record');
  156. $currentStudentTeacher = null;
  157. if ($currentRecordId) {
  158. $currentStudentTeacher = \App\Models\Student::query()
  159. ->select('teacher_id')
  160. ->find($currentRecordId);
  161. }
  162. // 如果是老师登录,只返回自己的选项
  163. if (auth()->user()?->isTeacher() ?? false) {
  164. $teacherId = auth()->user()->teacher?->teacher_id;
  165. $teacherName = auth()->user()->teacher?->user?->full_name
  166. ?? auth()->user()->teacher?->name
  167. ?? '当前老师';
  168. if ($teacherId) {
  169. return [$teacherId => $teacherName];
  170. }
  171. return [];
  172. }
  173. // 如果是管理员,返回所有老师
  174. $options = self::teacherOptions();
  175. // 确保当前记录的老师在选项里,避免验证器 in 失败
  176. if ($currentStudentTeacher && $currentStudentTeacher->teacher_id && !isset($options[$currentStudentTeacher->teacher_id])) {
  177. $fallbackLabel = '老师 ' . $currentStudentTeacher->teacher_id;
  178. $options[$currentStudentTeacher->teacher_id] = $fallbackLabel;
  179. }
  180. return $options;
  181. }
  182. protected static function gradeOptions(): array
  183. {
  184. // 使用缓存优化性能,缓存30分钟
  185. return cache()->remember('grade_options', 1800, function () {
  186. return Student::query()
  187. ->select('grade')
  188. ->distinct()
  189. ->whereNotNull('grade')
  190. ->orderBy('grade')
  191. ->pluck('grade', 'grade')
  192. ->toArray();
  193. });
  194. }
  195. protected static function classOptions(): array
  196. {
  197. // 使用缓存优化性能,缓存30分钟
  198. return Cache::remember('class_options', 1800, function () {
  199. return Student::query()
  200. ->select('class_name')
  201. ->whereNotNull('class_name')
  202. ->where('class_name', '!=', '')
  203. ->distinct()
  204. ->orderBy('class_name')
  205. ->pluck('class_name', 'class_name')
  206. ->toArray();
  207. });
  208. }
  209. /**
  210. * 清除相关缓存
  211. */
  212. public static function clearCaches(): void
  213. {
  214. Cache::forget('teacher_options');
  215. Cache::forget('grade_options');
  216. Cache::forget('class_options');
  217. }
  218. /**
  219. * 查询老师选项(解耦缓存,便于复用和自动重建)
  220. */
  221. protected static function queryTeacherOptions(): array
  222. {
  223. $byRole = Teacher::with(['user' => function ($query) {
  224. $query->select('user_id', 'full_name', 'role');
  225. }])
  226. ->whereHas('user', function ($query) {
  227. $query->where('role', 'teacher');
  228. })
  229. ->select('teacher_id', 'user_id', 'name')
  230. ->get()
  231. ->map(function ($teacher) {
  232. return [
  233. 'id' => $teacher->teacher_id,
  234. 'name' => $teacher->user->full_name ?? $teacher->name ?? $teacher->teacher_id,
  235. ];
  236. })
  237. ->pluck('name', 'id')
  238. ->toArray();
  239. if (!empty($byRole)) {
  240. return $byRole;
  241. }
  242. // 兜底:直接列出 teachers 表的记录,避免选项为空导致 in 校验报错
  243. return Teacher::query()
  244. ->select('teacher_id', 'name')
  245. ->get()
  246. ->mapWithKeys(function ($teacher) {
  247. $label = $teacher->name ?: ('老师 ' . $teacher->teacher_id);
  248. return [$teacher->teacher_id => $label];
  249. })
  250. ->toArray();
  251. }
  252. /**
  253. * 在保存前自动设置老师ID
  254. */
  255. public static function beforeSave(): void
  256. {
  257. $currentUser = auth()->user();
  258. if ($currentUser?->isTeacher() ?? false) {
  259. // 如果是老师,自动设置老师ID
  260. request()->merge(['teacher_id' => $currentUser->teacher?->teacher_id]);
  261. }
  262. }
  263. }