AlbumResource.php 8.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. <?php
  2. namespace App\Filament\Resources;
  3. use App\Filament\Resources\AlbumResource\Pages;
  4. use App\Models\Album;
  5. use App\Models\AlbumCategory;
  6. use App\Service\DeepLService;
  7. use Filament\Forms\Components\DatePicker;
  8. use Filament\Forms\Components\FileUpload;
  9. use Filament\Forms\Components\Group;
  10. use Filament\Forms\Components\Radio;
  11. use Filament\Forms\Components\Section;
  12. use Filament\Forms\Components\Select;
  13. use Filament\Forms\Components\TextInput;
  14. use Filament\Forms\Components\Toggle;
  15. use Filament\Forms\Form;
  16. use Filament\Forms\Get;
  17. use Filament\Resources\Resource;
  18. use Filament\Tables;
  19. use Filament\Tables\Columns\ImageColumn;
  20. use Filament\Tables\Columns\TextColumn;
  21. use Filament\Tables\Filters\SelectFilter;
  22. use Filament\Tables\Table;
  23. use Illuminate\Database\Eloquent\Builder;
  24. use Illuminate\Support\Facades\DB;
  25. use SolutionForest\FilamentTranslateField\Forms\Component\Translate;
  26. class AlbumResource extends Resource
  27. {
  28. protected static ?string $model = Album::class;
  29. protected static ?string $modelLabel = '影音管理';
  30. protected static ?string $navigationIcon = 'heroicon-o-photo';
  31. protected static ?string $navigationGroup = '最新消息';
  32. protected static ?string $navigationLabel = '影音管理';
  33. public static function form(Form $form): Form
  34. {
  35. return $form
  36. ->schema([
  37. Section::make('新增 影音')->schema([
  38. Group::make()->schema([
  39. Select::make('album_category_id')
  40. ->options(AlbumCategory::orderBy('order')->get()->pluck('name', 'id'))
  41. ->label('影音分類')
  42. ->required()
  43. ->columnSpan(1)
  44. ->native(false)
  45. ->live(),
  46. ])->columnSpanFull()->columns(2),
  47. Group::make()->schema([
  48. DatePicker::make('post_date')
  49. ->label('發布日期')
  50. ->closeOnDateSelection(),
  51. ])->columnSpanFull()->columns(2),
  52. FileUpload::make('news_banner')->label('列表大圖')
  53. ->image()
  54. ->optimize('webp')
  55. ->directory('album/img')
  56. ->imageResizeTargetWidth('1920')
  57. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  58. FileUpload::make('news_img_pc')->label('列表圖(desktop)')
  59. ->image()
  60. ->optimize('webp')
  61. ->directory('album/img')
  62. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  63. FileUpload::make('news_img_mobile')->label('列表圖(mobile)')
  64. ->image()
  65. ->optimize('webp')
  66. ->directory('album/img')
  67. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  68. Translate::make()->schema(fn (string $locale) => [
  69. TextInput::make('title')
  70. ->label('標題')
  71. ->columnSpan(1),
  72. ])
  73. ->locales(['zh_TW', 'en'])
  74. ->actions([
  75. app(DeepLService::class)->createTranslationAction('Main', ['title']),
  76. ])
  77. ->id('main')->columnSpanFull()->columns(3),
  78. Section::make('')->schema([
  79. Radio::make('upload_type')->label('')->options([
  80. 1 => '網址',
  81. 2 => '檔案',
  82. ])->columnSpanFull()->default(1)->live(),
  83. Group::make()->schema([
  84. TextInput::make('link_video')->label('網址')->nullable(),
  85. ])->visible(fn (Get $get): bool => $get('upload_type') == 1)->columnSpanFull(),
  86. Group::make()->schema([
  87. FileUpload::make('link_upload')->label('')->directory('album/video')
  88. ->acceptedFileTypes(['video/mp4', 'video/mpeg', 'video/quicktime', 'video/x-msvideo', 'video/x-ms-wmv'])
  89. ->helperText('建議影片寬高限制為:1920*1080px,出血寬度720px,大小限制為:100M以下')
  90. ->maxSize(102400)->nullable(),
  91. ])->visible(fn (Get $get): bool => $get('upload_type') == 2)->columnSpanFull(),
  92. ])->columnSpanFull(),
  93. Toggle::make('on_top')->inline()->label('置頂輪播')->columnSpanFull(),
  94. // Toggle::make('homepage_top')->inline()->label('在首頁輪播')->columnSpanFull(),
  95. ])->columns(3),
  96. ]);
  97. }
  98. public static function table(Table $table): Table
  99. {
  100. return $table
  101. ->columns([
  102. //
  103. TextColumn::make('albumCategory.name')->label('分類')->alignCenter(),
  104. TextColumn::make('title')->label('標題')->alignCenter(),
  105. TextColumn::make('post_date')->label('發布日期')->date()->alignCenter(),
  106. ImageColumn::make('news_img_pc')->label('列表圖(desktop)')->alignCenter(),
  107. TextColumn::make('list_audit_state')->label('狀態')->badge()
  108. ->color(fn (string $state): string => match ($state) {
  109. '暫存' => 'warning',
  110. '已發佈' => 'success',
  111. }),
  112. TextColumn::make('created_at')->label('建立時間')->dateTime()->alignCenter(),
  113. TextColumn::make('updated_at')->label('更新時間')->dateTime()->alignCenter(),
  114. ])
  115. ->filters([
  116. SelectFilter::make('post_date')->label('年份')
  117. ->options(Album::select(DB::raw("DATE_FORMAT(post_date, '%Y') as year"))->whereNotNull('post_date')->distinct()->pluck('year', 'year')->toArray())
  118. ->query(
  119. fn (array $data, Builder $query): Builder => $query->when(
  120. $data['value'],
  121. fn (Builder $query, $value): Builder => $query->where('post_date', 'like', $data['value'].'%')
  122. )
  123. ),
  124. SelectFilter::make('album_category_id')->label('分類')
  125. ->relationship('albumCategory', 'name')
  126. ->getOptionLabelFromRecordUsing(fn ($record, $livewire) => $record->getTranslation('name', 'zh_TW')),
  127. SelectFilter::make('visible')->label('狀態')
  128. ->options([
  129. 0 => '暫存',
  130. 1 => '已發佈',
  131. ])
  132. ->query(
  133. fn (array $data, Builder $query): Builder => $query->when(
  134. $data['value'],
  135. fn (Builder $query, $value): Builder => $query->where('visible', $data['value'])
  136. )
  137. ),
  138. ])
  139. ->actions([
  140. Tables\Actions\EditAction::make(),
  141. Tables\Actions\DeleteAction::make(),
  142. \Filament\Tables\Actions\Action::make('audit')
  143. ->label(fn ($record) => match ($record->visible) {
  144. 0 => '發佈',
  145. 1 => '下架',
  146. })
  147. ->color(fn ($record) => match ($record->visible) {
  148. 0 => 'warning',
  149. 1 => 'gray',
  150. })
  151. ->icon(fn ($record) => match ($record->visible) {
  152. 0 => 'heroicon-m-chevron-double-up',
  153. 1 => 'heroicon-m-chevron-double-down',
  154. })
  155. ->action(function ($record) {
  156. $record->visible = ! $record->visible;
  157. $record->save();
  158. })
  159. ->outlined()
  160. ->requiresConfirmation(),
  161. ])
  162. ->bulkActions([
  163. Tables\Actions\BulkActionGroup::make([
  164. Tables\Actions\DeleteBulkAction::make(),
  165. ]),
  166. ])
  167. ->reorderable('order')
  168. ->defaultSort('order');
  169. }
  170. public static function getRelations(): array
  171. {
  172. return [
  173. //
  174. ];
  175. }
  176. public static function getPages(): array
  177. {
  178. return [
  179. 'index' => Pages\ListAlbums::route('/'),
  180. 'create' => Pages\CreateAlbum::route('/create'),
  181. 'edit' => Pages\EditAlbum::route('/{record}/edit'),
  182. 'view' => Pages\ViewAlbum::route('/{record}/view'),
  183. ];
  184. }
  185. }