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.
80 lines
2.0 KiB
PHP
80 lines
2.0 KiB
PHP
<?php
|
|
|
|
namespace App\Models\Tenant;
|
|
|
|
use App\Models\Concerns\BelongsToTenant;
|
|
use App\Models\Concerns\Auditable;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
|
|
|
class Payment extends Model
|
|
{
|
|
use Auditable, BelongsToTenant, SoftDeletes;
|
|
|
|
public const METHODS = [
|
|
'cash' => 'Numerar',
|
|
'card' => 'Card',
|
|
'transfer' => 'Virament',
|
|
'mobile' => 'Mobile pay',
|
|
];
|
|
|
|
protected $fillable = [
|
|
'company_id', 'client_id', 'work_order_id', 'user_id',
|
|
'paid_at', 'amount', 'method', 'reference', 'notes',
|
|
];
|
|
|
|
protected $casts = [
|
|
'paid_at' => 'date',
|
|
'amount' => 'decimal:2',
|
|
];
|
|
|
|
public function client(): BelongsTo
|
|
{
|
|
return $this->belongsTo(Client::class);
|
|
}
|
|
|
|
public function workOrder(): BelongsTo
|
|
{
|
|
return $this->belongsTo(WorkOrder::class);
|
|
}
|
|
|
|
public function user(): BelongsTo
|
|
{
|
|
return $this->belongsTo(User::class);
|
|
}
|
|
|
|
/**
|
|
* After save/delete, recompute the work order's pay_status.
|
|
*/
|
|
protected static function booted(): void
|
|
{
|
|
$sync = function (self $payment) {
|
|
if (! $payment->work_order_id) {
|
|
return;
|
|
}
|
|
$wo = WorkOrder::withoutGlobalScopes()->find($payment->work_order_id);
|
|
if (! $wo) {
|
|
return;
|
|
}
|
|
$paid = (float) static::withoutGlobalScopes()
|
|
->where('work_order_id', $wo->id)
|
|
->whereNull('deleted_at')
|
|
->sum('amount');
|
|
$total = (float) $wo->total;
|
|
|
|
if ($paid <= 0) {
|
|
$wo->pay_status = 'unpaid';
|
|
} elseif ($paid + 0.01 < $total) {
|
|
$wo->pay_status = 'partial';
|
|
} else {
|
|
$wo->pay_status = 'paid';
|
|
}
|
|
$wo->save();
|
|
};
|
|
|
|
static::saved($sync);
|
|
static::deleted($sync);
|
|
}
|
|
}
|