feat: Complete checkout overhaul with Stripe PaymentIntent integration and abandoned cart recovery
This commit is contained in:
@@ -1,198 +1,3 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Http\Requests\Auth\LoginRequest;
|
||||
use App\Mail\Welcome;
|
||||
use App\Models\Package;
|
||||
use App\Models\PackagePurchase;
|
||||
use App\Models\Tenant;
|
||||
use App\Models\TenantPackage;
|
||||
use App\Models\User;
|
||||
use Illuminate\Auth\Events\Registered;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Hash;
|
||||
use Illuminate\Support\Facades\Mail;
|
||||
use Illuminate\Support\Str;
|
||||
use Illuminate\Validation\Rules;
|
||||
use Illuminate\Validation\ValidationException;
|
||||
use Inertia\Response;
|
||||
use Throwable;
|
||||
|
||||
class CheckoutController extends Controller
|
||||
{
|
||||
/**
|
||||
* Render the checkout wizard using the legacy marketing controller for now.
|
||||
*/
|
||||
public function show(Request $request, Package $package): Response
|
||||
{
|
||||
$marketingController = app(MarketingController::class);
|
||||
|
||||
return $marketingController->purchaseWizard($request, $package->getKey());
|
||||
}
|
||||
|
||||
public function login(LoginRequest $request): JsonResponse
|
||||
{
|
||||
app()->setLocale($request->input('locale', app()->getLocale()));
|
||||
|
||||
$request->authenticate();
|
||||
$request->session()->regenerate();
|
||||
|
||||
$user = $request->user()?->fresh();
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'user' => $this->transformUser($user),
|
||||
]);
|
||||
}
|
||||
|
||||
public function register(Request $request): JsonResponse
|
||||
{
|
||||
app()->setLocale($request->input('locale', app()->getLocale()));
|
||||
|
||||
try {
|
||||
$validated = $request->validate([
|
||||
'username' => ['required', 'string', 'max:255', 'unique:'.User::class],
|
||||
'email' => ['required', 'string', 'lowercase', 'email', 'max:255', 'unique:'.User::class],
|
||||
'password' => ['required', 'confirmed', Rules\Password::defaults()],
|
||||
'first_name' => ['required', 'string', 'max:255'],
|
||||
'last_name' => ['required', 'string', 'max:255'],
|
||||
'address' => ['required', 'string', 'max:500'],
|
||||
'phone' => ['required', 'string', 'max:20'],
|
||||
'privacy_consent' => ['accepted'],
|
||||
'package_id' => ['nullable', 'integer'],
|
||||
]);
|
||||
} catch (ValidationException $exception) {
|
||||
throw $exception;
|
||||
}
|
||||
|
||||
$shouldAutoVerify = app()->environment(['local', 'testing']);
|
||||
|
||||
DB::beginTransaction();
|
||||
|
||||
try {
|
||||
$user = User::create([
|
||||
'username' => $validated['username'],
|
||||
'email' => $validated['email'],
|
||||
'first_name' => $validated['first_name'],
|
||||
'last_name' => $validated['last_name'],
|
||||
'address' => $validated['address'],
|
||||
'phone' => $validated['phone'],
|
||||
'password' => bcrypt($validated['password']),
|
||||
'role' => 'user',
|
||||
'pending_purchase' => !empty($validated['package_id']),
|
||||
]);
|
||||
|
||||
if ($user->pending_purchase) {
|
||||
$request->session()->put('pending_user_id', $user->id);
|
||||
}
|
||||
|
||||
if ($shouldAutoVerify) {
|
||||
$user->forceFill(['email_verified_at' => now()])->save();
|
||||
}
|
||||
|
||||
$tenant = Tenant::create([
|
||||
'user_id' => $user->id,
|
||||
'name' => $validated['first_name'].' '.$validated['last_name'],
|
||||
'slug' => Str::slug($validated['first_name'].' '.$validated['last_name'].'-'.now()->timestamp),
|
||||
'email' => $validated['email'],
|
||||
'is_active' => true,
|
||||
'is_suspended' => false,
|
||||
'event_credits_balance' => 0,
|
||||
'subscription_tier' => 'free',
|
||||
'subscription_expires_at' => null,
|
||||
'settings' => json_encode([
|
||||
'branding' => [
|
||||
'logo_url' => null,
|
||||
'primary_color' => '#3B82F6',
|
||||
'secondary_color' => '#1F2937',
|
||||
'font_family' => 'Inter, sans-serif',
|
||||
],
|
||||
'features' => [
|
||||
'photo_likes_enabled' => false,
|
||||
'event_checklist' => false,
|
||||
'custom_domain' => false,
|
||||
'advanced_analytics' => false,
|
||||
],
|
||||
'custom_domain' => null,
|
||||
'contact_email' => $validated['email'],
|
||||
'event_default_type' => 'general',
|
||||
]),
|
||||
]);
|
||||
|
||||
event(new Registered($user));
|
||||
|
||||
Auth::login($user);
|
||||
$request->session()->regenerate();
|
||||
|
||||
DB::commit();
|
||||
|
||||
Mail::to($user)->queue(new Welcome($user));
|
||||
|
||||
$redirect = $shouldAutoVerify ? route('dashboard') : route('verification.notice');
|
||||
$pendingPurchase = $user->pending_purchase;
|
||||
|
||||
if (!empty($validated['package_id'])) {
|
||||
$package = Package::find($validated['package_id']);
|
||||
|
||||
if (!$package) {
|
||||
throw ValidationException::withMessages([
|
||||
'package_id' => __('validation.exists', ['attribute' => 'package'])
|
||||
]);
|
||||
}
|
||||
|
||||
if ((float) $package->price <= 0.0) {
|
||||
TenantPackage::create([
|
||||
'tenant_id' => $tenant->id,
|
||||
'package_id' => $package->id,
|
||||
'price' => 0,
|
||||
'active' => true,
|
||||
]);
|
||||
|
||||
PackagePurchase::create([
|
||||
'tenant_id' => $tenant->id,
|
||||
'package_id' => $package->id,
|
||||
'type' => $package->type === 'endcustomer' ? 'endcustomer_event' : 'reseller_subscription',
|
||||
'price' => 0,
|
||||
'purchased_at' => now(),
|
||||
'provider_id' => 'free',
|
||||
]);
|
||||
|
||||
$tenant->update(['subscription_status' => 'active']);
|
||||
$user->update(['role' => 'tenant_admin', 'pending_purchase' => false]);
|
||||
$pendingPurchase = false;
|
||||
$redirect = $shouldAutoVerify ? route('dashboard') : route('verification.notice');
|
||||
} else {
|
||||
$pendingPurchase = true;
|
||||
$redirect = route('buy.packages', $package->id);
|
||||
}
|
||||
}
|
||||
|
||||
$freshUser = $user->fresh();
|
||||
|
||||
return response()->json([
|
||||
'success' => true,
|
||||
'user' => $this->transformUser($freshUser),
|
||||
'pending_purchase' => $pendingPurchase,
|
||||
'redirect' => $redirect,
|
||||
]);
|
||||
} catch (ValidationException $validationException) {
|
||||
DB::rollBack();
|
||||
throw $validationException;
|
||||
} catch (Throwable $throwable) {
|
||||
DB::rollBack();
|
||||
report($throwable);
|
||||
|
||||
return response()->json([
|
||||
'success' => false,
|
||||
'message' => __('auth.registration_failed'),
|
||||
], 500);
|
||||
}
|
||||
}
|
||||
|
||||
private function transformUser(?User $user): ?array
|
||||
{
|
||||
if (!$user) {
|
||||
@@ -207,18 +12,64 @@ class CheckoutController extends Controller
|
||||
'email_verified_at' => $user->email_verified_at,
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Track an abandoned checkout for reminder emails
|
||||
*/
|
||||
public function trackAbandonedCheckout(Request $request): JsonResponse
|
||||
{
|
||||
$request->validate([
|
||||
'package_id' => 'required|integer|exists:packages,id',
|
||||
'last_step' => 'required|integer|min:1|max:4',
|
||||
'user_id' => 'nullable|integer|exists:users,id',
|
||||
'email' => 'nullable|email',
|
||||
]);
|
||||
|
||||
try {
|
||||
$userId = $request->user_id;
|
||||
$email = $request->email;
|
||||
|
||||
// Wenn kein user_id aber email, versuche User zu finden
|
||||
if (!$userId && $email) {
|
||||
$user = User::where('email', $email)->first();
|
||||
$userId = $user?->id;
|
||||
}
|
||||
|
||||
// Nur tracken wenn wir einen User haben
|
||||
if (!$userId) {
|
||||
return response()->json(['success' => false, 'message' => 'No user found to track']);
|
||||
}
|
||||
|
||||
$user = User::find($userId);
|
||||
if (!$user) {
|
||||
return response()->json(['success' => false, 'message' => 'User not found']);
|
||||
}
|
||||
|
||||
// Erstelle oder update abandoned checkout
|
||||
AbandonedCheckout::updateOrCreate(
|
||||
[
|
||||
'user_id' => $userId,
|
||||
'package_id' => $request->package_id,
|
||||
],
|
||||
[
|
||||
'email' => $user->email,
|
||||
'checkout_state' => null, // Später erweitern
|
||||
'last_step' => $request->last_step,
|
||||
'abandoned_at' => now(),
|
||||
'reminded_at' => null,
|
||||
'reminder_stage' => 'none',
|
||||
'expires_at' => now()->addDays(7), // 7 Tage gültig
|
||||
'converted' => false,
|
||||
]
|
||||
);
|
||||
|
||||
Log::info("Abandoned checkout tracked for user {$userId}, package {$request->package_id}, step {$request->last_step}");
|
||||
|
||||
return response()->json(['success' => true]);
|
||||
|
||||
} catch (\Exception $e) {
|
||||
Log::error('Failed to track abandoned checkout: ' . $e->getMessage());
|
||||
return response()->json(['success' => false, 'message' => 'Failed to track checkout'], 500);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
Reference in New Issue
Block a user