Faza 3.5+3.6+4+5: Marketing, Reports, Provisioning, PWA
═══ Faza 3.5: Marketing ═══
Schema: msg_templates, marketing_channels, calls
Modele cu logică:
- MessageTemplate::render($context) — substituie {key} tokens
- MarketingChannel: roi/conversion_rate/cost_per_lead computed attrs
- Call: duration_formatted helper
Resources Filament (group Marketing):
- MessageTemplateResource: 5 canale (telegram/whatsapp/viber/sms/email)
- MarketingChannelResource: budget vs revenue cu ROI live calculat
- CallResource: in/out/missed cu filtre azi/missed
═══ Faza 3.6: Analytics ═══
Custom Filament Page Reports cu 6 rapoarte tab-uite:
- Finanțe: încasări/cheltuieli/profit/datorii + breakdown pe metodă/categorie
- Încărcare: fișe deschise/închise + breakdown pe status
- Mecanici: ore lucrate, manopere, venit per mecanic
- Manopere top: cele mai frecvente cu nr/ore/venit
- Piese: top vândute + low-stock
- Clienți: noi în perioadă + lead-uri pe sursă
Selector perioadă: azi / săptămâna / luna / luna trecută / anul
═══ Faza 4: Central provisioning ═══
- CoolifyClient service (Coolify v4 REST API wrapper)
- CompanyProvisioner: creează Company + admin user + roles + adaugă
subdomeniul în Coolify FQDN + trigger redeploy automat
- CreateCompany page override → folosește provisioner, returnează
notificare cu credentialele admin
- Form CompanyResource extins cu admin_name/email/password (vizibil doar create)
- Action 'Suspendă' / 'Activează' pe table cu confirmation
Env vars necesare în Coolify pentru provisioning auto:
COOLIFY_API_URL=http://65.21.20.141:8000
COOLIFY_API_TOKEN=<token>
COOLIFY_APP_UUID=g13hlrpd5g44zxl5af3ktio2
═══ Faza 5: PWA + branding ═══
- Route /manifest.json dinamic per tenant (nume, theme color, icons)
- Route /sw.js — service worker minimal (cache shell + static)
- TenantPanelProvider renderHook HEAD_END — link manifest + theme-color
+ apple-mobile-web-app meta
- TenantPanelProvider renderHook BODY_END — registrare service worker
Seed extins:
- 5 template-uri mesaje (programare/auto-gata/reminder/ITP/felicitare)
- 5 canale marketing (Google Ads/FB/IG/Telegram/Recomandări)
- 2 apeluri demo
Total Filament tenant routes: 81.
This commit is contained in:
@@ -0,0 +1,59 @@
|
||||
<?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 Call extends Model
|
||||
{
|
||||
use BelongsToTenant, SoftDeletes;
|
||||
|
||||
public const DIRECTIONS = [
|
||||
'incoming' => 'Primit',
|
||||
'outgoing' => 'Efectuat',
|
||||
'missed' => 'Pierdut',
|
||||
];
|
||||
|
||||
public const STATUSES = [
|
||||
'answered' => 'Răspuns',
|
||||
'missed' => 'Pierdut',
|
||||
'busy' => 'Ocupat',
|
||||
'no_answer' => 'Fără răspuns',
|
||||
];
|
||||
|
||||
protected $fillable = [
|
||||
'company_id', 'client_id', 'lead_id', 'user_id',
|
||||
'direction', 'phone', 'called_at', 'duration_sec',
|
||||
'status', 'recording_url', 'notes', 'lead_created',
|
||||
];
|
||||
|
||||
protected $casts = [
|
||||
'called_at' => 'datetime',
|
||||
'lead_created' => 'boolean',
|
||||
];
|
||||
|
||||
public function client(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(Client::class);
|
||||
}
|
||||
|
||||
public function lead(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(Lead::class);
|
||||
}
|
||||
|
||||
public function user(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(User::class);
|
||||
}
|
||||
|
||||
public function getDurationFormattedAttribute(): string
|
||||
{
|
||||
$sec = (int) $this->duration_sec;
|
||||
if ($sec === 0) return '—';
|
||||
return sprintf('%d:%02d', intdiv($sec, 60), $sec % 60);
|
||||
}
|
||||
}
|
||||
@@ -32,6 +32,16 @@ class Client extends Model
|
||||
return $this->hasMany(Vehicle::class);
|
||||
}
|
||||
|
||||
public function workOrders(): HasMany
|
||||
{
|
||||
return $this->hasMany(WorkOrder::class);
|
||||
}
|
||||
|
||||
public function payments(): HasMany
|
||||
{
|
||||
return $this->hasMany(Payment::class);
|
||||
}
|
||||
|
||||
public function assignedTo(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(User::class, 'assigned_to');
|
||||
|
||||
@@ -0,0 +1,53 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models\Tenant;
|
||||
|
||||
use App\Models\Concerns\BelongsToTenant;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Illuminate\Database\Eloquent\SoftDeletes;
|
||||
|
||||
class MarketingChannel extends Model
|
||||
{
|
||||
use BelongsToTenant, SoftDeletes;
|
||||
|
||||
protected $fillable = [
|
||||
'company_id', 'name', 'icon', 'color',
|
||||
'budget_monthly', 'spent_monthly',
|
||||
'leads_count', 'converted_count', 'revenue',
|
||||
'is_active', 'notes',
|
||||
];
|
||||
|
||||
protected $casts = [
|
||||
'budget_monthly' => 'decimal:2',
|
||||
'spent_monthly' => 'decimal:2',
|
||||
'revenue' => 'decimal:2',
|
||||
'is_active' => 'boolean',
|
||||
];
|
||||
|
||||
public function getRoiAttribute(): float
|
||||
{
|
||||
$spent = (float) $this->spent_monthly;
|
||||
if ($spent <= 0) {
|
||||
return 0;
|
||||
}
|
||||
return round((((float) $this->revenue - $spent) / $spent) * 100, 2);
|
||||
}
|
||||
|
||||
public function getConversionRateAttribute(): float
|
||||
{
|
||||
$leads = (int) $this->leads_count;
|
||||
if ($leads <= 0) {
|
||||
return 0;
|
||||
}
|
||||
return round(((int) $this->converted_count / $leads) * 100, 2);
|
||||
}
|
||||
|
||||
public function getCostPerLeadAttribute(): float
|
||||
{
|
||||
$leads = (int) $this->leads_count;
|
||||
if ($leads <= 0) {
|
||||
return 0;
|
||||
}
|
||||
return round((float) $this->spent_monthly / $leads, 2);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,44 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models\Tenant;
|
||||
|
||||
use App\Models\Concerns\BelongsToTenant;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Illuminate\Database\Eloquent\SoftDeletes;
|
||||
|
||||
class MessageTemplate extends Model
|
||||
{
|
||||
use BelongsToTenant, SoftDeletes;
|
||||
|
||||
protected $table = 'msg_templates';
|
||||
|
||||
public const CHANNELS = [
|
||||
'telegram' => 'Telegram',
|
||||
'whatsapp' => 'WhatsApp',
|
||||
'viber' => 'Viber',
|
||||
'sms' => 'SMS',
|
||||
'email' => 'Email',
|
||||
];
|
||||
|
||||
protected $fillable = [
|
||||
'company_id', 'name', 'channel', 'subject',
|
||||
'body', 'variables', 'is_active',
|
||||
];
|
||||
|
||||
protected $casts = [
|
||||
'variables' => 'array',
|
||||
'is_active' => 'boolean',
|
||||
];
|
||||
|
||||
/**
|
||||
* Render template body with the given context, replacing {key} tokens.
|
||||
*/
|
||||
public function render(array $context = []): string
|
||||
{
|
||||
$body = $this->body;
|
||||
foreach ($context as $key => $val) {
|
||||
$body = str_replace('{' . $key . '}', (string) $val, $body);
|
||||
}
|
||||
return $body;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user