refactor: hapus menu approvals, pindahkan tombol setujui/tolak langsung ke halaman transaksi kas

This commit is contained in:
2026-04-03 07:21:53 +07:00
parent 764aa4d82f
commit 3c99dfdc26
5 changed files with 262 additions and 38 deletions
@@ -5,6 +5,7 @@ namespace App\Filament\Resources\Approvals;
use App\Filament\Resources\Approvals\Pages\CreateApproval;
use App\Filament\Resources\Approvals\Pages\EditApproval;
use App\Filament\Resources\Approvals\Pages\ListApprovals;
use App\Filament\Resources\Approvals\Pages\ViewApproval;
use App\Filament\Resources\Approvals\Schemas\ApprovalForm;
use App\Filament\Resources\Approvals\Tables\ApprovalsTable;
use App\Models\Approval;
@@ -17,7 +18,7 @@ class ApprovalResource extends Resource
protected static ?string $model = Approval::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-clipboard-document-check';
protected static string|\UnitEnum|null $navigationGroup = 'Keputusan';
protected static ?string $modelLabel = 'Persetujuan';
protected static bool $shouldRegisterNavigation = false;
public static function form(Schema $form): Schema
{
@@ -34,6 +35,7 @@ class ApprovalResource extends Resource
return [
'index' => ListApprovals::route('/'),
'create' => CreateApproval::route('/create'),
'view' => ViewApproval::route('/{record}'),
'edit' => EditApproval::route('/{record}/edit'),
];
}
@@ -0,0 +1,106 @@
<?php
namespace App\Filament\Resources\Approvals\Pages;
use App\Filament\Resources\Approvals\ApprovalResource;
use App\Models\Approval;
use App\Models\ApprovalItem;
use Filament\Actions\Action;
use Filament\Forms\Components\Textarea;
use Filament\Infolists\Components\RepeatableEntry;
use Filament\Infolists\Components\TextEntry;
use Filament\Resources\Pages\ViewRecord;
use Filament\Schemas\Components\Section;
use Filament\Schemas\Schema;
class ViewApproval extends ViewRecord
{
protected static string $resource = ApprovalResource::class;
protected function getHeaderActions(): array
{
return [
Action::make('approve')
->label('Setujui')
->icon('heroicon-o-check-circle')
->color('success')
->requiresConfirmation()
->visible(fn () => $this->record->status === 'pending'
&& ! $this->record->items()->where('user_id', auth()->id())->exists())
->form([Textarea::make('notes')->label('Catatan')->rows(2)])
->action(function (array $data): void {
ApprovalItem::create([
'approval_id' => $this->record->id,
'user_id' => auth()->id(),
'decision' => 'approve',
'notes' => $data['notes'] ?? null,
]);
$count = $this->record->items()->where('decision', 'approve')->count();
if ($count >= $this->record->required_approvals) {
$this->record->update(['status' => 'approved']);
}
$this->refreshFormData([]);
}),
Action::make('reject')
->label('Tolak')
->icon('heroicon-o-x-circle')
->color('danger')
->visible(fn () => $this->record->status === 'pending'
&& ! $this->record->items()->where('user_id', auth()->id())->exists())
->form([Textarea::make('notes')->label('Alasan Penolakan')->required()->rows(2)])
->action(function (array $data): void {
ApprovalItem::create([
'approval_id' => $this->record->id,
'user_id' => auth()->id(),
'decision' => 'reject',
'notes' => $data['notes'],
]);
$this->record->update(['status' => 'rejected']);
$this->refreshFormData([]);
}),
];
}
public function infolist(Schema $infolist): Schema
{
$record = $this->record;
$subject = $record->approvable;
$label = class_basename($record->model_type);
return $infolist->schema([
Section::make('Yang Dimintakan Persetujuan')->schema([
TextEntry::make('model_type')->label('Tipe')
->state($label),
TextEntry::make('subject_title')->label('Judul / Deskripsi')
->state(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}",
}),
TextEntry::make('status')->badge()
->color(fn ($state) => match ($state) {
'approved' => 'success',
'rejected' => 'danger',
default => 'warning',
}),
TextEntry::make('required_approvals')->label('Persetujuan Dibutuhkan'),
])->columns(2),
Section::make('Riwayat Keputusan')->schema([
TextEntry::make('decisions')
->label('')
->state(function () use ($record): string {
if ($record->items->isEmpty()) return 'Belum ada keputusan.';
return $record->items->map(fn ($item) =>
"{$item->user->name}: " . ($item->decision === 'approve' ? '✓ Setuju' : '✗ Tolak') .
($item->notes ? "{$item->notes}" : '')
)->join("\n");
})
->columnSpanFull(),
]),
]);
}
}
@@ -7,7 +7,7 @@ use App\Models\ApprovalItem;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
use Filament\Actions\ViewAction;
use Filament\Forms\Components\Textarea;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\SelectFilter;
@@ -20,35 +20,61 @@ class ApprovalsTable
return $table
->columns([
TextColumn::make('model_type')->label('Tipe')
->formatStateUsing(fn ($state) => class_basename($state)),
TextColumn::make('model_id')->label('ID'),
TextColumn::make('required_approvals')->label('Dibutuhkan'),
TextColumn::make('items_count')->counts('items')->label('Sudah Approve'),
->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'),
TextColumn::make('status')->badge()
->color(fn ($state) => match ($state) {
'approved' => 'success',
'rejected' => 'danger',
default => 'warning',
})
->formatStateUsing(fn ($state) => match ($state) {
'approved' => 'Disetujui',
'rejected' => 'Ditolak',
default => 'Menunggu',
}),
TextColumn::make('created_at')->label('Dibuat')->date('d M Y'),
TextColumn::make('created_at')->label('Dibuat')->date('d M Y')->sortable(),
])
->defaultSort('created_at', 'desc')
->filters([
SelectFilter::make('status')->options([
'pending' => 'Pending',
'pending' => 'Menunggu',
'approved' => 'Disetujui',
'rejected' => 'Ditolak',
]),
])
->recordActions([
ViewAction::make()->label('Detail'),
Action::make('approve')
->label('Setujui')
->icon('heroicon-o-check-circle')
->color('success')
->requiresConfirmation()
->visible(fn (Approval $record) => $record->status === 'pending')
->form([
Textarea::make('notes')->label('Catatan')->rows(2),
])
->visible(fn (Approval $record) => $record->status === 'pending'
&& ! $record->items()->where('user_id', auth()->id())->exists())
->form([Textarea::make('notes')->label('Catatan')->rows(2)])
->action(function (Approval $record, array $data): void {
ApprovalItem::create([
'approval_id' => $record->id,
@@ -56,31 +82,25 @@ class ApprovalsTable
'decision' => 'approve',
'notes' => $data['notes'] ?? null,
]);
$approveCount = $record->items()->where('decision', 'approve')->count();
if ($approveCount >= $record->required_approvals) {
$count = $record->items()->where('decision', 'approve')->count();
if ($count >= $record->required_approvals) {
$record->update(['status' => 'approved']);
}
\App\Models\ActivityLog::create([
'user_id' => auth()->id(),
'action' => 'approved',
'model_type' => Approval::class,
'model_id' => $record->id,
'description' => auth()->user()->name . " menyetujui " . class_basename($record->model_type) . " #{$record->model_id}",
'description' => auth()->user()->name . " menyetujui persetujuan #{$record->id}",
]);
}),
Action::make('reject')
->label('Tolak')
->icon('heroicon-o-x-circle')
->color('danger')
->requiresConfirmation()
->visible(fn (Approval $record) => $record->status === 'pending')
->form([
Textarea::make('notes')->label('Alasan Penolakan')->required()->rows(2),
])
->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)])
->action(function (Approval $record, array $data): void {
ApprovalItem::create([
'approval_id' => $record->id,
@@ -88,19 +108,15 @@ class ApprovalsTable
'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,
'description' => auth()->user()->name . " menolak " . class_basename($record->model_type) . " #{$record->model_id}",
'description' => auth()->user()->name . " menolak persetujuan #{$record->id}",
]);
}),
EditAction::make(),
])
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
}