AlbumResource.php 9.0KB

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