Files
Vasyka 06696727dd 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.
2026-05-07 09:52:01 +00:00

90 lines
3.3 KiB
PHP

<?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('/'),
];
}
}