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.
110 lines
3.1 KiB
PHP
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;
|
|
}
|
|
}
|