2026-04-03 04:22:34 +07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace App\Filament\Resources\Approvals\Tables;
|
|
|
|
|
|
2026-04-03 05:02:33 +07:00
|
|
|
use App\Models\Approval;
|
|
|
|
|
use App\Models\ApprovalItem;
|
|
|
|
|
use Filament\Actions\Action;
|
2026-04-03 04:22:34 +07:00
|
|
|
use Filament\Actions\BulkActionGroup;
|
|
|
|
|
use Filament\Actions\DeleteBulkAction;
|
2026-04-03 07:21:53 +07:00
|
|
|
use Filament\Actions\ViewAction;
|
2026-04-03 05:02:33 +07:00
|
|
|
use Filament\Forms\Components\Textarea;
|
2026-04-03 04:22:34 +07:00
|
|
|
use Filament\Tables\Columns\TextColumn;
|
|
|
|
|
use Filament\Tables\Filters\SelectFilter;
|
|
|
|
|
use Filament\Tables\Table;
|
|
|
|
|
|
|
|
|
|
class ApprovalsTable
|
|
|
|
|
{
|
|
|
|
|
public static function configure(Table $table): Table
|
|
|
|
|
{
|
|
|
|
|
return $table
|
|
|
|
|
->columns([
|
2026-04-03 05:02:33 +07:00
|
|
|
TextColumn::make('model_type')->label('Tipe')
|
2026-04-03 07:21:53 +07:00
|
|
|
->formatStateUsing(fn ($state) => match ($state) {
|
|
|
|
|
'App\\Models\\CashRecord' => '💰 Transaksi Kas',
|
|
|
|
|
'App\\Models\\Activity' => '📅 Kegiatan',
|
|
|
|
|
default => class_basename($state),
|
|
|
|
|
}),
|
|
|
|
|
TextColumn::make('subject')
|
|
|
|
|
->label('Deskripsi')
|
|
|
|
|
->state(function (Approval $record): string {
|
|
|
|
|
$subject = $record->approvable;
|
|
|
|
|
return match (true) {
|
|
|
|
|
$subject instanceof \App\Models\CashRecord =>
|
|
|
|
|
$subject->description . ' — Rp ' . number_format($subject->amount, 0, ',', '.'),
|
|
|
|
|
$subject instanceof \App\Models\Activity =>
|
|
|
|
|
$subject->title,
|
|
|
|
|
default => "#{$record->model_id}",
|
|
|
|
|
};
|
|
|
|
|
})
|
|
|
|
|
->limit(50),
|
|
|
|
|
TextColumn::make('progress')
|
|
|
|
|
->label('Progress')
|
|
|
|
|
->state(fn (Approval $record) =>
|
|
|
|
|
$record->items()->where('decision', 'approve')->count()
|
|
|
|
|
. ' / ' . $record->required_approvals . ' persetujuan')
|
|
|
|
|
->badge()->color('info'),
|
2026-04-03 04:22:34 +07:00
|
|
|
TextColumn::make('status')->badge()
|
|
|
|
|
->color(fn ($state) => match ($state) {
|
|
|
|
|
'approved' => 'success',
|
|
|
|
|
'rejected' => 'danger',
|
|
|
|
|
default => 'warning',
|
2026-04-03 07:21:53 +07:00
|
|
|
})
|
|
|
|
|
->formatStateUsing(fn ($state) => match ($state) {
|
|
|
|
|
'approved' => 'Disetujui',
|
|
|
|
|
'rejected' => 'Ditolak',
|
|
|
|
|
default => 'Menunggu',
|
2026-04-03 04:22:34 +07:00
|
|
|
}),
|
2026-04-03 07:21:53 +07:00
|
|
|
TextColumn::make('created_at')->label('Dibuat')->date('d M Y')->sortable(),
|
2026-04-03 04:22:34 +07:00
|
|
|
])
|
2026-04-03 07:21:53 +07:00
|
|
|
->defaultSort('created_at', 'desc')
|
2026-04-03 04:22:34 +07:00
|
|
|
->filters([
|
|
|
|
|
SelectFilter::make('status')->options([
|
2026-04-03 07:21:53 +07:00
|
|
|
'pending' => 'Menunggu',
|
2026-04-03 04:22:34 +07:00
|
|
|
'approved' => 'Disetujui',
|
|
|
|
|
'rejected' => 'Ditolak',
|
|
|
|
|
]),
|
|
|
|
|
])
|
2026-04-03 05:02:33 +07:00
|
|
|
->recordActions([
|
2026-04-03 07:21:53 +07:00
|
|
|
ViewAction::make()->label('Detail'),
|
2026-04-03 05:02:33 +07:00
|
|
|
Action::make('approve')
|
|
|
|
|
->label('Setujui')
|
|
|
|
|
->icon('heroicon-o-check-circle')
|
|
|
|
|
->color('success')
|
|
|
|
|
->requiresConfirmation()
|
2026-04-03 07:21:53 +07:00
|
|
|
->visible(fn (Approval $record) => $record->status === 'pending'
|
|
|
|
|
&& ! $record->items()->where('user_id', auth()->id())->exists())
|
|
|
|
|
->form([Textarea::make('notes')->label('Catatan')->rows(2)])
|
2026-04-03 05:02:33 +07:00
|
|
|
->action(function (Approval $record, array $data): void {
|
|
|
|
|
ApprovalItem::create([
|
|
|
|
|
'approval_id' => $record->id,
|
|
|
|
|
'user_id' => auth()->id(),
|
|
|
|
|
'decision' => 'approve',
|
|
|
|
|
'notes' => $data['notes'] ?? null,
|
|
|
|
|
]);
|
2026-04-03 07:21:53 +07:00
|
|
|
$count = $record->items()->where('decision', 'approve')->count();
|
|
|
|
|
if ($count >= $record->required_approvals) {
|
2026-04-03 05:02:33 +07:00
|
|
|
$record->update(['status' => 'approved']);
|
|
|
|
|
}
|
|
|
|
|
\App\Models\ActivityLog::create([
|
|
|
|
|
'user_id' => auth()->id(),
|
|
|
|
|
'action' => 'approved',
|
|
|
|
|
'model_type' => Approval::class,
|
|
|
|
|
'model_id' => $record->id,
|
2026-04-03 07:21:53 +07:00
|
|
|
'description' => auth()->user()->name . " menyetujui persetujuan #{$record->id}",
|
2026-04-03 05:02:33 +07:00
|
|
|
]);
|
|
|
|
|
}),
|
|
|
|
|
Action::make('reject')
|
|
|
|
|
->label('Tolak')
|
|
|
|
|
->icon('heroicon-o-x-circle')
|
|
|
|
|
->color('danger')
|
2026-04-03 07:21:53 +07:00
|
|
|
->visible(fn (Approval $record) => $record->status === 'pending'
|
|
|
|
|
&& ! $record->items()->where('user_id', auth()->id())->exists())
|
|
|
|
|
->form([Textarea::make('notes')->label('Alasan Penolakan')->required()->rows(2)])
|
2026-04-03 05:02:33 +07:00
|
|
|
->action(function (Approval $record, array $data): void {
|
|
|
|
|
ApprovalItem::create([
|
|
|
|
|
'approval_id' => $record->id,
|
|
|
|
|
'user_id' => auth()->id(),
|
|
|
|
|
'decision' => 'reject',
|
|
|
|
|
'notes' => $data['notes'],
|
|
|
|
|
]);
|
|
|
|
|
$record->update(['status' => 'rejected']);
|
|
|
|
|
\App\Models\ActivityLog::create([
|
|
|
|
|
'user_id' => auth()->id(),
|
|
|
|
|
'action' => 'rejected',
|
|
|
|
|
'model_type' => Approval::class,
|
|
|
|
|
'model_id' => $record->id,
|
2026-04-03 07:21:53 +07:00
|
|
|
'description' => auth()->user()->name . " menolak persetujuan #{$record->id}",
|
2026-04-03 05:02:33 +07:00
|
|
|
]);
|
|
|
|
|
}),
|
|
|
|
|
])
|
2026-04-03 04:22:34 +07:00
|
|
|
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
|
|
|
|
|
}
|
|
|
|
|
}
|