feat: pindah route ke /dashboard, tambah seeders, business rules via observers, action verifikasi & approval
This commit is contained in:
@@ -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()])]);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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}",
|
||||
]);
|
||||
}
|
||||
}
|
||||
@@ -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.');
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -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",
|
||||
]);
|
||||
}
|
||||
}
|
||||
@@ -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,
|
||||
|
||||
Reference in New Issue
Block a user