Files
Vasyka 8fdfc9ef85 feat: Part product images + seasonal tire-swap reminders
Part (HasMedia):
- Spatie media `image` single-file collection + imageUrl() helper
- PartResource form: image upload section (image editor, 2 MB max)
- Parts list: circular thumbnail column
- Shop catalog cards: square thumbnail + 📦 placeholder
- Shop part detail: 260px image alongside info, single column when no image

Seasonal tire-swap reminders:
- NotificationDispatcher::tireSeasonalSwap(TireSet) — Telegram first, email
  fallback (when set has a vehicle, via ServiceReminderMail with 'tire_swap'
  type and a size-aware note)
- tires:remind-seasonal artisan command, self-gating to Feb 15-Mar 15
  (notify winter sets stored) and Sep 15-Oct 15 (notify summer sets stored).
  60-day cooldown per client via service_reminders_sent. --force / --dry-run.
- Schedule: weekly Mon 09:30

Tests (6 new):
- outside window no-ops; spring window notifies winter; spring ignores summer;
  autumn notifies summer; cooldown blocks doubles; --force overrides window

Full suite: 106 passed.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-02 19:31:24 +00:00

57 lines
2.4 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; $img = $p->imageUrl(); @endphp
<div class="product">
<a href="/shop/part/{{ $p->id }}" class="product-thumb">
@if ($img)
<img src="{{ $img }}" alt="{{ $p->name }}" loading="lazy">
@else
<div class="product-thumb-empty">📦</div>
@endif
</a>
<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