fix: gabungkan MyPostResource ke PostResource dengan scope dan UI adaptif per role

This commit is contained in:
2026-04-03 07:01:46 +07:00
parent a7e10600d4
commit 764aa4d82f
10 changed files with 62 additions and 202 deletions
@@ -1,46 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts;
use App\Filament\Resources\MyPosts\Pages\CreateMyPost;
use App\Filament\Resources\MyPosts\Pages\EditMyPost;
use App\Filament\Resources\MyPosts\Pages\ListMyPosts;
use App\Models\Post;
use Filament\Resources\Resource;
use Filament\Schemas\Schema;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
class MyPostResource extends Resource
{
protected static ?string $model = Post::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-pencil-square';
protected static string|\UnitEnum|null $navigationGroup = 'Konten';
protected static ?string $modelLabel = 'Artikel Saya';
protected static ?string $slug = 'my-posts';
// Hanya tampilkan artikel milik user yang login
public static function getEloquentQuery(): Builder
{
return parent::getEloquentQuery()->where('author_id', auth()->id());
}
public static function form(Schema $form): Schema
{
return \App\Filament\Resources\MyPosts\Schemas\MyPostForm::configure($form);
}
public static function table(Table $table): Table
{
return \App\Filament\Resources\MyPosts\Tables\MyPostsTable::configure($table);
}
public static function getPages(): array
{
return [
'index' => ListMyPosts::route('/'),
'create' => CreateMyPost::route('/create'),
'edit' => EditMyPost::route('/{record}/edit'),
];
}
}
@@ -1,11 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts\Pages;
use App\Filament\Resources\MyPosts\MyPostResource;
use Filament\Resources\Pages\CreateRecord;
class CreateMyPost extends CreateRecord
{
protected static string $resource = MyPostResource::class;
}
@@ -1,19 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts\Pages;
use App\Filament\Resources\MyPosts\MyPostResource;
use Filament\Actions\DeleteAction;
use Filament\Resources\Pages\EditRecord;
class EditMyPost extends EditRecord
{
protected static string $resource = MyPostResource::class;
protected function getHeaderActions(): array
{
return [
DeleteAction::make(),
];
}
}
@@ -1,19 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts\Pages;
use App\Filament\Resources\MyPosts\MyPostResource;
use Filament\Actions\CreateAction;
use Filament\Resources\Pages\ListRecords;
class ListMyPosts extends ListRecords
{
protected static string $resource = MyPostResource::class;
protected function getHeaderActions(): array
{
return [
CreateAction::make(),
];
}
}
@@ -1,30 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts\Schemas;
use Filament\Forms\Components\RichEditor;
use Filament\Forms\Components\Select;
use Filament\Forms\Components\TextInput;
use Filament\Schemas\Schema;
use Illuminate\Support\Str;
class MyPostForm
{
public static function configure(Schema $schema): Schema
{
return $schema->components([
TextInput::make('title')->label('Judul')->required()
->live(onBlur: true)
->afterStateUpdated(fn ($state, $set) => $set('slug', Str::slug($state))),
TextInput::make('slug')->required()->unique(ignoreRecord: true),
Select::make('category')->label('Kategori')
->options([
'umum' => 'Umum',
'pengumuman' => 'Pengumuman',
'berita' => 'Berita',
])
->default('umum')->required(),
RichEditor::make('content')->label('Konten')->required()->columnSpanFull(),
]);
}
}
@@ -1,59 +0,0 @@
<?php
namespace App\Filament\Resources\MyPosts\Tables;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Table;
class MyPostsTable
{
public static function configure(Table $table): Table
{
return $table
->columns([
TextColumn::make('title')->label('Judul')->searchable()->sortable(),
TextColumn::make('category')->label('Kategori')->badge(),
TextColumn::make('status')->badge()
->color(fn ($state) => match ($state) {
'published' => 'success',
'pending' => 'warning',
'rejected' => 'danger',
default => 'gray',
}),
TextColumn::make('rejection_reason')->label('Alasan Penolakan')
->visible(fn ($record) => $record?->status === 'rejected')
->limit(40)->default('-'),
TextColumn::make('created_at')->label('Dibuat')->date('d M Y')->sortable(),
])
->recordActions([
// Hanya bisa edit jika masih draft atau rejected
EditAction::make()
->visible(fn ($record) => in_array($record->status, ['draft', 'rejected'])),
// Ajukan untuk review
Action::make('submit')
->label('Ajukan')
->icon('heroicon-o-paper-airplane')
->color('info')
->requiresConfirmation()
->visible(fn ($record) => in_array($record->status, ['draft', 'rejected']))
->action(fn ($record) => $record->update(['status' => 'pending', 'rejection_reason' => null])),
])
->toolbarActions([
BulkActionGroup::make([
DeleteBulkAction::make()
->before(function ($records) {
// Tidak bisa hapus yang sudah published
$records->each(function ($record) {
if ($record->status === 'published') {
throw new \Exception("Artikel yang sudah diterbitkan tidak dapat dihapus.");
}
});
}),
]),
]);
}
}
+21 -1
View File
@@ -11,13 +11,33 @@ use App\Models\Post;
use Filament\Resources\Resource; use Filament\Resources\Resource;
use Filament\Schemas\Schema; use Filament\Schemas\Schema;
use Filament\Tables\Table; use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
class PostResource extends Resource class PostResource extends Resource
{ {
protected static ?string $model = Post::class; protected static ?string $model = Post::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-newspaper'; protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-newspaper';
protected static string|\UnitEnum|null $navigationGroup = 'Konten'; protected static string|\UnitEnum|null $navigationGroup = 'Konten';
protected static ?string $modelLabel = 'Artikel';
// Label dinamis sesuai role
public static function getModelLabel(): string
{
return auth()->user()?->hasAnyRole(['super_admin', 'ketua', 'auditor'])
? 'Artikel'
: 'Artikel Saya';
}
// Scope: ketua/super_admin/auditor lihat semua, lainnya hanya milik sendiri
public static function getEloquentQuery(): Builder
{
$query = parent::getEloquentQuery();
if (auth()->user()?->hasAnyRole(['super_admin', 'ketua', 'auditor'])) {
return $query;
}
return $query->where('author_id', auth()->id());
}
public static function form(Schema $form): Schema public static function form(Schema $form): Schema
{ {
@@ -13,6 +13,8 @@ class PostForm
{ {
public static function configure(Schema $schema): Schema public static function configure(Schema $schema): Schema
{ {
$isAdmin = auth()->user()?->hasAnyRole(['super_admin', 'ketua']);
return $schema->components([ return $schema->components([
TextInput::make('title')->label('Judul')->required() TextInput::make('title')->label('Judul')->required()
->live(onBlur: true) ->live(onBlur: true)
@@ -26,6 +28,7 @@ class PostForm
]) ])
->default('umum')->required(), ->default('umum')->required(),
DateTimePicker::make('published_at')->label('Tanggal Publikasi') DateTimePicker::make('published_at')->label('Tanggal Publikasi')
->visible($isAdmin)
->helperText('Kosongkan untuk menyimpan sebagai draft'), ->helperText('Kosongkan untuk menyimpan sebagai draft'),
RichEditor::make('content')->label('Konten')->required()->columnSpanFull(), RichEditor::make('content')->label('Konten')->required()->columnSpanFull(),
]); ]);
@@ -15,6 +15,8 @@ class PostsTable
{ {
public static function configure(Table $table): Table public static function configure(Table $table): Table
{ {
$isAdmin = auth()->user()?->hasAnyRole(['super_admin', 'ketua']);
return $table return $table
->columns([ ->columns([
TextColumn::make('title')->label('Judul')->searchable()->sortable(), TextColumn::make('title')->label('Judul')->searchable()->sortable(),
@@ -31,7 +33,10 @@ class PostsTable
'rejected' => 'danger', 'rejected' => 'danger',
default => 'gray', default => 'gray',
}), }),
TextColumn::make('author.name')->label('Penulis'), TextColumn::make('author.name')->label('Penulis')->visible($isAdmin),
TextColumn::make('rejection_reason')->label('Alasan Penolakan')
->limit(40)->default('-')
->visible(fn ($record) => $record?->status === 'rejected'),
TextColumn::make('published_at')->label('Dipublikasi') TextColumn::make('published_at')->label('Dipublikasi')
->dateTime('d M Y')->default('-')->sortable(), ->dateTime('d M Y')->default('-')->sortable(),
]) ])
@@ -44,22 +49,34 @@ class PostsTable
]), ]),
]) ])
->recordActions([ ->recordActions([
// Untuk anggota/pengurus/bendahara: ajukan artikel
Action::make('submit')
->label('Ajukan')
->icon('heroicon-o-paper-airplane')
->color('info')
->requiresConfirmation()
->visible(fn ($record) => ! $isAdmin && in_array($record->status, ['draft', 'rejected']))
->action(fn ($record) => $record->update(['status' => 'pending', 'rejection_reason' => null])),
// Untuk admin: approve
Action::make('publish') Action::make('publish')
->label('Terbitkan') ->label('Terbitkan')
->icon('heroicon-o-check-circle') ->icon('heroicon-o-check-circle')
->color('success') ->color('success')
->requiresConfirmation() ->requiresConfirmation()
->visible(fn ($record) => $record->status === 'pending') ->visible(fn ($record) => $isAdmin && $record->status === 'pending')
->action(fn ($record) => $record->update([ ->action(fn ($record) => $record->update([
'status' => 'published', 'status' => 'published',
'published_at' => now(), 'published_at' => now(),
'reviewed_by' => auth()->id(), 'reviewed_by' => auth()->id(),
])), ])),
// Untuk admin: tolak
Action::make('reject') Action::make('reject')
->label('Tolak') ->label('Tolak')
->icon('heroicon-o-x-circle') ->icon('heroicon-o-x-circle')
->color('danger') ->color('danger')
->visible(fn ($record) => $record->status === 'pending') ->visible(fn ($record) => $isAdmin && $record->status === 'pending')
->form([ ->form([
Textarea::make('rejection_reason')->label('Alasan Penolakan')->required(), Textarea::make('rejection_reason')->label('Alasan Penolakan')->required(),
]) ])
@@ -68,7 +85,9 @@ class PostsTable
'reviewed_by' => auth()->id(), 'reviewed_by' => auth()->id(),
'rejection_reason' => $data['rejection_reason'], 'rejection_reason' => $data['rejection_reason'],
])), ])),
EditAction::make(),
EditAction::make()
->visible(fn ($record) => $isAdmin || in_array($record->status, ['draft', 'rejected'])),
]) ])
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]); ->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
} }
+10 -8
View File
@@ -23,24 +23,26 @@ class PermissionSeeder extends Seeder
->where('name', 'not like', '%Permission%') ->where('name', 'not like', '%Permission%')
->get()); ->get());
// Bendahara: hanya kas // Bendahara: hanya kas + artikel sendiri
$bendahara->syncPermissions(Permission::where('name', 'like', '%CashRecord%') $bendahara->syncPermissions(Permission::where('name', 'like', '%CashRecord%')
->orWhere('name', 'like', '%CashCategory%') ->orWhere('name', 'like', '%CashCategory%')
->orWhereIn('name', ['ViewAny:Post', 'View:Post', 'Create:Post', 'Update:Post', 'Delete:Post'])
->get()); ->get());
// Pengurus: kegiatan + lihat anggota & divisi // Pengurus: kegiatan + lihat anggota & divisi + artikel sendiri
$pengurus->syncPermissions(Permission::where('name', 'like', '%Activity%') $pengurus->syncPermissions(Permission::where('name', 'like', '%Activity%')
->orWhere('name', 'like', 'ViewAny:User') ->orWhereIn('name', [
->orWhere('name', 'like', 'View:User') 'ViewAny:User', 'View:User',
->orWhere('name', 'like', 'ViewAny:Division') 'ViewAny:Division', 'View:Division',
->orWhere('name', 'like', 'View:Division') 'ViewAny:Post', 'View:Post', 'Create:Post', 'Update:Post', 'Delete:Post',
])
->get()); ->get());
// Anggota: lihat kegiatan & voting + kelola artikel sendiri // Anggota: lihat kegiatan & voting + artikel sendiri
$anggota->syncPermissions(Permission::whereIn('name', [ $anggota->syncPermissions(Permission::whereIn('name', [
'ViewAny:Activity', 'View:Activity', 'ViewAny:Activity', 'View:Activity',
'ViewAny:Vote', 'View:Vote', 'ViewAny:Vote', 'View:Vote',
'ViewAny:MyPost', 'View:MyPost', 'Create:MyPost', 'Update:MyPost', 'Delete:MyPost', 'ViewAny:Post', 'View:Post', 'Create:Post', 'Update:Post', 'Delete:Post',
])->get()); ])->get());
// Auditor: read-only semua + akses audit // Auditor: read-only semua + akses audit