feat: pindah route ke /dashboard, tambah seeders, business rules via observers, action verifikasi & approval

This commit is contained in:
2026-04-03 04:34:21 +07:00
parent 8675c14f15
commit d42b23f604
12 changed files with 355 additions and 16 deletions
@@ -2,6 +2,7 @@
namespace App\Filament\Resources\Activities\Tables;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
@@ -35,7 +36,34 @@ class ActivitiesTable
'rejected' => 'Ditolak',
]),
])
->recordActions([EditAction::make()])
->recordActions([
EditAction::make(),
Action::make('submit')
->label('Ajukan')
->icon('heroicon-o-paper-airplane')
->color('info')
->requiresConfirmation()
->visible(fn ($record) => $record->status === 'draft')
->action(fn ($record) => $record->update(['status' => 'pending'])),
Action::make('approve')
->label('Setujui')
->icon('heroicon-o-check-circle')
->color('success')
->requiresConfirmation()
->visible(fn ($record) => $record->status === 'pending')
->action(fn ($record) => $record->update([
'status' => 'approved',
'approved_by' => auth()->id(),
'approved_at' => now(),
])),
Action::make('reject')
->label('Tolak')
->icon('heroicon-o-x-circle')
->color('danger')
->requiresConfirmation()
->visible(fn ($record) => $record->status === 'pending')
->action(fn ($record) => $record->update(['status' => 'rejected'])),
])
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
}
}
@@ -3,6 +3,7 @@
namespace App\Filament\Resources\CashRecords\Tables;
use App\Models\CashCategory;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction;
@@ -26,9 +27,23 @@ class CashRecordsTable
])
->filters([
SelectFilter::make('category_id')->label('Kategori')
->options(CashCategory::pluck('name', 'id')),
->options(\App\Models\CashCategory::pluck('name', 'id')),
])
->recordActions([
EditAction::make()->hidden(fn ($record) => $record->verified_at !== null),
Action::make('verify')
->label('Verifikasi')
->icon('heroicon-o-check-circle')
->color('success')
->requiresConfirmation()
->hidden(fn ($record) => $record->verified_at !== null)
->action(function ($record) {
$record->update([
'verified_by' => auth()->id(),
'verified_at' => now(),
]);
}),
])
->recordActions([EditAction::make()])
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
}
}
+52
View File
@@ -0,0 +1,52 @@
<?php
namespace App\Observers;
use App\Models\Activity;
use App\Models\ActivityLog;
use Illuminate\Support\Facades\Auth;
class ActivityObserver
{
public function updated(Activity $activity): void
{
if ($activity->wasChanged('status')) {
$old = $activity->getOriginal('status');
$new = $activity->status;
// Validasi workflow: draft→pending, pending→approved/rejected
$allowed = [
'draft' => ['pending'],
'pending' => ['approved', 'rejected'],
];
if (isset($allowed[$old]) && ! in_array($new, $allowed[$old])) {
throw new \Exception("Transisi status dari {$old} ke {$new} tidak diizinkan.");
}
// Wajib isi executed_at & execution_notes jika sudah approved dan mau ditandai selesai
if ($new === 'approved' && $activity->wasChanged('executed_at') && empty($activity->execution_notes)) {
throw new \Exception('Catatan pelaksanaan wajib diisi.');
}
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'status_changed',
'model_type' => Activity::class,
'model_id' => $activity->id,
'description' => "Status kegiatan '{$activity->title}' diubah dari {$old} menjadi {$new}",
]);
}
}
public function created(Activity $activity): void
{
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'created',
'model_type' => Activity::class,
'model_id' => $activity->id,
'description' => "Kegiatan baru dibuat: {$activity->title}",
]);
}
}
+46
View File
@@ -0,0 +1,46 @@
<?php
namespace App\Observers;
use App\Models\ActivityLog;
use App\Models\CashRecord;
use Illuminate\Support\Facades\Auth;
class CashRecordObserver
{
public function created(CashRecord $record): void
{
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'created',
'model_type' => CashRecord::class,
'model_id' => $record->id,
'description' => "Transaksi kas baru: {$record->description} sebesar Rp " . number_format($record->amount, 0, ',', '.'),
]);
}
public function updated(CashRecord $record): void
{
// Setelah diverifikasi, tidak bisa diubah lagi
if ($record->getOriginal('verified_at') !== null) {
throw new \Exception('Transaksi yang sudah diverifikasi tidak dapat diubah.');
}
if ($record->wasChanged('verified_by') && $record->verified_by !== null) {
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'verified',
'model_type' => CashRecord::class,
'model_id' => $record->id,
'description' => "Transaksi kas diverifikasi: {$record->description}",
]);
}
}
public function deleting(CashRecord $record): void
{
if ($record->verified_at !== null) {
throw new \Exception('Transaksi yang sudah diverifikasi tidak dapat dihapus.');
}
}
}
+44
View File
@@ -0,0 +1,44 @@
<?php
namespace App\Observers;
use App\Models\ActivityLog;
use App\Models\MemberStatusLog;
use App\Models\User;
use Illuminate\Support\Facades\Auth;
class UserObserver
{
public function updated(User $user): void
{
// Log perubahan status anggota
if ($user->wasChanged('status')) {
MemberStatusLog::create([
'member_id' => $user->id,
'changed_by' => Auth::id() ?? $user->id,
'old_status' => $user->getOriginal('status'),
'new_status' => $user->status,
'reason' => $user->inactive_reason,
]);
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'status_changed',
'model_type' => User::class,
'model_id' => $user->id,
'description' => "Status anggota {$user->name} diubah dari {$user->getOriginal('status')} menjadi {$user->status}",
]);
}
}
public function created(User $user): void
{
ActivityLog::create([
'user_id' => Auth::id(),
'action' => 'created',
'model_type' => User::class,
'model_id' => $user->id,
'description' => "Anggota baru {$user->name} ditambahkan",
]);
}
}
+9 -12
View File
@@ -2,23 +2,20 @@
namespace App\Providers;
use App\Models\Activity;
use App\Models\CashRecord;
use App\Models\User;
use App\Observers\ActivityObserver;
use App\Observers\CashRecordObserver;
use App\Observers\UserObserver;
use Illuminate\Support\ServiceProvider;
class AppServiceProvider extends ServiceProvider
{
/**
* Register any application services.
*/
public function register(): void
{
//
}
/**
* Bootstrap any application services.
*/
public function boot(): void
{
//
User::observe(UserObserver::class);
CashRecord::observe(CashRecordObserver::class);
Activity::observe(ActivityObserver::class);
}
}
@@ -27,7 +27,7 @@ class AdminPanelProvider extends PanelProvider
return $panel
->default()
->id('admin')
->path('admin')
->path('dashboard')
->login()
->colors([
'primary' => Color::Amber,