06696727dd
══════ 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.
90 lines
3.3 KiB
PHP
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('/'),
|
|
];
|
|
}
|
|
}
|