Faza 6: Activity log + Kanban + Payroll + cleanup

══════ Activity log (Spatie) ══════
- spatie/laravel-activitylog v5 instalat
- Migration cu company_id pentru tenant scoping
- Trait Auditable (App\Models\Concerns\Auditable):
  - LogOptions cu logFillable + logOnlyDirty + dontSubmitEmptyLogs
  - tapActivity auto-fill company_id + causer
  - Descrieri RO (creat/modificat/șters/restaurat)
- Aplicat pe: Client, Vehicle, Lead, Deal, WorkOrder, Payment, Expense
- ActivityResource (group Admin → Jurnal activitate)
  - Listă read-only, scope pe tenant, filtre by description/today

══════ Kanban Work Orders ══════
- Custom Filament page la /app/kanban (group Service)
- 6 coloane (new → diagnosis → agreement → in_work → awaiting_parts → ready)
- Drag-drop nativ HTML5 cu wire:click moveCard()
- Cards arată: număr fișă, client, auto, plate, master, total
- Link 'Deschide' direct la editare WO

══════ Payroll (Salarii) ══════
Schema:
- employee_profiles: user_id, position, base_salary, works_pct, parts_pct
- payroll_runs: period (YYYY-MM), base, works_revenue/pct, parts_margin/pct,
  bonus, fines, advance, total auto-calculat
- payroll_adjustments: bonus/fine/advance per period

PayrollCalculator service:
- compute($userId, $period) — calculează auto:
  - Manopere finalizate de mecanic în luna respectivă (sum total)
  - Marja pieselor montate de el (sell-buy * qty)
  - Bonus + fines + advance from adjustments
  - Total = base + works% + parts% + bonus - fines - advance

Resources Filament (group Finanțe):
- EmployeeProfileResource: profil cu % comisioane
- PayrollRunResource: salarii cu action 'Calculează luna curentă' (toți userii)
  + per-row 'Recalculează'; Sum summary pe total
- PayrollAdjustmentResource: gestionare bonus/penalizări/avansuri

══════ Cleanup ══════
- Șterse toate /__debug, /__seed, /__try-login, /__force-login, /__whoami,
  /__coolify-check (security)
- Routes/web.php conține doar / redirect, /manifest.json, /sw.js

