feat: tambah fitur iuran anggota (MemberDue)
This commit is contained in:
@@ -0,0 +1,40 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues;
|
||||||
|
|
||||||
|
use App\Filament\Resources\MemberDues\Pages\CreateMemberDue;
|
||||||
|
use App\Filament\Resources\MemberDues\Pages\EditMemberDue;
|
||||||
|
use App\Filament\Resources\MemberDues\Pages\ListMemberDues;
|
||||||
|
use App\Filament\Resources\MemberDues\Schemas\MemberDueForm;
|
||||||
|
use App\Filament\Resources\MemberDues\Tables\MemberDuesTable;
|
||||||
|
use App\Models\MemberDue;
|
||||||
|
use Filament\Resources\Resource;
|
||||||
|
use Filament\Schemas\Schema;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
|
||||||
|
class MemberDueResource extends Resource
|
||||||
|
{
|
||||||
|
protected static ?string $model = MemberDue::class;
|
||||||
|
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-currency-dollar';
|
||||||
|
protected static string|\UnitEnum|null $navigationGroup = 'Organisasi';
|
||||||
|
protected static ?string $modelLabel = 'Iuran Anggota';
|
||||||
|
|
||||||
|
public static function form(Schema $form): Schema
|
||||||
|
{
|
||||||
|
return MemberDueForm::configure($form);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function table(Table $table): Table
|
||||||
|
{
|
||||||
|
return MemberDuesTable::configure($table);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getPages(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'index' => ListMemberDues::route('/'),
|
||||||
|
'create' => CreateMemberDue::route('/create'),
|
||||||
|
'edit' => EditMemberDue::route('/{record}/edit'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\MemberDues\MemberDueResource;
|
||||||
|
use Filament\Resources\Pages\CreateRecord;
|
||||||
|
|
||||||
|
class CreateMemberDue extends CreateRecord
|
||||||
|
{
|
||||||
|
protected static string $resource = MemberDueResource::class;
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\MemberDues\MemberDueResource;
|
||||||
|
use Filament\Resources\Pages\EditRecord;
|
||||||
|
|
||||||
|
class EditMemberDue extends EditRecord
|
||||||
|
{
|
||||||
|
protected static string $resource = MemberDueResource::class;
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues\Pages;
|
||||||
|
|
||||||
|
use App\Filament\Resources\MemberDues\MemberDueResource;
|
||||||
|
use Filament\Actions\CreateAction;
|
||||||
|
use Filament\Resources\Pages\ListRecords;
|
||||||
|
|
||||||
|
class ListMemberDues extends ListRecords
|
||||||
|
{
|
||||||
|
protected static string $resource = MemberDueResource::class;
|
||||||
|
|
||||||
|
protected function getHeaderActions(): array
|
||||||
|
{
|
||||||
|
return [CreateAction::make()];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues\Schemas;
|
||||||
|
|
||||||
|
use App\Models\User;
|
||||||
|
use Filament\Forms\Components\DatePicker;
|
||||||
|
use Filament\Forms\Components\Select;
|
||||||
|
use Filament\Forms\Components\Textarea;
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Schemas\Schema;
|
||||||
|
|
||||||
|
class MemberDueForm
|
||||||
|
{
|
||||||
|
public static function configure(Schema $schema): Schema
|
||||||
|
{
|
||||||
|
return $schema->components([
|
||||||
|
Select::make('user_id')->label('Anggota')
|
||||||
|
->options(User::where('status', 'aktif')->pluck('name', 'id'))
|
||||||
|
->searchable()->required(),
|
||||||
|
TextInput::make('period')->label('Periode (YYYY-MM)')
|
||||||
|
->placeholder(now()->format('Y-m'))
|
||||||
|
->required(),
|
||||||
|
TextInput::make('amount')->label('Jumlah (Rp)')->numeric()->required(),
|
||||||
|
Select::make('status')->options(['lunas' => 'Lunas', 'belum' => 'Belum Lunas'])
|
||||||
|
->default('belum')->required()
|
||||||
|
->live(),
|
||||||
|
DatePicker::make('paid_at')->label('Tanggal Bayar')
|
||||||
|
->visible(fn ($get) => $get('status') === 'lunas'),
|
||||||
|
Textarea::make('notes')->label('Catatan')->rows(2)->columnSpanFull(),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Filament\Resources\MemberDues\Tables;
|
||||||
|
|
||||||
|
use Filament\Actions\BulkActionGroup;
|
||||||
|
use Filament\Actions\DeleteBulkAction;
|
||||||
|
use Filament\Actions\EditAction;
|
||||||
|
use Filament\Tables\Columns\BadgeColumn;
|
||||||
|
use Filament\Tables\Columns\TextColumn;
|
||||||
|
use Filament\Tables\Filters\SelectFilter;
|
||||||
|
use Filament\Tables\Table;
|
||||||
|
|
||||||
|
class MemberDuesTable
|
||||||
|
{
|
||||||
|
public static function configure(Table $table): Table
|
||||||
|
{
|
||||||
|
return $table
|
||||||
|
->columns([
|
||||||
|
TextColumn::make('user.name')->label('Anggota')->searchable()->sortable(),
|
||||||
|
TextColumn::make('period')->label('Periode')->sortable(),
|
||||||
|
TextColumn::make('amount')->label('Jumlah')->money('IDR')->sortable(),
|
||||||
|
TextColumn::make('status')->label('Status')->badge()
|
||||||
|
->color(fn ($state) => $state === 'lunas' ? 'success' : 'danger'),
|
||||||
|
TextColumn::make('paid_at')->label('Tgl Bayar')->date('d M Y')->placeholder('-'),
|
||||||
|
TextColumn::make('notes')->label('Catatan')->limit(30)->placeholder('-'),
|
||||||
|
])
|
||||||
|
->defaultSort('period', 'desc')
|
||||||
|
->filters([
|
||||||
|
SelectFilter::make('status')->options(['lunas' => 'Lunas', 'belum' => 'Belum Lunas']),
|
||||||
|
])
|
||||||
|
->recordActions([EditAction::make()])
|
||||||
|
->toolbarActions([BulkActionGroup::make([DeleteBulkAction::make()])]);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Models;
|
||||||
|
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
||||||
|
|
||||||
|
class MemberDue extends Model
|
||||||
|
{
|
||||||
|
protected $fillable = [
|
||||||
|
'user_id', 'amount', 'period', 'status', 'paid_at', 'notes', 'created_by',
|
||||||
|
];
|
||||||
|
|
||||||
|
protected $casts = [
|
||||||
|
'paid_at' => 'date',
|
||||||
|
];
|
||||||
|
|
||||||
|
protected static function booted(): void
|
||||||
|
{
|
||||||
|
static::creating(function (MemberDue $due) {
|
||||||
|
$due->created_by ??= auth()->id();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function user(): BelongsTo
|
||||||
|
{
|
||||||
|
return $this->belongsTo(User::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function creator(): BelongsTo
|
||||||
|
{
|
||||||
|
return $this->belongsTo(User::class, 'created_by');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use Illuminate\Database\Migrations\Migration;
|
||||||
|
use Illuminate\Database\Schema\Blueprint;
|
||||||
|
use Illuminate\Support\Facades\Schema;
|
||||||
|
|
||||||
|
return new class extends Migration
|
||||||
|
{
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
Schema::create('member_dues', function (Blueprint $table) {
|
||||||
|
$table->id();
|
||||||
|
$table->foreignId('user_id')->constrained('users');
|
||||||
|
$table->unsignedInteger('amount');
|
||||||
|
$table->string('period'); // format: YYYY-MM
|
||||||
|
$table->enum('status', ['lunas', 'belum'])->default('belum');
|
||||||
|
$table->date('paid_at')->nullable();
|
||||||
|
$table->text('notes')->nullable();
|
||||||
|
$table->foreignId('created_by')->constrained('users');
|
||||||
|
$table->timestamps();
|
||||||
|
|
||||||
|
$table->unique(['user_id', 'period']);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
Schema::dropIfExists('member_dues');
|
||||||
|
}
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user