feat(i18n): Complete localization of marketing frontend with react-i18next, prefixed URLs, JSON migrations, and automation

This commit is contained in:
Codex Agent
2025-10-03 13:05:13 +02:00
parent 1845d83583
commit 60f8de9162
46 changed files with 3454 additions and 590 deletions

View File

@@ -28,11 +28,10 @@ class MarketingRegisterController extends Controller
{
$package = $package_id ? Package::find($package_id) : null;
App::setLocale('de');
//App::setLocale('de');
return Inertia::render('Auth/Register', [
return Inertia::render('auth/register', [
'package' => $package,
'privacyHtml' => view('legal.datenschutz')->render(),
]);
}
@@ -157,3 +156,4 @@ class MarketingRegisterController extends Controller

View File

@@ -33,11 +33,9 @@ class MarketingController extends Controller
public function index()
{
$packages = [
['id' => 'basic', 'name' => 'Basic', 'events' => 1, 'price' => 0, 'description' => '1 Event, 100 Fotos, Grundfunktionen'],
['id' => 'standard', 'name' => 'Standard', 'events' => 10, 'price' => 99, 'description' => '10 Events, Unbegrenzt Fotos, Erweiterte Features'],
['id' => 'premium', 'name' => 'Premium', 'events' => 50, 'price' => 199, 'description' => '50 Events, Support & Custom, Alle Features'],
];
$packages = Package::where('type', 'endcustomer')->orderBy('price')->get()->map(function ($p) {
return $p->append(['features', 'limits']);
});
return Inertia::render('marketing/Home', compact('packages'));
}
@@ -417,7 +415,7 @@ class MarketingController extends Controller
public function occasionsType($locale, $type)
{
$validTypes = ['weddings', 'birthdays', 'corporate-events', 'family-celebrations'];
$validTypes = ['hochzeit', 'geburtstag', 'firmenevent'];
if (!in_array($type, $validTypes)) {
abort(404, 'Invalid occasion type');
}

View File

@@ -4,63 +4,43 @@ namespace App\Http\Controllers;
use App\Models\User;
use Illuminate\Http\Request;
use Inertia\Inertia;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Validation\Rules;
use Illuminate\View\View;
use Illuminate\Http\RedirectResponse;
class ProfileController extends Controller
{
/**
* Display the user's profile form.
*/
public function edit(Request $request): View
public function index()
{
return view('profile.edit', [
'user' => $request->user(),
$user = Auth::user()->load('purchases.packages');
return Inertia::render('Profile/Index', [
'user' => $user,
]);
}
/**
* Update the user's profile information.
*/
public function update(Request $request, User $user): RedirectResponse
public function account()
{
// Authorized via auth middleware
$user = Auth::user()->load('purchases.packages');
if (request()->isMethod('post')) {
$validated = request()->validate([
'name' => 'required|string|max:255',
'email' => 'required|email|unique:users,email,' . $user->id,
]);
$request->validate([
'username' => ['required', 'string', 'max:255', 'alpha_dash', 'unique:users,username,' . $user->id],
'email' => ['required', 'string', 'email', 'max:255', 'unique:users,email,' . $user->id],
'first_name' => ['required', 'string', 'max:255'],
'last_name' => ['required', 'string', 'max:255'],
'address' => ['required', 'string'],
'phone' => ['required', 'string', 'max:20'],
$user->update($validated);
return back()->with('success', 'Profil aktualisiert.');
}
return Inertia::render('Profile/Account', [
'user' => $user,
]);
$user->update($request->only([
'username', 'email', 'first_name', 'last_name', 'address', 'phone'
]));
return back()->with('status', 'profile-updated');
}
/**
* Update the user's password.
*/
public function updatePassword(Request $request, User $user): RedirectResponse
public function orders()
{
// Authorized via auth middleware
$request->validate([
'current_password' => ['required', 'current_password'],
'password' => ['required', 'confirmed', Rules\Password::defaults()],
$user = Auth::user()->load('purchases.packages');
return Inertia::render('Profile/Orders', [
'purchases' => $user->purchases,
]);
$user->update([
'password' => Hash::make($request->password),
]);
return back()->with('status', 'password-updated');
}
}

70
app/Http/Kernel.php Normal file
View File

@@ -0,0 +1,70 @@
<?php
namespace App\Http;
use Illuminate\Foundation\Http\Kernel as HttpKernel;
class Kernel extends HttpKernel
{
/**
* The application's global HTTP middleware stack.
*
* These middleware are run during every request to your application.
*
* @var array<int, class-string|string>
*/
protected $middleware = [
// \App\Http\Middleware\TrustHosts::class,
\App\Http\Middleware\TrustProxies::class,
\Illuminate\Http\Middleware\HandleCors::class,
\App\Http\Middleware\PreventRequestsDuringMaintenance::class,
\Illuminate\Foundation\Http\Middleware\ValidatePostSize::class,
\App\Http\Middleware\TrimStrings::class,
\Illuminate\Foundation\Http\Middleware\ConvertEmptyStringsToNull::class,
];
/**
* The application's route middleware groups.
*
* @var array<string, array<int, class-string|string>>
*/
protected $middlewareGroups = [
'web' => [
\App\Http\Middleware\EncryptCookies::class,
\Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
\Illuminate\Session\Middleware\StartSession::class,
\Illuminate\View\Middleware\ShareErrorsFromSession::class,
\App\Http\Middleware\VerifyCsrfToken::class,
\Illuminate\Routing\Middleware\SubstituteBindings::class,
\App\Http\Middleware\HandleInertiaRequests::class,
\Illuminate\Http\Middleware\AddLinkHeadersForPreloadedAssets::class,
],
'api' => [
// \Laravel\Sanctum\Http\Middleware\EnsureFrontendRequestsAreStateful::class,
'throttle:api',
\Illuminate\Routing\Middleware\SubstituteBindings::class,
],
];
/**
* The application's route middleware.
*
* These middleware may be assigned to groups or used individually.
*
* @var array<string, class-string|string>
*/
protected $routeMiddleware = [
'auth' => \App\Http\Middleware\Authenticate::class,
'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
'auth.session' => \Illuminate\Session\Middleware\AuthenticateSession::class,
'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class,
'can' => \Illuminate\Auth\Middleware\Authorize::class,
'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
'password.confirm' => \Illuminate\Auth\Middleware\RequirePassword::class,
'signed' => \App\Http\Middleware\ValidateSignature::class,
'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class,
'verified' => \Illuminate\Auth\Middleware\EnsureEmailIsVerified::class,
'locale' => \App\Http\Middleware\SetLocale::class,
];
}

View File

@@ -61,6 +61,12 @@ class HandleInertiaRequests extends Middleware
],
'supportedLocales' => $supportedLocales,
'sidebarOpen' => ! $request->hasCookie('sidebar_state') || $request->cookie('sidebar_state') === 'true',
'locale' => app()->getLocale(),
'translations' => [
'marketing' => __('marketing'),
'auth' => __('auth'),
'profile' => __('profile'),
],
];
}
}

View File

@@ -4,6 +4,8 @@ namespace App\Http\Middleware;
use Closure;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\App;
use Illuminate\Support\Facades\Session;
use Symfony\Component\HttpFoundation\Response;
class SetLocale
@@ -15,16 +17,25 @@ class SetLocale
*/
public function handle(Request $request, Closure $next): Response
{
$locale = $request->segment(1);
$locale = $request->segment(1); // Erste URL-Segment als Locale (z.B. /de/packages -> 'de')
// Unterstützte Sprachen
$supportedLocales = ['de', 'en'];
if (in_array($locale, $supportedLocales)) {
app()->setLocale($locale);
session()->put('locale', $locale);
// Locale setzen
App::setLocale($locale);
Session::put('locale', $locale);
} else {
$locale = session('locale', config('app.locale', 'de'));
app()->setLocale($locale);
// Fallback zu 'de'
$defaultLocale = 'de';
App::setLocale($defaultLocale);
Session::put('locale', $defaultLocale);
// Redirect zu default Locale, wenn keine Locale in URL
if (!$locale) {
return redirect("/{$defaultLocale}" . $request->getRequestUri());
}
}
return $next($request);