AlbumResource.php 8.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  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. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  57. FileUpload::make('news_img_pc')->label('列表圖(desktop)')
  58. ->image()
  59. ->optimize('webp')
  60. ->directory('album/img')
  61. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  62. FileUpload::make('news_img_mobile')->label('列表圖(mobile)')
  63. ->image()
  64. ->optimize('webp')
  65. ->directory('album/img')
  66. ->acceptedFileTypes(['image/jpeg', 'image/jpg', 'image/png', 'image/webp'])->required()->imageEditor(),
  67. Translate::make()->schema(fn (string $locale) => [
  68. TextInput::make('title')
  69. ->label('標題')
  70. ->columnSpan(1),
  71. ])
  72. ->locales(['zh_TW', 'en'])
  73. ->actions([
  74. app(DeepLService::class)->createTranslationAction('Main', ['title']),
  75. ])
  76. ->id('main')->columnSpanFull()->columns(3),
  77. Section::make('')->schema([
  78. Radio::make('upload_type')->label('')->options([
  79. 1 => '網址',
  80. 2 => '檔案',
  81. ])->columnSpanFull()->default(1)->live(),
  82. Group::make()->schema([
  83. TextInput::make('link_video')->label('網址')->nullable(),
  84. ])->visible(fn (Get $get): bool => $get('upload_type') == 1)->columnSpanFull(),
  85. Group::make()->schema([
  86. FileUpload::make('link_upload')->label('')->directory('album/video')
  87. ->acceptedFileTypes(['video/mp4', 'video/mpeg', 'video/quicktime', 'video/x-msvideo', 'video/x-ms-wmv'])
  88. ->helperText('建議影片寬高限制為:1920*1080px,出血寬度720px,大小限制為:100M以下')
  89. ->maxSize(102400)->nullable(),
  90. ])->visible(fn (Get $get): bool => $get('upload_type') == 2)->columnSpanFull(),
  91. ])->columnSpanFull(),
  92. Toggle::make('on_top')->inline()->label('置頂輪播')->columnSpanFull(),
  93. // Toggle::make('homepage_top')->inline()->label('在首頁輪播')->columnSpanFull(),
  94. ])->columns(3),
  95. ]);
  96. }
  97. public static function table(Table $table): Table
  98. {
  99. return $table
  100. ->columns([
  101. //
  102. TextColumn::make('albumCategory.name')->label('分類')->alignCenter(),
  103. TextColumn::make('title')->label('標題')->alignCenter(),
  104. TextColumn::make('post_date')->label('發布日期')->date()->alignCenter(),
  105. ImageColumn::make('news_img_pc')->label('列表圖(desktop)')->alignCenter(),
  106. TextColumn::make('list_audit_state')->label('狀態')->badge()
  107. ->color(fn (string $state): string => match ($state) {
  108. '暫存' => 'warning',
  109. '已發佈' => 'success',
  110. }),
  111. TextColumn::make('created_at')->label('建立時間')->dateTime()->alignCenter(),
  112. TextColumn::make('updated_at')->label('更新時間')->dateTime()->alignCenter(),
  113. ])
  114. ->filters([
  115. SelectFilter::make('post_date')->label('年份')
  116. ->options(Album::select(DB::raw("DATE_FORMAT(post_date, '%Y') as year"))->whereNotNull('post_date')->distinct()->pluck('year', 'year')->toArray())
  117. ->query(
  118. fn (array $data, Builder $query): Builder => $query->when(
  119. $data['value'],
  120. fn (Builder $query, $value): Builder => $query->where('post_date', 'like', $data['value'].'%')
  121. )
  122. ),
  123. SelectFilter::make('album_category_id')->label('分類')
  124. ->relationship('albumCategory', 'name')
  125. ->getOptionLabelFromRecordUsing(fn ($record, $livewire) => $record->getTranslation('name', 'zh_TW')),
  126. SelectFilter::make('visible')->label('狀態')
  127. ->options([
  128. 0 => '暫存',
  129. 1 => '已發佈',
  130. ])
  131. ->query(
  132. fn (array $data, Builder $query): Builder => $query->when(
  133. $data['value'],
  134. fn (Builder $query, $value): Builder => $query->where('visible', $data['value'])
  135. )
  136. ),
  137. ])
  138. ->actions([
  139. Tables\Actions\EditAction::make(),
  140. Tables\Actions\DeleteAction::make(),
  141. \Filament\Tables\Actions\Action::make('audit')
  142. ->label(fn ($record) => match ($record->visible) {
  143. 0 => '發佈',
  144. 1 => '下架',
  145. })
  146. ->color(fn ($record) => match ($record->visible) {
  147. 0 => 'warning',
  148. 1 => 'gray',
  149. })
  150. ->icon(fn ($record) => match ($record->visible) {
  151. 0 => 'heroicon-m-chevron-double-up',
  152. 1 => 'heroicon-m-chevron-double-down',
  153. })
  154. ->action(function ($record) {
  155. $record->visible = ! $record->visible;
  156. $record->save();
  157. })
  158. ->outlined()
  159. ->requiresConfirmation(),
  160. ])
  161. ->bulkActions([
  162. Tables\Actions\BulkActionGroup::make([
  163. Tables\Actions\DeleteBulkAction::make(),
  164. ]),
  165. ])
  166. ->reorderable('order')
  167. ->defaultSort('order');
  168. }
  169. public static function getRelations(): array
  170. {
  171. return [
  172. //
  173. ];
  174. }
  175. public static function getPages(): array
  176. {
  177. return [
  178. 'index' => Pages\ListAlbums::route('/'),
  179. 'create' => Pages\CreateAlbum::route('/create'),
  180. 'edit' => Pages\EditAlbum::route('/{record}/edit'),
  181. 'view' => Pages\ViewAlbum::route('/{record}/view'),
  182. ];
  183. }
  184. }