e48ef1b755
Mechanic Workflow (Stage 7): - /app/mechanic Filament page filtered to master_id = auth user - Kanban 4 columns (in_work / awaiting_parts / ready / recent), each card shows WO#, plate, client, complaint summary, photo presence - 2 KPI tiles (active now / closed today) - Mobile-responsive grid (auto-fit, minmax 260px) WarehouseService: - issueNow(WorkOrderPart) — consume reservations immediately scoped to one line, without closing the WO (mechanic physically takes part now) - returnPart(WorkOrderPart, qty?, notes?) — refund to stock as new batch at original buy_price, writes `return` event, capped at consumed total WO PartsRelationManager: - "Eliberează" action — visible when active reservation exists - "Restituire" action — visible when consumed reservation exists, with qty modal + notes Scan Center (Stage 14): - PartResource "QR" action — per-part SVG QR with payload PART:<article|id> - BulkAction "Tipărește etichete QR" → /parts/labels?ids=N,M (HTML A4 sheet, 3-col grid, print CSS hides toolbar) - /app/scan Filament page using html5-qrcode 2.3.8 (CDN), auto-picks back camera, decodes → Livewire dispatches scanner-decoded → resolveAndRedirect - Lookup matches PART:N prefix, parts.article, parts.barcode, or numeric id - Manual input fallback for browsers without camera Tests (6 new): - WarehouseIssueReturnTest (3): issueNow consumes immediately; returnPart creates positive batch + return event; over-return is capped - ScannerLookupTest (3): PART: prefix lookup, raw barcode lookup, unknown miss Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
80 lines
2.2 KiB
PHP
80 lines
2.2 KiB
PHP
<?php
|
|
|
|
namespace App\Filament\Tenant\Pages;
|
|
|
|
use App\Models\Tenant\Part;
|
|
use Filament\Notifications\Notification;
|
|
use Filament\Pages\Page;
|
|
use Livewire\Attributes\On;
|
|
|
|
/**
|
|
* Mobile scanner: opens camera in the browser, decodes QR/barcode, looks up
|
|
* Part by:
|
|
* - `PART:<article|id>` payload (our own QR labels)
|
|
* - exact barcode match on parts.barcode
|
|
* - exact article match on parts.article
|
|
* On match → redirect to Part edit page.
|
|
*/
|
|
class Scanner extends Page
|
|
{
|
|
protected static string|\BackedEnum|null $navigationIcon = 'heroicon-o-qr-code';
|
|
|
|
protected static ?string $navigationLabel = 'Scaner';
|
|
|
|
protected static string|\UnitEnum|null $navigationGroup = 'Depozit';
|
|
|
|
protected static ?int $navigationSort = 39;
|
|
|
|
protected static ?string $title = 'Scaner cod QR / Bare';
|
|
|
|
protected string $view = 'filament.tenant.pages.scanner';
|
|
|
|
public string $manual = '';
|
|
|
|
#[On('scanner-decoded')]
|
|
public function decoded(string $text): void
|
|
{
|
|
$this->resolveAndRedirect(trim($text));
|
|
}
|
|
|
|
public function submitManual(): void
|
|
{
|
|
if (trim($this->manual) === '') return;
|
|
$this->resolveAndRedirect(trim($this->manual));
|
|
}
|
|
|
|
protected function resolveAndRedirect(string $code): void
|
|
{
|
|
$clean = $code;
|
|
if (str_starts_with($clean, 'PART:')) {
|
|
$clean = substr($clean, 5);
|
|
}
|
|
|
|
$part = Part::where(function ($q) use ($clean, $code) {
|
|
$q->where('article', $clean)
|
|
->orWhere('barcode', $clean)
|
|
->orWhere('barcode', $code);
|
|
if (ctype_digit($clean)) $q->orWhere('id', (int) $clean);
|
|
})
|
|
->first();
|
|
|
|
if (! $part) {
|
|
Notification::make()
|
|
->title('Cod necunoscut')
|
|
->body('Nu am găsit nicio piesă pentru: ' . $code)
|
|
->warning()
|
|
->send();
|
|
return;
|
|
}
|
|
|
|
Notification::make()
|
|
->title('Piesă găsită: ' . $part->name)
|
|
->success()
|
|
->send();
|
|
|
|
$this->redirect(
|
|
route('filament.tenant.resources.parts.edit', ['record' => $part->id])
|
|
);
|
|
}
|
|
}
|