Faza 3.1: CRM core — Leads, Deals, Appointments, Settings, Widgets, Users

Spatie Permission cu teams (team_foreign_key=company_id, teams=true):
- migrations create_permission_tables (model_has_roles cu company_id scope)
- HasRoles trait pe User
- ResolveTenant middleware setează permissions team_id la tenant.id
- Seed: 7 roluri default per tenant (admin/manager/receptionist/mechanic/parts_manager/accountant/marketer)

Module noi:
- Leads (cereri): name, phone, car/model, source, UTM, status, budget, assigned_to,
  acțiune "Convertește" → creează automat Client + Deal
- Deals (pipeline): client/vehicle, stage (8 stage-uri), price, source, lost_reason
- Posts + Appointments: post_id (boxă), master_id, date+time_start+time_end, status, color
- UserResource (tenant): CRUD users cu role/status/locale; canViewAny doar pentru admin

Custom Filament page "Setări" (tenant):
- Brand & contact (display_name, city, phone, email)
- Localizare (limba RO/RU/EN, currency, theme color picker)
- Servicii & tarif (labor_rate)
- Liste configurabile (services, cars) — păstrate în companies.settings JSON

Widgets dashboard:
- Tenant: StatsOverview (Clienți, Mașini, Cereri noi, Deal-uri active, Programări azi)
- Central: PlatformStats (Companii total/active/trial, Expiră în 7 zile)

Seed extins demo PSauto:
- 3 posturi (Pod 1/2/3 cu culori)
- 2 lead-uri demo (Alex Grosu Telegram, Irina Cojocaru WhatsApp)
- 3 deal-uri demo (BMW done, Audi in_work, Porsche agree)
- 2 programări (azi + mâine)

Filament v5 fixes:
- $navigationGroup type → string|UnitEnum|null (parent stricter signature)
- Toate resources noi au tipurile corecte
This commit is contained in:
2026-05-06 17:36:32 +00:00
parent 4b1635d045
commit c9cb3560ef
34 changed files with 1742 additions and 3 deletions
+55
View File
@@ -0,0 +1,55 @@
<?php
namespace App\Models\Tenant;
use App\Models\Concerns\BelongsToTenant;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
class Appointment extends Model
{
use BelongsToTenant;
public const STATUSES = [
'scheduled' => 'Programat',
'arrived' => 'Sosit',
'done' => 'Finalizat',
'cancelled' => 'Anulat',
'no_show' => 'Neprezentat',
];
protected $fillable = [
'company_id', 'post_id', 'client_id', 'vehicle_id', 'master_id', 'deal_id',
'date', 'time_start', 'time_end',
'title', 'color', 'status', 'notes',
];
protected $casts = [
'date' => 'date',
];
public function post(): BelongsTo
{
return $this->belongsTo(Post::class);
}
public function client(): BelongsTo
{
return $this->belongsTo(Client::class);
}
public function vehicle(): BelongsTo
{
return $this->belongsTo(Vehicle::class);
}
public function master(): BelongsTo
{
return $this->belongsTo(User::class, 'master_id');
}
public function deal(): BelongsTo
{
return $this->belongsTo(Deal::class);
}
}
+56
View File
@@ -0,0 +1,56 @@
<?php
namespace App\Models\Tenant;
use App\Models\Concerns\BelongsToTenant;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\SoftDeletes;
class Deal extends Model
{
use BelongsToTenant, SoftDeletes;
public const STAGES = [
'new' => 'Nou',
'contact' => 'Contact',
'agree' => 'Aprobare',
'scheduled' => 'Programat',
'arrived' => 'Sosit',
'in_work' => 'În lucru',
'done' => 'Finalizat',
'lost' => 'Pierdut',
];
protected $fillable = [
'company_id', 'client_id', 'vehicle_id',
'name', 'price', 'stage', 'source', 'note',
'assigned_to', 'won_at', 'lost_at', 'lost_reason',
];
protected $casts = [
'price' => 'decimal:2',
'won_at' => 'datetime',
'lost_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 isOpen(): bool
{
return ! in_array($this->stage, ['done', 'lost'], true);
}
}
+108
View File
@@ -0,0 +1,108 @@
<?php
namespace App\Models\Tenant;
use App\Models\Concerns\BelongsToTenant;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\SoftDeletes;
class Lead extends Model
{
use 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;
}
}
+23
View File
@@ -0,0 +1,23 @@
<?php
namespace App\Models\Tenant;
use App\Models\Concerns\BelongsToTenant;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\HasMany;
class Post extends Model
{
use BelongsToTenant;
protected $fillable = ['company_id', 'name', 'color', 'is_active', 'sort_order'];
protected $casts = [
'is_active' => 'boolean',
];
public function appointments(): HasMany
{
return $this->hasMany(Appointment::class);
}
}
+5 -1
View File
@@ -9,6 +9,7 @@ use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\SoftDeletes;
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Spatie\Permission\Traits\HasRoles;
/**
* Tenant-bound user. Belongs to exactly one Company.
@@ -17,7 +18,10 @@ use Illuminate\Notifications\Notifiable;
*/
class User extends Authenticatable implements FilamentUser
{
use BelongsToTenant, HasFactory, Notifiable, SoftDeletes;
use BelongsToTenant, HasFactory, HasRoles, Notifiable, SoftDeletes;
/** Spatie Permission scope key matches the team_foreign_key (company_id). */
protected $guard_name = 'web';
protected $fillable = [
'company_id', 'name', 'email', 'phone', 'avatar_url',