954ba8f059
Schema: - online_orders (token-tracked, status workflow, delivery method/fee) - online_order_items (price snapshot, fulfilled flag) - part_cross_refs (OEM/equivalent codes for search) - parts.is_published (shop visibility) Storefront (ShopController, tenant subdomain, /shop): - Catalog with search across name/article/brand/cross-refs, category + in-stock filters, live stock, white-label themed layout - Part detail page with cross-ref codes - VIN search → VinDecoder → guided catalog search - Session cart (per-tenant key), guest checkout, order confirmation page - Respects settings.shop.enabled (404 when off); tenant-guarded Part::searchPublished matches cross-ref articles via whereHas. Order notifications (ShopOrderNotifier, best-effort): - Staff: Web Push to active users - Customer: Telegram if phone matches a linked client Filament (tenant): - OnlineOrderResource under "Magazin" nav group, status workflow, items relation, "Onorează" action issues stock via WarehouseService (FIFO) - PartResource: is_published toggle + column + bulk publish/unpublish + CrossRefsRelationManager - Settings: shop section (enable, delivery methods, fee, free-over) - Landing page: shop button when enabled Tests (6 new): - catalog 404 when disabled; lists published only; cross-ref search; order placement (token + items + total); fulfill issues stock; cross-tenant token isolation Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
50 lines
2.1 KiB
PHP
50 lines
2.1 KiB
PHP
@extends('shop.layout')
|
|
@section('title', 'Catalog piese')
|
|
@section('content')
|
|
@php $currency = $tenant->settings['currency'] ?? 'MDL'; @endphp
|
|
|
|
<form method="GET" action="/shop" class="filters">
|
|
<input type="text" name="q" value="{{ $term }}" placeholder="Caută denumire, cod, brand, cod cross…">
|
|
<select name="cat" onchange="this.form.submit()">
|
|
<option value="">Toate categoriile</option>
|
|
@foreach ($categories as $c)
|
|
<option value="{{ $c }}" {{ $category === $c ? 'selected' : '' }}>{{ $c }}</option>
|
|
@endforeach
|
|
</select>
|
|
<label style="display:flex;align-items:center;gap:6px;font-size:14px;">
|
|
<input type="checkbox" name="in_stock" value="1" {{ $inStock ? 'checked' : '' }} onchange="this.form.submit()">
|
|
Doar în stoc
|
|
</label>
|
|
<button class="btn" type="submit">Caută</button>
|
|
</form>
|
|
|
|
@if ($parts->isEmpty())
|
|
<div class="card" style="text-align:center;padding:48px;">
|
|
<p class="muted">Nicio piesă găsită{{ $term ? ' pentru „' . $term . '”' : '' }}.</p>
|
|
</div>
|
|
@else
|
|
<div class="grid">
|
|
@foreach ($parts as $p)
|
|
@php $stock = (float) $p->qty; @endphp
|
|
<div class="product">
|
|
<a href="/shop/part/{{ $p->id }}">
|
|
<h3>{{ $p->name }}</h3>
|
|
</a>
|
|
<div class="meta">
|
|
{{ $p->brand ? $p->brand . ' · ' : '' }}{{ $p->article ?? '' }}
|
|
</div>
|
|
<div class="stock {{ $stock > 0 ? 'in' : 'out' }}">
|
|
{{ $stock > 0 ? '● În stoc' : '○ La comandă' }}
|
|
</div>
|
|
<div class="price">{{ number_format((float) $p->sell_price, 2) }} {{ $currency }}</div>
|
|
<form method="POST" action="/shop/part/{{ $p->id }}/add" style="margin-top:10px;">
|
|
@csrf
|
|
<button class="btn block" type="submit">Adaugă în coș</button>
|
|
</form>
|
|
</div>
|
|
@endforeach
|
|
</div>
|
|
<div style="margin-top:20px;">{{ $parts->links() }}</div>
|
|
@endif
|
|
@endsection
|