Batch 1: Procentaj + Finanțe consolidat + Recomandări
═══ Procentaj (markup rules) ═══
- markup_rules table cu type (category/brand/range), key, range_from/to, markup_pct, priority
- MarkupRule::bestForPart($part) — rezolvare brand → category → range → 30% default
- MarkupRule::applyToPart($part) — recalc sell_price = buy_price × (1 + pct/100)
- Filament resource sub Depozit cu form dinamic per tip
- Action 'Aplică toate regulile la stoc' — recalc tot catalogul (chunk 100)
═══ Finanțe consolidat ═══
- Custom Page /app/finance cu 4 tab-uri:
• Overview: încasări/cheltuieli/profit/datorii (4 cards)
• Cashflow: bar chart per zi (verde=in, roșu=out) + Net total
• P&L: venituri (manopere + piese) vs costuri (cost piese + cheltuieli pe categorie)
+ profit net + marjă %
• Balance: active (cash net + datorii + stoc), all-time totals
- Period filter: lună / luna trecută / an / 30 zile
═══ Recomandări ═══
- Custom Page /app/recommendations 4 sectiuni:
• Clienți pierduți (>6 luni fără WO + are istoric)
• Mașini km>100k (sugestie revizie)
• Fișe neplătite (rest > 0)
• VIP fără contact >30 zile
Total tenant routes: 100.
This commit is contained in:
@@ -0,0 +1,64 @@
|
||||
<?php
|
||||
|
||||
namespace App\Filament\Tenant\Pages;
|
||||
|
||||
use App\Models\Tenant\Client;
|
||||
use App\Models\Tenant\Vehicle;
|
||||
use App\Models\Tenant\WorkOrder;
|
||||
use Filament\Pages\Page;
|
||||
|
||||
class Recommendations extends Page
|
||||
{
|
||||
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-light-bulb';
|
||||
|
||||
protected static ?string $navigationLabel = 'Recomandări';
|
||||
|
||||
protected static string|\UnitEnum|null $navigationGroup = 'Analiză';
|
||||
|
||||
protected static ?int $navigationSort = 72;
|
||||
|
||||
protected static ?string $title = 'Recomandări — clienți de urmărit';
|
||||
|
||||
protected string $view = 'filament.tenant.pages.recommendations';
|
||||
|
||||
public function lostClients(): \Illuminate\Support\Collection
|
||||
{
|
||||
// Clients with no WO in last 6 months and previously >= 1 visit.
|
||||
$cutoff = now()->subMonths(6);
|
||||
return Client::with(['workOrders' => fn ($q) => $q->latest('opened_at')->limit(1)])
|
||||
->whereHas('workOrders', fn ($q) => $q->where('opened_at', '<', $cutoff))
|
||||
->whereDoesntHave('workOrders', fn ($q) => $q->where('opened_at', '>=', $cutoff))
|
||||
->where('status', '!=', 'lost')
|
||||
->limit(30)
|
||||
->get();
|
||||
}
|
||||
|
||||
public function highMileageVehicles(): \Illuminate\Support\Collection
|
||||
{
|
||||
return Vehicle::with('client')
|
||||
->where('mileage', '>', 100000)
|
||||
->orderByDesc('mileage')
|
||||
->limit(20)
|
||||
->get();
|
||||
}
|
||||
|
||||
public function unpaidWO(): \Illuminate\Support\Collection
|
||||
{
|
||||
return WorkOrder::with(['client', 'vehicle'])
|
||||
->where('pay_status', '!=', 'paid')
|
||||
->whereNotIn('status', ['cancelled'])
|
||||
->where('total', '>', 0)
|
||||
->orderByDesc('opened_at')
|
||||
->limit(30)
|
||||
->get();
|
||||
}
|
||||
|
||||
public function vipNeedingTouchup(): \Illuminate\Support\Collection
|
||||
{
|
||||
// VIP clients with no contact in 30+ days.
|
||||
return Client::where('status', 'vip')
|
||||
->where(fn ($q) => $q->whereNull('last_contact_at')->orWhere('last_contact_at', '<', now()->subDays(30)))
|
||||
->limit(20)
|
||||
->get();
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user