Files
autocrm/app/Models/Tenant/Lead.php
T
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

110 lines
3.1 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 Lead extends Model
{
use Auditable, BelongsToTenant, SoftDeletes;
public const STATUSES = [
'new' => 'Nou',
'contacted' => 'Contactat',
'no_answer' => 'Fără răspuns',
'scheduled' => 'Programat',
'converted' => 'Convertit',
'lost' => 'Pierdut',
];
public const SOURCES = [
'manual' => 'Manual',
'call' => 'Apel',
'site' => 'Site',
'telegram' => 'Telegram',
'whatsapp' => 'WhatsApp',
'viber' => 'Viber',
'facebook' => 'Facebook',
'instagram' => 'Instagram',
'tiktok' => 'TikTok',
'google' => 'Google',
'google_maps' => 'Google Maps',
'seo' => 'SEO',
'recommend' => 'Recomandare',
];
protected $fillable = [
'company_id', 'client_id', 'vehicle_id',
'name', 'phone', 'email', 'car', 'model', 'message',
'source', 'marketing_channel',
'utm_source', 'utm_medium', 'utm_campaign', 'utm_term', 'utm_content',
'status', 'budget', 'assigned_to', 'deal_id',
'contacted_at', 'converted_at', 'notes',
];
protected $casts = [
'budget' => 'decimal:2',
'contacted_at' => 'datetime',
'converted_at' => 'datetime',
];
public function client(): BelongsTo
{
return $this->belongsTo(Client::class);
}
public function vehicle(): BelongsTo
{
return $this->belongsTo(Vehicle::class);
}
public function assignedTo(): BelongsTo
{
return $this->belongsTo(User::class, 'assigned_to');
}
public function deal(): BelongsTo
{
return $this->belongsTo(Deal::class);
}
/** Convert lead → client + deal (idempotent if already converted). */
public function convert(?array $dealAttrs = null): Deal
{
if ($this->deal_id) {
return $this->deal;
}
$client = $this->client_id
? $this->client
: Client::firstOrCreate(
['company_id' => $this->company_id, 'phone' => $this->phone],
['type' => 'individual', 'name' => $this->name, 'email' => $this->email, 'source' => $this->source]
);
$deal = Deal::create(array_merge([
'company_id' => $this->company_id,
'client_id' => $client->id,
'name' => trim(($this->car ?? '') . ' ' . ($this->model ?? '')) ?: $this->name,
'price' => $this->budget,
'stage' => 'new',
'source' => $this->source,
'note' => $this->message,
'assigned_to' => $this->assigned_to,
], $dealAttrs ?? []));
$this->update([
'client_id' => $client->id,
'deal_id' => $deal->id,
'status' => 'converted',
'converted_at' => now(),
]);
return $deal;
}
}