12-05-2026 admin, Panel Displays
This commit is contained in:
parent
0762e3beac
commit
6a65354f4c
43 changed files with 3273 additions and 410 deletions
|
|
@ -1,14 +1,51 @@
|
|||
<?php
|
||||
|
||||
use App\Models\Display;
|
||||
use App\Models\DisplayPlaylist;
|
||||
use App\Models\DisplayPlaylistItem;
|
||||
use App\Models\DisplayVersion;
|
||||
use App\Models\DisplayVersionItem;
|
||||
use Illuminate\Support\Facades\URL;
|
||||
|
||||
beforeEach(function () {
|
||||
$portalDomain = config('domains.domain_portal');
|
||||
url()->forceRootUrl('https://'.$portalDomain);
|
||||
URL::forceRootUrl('https://'.$portalDomain);
|
||||
});
|
||||
|
||||
function publishDisplayModules(Display $display, array $versionIds): DisplayPlaylist
|
||||
{
|
||||
$playlist = DisplayPlaylist::factory()->published()->create([
|
||||
'display_id' => $display->id,
|
||||
]);
|
||||
|
||||
foreach (array_values($versionIds) as $sortOrder => $versionId) {
|
||||
DisplayPlaylistItem::factory()->create([
|
||||
'display_playlist_id' => $playlist->id,
|
||||
'display_version_id' => $versionId,
|
||||
'sort_order' => $sortOrder,
|
||||
]);
|
||||
}
|
||||
|
||||
return $playlist;
|
||||
}
|
||||
|
||||
function draftDisplayModules(Display $display, array $versionIds): DisplayPlaylist
|
||||
{
|
||||
$playlist = DisplayPlaylist::factory()->draft()->create([
|
||||
'display_id' => $display->id,
|
||||
]);
|
||||
|
||||
foreach (array_values($versionIds) as $sortOrder => $versionId) {
|
||||
DisplayPlaylistItem::factory()->create([
|
||||
'display_playlist_id' => $playlist->id,
|
||||
'display_version_id' => $versionId,
|
||||
'sort_order' => $sortOrder,
|
||||
]);
|
||||
}
|
||||
|
||||
return $playlist;
|
||||
}
|
||||
|
||||
test('returns playlist with video-display config', function () {
|
||||
$version = DisplayVersion::factory()->create(['type' => 'video-display']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
|
|
@ -22,7 +59,7 @@ test('returns playlist with video-display config', function () {
|
|||
'content' => ['headline' => 'Hello', 'subline' => 'World', 'url' => null],
|
||||
]);
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach($version->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
|
|
@ -33,6 +70,26 @@ test('returns playlist with video-display config', function () {
|
|||
$response->assertJsonPath('playlist.0.footerContent.0.headline', 'Hello');
|
||||
});
|
||||
|
||||
test('returns display media video urls without legacy asset prefix', function () {
|
||||
$version = DisplayVersion::factory()->create(['type' => 'video-display']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $version->id,
|
||||
'item_type' => 'video',
|
||||
'content' => [
|
||||
'filename' => '/storage/display-media/2026/05/video.mp4',
|
||||
'title' => 'Uploaded Video',
|
||||
'position' => 25,
|
||||
],
|
||||
]);
|
||||
$display = Display::factory()->create();
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonPath('playlist.0.videoPlaylist.0.src', '/storage/display-media/2026/05/video.mp4');
|
||||
});
|
||||
|
||||
test('returns playlist with b2in config', function () {
|
||||
$version = DisplayVersion::factory()->create([
|
||||
'type' => 'b2in',
|
||||
|
|
@ -51,7 +108,7 @@ test('returns playlist with b2in config', function () {
|
|||
],
|
||||
]);
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach($version->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
|
|
@ -90,7 +147,7 @@ test('returns playlist with offers config', function () {
|
|||
],
|
||||
]);
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach($version->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
|
|
@ -118,10 +175,7 @@ test('returns playlist with multiple versions in order', function () {
|
|||
]);
|
||||
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach([
|
||||
$videoVersion->id => ['sort_order' => 0],
|
||||
$b2inVersion->id => ['sort_order' => 1],
|
||||
]);
|
||||
publishDisplayModules($display, [$videoVersion->id, $b2inVersion->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
|
|
@ -134,7 +188,7 @@ test('returns playlist with multiple versions in order', function () {
|
|||
test('returns 404 for inactive display', function () {
|
||||
$version = DisplayVersion::factory()->create();
|
||||
$display = Display::factory()->create(['is_active' => false]);
|
||||
$display->versions()->attach($version->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
|
|
@ -152,7 +206,7 @@ test('returns 404 for display without versions', function () {
|
|||
test('check endpoint returns only updated_at', function () {
|
||||
$version = DisplayVersion::factory()->create();
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach($version->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$version->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/check");
|
||||
|
||||
|
|
@ -160,6 +214,206 @@ test('check endpoint returns only updated_at', function () {
|
|||
$response->assertJsonStructure(['updated_at']);
|
||||
});
|
||||
|
||||
test('display config ignores legacy pivot and reads published playlist', function () {
|
||||
$legacyVersion = DisplayVersion::factory()->create(['type' => 'video-display', 'name' => 'Legacy']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $legacyVersion->id,
|
||||
'item_type' => 'video',
|
||||
'content' => ['filename' => 'legacy.mp4', 'title' => 'Legacy', 'position' => 25],
|
||||
]);
|
||||
|
||||
$publishedVersion = DisplayVersion::factory()->create(['type' => 'video-display', 'name' => 'Published']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $publishedVersion->id,
|
||||
'item_type' => 'video',
|
||||
'content' => ['filename' => 'published.mp4', 'title' => 'Published', 'position' => 25],
|
||||
]);
|
||||
|
||||
$display = Display::factory()->create();
|
||||
$display->versions()->attach($legacyVersion->id, ['sort_order' => 0]);
|
||||
publishDisplayModules($display, [$publishedVersion->id]);
|
||||
|
||||
$response = $this->getJson("/api/display/{$display->id}/config");
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonPath('playlist.0.version_name', 'Published');
|
||||
$response->assertJsonPath('playlist.0.videoPlaylist.0.src', 'assets/published.mp4');
|
||||
});
|
||||
|
||||
test('draft preview returns playlist by preview token', function () {
|
||||
$liveVersion = DisplayVersion::factory()->create(['type' => 'video-display', 'name' => 'Live']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $liveVersion->id,
|
||||
'item_type' => 'video',
|
||||
'content' => ['filename' => 'live.mp4', 'title' => 'Live', 'position' => 25],
|
||||
]);
|
||||
|
||||
$draftVersion = DisplayVersion::factory()->create(['type' => 'video-display', 'name' => 'Draft']);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $draftVersion->id,
|
||||
'item_type' => 'video',
|
||||
'content' => ['filename' => 'draft.mp4', 'title' => 'Draft', 'position' => 25],
|
||||
]);
|
||||
|
||||
$display = Display::factory()->create(['preview_token' => 'preview-token-123']);
|
||||
publishDisplayModules($display, [$liveVersion->id]);
|
||||
draftDisplayModules($display, [$draftVersion->id]);
|
||||
|
||||
$response = $this->getJson('/api/display/preview/preview-token-123');
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonPath('playlist.0.version_name', 'Draft');
|
||||
$response->assertJsonPath('playlist.0.videoPlaylist.0.src', 'assets/draft.mp4');
|
||||
});
|
||||
|
||||
test('draft preview requires configured draft playlist', function () {
|
||||
$display = Display::factory()->create(['preview_token' => 'empty-preview-token']);
|
||||
|
||||
$response = $this->getJson('/api/display/preview/empty-preview-token');
|
||||
|
||||
$response->assertNotFound();
|
||||
});
|
||||
|
||||
test('module preview returns a single module config', function () {
|
||||
$version = DisplayVersion::factory()->create([
|
||||
'type' => 'b2in',
|
||||
'name' => 'Preview Module',
|
||||
'settings' => ['theme' => 'light'],
|
||||
]);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $version->id,
|
||||
'item_type' => 'media',
|
||||
'content' => [
|
||||
'category' => 'moebel',
|
||||
'media_type' => 'image',
|
||||
'media_url' => '../assets/module.jpg',
|
||||
'headline' => 'Modul',
|
||||
'subline' => 'Vorschau',
|
||||
'duration_seconds' => 8,
|
||||
],
|
||||
]);
|
||||
|
||||
$response = $this->getJson("/api/display/module/{$version->id}/preview");
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonCount(1, 'playlist');
|
||||
$response->assertJsonPath('playlist.0.version_name', 'Preview Module');
|
||||
$response->assertJsonPath('playlist.0.items.0.category', 'moebel');
|
||||
});
|
||||
|
||||
test('module preview exposes configurable player chrome settings', function () {
|
||||
$version = DisplayVersion::factory()->create([
|
||||
'type' => 'offers',
|
||||
'name' => 'Custom Chrome',
|
||||
'settings' => [
|
||||
'logo_url' => '/storage/display-media/logo.svg',
|
||||
'brand_text' => 'Musterstadt',
|
||||
'footer_claim' => 'Beratung im Schauraum',
|
||||
'footer_url' => 'cabinet-bielefeld.de',
|
||||
'qr_default_title' => 'Mehr Infos',
|
||||
'qr_subtitle' => 'Jetzt scannen',
|
||||
],
|
||||
]);
|
||||
DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $version->id,
|
||||
'item_type' => 'slide',
|
||||
'content' => [
|
||||
'type' => 'intro',
|
||||
'title' => 'Intro',
|
||||
'image_url' => '../assets/intro.jpg',
|
||||
],
|
||||
]);
|
||||
|
||||
$response = $this->getJson("/api/display/module/{$version->id}/preview");
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonPath('playlist.0.settings.logo_url', '/storage/display-media/logo.svg');
|
||||
$response->assertJsonPath('playlist.0.settings.brand_text', 'Musterstadt');
|
||||
$response->assertJsonPath('playlist.0.settings.footer_claim', 'Beratung im Schauraum');
|
||||
$response->assertJsonPath('playlist.0.settings.footer_url', 'cabinet-bielefeld.de');
|
||||
$response->assertJsonPath('playlist.0.settings.qr_default_title', 'Mehr Infos');
|
||||
$response->assertJsonPath('playlist.0.settings.qr_subtitle', 'Jetzt scannen');
|
||||
});
|
||||
|
||||
test('module item preview returns only the selected slide', function () {
|
||||
$version = DisplayVersion::factory()->create([
|
||||
'type' => 'offers',
|
||||
'name' => 'Offers Module',
|
||||
]);
|
||||
$firstSlide = DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $version->id,
|
||||
'item_type' => 'slide',
|
||||
'content' => [
|
||||
'type' => 'product-hero',
|
||||
'title' => 'First Slide',
|
||||
'image_url' => '../assets/first.jpg',
|
||||
],
|
||||
'sort_order' => 0,
|
||||
]);
|
||||
$secondSlide = DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $version->id,
|
||||
'item_type' => 'slide',
|
||||
'content' => [
|
||||
'type' => 'product-details',
|
||||
'title' => 'Second Slide',
|
||||
'image_url' => '../assets/second.jpg',
|
||||
],
|
||||
'sort_order' => 1,
|
||||
]);
|
||||
|
||||
$response = $this->getJson("/api/display/module/{$version->id}/item/{$secondSlide->id}/preview");
|
||||
|
||||
$response->assertSuccessful();
|
||||
$response->assertJsonCount(1, 'playlist');
|
||||
$response->assertJsonCount(1, 'playlist.0.slides');
|
||||
$response->assertJsonPath('playlist.0.slides.0.title', 'Second Slide');
|
||||
$response->assertJsonMissingPath('playlist.0.slides.1');
|
||||
|
||||
expect($firstSlide->fresh()->content['title'])->toBe('First Slide');
|
||||
});
|
||||
|
||||
test('module item preview requires the item to belong to the module', function () {
|
||||
$version = DisplayVersion::factory()->create(['type' => 'offers']);
|
||||
$otherVersion = DisplayVersion::factory()->create(['type' => 'offers']);
|
||||
$item = DisplayVersionItem::factory()->create([
|
||||
'display_version_id' => $otherVersion->id,
|
||||
'item_type' => 'slide',
|
||||
]);
|
||||
|
||||
$this->getJson("/api/display/module/{$version->id}/item/{$item->id}/preview")
|
||||
->assertNotFound();
|
||||
});
|
||||
|
||||
test('display player keeps previews in a strict 9 by 16 viewport', function () {
|
||||
$player = file_get_contents(public_path('_cabinet/display/index.html'));
|
||||
|
||||
expect($player)
|
||||
->toContain('width: min(100vw, calc(100vh * 9 / 16));')
|
||||
->toContain('height: min(100vh, calc(100vw * 16 / 9));')
|
||||
->toContain('container-type: size;')
|
||||
->toContain('translate(${offsetX}px, ${offsetY}px) scale(${scale})')
|
||||
->toContain('this.settings.logo_url')
|
||||
->toContain('this.settings.footer_claim')
|
||||
->toContain('this.settings.footer_url')
|
||||
->toContain('this.settings.header_logo_url')
|
||||
->toContain('this.settings.qr_label');
|
||||
});
|
||||
|
||||
test('preview player pages are reachable for valid display and module previews', function () {
|
||||
$display = Display::factory()->create(['preview_token' => 'page-preview-token']);
|
||||
$version = DisplayVersion::factory()->create();
|
||||
$item = DisplayVersionItem::factory()->create(['display_version_id' => $version->id]);
|
||||
|
||||
$this->get('/preview/page-preview-token')
|
||||
->assertSuccessful();
|
||||
|
||||
$this->get("/preview/module/{$version->id}")
|
||||
->assertSuccessful();
|
||||
|
||||
$this->get("/preview/module/{$version->id}/item/{$item->id}")
|
||||
->assertSuccessful();
|
||||
});
|
||||
|
||||
test('existing display config api still works', function () {
|
||||
$response = $this->getJson('/api/display/config');
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue