diff --git a/app/Filament/Exports/EmployeeMasterExporter.php b/app/Filament/Exports/EmployeeMasterExporter.php new file mode 100644 index 0000000..a3492cd --- /dev/null +++ b/app/Filament/Exports/EmployeeMasterExporter.php @@ -0,0 +1,44 @@ +label('ID'), + ExportColumn::make('plant.name'), + ExportColumn::make('name'), + ExportColumn::make('code'), + ExportColumn::make('department'), + ExportColumn::make('designation'), + ExportColumn::make('email'), + ExportColumn::make('mobile_number'), + ExportColumn::make('created_at'), + ExportColumn::make('updated_at'), + ExportColumn::make('created_by'), + ExportColumn::make('updated_by'), + ExportColumn::make('deleted_at'), + ]; + } + + public static function getCompletedNotificationBody(Export $export): string + { + $body = 'Your employee master export has completed and ' . number_format($export->successful_rows) . ' ' . str('row')->plural($export->successful_rows) . ' exported.'; + + if ($failedRowsCount = $export->getFailedRowsCount()) { + $body .= ' ' . number_format($failedRowsCount) . ' ' . str('row')->plural($failedRowsCount) . ' failed to export.'; + } + + return $body; + } +} diff --git a/app/Filament/Imports/EmployeeMasterImporter.php b/app/Filament/Imports/EmployeeMasterImporter.php new file mode 100644 index 0000000..64bc34f --- /dev/null +++ b/app/Filament/Imports/EmployeeMasterImporter.php @@ -0,0 +1,53 @@ +requiredMapping() + ->relationship() + ->rules(['required']), + ImportColumn::make('name'), + ImportColumn::make('code'), + ImportColumn::make('department'), + ImportColumn::make('designation'), + ImportColumn::make('email') + ->rules(['email']), + ImportColumn::make('mobile_number'), + ImportColumn::make('created_by'), + ImportColumn::make('updated_by'), + ]; + } + + public function resolveRecord(): ?EmployeeMaster + { + // return EmployeeMaster::firstOrNew([ + // // Update existing records, matching them by `$this->data['column_name']` + // 'email' => $this->data['email'], + // ]); + + return new EmployeeMaster(); + } + + public static function getCompletedNotificationBody(Import $import): string + { + $body = 'Your employee master import has completed and ' . number_format($import->successful_rows) . ' ' . str('row')->plural($import->successful_rows) . ' imported.'; + + if ($failedRowsCount = $import->getFailedRowsCount()) { + $body .= ' ' . number_format($failedRowsCount) . ' ' . str('row')->plural($failedRowsCount) . ' failed to import.'; + } + + return $body; + } +} diff --git a/app/Filament/Resources/EmployeeMasterResource.php b/app/Filament/Resources/EmployeeMasterResource.php new file mode 100644 index 0000000..882011e --- /dev/null +++ b/app/Filament/Resources/EmployeeMasterResource.php @@ -0,0 +1,217 @@ +schema([ + Forms\Components\Select::make('plant_id') + ->label('Plant') + ->relationship('plant', 'name') + ->required(), + Forms\Components\TextInput::make('name') + ->label('Name') + ->required() + ->reactive() + ->extraInputAttributes([ + 'oninput' => 'this.value = this.value.replace(/[^a-zA-Z\s]/g, "")', + ]), + Forms\Components\TextInput::make('code') + ->label('ID') + ->extraInputAttributes([ + 'oninput' => 'this.value = this.value.replace(/[^a-zA-Z0-9]/g, "")',]) + ->required() + ->unique( + table: 'employee_masters', + column: 'code', + ignoreRecord: true + ) + ->validationMessages([ + 'unique' => 'Duplicate employee code already exists.', + ]), + Forms\Components\TextInput::make('department') + ->label('Department') + ->required(), + Forms\Components\TextInput::make('designation') + ->label('Designation') + ->extraInputAttributes([ + 'oninput' => 'this.value = this.value.replace(/[^a-zA-Z\s]/g, "")',]) + ->required(), + Forms\Components\TextInput::make('email') + ->label('Email') + ->email() + ->required(), + Forms\Components\TextInput::make('mobile_number') + ->label('Mobile Number') + ->length(10) + ->reactive() + ->extraInputAttributes([ + 'oninput' => 'this.value = this.value.replace(/[^0-9]/g, "").slice(0, 10)', // blocks non-numbers + limits to 10 chars + 'maxlength' => 10, + ]) + ->required() + ->unique( + table: 'employee_masters', + column: 'mobile_number', + ignoreRecord: true + ) + ->validationMessages([ + 'unique' => 'Duplicate mobile number already exists.', + ]), + Forms\Components\Hidden::make('created_by') + ->label('Created by') + ->default(Filament::auth()->user()?->name ?? ''), + Forms\Components\Hidden::make('updated_by') + ->label('Updated by') + ->default(Filament::auth()->user()?->name ?? ''), + ]); + } + + public static function table(Table $table): Table + { + return $table + ->columns([ + Tables\Columns\TextColumn::make('No.') + ->label('No.') + ->getStateUsing(function ($record, $livewire, $column, $rowLoop) { + $paginator = $livewire->getTableRecords(); + $perPage = method_exists($paginator, 'perPage') ? $paginator->perPage() : 10; + $currentPage = method_exists($paginator, 'currentPage') ? $paginator->currentPage() : 1; + return ($currentPage - 1) * $perPage + $rowLoop->iteration; + }) + ->alignCenter(), + Tables\Columns\TextColumn::make('plant.name') + ->numeric() + ->sortable() + ->alignCenter(), + Tables\Columns\TextColumn::make('name') + ->label('Name') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('code') + ->label('Employee ID') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('department') + ->label('Department') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('designation') + ->label('Designation') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('email') + ->label('Email') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('mobile_number') + ->label('Mobile Number') + ->sortable() + ->alignCenter() + ->searchable(), + Tables\Columns\TextColumn::make('created_at') + ->dateTime() + ->sortable() + ->alignCenter(), + Tables\Columns\TextColumn::make('updated_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true) + ->alignCenter(), + Tables\Columns\TextColumn::make('created_by') + ->label('Created by') + ->sortable() + ->alignCenter(), + Tables\Columns\TextColumn::make('updated_by') + ->label('Updated by') + ->sortable() + ->toggleable(isToggledHiddenByDefault: true) + ->alignCenter(), + Tables\Columns\TextColumn::make('deleted_at') + ->dateTime() + ->sortable() + ->toggleable(isToggledHiddenByDefault: true) + ->alignCenter(), + ]) + ->filters([ + Tables\Filters\TrashedFilter::make(), + ]) + ->actions([ + Tables\Actions\ViewAction::make(), + Tables\Actions\EditAction::make(), + ]) + ->bulkActions([ + Tables\Actions\BulkActionGroup::make([ + Tables\Actions\DeleteBulkAction::make(), + Tables\Actions\ForceDeleteBulkAction::make(), + Tables\Actions\RestoreBulkAction::make(), + ]), + ]) + ->headerActions([ + ImportAction::make() + ->importer(EmployeeMasterImporter::class) + ->visible(function() { + return Filament::auth()->user()->can('view import employee master'); + }), + ExportAction::make() + ->exporter(EmployeeMasterExporter::class) + ->visible(function() { + return Filament::auth()->user()->can('view export employee master'); + }), + ]); + } + + public static function getRelations(): array + { + return [ + // + ]; + } + + public static function getPages(): array + { + return [ + 'index' => Pages\ListEmployeeMasters::route('/'), + 'create' => Pages\CreateEmployeeMaster::route('/create'), + 'view' => Pages\ViewEmployeeMaster::route('/{record}'), + 'edit' => Pages\EditEmployeeMaster::route('/{record}/edit'), + ]; + } + + public static function getEloquentQuery(): Builder + { + return parent::getEloquentQuery() + ->withoutGlobalScopes([ + SoftDeletingScope::class, + ]); + } +} diff --git a/app/Filament/Resources/EmployeeMasterResource/Pages/CreateEmployeeMaster.php b/app/Filament/Resources/EmployeeMasterResource/Pages/CreateEmployeeMaster.php new file mode 100644 index 0000000..d1f36eb --- /dev/null +++ b/app/Filament/Resources/EmployeeMasterResource/Pages/CreateEmployeeMaster.php @@ -0,0 +1,12 @@ +belongsTo(Plant::class); + } +} diff --git a/database/migrations/2026_05_25_110441_create_employee_masters_table.php b/database/migrations/2026_05_25_110441_create_employee_masters_table.php new file mode 100644 index 0000000..df4e2c6 --- /dev/null +++ b/database/migrations/2026_05_25_110441_create_employee_masters_table.php @@ -0,0 +1,45 @@ +