diff --git a/app/Filament/Tenant/Resources/WorkOrderResource.php b/app/Filament/Tenant/Resources/WorkOrderResource.php
index 8055bad..76bad9c 100644
--- a/app/Filament/Tenant/Resources/WorkOrderResource.php
+++ b/app/Filament/Tenant/Resources/WorkOrderResource.php
@@ -101,6 +101,35 @@ class WorkOrderResource extends Resource
Forms\Components\Textarea::make('diagnosis')->label('Diagnostic')->rows(3)->columnSpanFull(),
Forms\Components\Textarea::make('recommendations')->label('Recomandări')->rows(2)->columnSpanFull(),
]),
+ Schemas\Components\Section::make('Foto')
+ ->collapsible()
+ ->schema([
+ \Filament\Forms\Components\SpatieMediaLibraryFileUpload::make('photos')
+ ->label('Fotografii')
+ ->collection('photos')
+ ->multiple()
+ ->reorderable()
+ ->image()
+ ->imageEditor()
+ ->maxFiles(20)
+ ->columnSpanFull(),
+ ]),
+ Schemas\Components\Section::make('Tracking & ETA')
+ ->columns(3)
+ ->collapsible()
+ ->schema([
+ Forms\Components\DateTimePicker::make('eta_at')
+ ->label('Gata estimat (ETA)')
+ ->seconds(false),
+ Forms\Components\TextInput::make('tracking_token')
+ ->label('Token public')
+ ->disabled()
+ ->dehydrated(false)
+ ->columnSpan(2)
+ ->helperText(fn (?WorkOrder $record) => $record?->tracking_token
+ ? 'Link client: ' . $record->trackingUrl()
+ : 'Se generează la salvare'),
+ ]),
Schemas\Components\Section::make('Plată & total')
->columns(3)
->schema([
diff --git a/app/Filament/Tenant/Resources/WorkOrderResource/Pages/EditWorkOrder.php b/app/Filament/Tenant/Resources/WorkOrderResource/Pages/EditWorkOrder.php
index 63cc074..1615a20 100644
--- a/app/Filament/Tenant/Resources/WorkOrderResource/Pages/EditWorkOrder.php
+++ b/app/Filament/Tenant/Resources/WorkOrderResource/Pages/EditWorkOrder.php
@@ -15,6 +15,16 @@ class EditWorkOrder extends EditRecord
protected function getHeaderActions(): array
{
return [
+ Actions\Action::make('tracking')
+ ->label('Link client (QR)')
+ ->icon('heroicon-m-qr-code')
+ ->color('primary')
+ ->modalHeading(fn () => 'Tracking client — WO #' . $this->record->number)
+ ->modalSubmitAction(false)
+ ->modalCancelActionLabel('Închide')
+ ->modalContent(fn () => view('filament.tenant.tracking-qr', [
+ 'wo' => $this->record,
+ ])),
Actions\Action::make('pdf')
->label('Descarcă PDF')
->icon('heroicon-m-document-arrow-down')
diff --git a/app/Http/Controllers/TrackingController.php b/app/Http/Controllers/TrackingController.php
new file mode 100644
index 0000000..8a3bf93
--- /dev/null
+++ b/app/Http/Controllers/TrackingController.php
@@ -0,0 +1,66 @@
+current();
+ if (! $tenant) {
+ throw new NotFoundHttpException('Tracking only available on tenant subdomain.');
+ }
+
+ $wo = WorkOrder::with(['client', 'vehicle', 'master', 'media'])
+ ->where('tracking_token', $token)
+ ->first();
+
+ if (! $wo) {
+ throw new NotFoundHttpException('Fișa nu a fost găsită.');
+ }
+
+ return view('tracking.show', [
+ 'wo' => $wo,
+ 'tenant' => $tenant,
+ 'photos' => $wo->getMedia('photos'),
+ ]);
+ }
+
+ public function qr(Request $request, string $token)
+ {
+ $tenant = app(TenantManager::class)->current();
+ if (! $tenant) {
+ throw new NotFoundHttpException();
+ }
+ $wo = WorkOrder::where('tracking_token', $token)->first();
+ if (! $wo) {
+ throw new NotFoundHttpException();
+ }
+
+ $options = new \chillerlan\QRCode\QROptions([
+ 'outputType' => \chillerlan\QRCode\QRCode::OUTPUT_MARKUP_SVG,
+ 'eccLevel' => \chillerlan\QRCode\QRCode::ECC_M,
+ 'scale' => 6,
+ 'imageBase64' => false,
+ 'svgViewBoxSize' => 200,
+ 'addQuietzone' => true,
+ ]);
+
+ $svg = (new \chillerlan\QRCode\QRCode($options))->render($wo->trackingUrl());
+
+ return response($svg, 200, [
+ 'Content-Type' => 'image/svg+xml',
+ 'Cache-Control' => 'public, max-age=3600',
+ ]);
+ }
+}
diff --git a/app/Models/Concerns/Auditable.php b/app/Models/Concerns/Auditable.php
index c45cd58..c7da127 100644
--- a/app/Models/Concerns/Auditable.php
+++ b/app/Models/Concerns/Auditable.php
@@ -22,7 +22,7 @@ trait Auditable
return LogOptions::defaults()
->logFillable()
->logOnlyDirty()
- ->dontSubmitEmptyLogs()
+ ->dontLogEmptyChanges()
->setDescriptionForEvent(fn (string $event) => match ($event) {
'created' => 'creat',
'updated' => 'modificat',
diff --git a/app/Models/Tenant/WorkOrder.php b/app/Models/Tenant/WorkOrder.php
index 186782b..e8d9fe1 100644
--- a/app/Models/Tenant/WorkOrder.php
+++ b/app/Models/Tenant/WorkOrder.php
@@ -8,10 +8,12 @@ use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\SoftDeletes;
+use Spatie\MediaLibrary\HasMedia;
+use Spatie\MediaLibrary\InteractsWithMedia;
-class WorkOrder extends Model
+class WorkOrder extends Model implements HasMedia
{
- use Auditable, BelongsToTenant, SoftDeletes;
+ use Auditable, BelongsToTenant, InteractsWithMedia, SoftDeletes;
public const STATUSES = [
'new' => 'Nou',
@@ -38,17 +40,29 @@ class WorkOrder extends Model
'complaint', 'diagnosis', 'recommendations',
'status', 'pay_status', 'approved', 'approved_at',
'discount_pct', 'total',
+ 'eta_at', 'tracking_token',
];
protected $casts = [
'opened_at' => 'date',
'closed_at' => 'date',
'approved_at' => 'datetime',
+ 'eta_at' => 'datetime',
'approved' => 'boolean',
'discount_pct' => 'decimal:2',
'total' => 'decimal:2',
];
+ public function registerMediaCollections(): void
+ {
+ $this->addMediaCollection('photos');
+ }
+
+ public function trackingUrl(): string
+ {
+ return url('/t/' . $this->tracking_token);
+ }
+
public function client(): BelongsTo
{
return $this->belongsTo(Client::class);
@@ -112,6 +126,12 @@ class WorkOrder extends Model
/** Auto-send 'ready' email + broadcast WS event on status change. */
protected static function booted(): void
{
+ static::creating(function (self $wo) {
+ if (empty($wo->tracking_token)) {
+ $wo->tracking_token = \Illuminate\Support\Str::random(24);
+ }
+ });
+
static::updated(function (self $wo) {
if (
$wo->wasChanged('status')
diff --git a/database/migrations/2026_05_27_120000_add_eta_and_tracking_to_work_orders.php b/database/migrations/2026_05_27_120000_add_eta_and_tracking_to_work_orders.php
new file mode 100644
index 0000000..1fa9f9a
--- /dev/null
+++ b/database/migrations/2026_05_27_120000_add_eta_and_tracking_to_work_orders.php
@@ -0,0 +1,35 @@
+dateTime('eta_at')->nullable()->after('closed_at');
+ $t->string('tracking_token', 32)->nullable()->after('eta_at');
+ $t->unique('tracking_token');
+ });
+
+ WorkOrder::withoutGlobalScopes()
+ ->whereNull('tracking_token')
+ ->cursor()
+ ->each(function (WorkOrder $wo) {
+ $wo->tracking_token = Str::random(24);
+ $wo->saveQuietly();
+ });
+ }
+
+ public function down(): void
+ {
+ Schema::table('work_orders', function (Blueprint $t) {
+ $t->dropUnique(['tracking_token']);
+ $t->dropColumn(['eta_at', 'tracking_token']);
+ });
+ }
+};
diff --git a/phpunit.xml b/phpunit.xml
index e7f0a48..10790e1 100644
--- a/phpunit.xml
+++ b/phpunit.xml
@@ -32,5 +32,6 @@
+ Trimite acest link clientului — vede statusul fișei, ETA și fotografiile fără să se logheze. +
+ ++ Gata estimat: {{ $wo->eta_at->isoFormat('D MMM YYYY, HH:mm') }} +
+ @endif +{{ $wo->complaint }}
+