Total Filament tenant routes: 92.
This commit is contained in:
2026-05-07 09:52:01 +00:00
parent 2b4fa666ad
commit 06696727dd
34 changed files with 1180 additions and 27 deletions
@@ -0,0 +1,89 @@
<?php
namespace App\Filament\Tenant\Resources;
use App\Filament\Tenant\Resources\ActivityResource\Pages;
use App\Tenancy\TenantManager;
use Filament\Resources\Resource;
use Filament\Tables;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Spatie\Activitylog\Models\Activity;
class ActivityResource extends Resource
{
protected static ?string $model = Activity::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-list-bullet';
protected static ?string $navigationLabel = 'Jurnal activitate';
protected static string|\UnitEnum|null $navigationGroup = 'Admin';
protected static ?string $modelLabel = 'eveniment';
protected static ?string $pluralModelLabel = 'jurnal';
protected static ?int $navigationSort = 95;
public static function canCreate(): bool
{
return false;
}
public static function getEloquentQuery(): Builder
{
$tenantId = app(TenantManager::class)->currentId();
return parent::getEloquentQuery()
->when($tenantId, fn ($q) => $q->where('company_id', $tenantId));
}
public static function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('created_at')->label('Când')->dateTime('d.m.Y H:i')->sortable(),
Tables\Columns\TextColumn::make('description')->label('Acțiune')->badge()
->colors([
'success' => ['creat'],
'info' => ['modificat'],
'danger' => ['șters'],
'warning' => ['restaurat'],
]),
Tables\Columns\TextColumn::make('subject_type')
->label('Tip')
->formatStateUsing(fn ($s) => $s ? class_basename($s) : '—'),
Tables\Columns\TextColumn::make('subject_id')->label('ID')->placeholder('—'),
Tables\Columns\TextColumn::make('causer.name')->label('De către')->placeholder('Sistem'),
Tables\Columns\TextColumn::make('attribute_changes')
->label('Detalii')
->formatStateUsing(function ($state) {
if (! $state) return '—';
$arr = is_string($state) ? json_decode($state, true) : $state;
$changes = $arr['attributes'] ?? [];
return collect($changes)
->map(fn ($v, $k) => "{$k}: " . (is_scalar($v) ? \Illuminate\Support\Str::limit((string)$v, 30) : '—'))
->take(3)->implode(', ');
})
->wrap(),
])
->filters([
Tables\Filters\SelectFilter::make('description')
->label('Acțiune')
->options(['creat' => 'creat', 'modificat' => 'modificat', 'șters' => 'șters']),
Tables\Filters\Filter::make('today')
->label('Astăzi')
->query(fn ($q) => $q->whereDate('created_at', today())),
])
->defaultSort('created_at', 'desc')
->actions([])
->paginated([25, 50, 100]);
}
public static function getPages(): array
{
return [
'index' => Pages\ListActivities::route('/'),
];
}
}
@@ -0,0 +1,11 @@
<?php
namespace App\Filament\Tenant\Resources\ActivityResource\Pages;
use App\Filament\Tenant\Resources\ActivityResource;
use Filament\Resources\Pages\ListRecords;
class ListActivities extends ListRecords
{
protected static string $resource = ActivityResource::class;
}
@@ -0,0 +1,94 @@
<?php
namespace App\Filament\Tenant\Resources;
use App\Filament\Tenant\Resources\EmployeeProfileResource\Pages;
use App\Models\Tenant\EmployeeProfile;
use App\Models\Tenant\User;
use Filament\Actions;
use Filament\Forms;
use Filament\Resources\Resource;
use Filament\Schemas;
use Filament\Schemas\Schema;
use Filament\Tables;
use Filament\Tables\Table;
class EmployeeProfileResource extends Resource
{
protected static ?string $model = EmployeeProfile::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-identification';
protected static ?string $navigationLabel = 'Angajați';
protected static string|\UnitEnum|null $navigationGroup = 'Finanțe';
protected static ?string $modelLabel = 'angajat';
protected static ?string $pluralModelLabel = 'angajați';
protected static ?int $navigationSort = 52;
public static function form(Schema $schema): Schema
{
return $schema->components([
Schemas\Components\Section::make('Profil')
->columns(2)
->schema([
Forms\Components\Select::make('user_id')
->label('Utilizator')
->options(fn () => User::orderBy('name')->pluck('name', 'id'))
->searchable()
->required()
->unique(ignoreRecord: true),
Forms\Components\TextInput::make('position')->label('Funcție')->maxLength(120),
Forms\Components\DatePicker::make('hire_date')->label('Angajat din'),
]),
Schemas\Components\Section::make('Salariu & comisioane')
->columns(3)
->schema([
Forms\Components\TextInput::make('base_salary')->label('Salariu bază')->numeric()->default(0),
Forms\Components\TextInput::make('works_pct')
->label('% manopere')
->numeric()->default(0)
->suffix('%')
->helperText('% din venitul manoperelor finalizate.'),
Forms\Components\TextInput::make('parts_pct')
->label('% marja piese')
->numeric()->default(0)
->suffix('%')
->helperText('% din marja (sell-buy) pieselor montate.'),
]),
Forms\Components\Textarea::make('notes')->label('Notițe')->columnSpanFull()->rows(2),
]);
}
public static function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('user.name')->label('Nume')->searchable()->sortable(),
Tables\Columns\TextColumn::make('position')->label('Funcție')->placeholder('—'),
Tables\Columns\TextColumn::make('base_salary')->money('MDL')->alignRight(),
Tables\Columns\TextColumn::make('works_pct')->label('% Manopere')
->formatStateUsing(fn ($s) => $s . '%')->alignRight(),
Tables\Columns\TextColumn::make('parts_pct')->label('% Piese')
->formatStateUsing(fn ($s) => $s . '%')->alignRight(),
Tables\Columns\TextColumn::make('hire_date')->date('d.m.Y')->placeholder('—'),
])
->actions([
Actions\EditAction::make(),
Actions\DeleteAction::make(),
])
->defaultSort('user_id');
}
public static function getPages(): array
{
return [
'index' => Pages\ListEmployeeProfiles::route('/'),
'create' => Pages\CreateEmployeeProfile::route('/create'),
'edit' => Pages\EditEmployeeProfile::route('/{record}/edit'),
];
}
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\EmployeeProfileResource\Pages;
use App\Filament\Tenant\Resources\EmployeeProfileResource;
use Filament\Actions;
use Filament\Resources\Pages\CreateRecord;
class CreateEmployeeProfile extends CreateRecord
{
protected static string $resource = EmployeeProfileResource::class;
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\EmployeeProfileResource\Pages;
use App\Filament\Tenant\Resources\EmployeeProfileResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditEmployeeProfile extends EditRecord
{
protected static string $resource = EmployeeProfileResource::class;
protected function getHeaderActions(): array { return [Actions\DeleteAction::make()]; }
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\EmployeeProfileResource\Pages;
use App\Filament\Tenant\Resources\EmployeeProfileResource;
use Filament\Actions;
use Filament\Resources\Pages\ListRecords;
class ListEmployeeProfiles extends ListRecords
{
protected static string $resource = EmployeeProfileResource::class;
protected function getHeaderActions(): array { return [Actions\CreateAction::make()]; }
}
@@ -0,0 +1,99 @@
<?php
namespace App\Filament\Tenant\Resources;
use App\Filament\Tenant\Resources\PayrollAdjustmentResource\Pages;
use App\Models\Tenant\PayrollAdjustment;
use App\Models\Tenant\User;
use Filament\Actions;
use Filament\Forms;
use Filament\Resources\Resource;
use Filament\Schemas;
use Filament\Schemas\Schema;
use Filament\Tables;
use Filament\Tables\Table;
class PayrollAdjustmentResource extends Resource
{
protected static ?string $model = PayrollAdjustment::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-adjustments-horizontal';
protected static ?string $navigationLabel = 'Bonusuri & avansuri';
protected static string|\UnitEnum|null $navigationGroup = 'Finanțe';
protected static ?string $modelLabel = 'ajustare';
protected static ?string $pluralModelLabel = 'bonusuri/avansuri';
protected static ?int $navigationSort = 54;
public static function form(Schema $schema): Schema
{
return $schema->components([
Schemas\Components\Section::make('Detalii')
->columns(2)
->schema([
Forms\Components\Select::make('user_id')
->label('Utilizator')
->options(fn () => User::orderBy('name')->pluck('name', 'id'))
->searchable()
->required(),
Forms\Components\Select::make('type')
->options(PayrollAdjustment::TYPES)
->default('bonus')
->required(),
Forms\Components\TextInput::make('amount')->label('Sumă')->numeric()->required(),
Forms\Components\TextInput::make('period')
->label('Perioadă (YYYY-MM)')
->placeholder(now()->format('Y-m'))
->regex('/^\d{4}-\d{2}$/')
->required(),
Forms\Components\DatePicker::make('date')->default(today())->required(),
Forms\Components\TextInput::make('reason')->label('Motiv')->maxLength(160),
]),
]);
}
public static function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('date')->date('d.m.Y')->sortable(),
Tables\Columns\TextColumn::make('user.name')->label('Utilizator')->searchable(),
Tables\Columns\TextColumn::make('type')
->formatStateUsing(fn ($s) => PayrollAdjustment::TYPES[$s] ?? $s)
->badge()
->colors([
'success' => ['bonus'],
'danger' => ['fine'],
'warning' => ['advance'],
]),
Tables\Columns\TextColumn::make('period'),
Tables\Columns\TextColumn::make('amount')->money('MDL')->alignRight(),
Tables\Columns\TextColumn::make('reason')->limit(40),
Tables\Columns\IconColumn::make('applied')->boolean()->label('Aplicat'),
])
->filters([
Tables\Filters\SelectFilter::make('type')->options(PayrollAdjustment::TYPES),
Tables\Filters\SelectFilter::make('user_id')
->label('Utilizator')
->options(fn () => User::pluck('name', 'id')),
])
->actions([
Actions\EditAction::make(),
Actions\DeleteAction::make(),
])
->defaultSort('date', 'desc');
}
public static function getPages(): array
{
return [
'index' => Pages\ListPayrollAdjustments::route('/'),
'create' => Pages\CreatePayrollAdjustment::route('/create'),
'edit' => Pages\EditPayrollAdjustment::route('/{record}/edit'),
];
}
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollAdjustmentResource\Pages;
use App\Filament\Tenant\Resources\PayrollAdjustmentResource;
use Filament\Actions;
use Filament\Resources\Pages\CreateRecord;
class CreatePayrollAdjustment extends CreateRecord
{
protected static string $resource = PayrollAdjustmentResource::class;
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollAdjustmentResource\Pages;
use App\Filament\Tenant\Resources\PayrollAdjustmentResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditPayrollAdjustment extends EditRecord
{
protected static string $resource = PayrollAdjustmentResource::class;
protected function getHeaderActions(): array { return [Actions\DeleteAction::make()]; }
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollAdjustmentResource\Pages;
use App\Filament\Tenant\Resources\PayrollAdjustmentResource;
use Filament\Actions;
use Filament\Resources\Pages\ListRecords;
class ListPayrollAdjustments extends ListRecords
{
protected static string $resource = PayrollAdjustmentResource::class;
protected function getHeaderActions(): array { return [Actions\CreateAction::make()]; }
}
@@ -0,0 +1,132 @@
<?php
namespace App\Filament\Tenant\Resources;
use App\Filament\Tenant\Resources\PayrollRunResource\Pages;
use App\Models\Tenant\PayrollRun;
use App\Models\Tenant\User;
use App\Services\PayrollCalculator;
use Filament\Actions;
use Filament\Forms;
use Filament\Notifications\Notification;
use Filament\Resources\Resource;
use Filament\Schemas;
use Filament\Schemas\Schema;
use Filament\Tables;
use Filament\Tables\Table;
class PayrollRunResource extends Resource
{
protected static ?string $model = PayrollRun::class;
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-banknotes';
protected static ?string $navigationLabel = 'Salarii';
protected static string|\UnitEnum|null $navigationGroup = 'Finanțe';
protected static ?string $modelLabel = 'salariu';
protected static ?string $pluralModelLabel = 'salarii';
protected static ?int $navigationSort = 53;
public static function form(Schema $schema): Schema
{
return $schema->components([
Schemas\Components\Section::make('Detalii')
->columns(2)
->schema([
Forms\Components\Select::make('user_id')
->label('Utilizator')
->options(fn () => User::orderBy('name')->pluck('name', 'id'))
->searchable()
->required(),
Forms\Components\TextInput::make('period')
->label('Perioada (YYYY-MM)')
->placeholder(now()->format('Y-m'))
->required()
->regex('/^\d{4}-\d{2}$/'),
]),
Schemas\Components\Section::make('Calcul')
->columns(3)
->schema([
Forms\Components\TextInput::make('base')->label('Bază')->numeric()->default(0),
Forms\Components\TextInput::make('works_revenue')->label('Venit manopere')->numeric()->disabled(),
Forms\Components\TextInput::make('works_pct_amount')->label('Comision manopere')->numeric()->disabled(),
Forms\Components\TextInput::make('parts_margin')->label('Marja piese')->numeric()->disabled(),
Forms\Components\TextInput::make('parts_pct_amount')->label('Comision piese')->numeric()->disabled(),
Forms\Components\TextInput::make('bonus')->numeric()->default(0),
Forms\Components\TextInput::make('fines')->label('Penalizări')->numeric()->default(0),
Forms\Components\TextInput::make('advance')->label('Avans')->numeric()->default(0),
Forms\Components\TextInput::make('total')->label('Total net')->numeric()->disabled(),
]),
Schemas\Components\Section::make('Plată')
->columns(2)
->schema([
Forms\Components\Toggle::make('paid')->label('Achitat'),
Forms\Components\DatePicker::make('paid_at')->label('Data plății'),
]),
Forms\Components\Textarea::make('notes')->label('Notițe')->columnSpanFull()->rows(2),
]);
}
public static function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('period')->label('Perioadă')->sortable(),
Tables\Columns\TextColumn::make('user.name')->label('Utilizator')->searchable(),
Tables\Columns\TextColumn::make('base')->money('MDL')->alignRight(),
Tables\Columns\TextColumn::make('works_pct_amount')->label('% manopere')->money('MDL')->alignRight(),
Tables\Columns\TextColumn::make('parts_pct_amount')->label('% piese')->money('MDL')->alignRight(),
Tables\Columns\TextColumn::make('bonus')->money('MDL')->alignRight()->color('success'),
Tables\Columns\TextColumn::make('fines')->money('MDL')->alignRight()->color('danger'),
Tables\Columns\TextColumn::make('advance')->money('MDL')->alignRight()->color('warning'),
Tables\Columns\TextColumn::make('total')->label('Total')->money('MDL')->alignRight()->weight('bold')
->summarize(Tables\Columns\Summarizers\Sum::make()->money('MDL')),
Tables\Columns\IconColumn::make('paid')->boolean(),
])
->headerActions([
Actions\Action::make('compute_all')
->label('Calculează luna curentă')
->icon('heroicon-m-calculator')
->color('primary')
->action(function () {
$period = now()->format('Y-m');
$count = 0;
foreach (User::pluck('id') as $uid) {
app(PayrollCalculator::class)->compute($uid, $period);
$count++;
}
Notification::make()
->title("Calculat salariul {$period} pentru {$count} utilizatori")
->success()->send();
}),
])
->filters([
Tables\Filters\SelectFilter::make('period')
->label('Perioadă')
->options(fn () => PayrollRun::distinct()->pluck('period', 'period')->toArray()),
Tables\Filters\TernaryFilter::make('paid')->label('Achitat'),
])
->actions([
Actions\Action::make('recompute')
->label('Recalculează')
->icon('heroicon-m-arrow-path')
->action(fn (PayrollRun $r) => app(PayrollCalculator::class)->compute($r->user_id, $r->period)),
Actions\EditAction::make(),
Actions\DeleteAction::make(),
])
->defaultSort('period', 'desc');
}
public static function getPages(): array
{
return [
'index' => Pages\ListPayrollRuns::route('/'),
'create' => Pages\CreatePayrollRun::route('/create'),
'edit' => Pages\EditPayrollRun::route('/{record}/edit'),
];
}
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollRunResource\Pages;
use App\Filament\Tenant\Resources\PayrollRunResource;
use Filament\Actions;
use Filament\Resources\Pages\CreateRecord;
class CreatePayrollRun extends CreateRecord
{
protected static string $resource = PayrollRunResource::class;
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollRunResource\Pages;
use App\Filament\Tenant\Resources\PayrollRunResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditPayrollRun extends EditRecord
{
protected static string $resource = PayrollRunResource::class;
protected function getHeaderActions(): array { return [Actions\DeleteAction::make()]; }
}
@@ -0,0 +1,14 @@
<?php
namespace App\Filament\Tenant\Resources\PayrollRunResource\Pages;
use App\Filament\Tenant\Resources\PayrollRunResource;
use Filament\Actions;
use Filament\Resources\Pages\ListRecords;
class ListPayrollRuns extends ListRecords
{
protected static string $resource = PayrollRunResource::class;
protected function getHeaderActions(): array { return [Actions\CreateAction::make()]; }
}