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:
@@ -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('/'),
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user