## Summary - Fixes misleading “queued / running in background” message when Review Pack generation request reuses an existing ready pack (fingerprint dedupe). - Improves resilience of Filament/Livewire interactions by ensuring the Livewire intercept shim applies after Livewire initializes. - Aligns Review Pack operation notifications with Ops-UX patterns (queued + completed notifications) and removes the old ReviewPackStatusNotification. ## Key Changes - Review Pack generate action now: - Shows queued toast only when a new pack is actually created/queued. - Shows a “Review pack already available” success notification with a link when dedupe returns an existing pack. ## Tests - `vendor/bin/sail artisan test --compact tests/Feature/ReviewPack/ReviewPackGenerationTest.php` - `vendor/bin/sail artisan test --compact tests/Feature/ReviewPack/ReviewPackResourceTest.php` - `vendor/bin/sail artisan test --compact tests/Feature/LivewireInterceptShimTest.php` ## Notes - No global search behavior changes for ReviewPacks (still excluded). - Destructive actions remain confirmation-gated (`->requiresConfirmation()`). Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de> Reviewed-on: #133
165 lines
5.9 KiB
PHP
165 lines
5.9 KiB
PHP
@php
|
|
use App\Support\Badges\BadgeCatalog;
|
|
use App\Support\Badges\BadgeDomain;
|
|
use App\Support\ReviewPackStatus;
|
|
|
|
/** @var ?\App\Models\Tenant $tenant */
|
|
/** @var ?\App\Models\ReviewPack $pack */
|
|
/** @var ?ReviewPackStatus $statusEnum */
|
|
/** @var bool $canView */
|
|
/** @var bool $canManage */
|
|
/** @var ?string $downloadUrl */
|
|
/** @var ?string $failedReason */
|
|
|
|
$badgeSpec = $statusEnum ? BadgeCatalog::spec(BadgeDomain::ReviewPackStatus, $statusEnum->value) : null;
|
|
@endphp
|
|
|
|
<x-filament::section heading="Review Pack">
|
|
@if (! $pack)
|
|
{{-- State 1: No pack --}}
|
|
<div class="flex flex-col items-center gap-3 py-4 text-center">
|
|
<x-heroicon-o-document-arrow-down class="h-8 w-8 text-gray-400 dark:text-gray-500" />
|
|
<div class="text-sm text-gray-500 dark:text-gray-400">
|
|
No review pack generated yet.
|
|
</div>
|
|
|
|
@if ($canManage)
|
|
<x-filament::button
|
|
size="sm"
|
|
wire:click="generatePack"
|
|
wire:loading.attr="disabled"
|
|
>
|
|
Generate pack
|
|
</x-filament::button>
|
|
@endif
|
|
</div>
|
|
@elseif ($statusEnum === ReviewPackStatus::Queued || $statusEnum === ReviewPackStatus::Generating)
|
|
{{-- State 2: Queued / Generating --}}
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex items-center gap-2">
|
|
<x-filament::badge
|
|
:color="$badgeSpec?->color"
|
|
:icon="$badgeSpec?->icon"
|
|
>
|
|
{{ $badgeSpec?->label ?? '—' }}
|
|
</x-filament::badge>
|
|
</div>
|
|
|
|
<div class="flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400">
|
|
<x-filament::loading-indicator class="h-4 w-4" />
|
|
Generation in progress…
|
|
</div>
|
|
|
|
<div class="text-xs text-gray-400 dark:text-gray-500">
|
|
Started {{ $pack->created_at?->diffForHumans() ?? '—' }}
|
|
</div>
|
|
</div>
|
|
@elseif ($statusEnum === ReviewPackStatus::Ready)
|
|
{{-- State 3: Ready --}}
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex items-center gap-2">
|
|
<x-filament::badge
|
|
:color="$badgeSpec?->color"
|
|
:icon="$badgeSpec?->icon"
|
|
>
|
|
{{ $badgeSpec?->label ?? '—' }}
|
|
</x-filament::badge>
|
|
</div>
|
|
|
|
<dl class="grid grid-cols-2 gap-x-4 gap-y-1 text-sm">
|
|
<dt class="text-gray-500 dark:text-gray-400">Generated</dt>
|
|
<dd>{{ $pack->generated_at?->format('M j, Y H:i') ?? '—' }}</dd>
|
|
|
|
<dt class="text-gray-500 dark:text-gray-400">Expires</dt>
|
|
<dd>{{ $pack->expires_at?->format('M j, Y') ?? '—' }}</dd>
|
|
|
|
<dt class="text-gray-500 dark:text-gray-400">Size</dt>
|
|
<dd>{{ $pack->file_size ? Number::fileSize($pack->file_size) : '—' }}</dd>
|
|
</dl>
|
|
|
|
<div class="flex items-center gap-2">
|
|
@if ($canView && $downloadUrl)
|
|
<x-filament::button
|
|
size="sm"
|
|
tag="a"
|
|
:href="$downloadUrl"
|
|
target="_blank"
|
|
icon="heroicon-o-arrow-down-tray"
|
|
>
|
|
Download
|
|
</x-filament::button>
|
|
@endif
|
|
|
|
@if ($canManage)
|
|
<x-filament::button
|
|
size="sm"
|
|
color="gray"
|
|
wire:click="generatePack"
|
|
wire:loading.attr="disabled"
|
|
>
|
|
Generate new
|
|
</x-filament::button>
|
|
@endif
|
|
</div>
|
|
</div>
|
|
@elseif ($statusEnum === ReviewPackStatus::Failed)
|
|
{{-- State 4: Failed --}}
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex items-center gap-2">
|
|
<x-filament::badge
|
|
:color="$badgeSpec?->color"
|
|
:icon="$badgeSpec?->icon"
|
|
>
|
|
{{ $badgeSpec?->label ?? '—' }}
|
|
</x-filament::badge>
|
|
</div>
|
|
|
|
@if ($failedReason)
|
|
<div class="text-sm text-danger-600 dark:text-danger-400">
|
|
{{ $failedReason }}
|
|
</div>
|
|
@endif
|
|
|
|
<div class="text-xs text-gray-400 dark:text-gray-500">
|
|
{{ $pack->updated_at?->diffForHumans() ?? '—' }}
|
|
</div>
|
|
|
|
@if ($canManage)
|
|
<x-filament::button
|
|
size="sm"
|
|
wire:click="generatePack"
|
|
wire:loading.attr="disabled"
|
|
>
|
|
Retry
|
|
</x-filament::button>
|
|
@endif
|
|
</div>
|
|
@elseif ($statusEnum === ReviewPackStatus::Expired)
|
|
{{-- State 5: Expired --}}
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex items-center gap-2">
|
|
<x-filament::badge
|
|
:color="$badgeSpec?->color"
|
|
:icon="$badgeSpec?->icon"
|
|
>
|
|
{{ $badgeSpec?->label ?? '—' }}
|
|
</x-filament::badge>
|
|
</div>
|
|
|
|
<div class="text-sm text-gray-500 dark:text-gray-400">
|
|
Expired {{ $pack->expires_at?->diffForHumans() ?? '—' }}
|
|
</div>
|
|
|
|
@if ($canManage)
|
|
<x-filament::button
|
|
size="sm"
|
|
wire:click="generatePack"
|
|
wire:loading.attr="disabled"
|
|
>
|
|
Generate new
|
|
</x-filament::button>
|
|
@endif
|
|
</div>
|
|
@endif
|
|
</x-filament::section>
|