From ae0cddc27013e6379005e3e0e40c1ba58dddc7d5 Mon Sep 17 00:00:00 2001 From: tuxarmy Date: Sat, 4 Apr 2026 06:44:54 +0700 Subject: [PATCH] feat: tambah sistem poin anggota (kehadiran +10, artikel +5) - Model MemberPoint + migration - PostObserver: +5 poin saat artikel dipublish - ActivityObserver: +10 poin saat peserta hadir di kegiatan - MemberPointResource: tampil di grup Organisasi - MemberPointSeeder + update ActivitySeeder dengan pivot status kehadiran - Update PermissionSeeder: anggota bisa lihat poin --- .../MemberPoints/MemberPointResource.php | 49 +++++++++++++++++++ .../MemberPoints/Pages/CreateMemberPoint.php | 11 +++++ .../MemberPoints/Pages/EditMemberPoint.php | 19 +++++++ .../MemberPoints/Pages/ListMemberPoints.php | 19 +++++++ .../MemberPoints/Schemas/MemberPointForm.php | 16 ++++++ .../MemberPoints/Tables/MemberPointsTable.php | 36 ++++++++++++++ app/Models/MemberPoint.php | 16 ++++++ app/Models/User.php | 11 +++++ app/Observers/ActivityObserver.php | 18 +++++++ app/Observers/PostObserver.php | 22 +++++++++ app/Providers/AppServiceProvider.php | 3 ++ ...4_03_233440_create_member_points_table.php | 26 ++++++++++ database/seeders/ActivitySeeder.php | 7 ++- database/seeders/DatabaseSeeder.php | 1 + database/seeders/MemberPointSeeder.php | 33 +++++++++++++ database/seeders/PermissionSeeder.php | 1 + 16 files changed, 287 insertions(+), 1 deletion(-) create mode 100644 app/Filament/Resources/MemberPoints/MemberPointResource.php create mode 100644 app/Filament/Resources/MemberPoints/Pages/CreateMemberPoint.php create mode 100644 app/Filament/Resources/MemberPoints/Pages/EditMemberPoint.php create mode 100644 app/Filament/Resources/MemberPoints/Pages/ListMemberPoints.php create mode 100644 app/Filament/Resources/MemberPoints/Schemas/MemberPointForm.php create mode 100644 app/Filament/Resources/MemberPoints/Tables/MemberPointsTable.php create mode 100644 app/Models/MemberPoint.php create mode 100644 app/Observers/PostObserver.php create mode 100644 database/migrations/2026_04_03_233440_create_member_points_table.php create mode 100644 database/seeders/MemberPointSeeder.php diff --git a/app/Filament/Resources/MemberPoints/MemberPointResource.php b/app/Filament/Resources/MemberPoints/MemberPointResource.php new file mode 100644 index 0000000..ccad068 --- /dev/null +++ b/app/Filament/Resources/MemberPoints/MemberPointResource.php @@ -0,0 +1,49 @@ + ListMemberPoints::route('/'), + 'create' => CreateMemberPoint::route('/create'), + 'edit' => EditMemberPoint::route('/{record}/edit'), + ]; + } +} diff --git a/app/Filament/Resources/MemberPoints/Pages/CreateMemberPoint.php b/app/Filament/Resources/MemberPoints/Pages/CreateMemberPoint.php new file mode 100644 index 0000000..4dce7d2 --- /dev/null +++ b/app/Filament/Resources/MemberPoints/Pages/CreateMemberPoint.php @@ -0,0 +1,11 @@ +components([ + // + ]); + } +} diff --git a/app/Filament/Resources/MemberPoints/Tables/MemberPointsTable.php b/app/Filament/Resources/MemberPoints/Tables/MemberPointsTable.php new file mode 100644 index 0000000..1b10aa8 --- /dev/null +++ b/app/Filament/Resources/MemberPoints/Tables/MemberPointsTable.php @@ -0,0 +1,36 @@ +columns([ + TextColumn::make('user.name')->label('Anggota')->searchable()->sortable(), + TextColumn::make('points')->label('Poin')->sortable() + ->color(fn ($state) => $state > 0 ? 'success' : 'danger'), + TextColumn::make('reason')->label('Keterangan')->limit(50), + TextColumn::make('source_type')->label('Sumber')->badge() + ->color(fn ($state) => match ($state) { + 'activity' => 'info', + 'post' => 'warning', + default => 'gray', + }), + TextColumn::make('created_at')->label('Tanggal')->date('d M Y')->sortable(), + ]) + ->defaultSort('created_at', 'desc') + ->filters([ + SelectFilter::make('source_type')->label('Sumber') + ->options(['activity' => 'Kegiatan', 'post' => 'Artikel']), + ]) + ->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]); + } +} diff --git a/app/Models/MemberPoint.php b/app/Models/MemberPoint.php new file mode 100644 index 0000000..9acf684 --- /dev/null +++ b/app/Models/MemberPoint.php @@ -0,0 +1,16 @@ +belongsTo(User::class); + } +} diff --git a/app/Models/User.php b/app/Models/User.php index 9c94f83..5de451b 100644 --- a/app/Models/User.php +++ b/app/Models/User.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Models\MemberPoint; use Database\Factories\UserFactory; use Filament\Models\Contracts\FilamentUser; use Filament\Panel; @@ -54,6 +55,16 @@ class User extends Authenticatable implements FilamentUser return $this->hasMany(CashRecord::class, 'created_by'); } + public function points(): HasMany + { + return $this->hasMany(MemberPoint::class); + } + + public function totalPoints(): int + { + return $this->points()->sum('points'); + } + public function canAccessPanel(Panel $panel): bool { return true; diff --git a/app/Observers/ActivityObserver.php b/app/Observers/ActivityObserver.php index a8a3ace..2f9c812 100644 --- a/app/Observers/ActivityObserver.php +++ b/app/Observers/ActivityObserver.php @@ -4,6 +4,7 @@ namespace App\Observers; use App\Models\Activity; use App\Models\ActivityLog; +use App\Models\MemberPoint; use App\Services\NotificationService; use Illuminate\Support\Facades\Auth; @@ -66,4 +67,21 @@ class ActivityObserver 'description' => "Kegiatan baru dibuat: {$activity->title}", ]); } + + public function pivotAttached(Activity $activity, string $relationName, array $pivotIds, array $pivotIdsAttributes): void + { + if ($relationName !== 'participants') return; + + foreach ($pivotIdsAttributes as $userId => $attrs) { + if (($attrs['status'] ?? 'hadir') === 'hadir') { + MemberPoint::create([ + 'user_id' => $userId, + 'points' => 10, + 'reason' => "Hadir di kegiatan: {$activity->title}", + 'source_type' => 'activity', + 'source_id' => $activity->id, + ]); + } + } + } } diff --git a/app/Observers/PostObserver.php b/app/Observers/PostObserver.php new file mode 100644 index 0000000..57526a1 --- /dev/null +++ b/app/Observers/PostObserver.php @@ -0,0 +1,22 @@ +wasChanged('status') && $post->status === 'published') { + MemberPoint::create([ + 'user_id' => $post->user_id, + 'points' => 5, + 'reason' => "Artikel dipublikasi: {$post->title}", + 'source_type' => 'post', + 'source_id' => $post->id, + ]); + } + } +} diff --git a/app/Providers/AppServiceProvider.php b/app/Providers/AppServiceProvider.php index 7a4cc68..72104b9 100644 --- a/app/Providers/AppServiceProvider.php +++ b/app/Providers/AppServiceProvider.php @@ -4,10 +4,12 @@ namespace App\Providers; use App\Models\Activity; use App\Models\CashRecord; +use App\Models\Post; use App\Models\User; use App\Models\Vote; use App\Observers\ActivityObserver; use App\Observers\CashRecordObserver; +use App\Observers\PostObserver; use App\Observers\UserObserver; use App\Observers\VoteObserver; use Illuminate\Support\ServiceProvider; @@ -20,5 +22,6 @@ class AppServiceProvider extends ServiceProvider CashRecord::observe(CashRecordObserver::class); Activity::observe(ActivityObserver::class); Vote::observe(VoteObserver::class); + Post::observe(PostObserver::class); } } diff --git a/database/migrations/2026_04_03_233440_create_member_points_table.php b/database/migrations/2026_04_03_233440_create_member_points_table.php new file mode 100644 index 0000000..dfcd7a8 --- /dev/null +++ b/database/migrations/2026_04_03_233440_create_member_points_table.php @@ -0,0 +1,26 @@ +id(); + $table->foreignId('user_id')->constrained('users'); + $table->integer('points'); + $table->string('reason'); + $table->string('source_type')->nullable(); // activity, post + $table->unsignedBigInteger('source_id')->nullable(); + $table->timestamps(); + }); + } + + public function down(): void + { + Schema::dropIfExists('member_points'); + } +}; diff --git a/database/seeders/ActivitySeeder.php b/database/seeders/ActivitySeeder.php index ab4ebd0..189f51c 100644 --- a/database/seeders/ActivitySeeder.php +++ b/database/seeders/ActivitySeeder.php @@ -46,7 +46,12 @@ class ActivitySeeder extends Seeder foreach ($activities as $data) { $activity = Activity::create(array_merge($data, ['created_by' => $pengurus?->id])); - $activity->participants()->sync($anggota->pluck('id')); + + // Attach peserta dengan status kehadiran + $syncData = $anggota->mapWithKeys(fn ($user) => [ + $user->id => ['status' => 'hadir', 'notes' => null] + ])->toArray(); + $activity->participants()->sync($syncData); } } } diff --git a/database/seeders/DatabaseSeeder.php b/database/seeders/DatabaseSeeder.php index 41fbe4b..546804e 100644 --- a/database/seeders/DatabaseSeeder.php +++ b/database/seeders/DatabaseSeeder.php @@ -19,6 +19,7 @@ class DatabaseSeeder extends Seeder VoteSeeder::class, PostSeeder::class, AuditSeeder::class, + MemberPointSeeder::class, ]); } } diff --git a/database/seeders/MemberPointSeeder.php b/database/seeders/MemberPointSeeder.php new file mode 100644 index 0000000..5166db1 --- /dev/null +++ b/database/seeders/MemberPointSeeder.php @@ -0,0 +1,33 @@ +each(function ($activity) { + $activity->participants()->wherePivot('status', 'hadir')->each(function ($user) use ($activity) { + MemberPoint::firstOrCreate( + ['user_id' => $user->id, 'source_type' => 'activity', 'source_id' => $activity->id], + ['points' => 10, 'reason' => "Hadir di kegiatan: {$activity->title}"] + ); + }); + }); + + // Poin dari artikel yang sudah published + Post::where('status', 'published')->each(function ($post) { + MemberPoint::firstOrCreate( + ['user_id' => $post->user_id, 'source_type' => 'post', 'source_id' => $post->id], + ['points' => 5, 'reason' => "Artikel dipublikasi: {$post->title}"] + ); + }); + } +} diff --git a/database/seeders/PermissionSeeder.php b/database/seeders/PermissionSeeder.php index 3cf86d0..577741b 100644 --- a/database/seeders/PermissionSeeder.php +++ b/database/seeders/PermissionSeeder.php @@ -44,6 +44,7 @@ class PermissionSeeder extends Seeder 'ViewAny:Activity', 'View:Activity', 'ViewAny:Vote', 'View:Vote', 'ViewAny:Post', 'View:Post', 'Create:Post', 'Update:Post', 'Delete:Post', + 'ViewAny:MemberPoint', 'View:MemberPoint', ])->get()); // Auditor: read-only semua + akses audit