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:
@@ -13,7 +13,10 @@ use App\Models\Tenant\Lead;
|
||||
use App\Models\Tenant\Post;
|
||||
use App\Models\Tenant\User;
|
||||
use App\Models\Tenant\Vehicle;
|
||||
use App\Models\Tenant\Call;
|
||||
use App\Models\Tenant\Expense;
|
||||
use App\Models\Tenant\MarketingChannel;
|
||||
use App\Models\Tenant\MessageTemplate;
|
||||
use App\Models\Tenant\Part;
|
||||
use App\Models\Tenant\Payment;
|
||||
use App\Models\Tenant\Purchase;
|
||||
@@ -424,6 +427,52 @@ class DatabaseSeeder extends Seeder
|
||||
['client_id' => $c1->id, 'method' => 'cash', 'reference' => 'CHIT-001', 'user_id' => $admin->id]
|
||||
);
|
||||
|
||||
// ─── Template-uri mesaje demo ───────────────────────────
|
||||
$templates = [
|
||||
['Programare confirmată', 'telegram', 'Salut, {name}! Programarea pentru {service} este confirmată pe {date} la {time}. Vă așteptăm! 🚗'],
|
||||
['Auto gata de ridicare', 'telegram', 'Salut, {name}! Mașina dvs. {car} este gata. Total: {amount} MDL. ✅'],
|
||||
['Reminder revizie', 'whatsapp', 'Salut, {name}! Vă reamintim — pentru {car} se apropie revizia (kilometraj {mileage}). Programați-vă din timp! 🔧'],
|
||||
['Reminder ITP', 'sms', '{name}, ITP-ul mașinii {car} expiră luna aceasta. Programați-vă: 022-123-456'],
|
||||
['Felicitare zi naștere', 'telegram', '🎉 La mulți ani, {name}! Discount 15% la orice manoperă luna aceasta.'],
|
||||
];
|
||||
foreach ($templates as [$name, $channel, $body]) {
|
||||
MessageTemplate::firstOrCreate(
|
||||
['company_id' => $psauto->id, 'name' => $name],
|
||||
['channel' => $channel, 'body' => $body, 'is_active' => true]
|
||||
);
|
||||
}
|
||||
|
||||
// ─── Canale marketing demo ──────────────────────────────
|
||||
$channels = [
|
||||
['Google Ads', '🔍', '#EA4335', 5000, 4200, 28, 12, 48000],
|
||||
['Facebook', '📘', '#1877F2', 3000, 2800, 18, 7, 22000],
|
||||
['Instagram', '📸', '#C13584', 2000, 1900, 22, 9, 31000],
|
||||
['Telegram', '✈️', '#229ED9', 500, 200, 15, 8, 26000],
|
||||
['Recomandări', '⭐', '#F59E0B', 0, 0, 35, 25, 95000],
|
||||
];
|
||||
foreach ($channels as [$name, $icon, $color, $budget, $spent, $leads, $conv, $revenue]) {
|
||||
MarketingChannel::firstOrCreate(
|
||||
['company_id' => $psauto->id, 'name' => $name],
|
||||
[
|
||||
'icon' => $icon, 'color' => $color,
|
||||
'budget_monthly' => $budget, 'spent_monthly' => $spent,
|
||||
'leads_count' => $leads, 'converted_count' => $conv,
|
||||
'revenue' => $revenue, 'is_active' => true,
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
// ─── Apeluri demo ───────────────────────────────────────
|
||||
Call::firstOrCreate(
|
||||
['company_id' => $psauto->id, 'phone' => '+373 69 100001', 'called_at' => today()->subHours(3)],
|
||||
['direction' => 'incoming', 'duration_sec' => 185, 'status' => 'answered',
|
||||
'client_id' => $c1->id, 'user_id' => $admin->id, 'notes' => 'Programare diagnostic']
|
||||
);
|
||||
Call::firstOrCreate(
|
||||
['company_id' => $psauto->id, 'phone' => '+373 79 602002', 'called_at' => today()->subHours(5)],
|
||||
['direction' => 'missed', 'duration_sec' => 0, 'status' => 'missed', 'notes' => '']
|
||||
);
|
||||
|
||||
// ─── Cheltuieli demo ────────────────────────────────────
|
||||
$expensesData = [
|
||||
['salary', 'Salariu Vasile Ivanov', 8000, today()->startOfMonth(), 'cash'],
|
||||
|
||||
Reference in New Issue
Block a user