Frontend: Editorial-Relaunch der öffentlichen Strecke + Ausgaben-Routing

Öffentliche Seiten auf gemeinsames Editorial-Design (x-web.site-header/-footer,
Design-Tokens) und Ausgaben-Präfix /{edition}/ (de|en) umgestellt.

- Routing: neue Middleware SetEdition (Locale + URL::defaults), /{edition}-Gruppe
  in routes/web.php, Root-Redirect auf /de, 301 für Legacy-.html-URLs,
  Baseline-Default in AppServiceProvider.
- Neue URL-Schemata: /{edition}/press-release/{slug}, /{edition}/category/{slug}.
- Ausgabe = Sprache: DE/EN-Umschalter (Region/CH/AT entfernt); EditorialClock
  und Livewire-Komponenten sprachdynamisch.
- Detail-, Kategorie- und Veröffentlichen-Seite mit echten Daten neu aufgebaut.
- Suche aktiviert: Volt-Komponente livewire/web/search (Titel/Text/Keywords +
  Firma + Rubrik, Filter, Sortierung, Pagination, URL-Parameter q/category/sort).
- Rubriken-Navigation statt Übersichtsseite: Helper CategoryNavigation;
  web/kategorien.blade.php + Route entfernt (Legacy-301).
- Tests: Edition-Routing, Kategorie-Seite/-Navigation, Detail, Veröffentlichen,
  Suche, EditorialClock. Doku in "Echte öffentliche Unterseiten.md" aktualisiert.

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
Kevin Adametz 2026-06-16 16:39:28 +00:00
parent a0547208d3
commit 253141c6dc
64 changed files with 4457 additions and 2971 deletions

View file

@ -0,0 +1,47 @@
<?php
namespace App\Http\Middleware;
use Closure;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\URL;
use Symfony\Component\HttpFoundation\Response;
/**
* Setzt die aktive Ausgabe (Sprache) für den Public-Frontend-Request.
*
* Das Frontend wird unter einem Sprach-Präfix ausgeliefert: `/de/...` und
* `/en/...`. Die Ausgabe entspricht 1:1 der Inhalts-Sprache.
*
* Aufgaben:
* 1. Ausgabe aus dem ersten URL-Segment lesen (de|en), sonst Fallback `de`.
* 2. `URL::defaults(['edition' => …])` setzen, damit benannte Routen
* (`route('kategorie', …)`) das Präfix automatisch erhalten auch auf
* Seiten ohne Edition-Präfix (Hub/Auth) bleibt so ein gültiger Default.
* 3. App-Locale nur für echte Edition-URLs übernehmen, damit Admin-/Hub-Seiten
* ihre eigene Locale behalten.
*/
class SetEdition
{
/**
* @var array<int, string>
*/
public const EDITIONS = ['de', 'en'];
public const DEFAULT_EDITION = 'de';
public function handle(Request $request, Closure $next): Response
{
$segment = $request->segment(1);
$isEdition = in_array($segment, self::EDITIONS, true);
$edition = $isEdition ? $segment : self::DEFAULT_EDITION;
URL::defaults(['edition' => $edition]);
if ($isEdition) {
app()->setLocale($edition);
}
return $next($request);
}
}

View file

@ -6,6 +6,7 @@ use App\Contracts\NewsletterSyncClient;
use App\Helpers\ThemeHelper;
use App\Http\Middleware\EnsureUserIsAdmin;
use App\Http\Middleware\LogSlowAdminRequests;
use App\Http\Middleware\SetEdition;
use App\Listeners\ActivateUserAfterVerification;
use App\Listeners\SyncCompanyMembershipsOnLogin;
use App\Models\AdminPreset;
@ -59,6 +60,13 @@ class AppServiceProvider extends ServiceProvider
URL::forceScheme('https');
}
// Baseline für das Ausgabe-Präfix (de|en). Die SetEdition-Middleware
// überschreibt dies pro Request mit der tatsächlichen Ausgabe; der
// Default hier stellt sicher, dass route('…')-Aufrufe das {edition}-
// Segment auch außerhalb der HTTP-Middleware erhalten (Livewire-Tests,
// Konsole, Mails).
URL::defaults(['edition' => SetEdition::DEFAULT_EDITION]);
// Registrierung → Verifizierungsmail; bestätigter Klick → Aktivierung.
Event::listen(Registered::class, SendEmailVerificationNotification::class);
Event::listen(Verified::class, ActivateUserAfterVerification::class);

