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; namespace App\Filament\Resources\Activities\Tables;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup; use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction; use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction; use Filament\Actions\EditAction;
@@ -35,7 +36,34 @@ class ActivitiesTable
'rejected' => 'Ditolak', '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()])]); ->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
} }
} }
@@ -3,6 +3,7 @@
namespace App\Filament\Resources\CashRecords\Tables; namespace App\Filament\Resources\CashRecords\Tables;
use App\Models\CashCategory; use App\Models\CashCategory;
use Filament\Actions\Action;
use Filament\Actions\BulkActionGroup; use Filament\Actions\BulkActionGroup;
use Filament\Actions\DeleteBulkAction; use Filament\Actions\DeleteBulkAction;
use Filament\Actions\EditAction; use Filament\Actions\EditAction;
@@ -26,9 +27,23 @@ class CashRecordsTable
]) ])
->filters([ ->filters([
SelectFilter::make('category_id')->label('Kategori') 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()])]); ->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; 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; use Illuminate\Support\ServiceProvider;
class AppServiceProvider extends ServiceProvider class AppServiceProvider extends ServiceProvider
{ {
/**
* Register any application services.
*/
public function register(): void
{
//
}
/**
* Bootstrap any application services.
*/
public function boot(): void 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 return $panel
->default() ->default()
->id('admin') ->id('admin')
->path('admin') ->path('dashboard')
->login() ->login()
->colors([ ->colors([
'primary' => Color::Amber, 'primary' => Color::Amber,
+52
View File
@@ -0,0 +1,52 @@
<?php
namespace Database\Seeders;
use App\Models\Activity;
use App\Models\User;
use Illuminate\Database\Seeder;
class ActivitySeeder extends Seeder
{
public function run(): void
{
$pengurus = User::role('pengurus')->first();
$ketua = User::role('ketua')->first();
$anggota = User::role('anggota')->get();
$activities = [
[
'title' => 'Kerja Bakti Desa',
'description' => 'Kegiatan bersih-bersih lingkungan desa',
'start_date' => now()->addDays(7),
'end_date' => now()->addDays(7),
'status' => 'approved',
'approved_by' => $ketua?->id,
'approved_at' => now(),
],
[
'title' => 'Pelatihan Kewirausahaan',
'description' => 'Pelatihan usaha kecil untuk pemuda desa',
'start_date' => now()->addDays(14),
'end_date' => now()->addDays(15),
'status' => 'pending',
],
[
'title' => 'Turnamen Voli Antar RT',
'description' => 'Kompetisi voli untuk mempererat silaturahmi',
'start_date' => now()->subDays(10),
'end_date' => now()->subDays(8),
'status' => 'approved',
'approved_by' => $ketua?->id,
'approved_at' => now()->subDays(15),
'executed_at' => now()->subDays(10),
'execution_notes' => 'Kegiatan berjalan lancar, diikuti 8 tim.',
],
];
foreach ($activities as $data) {
$activity = Activity::create(array_merge($data, ['created_by' => $pengurus?->id]));
$activity->participants()->sync($anggota->pluck('id'));
}
}
}
+36
View File
@@ -0,0 +1,36 @@
<?php
namespace Database\Seeders;
use App\Models\CashCategory;
use App\Models\CashRecord;
use App\Models\User;
use Illuminate\Database\Seeder;
class CashSeeder extends Seeder
{
public function run(): void
{
$pemasukan = CashCategory::firstOrCreate(['name' => 'pemasukan']);
$pengeluaran = CashCategory::firstOrCreate(['name' => 'pengeluaran']);
$bendahara = User::role('bendahara')->first();
$ketua = User::role('ketua')->first();
$records = [
['category_id' => $pemasukan->id, 'amount' => 500000, 'description' => 'Iuran anggota bulan Januari', 'date' => now()->subMonths(2)],
['category_id' => $pemasukan->id, 'amount' => 500000, 'description' => 'Iuran anggota bulan Februari', 'date' => now()->subMonth()],
['category_id' => $pemasukan->id, 'amount' => 1000000, 'description' => 'Donasi kegiatan kerja bakti', 'date' => now()->subDays(20)],
['category_id' => $pengeluaran->id, 'amount' => 250000, 'description' => 'Pembelian alat kebersihan', 'date' => now()->subDays(18)],
['category_id' => $pengeluaran->id, 'amount' => 150000, 'description' => 'Konsumsi rapat bulanan', 'date' => now()->subDays(5)],
];
foreach ($records as $data) {
CashRecord::create(array_merge($data, [
'created_by' => $bendahara?->id,
'verified_by' => $ketua?->id,
'verified_at' => now()->subDays(1),
]));
}
}
}
+4
View File
@@ -10,6 +10,10 @@ class DatabaseSeeder extends Seeder
{ {
$this->call([ $this->call([
RolesAndPermissionsSeeder::class, RolesAndPermissionsSeeder::class,
DivisionSeeder::class,
UserSeeder::class,
ActivitySeeder::class,
CashSeeder::class,
]); ]);
} }
} }
+24
View File
@@ -0,0 +1,24 @@
<?php
namespace Database\Seeders;
use App\Models\Division;
use Illuminate\Database\Seeder;
class DivisionSeeder extends Seeder
{
public function run(): void
{
$divisions = [
['name' => 'Humas', 'description' => 'Hubungan masyarakat dan komunikasi'],
['name' => 'Pendidikan', 'description' => 'Bidang pendidikan dan pelatihan'],
['name' => 'Olahraga', 'description' => 'Bidang olahraga dan kesehatan'],
['name' => 'Seni & Budaya', 'description' => 'Bidang seni dan pelestarian budaya'],
['name' => 'Lingkungan', 'description' => 'Bidang lingkungan hidup'],
];
foreach ($divisions as $division) {
Division::firstOrCreate(['name' => $division['name']], $division);
}
}
}
+41
View File
@@ -0,0 +1,41 @@
<?php
namespace Database\Seeders;
use App\Models\Division;
use App\Models\User;
use Illuminate\Database\Seeder;
use Spatie\Permission\Models\Role;
class UserSeeder extends Seeder
{
public function run(): void
{
$divisions = Division::pluck('id', 'name');
$users = [
['name' => 'Budi Santoso', 'email' => 'ketua@persegi.id', 'role' => 'ketua', 'division' => 'Humas'],
['name' => 'Siti Rahayu', 'email' => 'bendahara@persegi.id','role' => 'bendahara', 'division' => 'Pendidikan'],
['name' => 'Ahmad Fauzi', 'email' => 'pengurus@persegi.id', 'role' => 'pengurus', 'division' => 'Olahraga'],
['name' => 'Dewi Lestari', 'email' => 'auditor@persegi.id', 'role' => 'auditor', 'division' => 'Seni & Budaya'],
['name' => 'Rizky Pratama', 'email' => 'anggota1@persegi.id', 'role' => 'anggota', 'division' => 'Lingkungan'],
['name' => 'Nur Hidayah', 'email' => 'anggota2@persegi.id', 'role' => 'anggota', 'division' => 'Humas'],
];
foreach ($users as $data) {
$user = User::firstOrCreate(
['email' => $data['email']],
[
'name' => $data['name'],
'password' => bcrypt('password'),
'phone' => '08' . rand(100000000, 999999999),
'address' => 'Desa Karangdadap, Kalibagor, Banyumas',
'division_id' => $divisions[$data['division']] ?? null,
'status' => 'aktif',
]
);
$user->syncRoles([$data['role']]);
}
}
}