change role to "user" for new registrations, fixed some registration form errors and implemented a reg-test

This commit is contained in:
Codex Agent
2025-10-02 15:06:50 +02:00
parent 7475210893
commit 1845d83583
13 changed files with 416 additions and 191 deletions

View File

@@ -1,8 +1,7 @@
import React, { useState } from 'react';
import { useForm, router } from '@inertiajs/react';
import { Head } from '@inertiajs/react';
import React, { useEffect, useState } from 'react';
import { useForm } from '@inertiajs/react';
import { LoaderCircle, User, Mail, Phone, Lock, Home, MapPin } from 'lucide-react';
import { Dialog, DialogContent, DialogTrigger } from '@/components/ui/dialog';
import { Dialog, DialogContent } from '@/components/ui/dialog';
interface RegisterProps {
package?: {
@@ -18,8 +17,9 @@ import MarketingLayout from '@/layouts/marketing/MarketingLayout';
export default function Register({ package: initialPackage, privacyHtml }: RegisterProps) {
const [privacyOpen, setPrivacyOpen] = useState(false);
const [hasTriedSubmit, setHasTriedSubmit] = useState(false);
const { data, setData, post, processing, errors, setError } = useForm({
const { data, setData, post, processing, errors, clearErrors } = useForm({
username: '',
email: '',
password: '',
@@ -32,45 +32,33 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
package_id: initialPackage?.id || null,
});
React.useEffect(() => {
if (Object.keys(errors).length > 0) {
console.log('Validation errors received:', errors);
}
if (!processing) {
console.log('Registration processing completed');
}
}, [errors, processing, data]);
React.useEffect(() => {
if (Object.keys(errors).length > 0) {
window.scrollTo({ top: 0, behavior: 'smooth' });
}
}, [errors]);
React.useEffect(() => {
if (Object.keys(errors).length > 0) {
// Force re-render or scroll to errors
window.scrollTo({ top: 0, behavior: 'smooth' });
}
}, [errors]);
const submit = (e: React.FormEvent) => {
e.preventDefault();
console.log('Submitting registration form with data:', data);
router.post('/register', data, {
preserveState: true,
forceFormData: true,
onSuccess: () => {
console.log('Registration successful');
},
onError: (errors) => {
console.log('Registration errors:', errors);
setError(errors);
},
setHasTriedSubmit(true);
post('/register', {
preserveScroll: true,
});
console.log('POST to /register initiated');
};
useEffect(() => {
if (!hasTriedSubmit) {
return;
}
const errorKeys = Object.keys(errors);
if (errorKeys.length === 0) {
return;
}
const firstError = errorKeys[0];
const field = document.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(`[name="${firstError}"]`);
if (field) {
field.scrollIntoView({ behavior: 'smooth', block: 'center' });
field.focus();
}
}, [errors, hasTriedSubmit]);
return (
<MarketingLayout title="Registrieren">
<div className="min-h-screen flex items-center justify-center bg-gray-50 py-12 px-4 sm:px-6 lg:px-8">
@@ -93,7 +81,7 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
</div>
)}
</div>
<form key={`form-${processing ? 'submitting' : 'idle'}-${Object.keys(errors).length}`} onSubmit={submit} className="mt-8 space-y-6">
<form onSubmit={submit} className="mt-8 space-y-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="md:col-span-1">
<label htmlFor="first_name" className="block text-sm font-medium text-gray-700 mb-1">
@@ -107,7 +95,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="text"
required
value={data.first_name}
onChange={(e) => setData('first_name', e.target.value)}
onChange={(e) => {
setData('first_name', e.target.value);
if (e.target.value.trim() && errors.first_name) {
clearErrors('first_name');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.first_name ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Vorname"
/>
@@ -127,7 +120,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="text"
required
value={data.last_name}
onChange={(e) => setData('last_name', e.target.value)}
onChange={(e) => {
setData('last_name', e.target.value);
if (e.target.value.trim() && errors.last_name) {
clearErrors('last_name');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.last_name ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Nachname"
/>
@@ -147,7 +145,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="email"
required
value={data.email}
onChange={(e) => setData('email', e.target.value)}
onChange={(e) => {
setData('email', e.target.value);
if (e.target.value.trim() && errors.email) {
clearErrors('email');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.email ? 'border-red-500' : 'border-gray-300'}`}
placeholder="email@example.com"
/>
@@ -167,7 +170,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="text"
required
value={data.address}
onChange={(e) => setData('address', e.target.value)}
onChange={(e) => {
setData('address', e.target.value);
if (e.target.value.trim() && errors.address) {
clearErrors('address');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.address ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Adresse"
/>
@@ -187,7 +195,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="tel"
required
value={data.phone}
onChange={(e) => setData('phone', e.target.value)}
onChange={(e) => {
setData('phone', e.target.value);
if (e.target.value.trim() && errors.phone) {
clearErrors('phone');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.phone ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Telefonnummer"
/>
@@ -207,7 +220,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="text"
required
value={data.username}
onChange={(e) => setData('username', e.target.value)}
onChange={(e) => {
setData('username', e.target.value);
if (e.target.value.trim() && errors.username) {
clearErrors('username');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.username ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Benutzername"
/>
@@ -227,7 +245,15 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="password"
required
value={data.password}
onChange={(e) => setData('password', e.target.value)}
onChange={(e) => {
setData('password', e.target.value);
if (e.target.value.trim() && errors.password) {
clearErrors('password');
}
if (data.password_confirmation && e.target.value === data.password_confirmation) {
clearErrors('password_confirmation');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.password ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Passwort"
/>
@@ -247,12 +273,20 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="password"
required
value={data.password_confirmation}
onChange={(e) => setData('password_confirmation', e.target.value)}
onChange={(e) => {
setData('password_confirmation', e.target.value);
if (e.target.value.trim() && errors.password_confirmation) {
clearErrors('password_confirmation');
}
if (data.password && e.target.value === data.password) {
clearErrors('password_confirmation');
}
}}
className={`block w-full pl-10 pr-3 py-3 border rounded-md shadow-sm placeholder-gray-400 focus:outline-none focus:ring-2 focus:ring-[#FFB6C1] focus:border-[#FFB6C1] sm:text-sm ${errors.password_confirmation ? 'border-red-500' : 'border-gray-300'}`}
placeholder="Passwort bestätigen"
/>
</div>
{errors.password_confirmation && <p className="text-sm text-red-600 mt-1">{errors.password_confirmation}</p>}
{errors.password_confirmation && <p key={`error-password_confirmation`} className="text-sm text-red-600 mt-1">{errors.password_confirmation}</p>}
</div>
<div className="md:col-span-2 flex items-start">
@@ -262,7 +296,12 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
type="checkbox"
required
checked={data.privacy_consent}
onChange={(e) => setData('privacy_consent', e.target.checked)}
onChange={(e) => {
setData('privacy_consent', e.target.checked);
if (e.target.checked && errors.privacy_consent) {
clearErrors('privacy_consent');
}
}}
className="h-4 w-4 text-[#FFB6C1] focus:ring-[#FFB6C1] border-gray-300 rounded"
/>
<label htmlFor="privacy_consent" className="ml-2 block text-sm text-gray-900">
@@ -281,14 +320,15 @@ export default function Register({ package: initialPackage, privacyHtml }: Regis
</div>
{Object.keys(errors).length > 0 && (
<div className="p-4 bg-red-50 border border-red-200 rounded-md">
<p className="text-sm text-red-800">
<div key={`general-errors-${Object.keys(errors).join('-')}`} className="p-4 bg-red-50 border border-red-200 rounded-md mb-6">
<h4 className="text-sm font-medium text-red-800 mb-2">Fehler bei der Registrierung:</h4>
<ul className="text-sm text-red-800 space-y-1">
{Object.entries(errors).map(([key, value]) => (
<span key={key}>
{value}
</span>
<li key={key} className="flex items-start">
<span className="font-medium">{key.replace('_', ' ')}:</span> {value}
</li>
))}
</p>
</ul>
</div>
)}