View file

@ -0,0 +1,67 @@
<?php
namespace App\Support;
use App\Enums\Portal;
use App\Enums\PressReleaseStatus;
use App\Models\Category;
use App\Scopes\PortalScope;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Support\Collection;
/**
* Liefert die Rubriken-Navigation der aktiven Ausgabe (Portal × Sprache).
*
* Es gibt bewusst keine separate Rubriken-Übersichtsseite: Die Navigation
* führt direkt auf die jeweilige Kategorieseite (`route('kategorie', …)`),
* auf der die Pressemitteilungen dieser Rubrik erscheinen.
*/
class CategoryNavigation
{
/**
* Top-Level-Rubriken der aktiven Ausgabe als Navigationsitems.
*
* @return Collection<int, array{label:string, slug:string, href:string}>
*/
public static function items(int $limit = 9): Collection
{
$portal = Portal::tryFrom((string) config('app.theme')) ?? Portal::Businessportal24;
$language = app()->getLocale();
$portalValues = [$portal->value, Portal::Both->value];
return Category::query()
->with(['translations' => fn ($query) => $query->where('locale', $language)])
->withCount([
'pressReleases as published_count' => function (Builder $query) use ($portalValues, $language): void {
$query
->withoutGlobalScope(PortalScope::class)
->whereIn('portal', $portalValues)
->where('status', PressReleaseStatus::Published)
->where('language', $language)
->whereNotNull('published_at');
},
])
->whereIn('portal', $portalValues)
->where('is_active', true)
->whereNull('parent_id')
->orderByDesc('published_count')
->orderBy('id')
->limit($limit)
->get()
->map(function (Category $category): ?array {
$translation = $category->translations->first();
if (! $translation || ! $translation->slug) {
return null;
}
return [
'label' => $translation->name,
'slug' => $translation->slug,
'href' => route('kategorie', ['slug' => $translation->slug]),
];
})
->filter()
->values();
}
}

View file

@ -0,0 +1,48 @@
<?php
namespace App\Support;
use App\Enums\PressReleaseStatus;
use App\Models\PressRelease;
use App\Scopes\PortalScope;
use Illuminate\Support\Carbon;
/**
* Liefert den „Redaktions-Stichtag" für aktualitätsbezogene Startseiten-Abfragen.
*
* Das migrierte Archiv ist historisch: Der jüngste veröffentlichte Datensatz
* kann Wochen/Monate in der Vergangenheit liegen. Würde man „heute / letzte 7
* Tage" hart gegen das echte now() rechnen, liefern alle Aktualitäts-Module
* 0 Treffer und fallen auf Mock-Daten zurück.
*
* Stattdessen wird der Stichtag an den jüngsten verfügbaren Veröffentlichungs-
* zeitpunkt gekoppelt (gedeckelt auf now()): Liegt das jüngste published_at in
* der Vergangenheit, dient es als Referenz-„Jetzt"; sobald frische
* Pressemitteilungen mit aktuellem Datum einlaufen, greift automatisch wieder
* das echte now().
*/
class EditorialClock
{
/**
* @param array<int, string> $portalValues
*/
public static function reference(array $portalValues, string $language = 'de'): Carbon
{
$latest = PressRelease::query()
->withoutGlobalScope(PortalScope::class)
->whereIn('portal', $portalValues)
->where('status', PressReleaseStatus::Published)
->where('language', $language)
->whereNotNull('published_at')
->where('published_at', '<=', now())
->max('published_at');
if (! $latest) {
return now();
}
$latest = Carbon::parse($latest);
return $latest->lt(now()) ? $latest : now();
}
}