StudentResource.php 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  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. ->default('未分配')
  104. ->sortable()
  105. ->searchable()
  106. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时不显示老师列
  107. ])
  108. ->filters([
  109. Tables\Filters\SelectFilter::make('grade')
  110. ->label('年级')
  111. ->options(fn () => self::gradeOptions())
  112. ->placeholder('全部年级'),
  113. Tables\Filters\SelectFilter::make('class_name')
  114. ->label('班级')
  115. ->options(fn () => self::classOptions())
  116. ->placeholder('全部班级'),
  117. Tables\Filters\SelectFilter::make('teacher_id')
  118. ->label('指导老师')
  119. ->options(fn () => self::teacherOptionsForCurrentUser())
  120. ->placeholder('全部老师')
  121. ->visible(fn () => !($currentUser?->isTeacher() ?? false)), // 老师登录时隐藏老师筛选
  122. ])
  123. ->actions([])
  124. ->bulkActions([])
  125. ->emptyStateHeading('暂无学生记录')
  126. ->emptyStateDescription('开始创建你的第一个学生吧')
  127. ->emptyStateActions([]);
  128. }
  129. public static function getPages(): array
  130. {
  131. return [
  132. 'index' => Pages\ListStudents::route('/'),
  133. 'create' => Pages\CreateStudent::route('/create'),
  134. 'view' => Pages\ViewStudent::route('/{record}'),
  135. 'edit' => Pages\EditStudent::route('/{record}/edit'),
  136. ];
  137. }
  138. protected static function teacherOptions(): array
  139. {
  140. // 使用缓存优化性能,缓存1小时
  141. return cache()->remember('teacher_options', 3600, function () {
  142. return Teacher::with(['user' => function ($query) {
  143. $query->select('user_id', 'full_name', 'role');
  144. }])
  145. ->whereHas('user', function ($query) {
  146. $query->where('role', 'teacher');
  147. })
  148. ->select('teacher_id', 'user_id', 'name')
  149. ->get()
  150. ->map(function ($teacher) {
  151. return [
  152. 'id' => $teacher->teacher_id,
  153. 'name' => $teacher->user->full_name ?? $teacher->name
  154. ];
  155. })
  156. ->pluck('name', 'id')
  157. ->toArray();
  158. });
  159. }
  160. protected static function teacherOptionsForCurrentUser(): array
  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. return self::teacherOptions();
  175. }
  176. protected static function gradeOptions(): array
  177. {
  178. // 使用缓存优化性能,缓存30分钟
  179. return cache()->remember('grade_options', 1800, function () {
  180. return Student::query()
  181. ->select('grade')
  182. ->distinct()
  183. ->whereNotNull('grade')
  184. ->orderBy('grade')
  185. ->pluck('grade', 'grade')
  186. ->toArray();
  187. });
  188. }
  189. protected static function classOptions(): array
  190. {
  191. // 使用缓存优化性能,缓存30分钟
  192. return Cache::remember('class_options', 1800, function () {
  193. return Student::query()
  194. ->select('class_name')
  195. ->whereNotNull('class_name')
  196. ->where('class_name', '!=', '')
  197. ->distinct()
  198. ->orderBy('class_name')
  199. ->pluck('class_name', 'class_name')
  200. ->toArray();
  201. });
  202. }
  203. /**
  204. * 清除相关缓存
  205. */
  206. public static function clearCaches(): void
  207. {
  208. Cache::forget('teacher_options');
  209. Cache::forget('grade_options');
  210. Cache::forget('class_options');
  211. }
  212. /**
  213. * 在保存前自动设置老师ID
  214. */
  215. public static function beforeSave(): void
  216. {
  217. $currentUser = auth()->user();
  218. if ($currentUser?->isTeacher() ?? false) {
  219. // 如果是老师,自动设置老师ID
  220. request()->merge(['teacher_id' => $currentUser->teacher?->teacher_id]);
  221. }
  222. }
  223. }