'use client'; import { useState, useRef, useMemo } from 'react'; import { motion, AnimatePresence } from 'framer-motion'; import { useRouter } from 'next/navigation'; import Link from 'next/link'; import Image from 'next/image'; import { User, Mail, Phone, Lock, Eye, EyeOff, MessageCircle, Camera, X, CheckCircle, XCircle, ArrowLeft, Loader2, Shield, KeyRound, MapPin, FileText, BadgeCheck, Briefcase, Calendar } from 'lucide-react'; import toast, { Toaster } from 'react-hot-toast'; import { registerRealEstateAgent, loginWithEmail, sendEmailOTP, verifyEmail } from '../../utils/api'; import AuthService from '../../services/AuthService'; export default function AgentRegisterPage() { const router = useRouter(); const [step, setStep] = useState(1); const [showOtpModal, setShowOtpModal] = useState(false); const [showPassword, setShowPassword] = useState(false); const [showConfirmPassword, setShowConfirmPassword] = useState(false); const [isLoading, setIsLoading] = useState(false); const [formData, setFormData] = useState({ firstName: '', lastName: '', email: '', phone: '', whatsapp: '', nationalNumber: '', password: '', confirmPassword: '', agencyAddress: '', licenseNumber: '', agreeTerms: false }); const [idImages, setIdImages] = useState({ front: null, back: null, license: null }); const [idImagePreviews, setIdImagePreviews] = useState({ front: '', back: '', license: '' }); const [otpCode, setOtpCode] = useState(''); const [errors, setErrors] = useState({}); const fileInputFrontRef = useRef(null); const fileInputBackRef = useRef(null); const fileInputLicenseStep3Ref = useRef(null); const validateEmail = (email) => /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email); const validatePhone = (phone) => /^(09|05)[0-9]{8}$/.test(phone); const handleImageUpload = (side, file) => { if (!file) return; if (!file.type.startsWith('image/')) { toast.error('الرجاء اختيار صورة صالحة'); return; } if (file.size > 5 * 1024 * 1024) { toast.error('حجم الصورة يجب أن يكون أقل من 5 ميجابايت'); return; } const reader = new FileReader(); reader.onloadend = () => { setIdImagePreviews((prev) => ({ ...prev, [side]: reader.result })); }; reader.readAsDataURL(file); setIdImages((prev) => ({ ...prev, [side]: file })); toast.success('تم رفع الصورة بنجاح', { style: { background: '#dcfce7', color: '#166534' } }); }; const validateStep1 = () => { const newErrors = {}; if (!formData.firstName) newErrors.firstName = 'الاسم الأول مطلوب'; if (!formData.lastName) newErrors.lastName = 'اسم العائلة مطلوب'; if (!formData.email) newErrors.email = 'البريد الإلكتروني مطلوب'; else if (!validateEmail(formData.email)) newErrors.email = 'البريد الإلكتروني غير صالح'; if (!formData.phone) newErrors.phone = 'رقم الهاتف مطلوب'; else if (!validatePhone(formData.phone)) newErrors.phone = 'رقم الهاتف غير صالح (يجب أن يبدأ 09 أو 05)'; if (!formData.password) newErrors.password = 'كلمة المرور مطلوبة'; else if (formData.password.length < 6) newErrors.password = 'كلمة المرور يجب أن تكون 6 أحرف على الأقل'; if (!formData.confirmPassword) newErrors.confirmPassword = 'تأكيد كلمة المرور مطلوب'; else if (formData.password !== formData.confirmPassword) newErrors.confirmPassword = 'كلمات المرور غير متطابقة'; setErrors(newErrors); return Object.keys(newErrors).length === 0; }; const validateStep2 = () => { const newErrors = {}; if (!formData.nationalNumber) newErrors.nationalNumber = 'الرقم الوطني مطلوب'; if (!formData.whatsapp) newErrors.whatsapp = 'رقم الواتساب مطلوب'; else if (!validatePhone(formData.whatsapp)) newErrors.whatsapp = 'رقم الواتساب غير صالح (يجب أن يبدأ 09 أو 05)'; if (!idImages.front) newErrors.front = 'صورة الوجه الأمامي للهوية مطلوبة'; if (!idImages.back) newErrors.back = 'صورة الوجه الخلفي للهوية مطلوبة'; setErrors(newErrors); return Object.keys(newErrors).length === 0; }; const validateStep3 = () => { const newErrors = {}; if (!formData.agencyAddress) newErrors.agencyAddress = 'عنوان الوكالة مطلوب'; if (!formData.licenseNumber) newErrors.licenseNumber = 'رقم الترخيص مطلوب'; if (!idImages.license) newErrors.license = 'صورة الترخيص مطلوبة'; setErrors(newErrors); return Object.keys(newErrors).length === 0; }; const handleNextStep = () => { if (step === 1 && validateStep1()) { setStep(2); window.scrollTo({ top: 0, behavior: 'smooth' }); } else if (step === 2 && validateStep2()) { setStep(3); window.scrollTo({ top: 0, behavior: 'smooth' }); } else if (step === 3 && validateStep3()) { setStep(4); window.scrollTo({ top: 0, behavior: 'smooth' }); } else { toast.error('يرجى تصحيح الأخطاء في النموذج'); } }; const buildFormData = () => { const fd = new FormData(); // مطابق للـ API fd.append('Address', formData.agencyAddress); fd.append('LicenseNumber', formData.licenseNumber); fd.append('Owner.FirstName', formData.firstName); fd.append('Owner.LastName', formData.lastName); fd.append('Owner.Email', formData.email); fd.append('Owner.Password', formData.password); fd.append('Owner.PhoneNumber', formData.phone); fd.append('Owner.Phone', formData.phone); fd.append('Owner.NationalNumber', formData.nationalNumber); fd.append('Owner.WhatsAppNumber', formData.whatsapp); fd.append('Owner.Language', '0'); // مطابق لأسماء الحقول الظاهرة في swagger if (idImages.front) fd.append('FrontIdCarImage', idImages.front); if (idImages.back) fd.append('RearIdCarImage', idImages.back); if (idImages.license) fd.append('license', idImages.license); return fd; }; const handleSubmit = async (e) => { e.preventDefault(); if (!formData.agreeTerms) { toast.error('يجب الموافقة على الشروط والأحكام'); return; } setIsLoading(true); try { const fd = buildFormData(); const res = await registerRealEstateAgent(fd); if (res?.ok || res?.status === 200) { const tempToken = res?.data; if (tempToken) AuthService.addToken(tempToken); toast.success(res?.message || 'تم إنشاء الحساب! يرجى التحقق من بريدك الإلكتروني', { duration: 4000 }); const loginRes = await loginWithEmail(formData.email, formData.password); if (loginRes?.status === 206) { const otpToken = loginRes?.data; if (otpToken) AuthService.addToken(otpToken); toast(loginRes?.message || 'تم إرسال رمز التحقق إلى بريدك الإلكتروني', { icon: '📧' }); setShowOtpModal(true); } else if (loginRes?.status === 200) { const loginToken = loginRes?.data; if (loginToken) AuthService.addToken(loginToken); toast.success(loginRes?.message || 'تم تسجيل الدخول بنجاح!'); router.push('/'); } else { toast.success('تم إنشاء الحساب بنجاح! يرجى تسجيل الدخول'); setTimeout(() => router.push('/login'), 1500); } } else { toast.error(res?.message || res?.data?.message || 'فشل في إنشاء الحساب'); } } catch (err) { toast.error(err?.message || 'حدث خطأ أثناء التسجيل'); } finally { setIsLoading(false); } }; const handleVerifyOTP = async () => { if (!otpCode || otpCode.length < 4) { toast.error('يرجى إدخال رمز التحقق'); return; } setIsLoading(true); try { const res = await verifyEmail(otpCode); if (res?.status === 200) { AuthService.deleteToken(); toast.success(res?.message || 'تم التحقق من البريد الإلكتروني بنجاح!'); setShowOtpModal(false); setTimeout(() => router.push('/login'), 1500); } else { toast.error(res?.message || res?.data?.message || 'رمز التحقق غير صحيح'); } } catch (err) { toast.error(err?.message || 'حدث خطأ أثناء التحقق'); } finally { setIsLoading(false); } }; const handleResendOTP = async () => { setIsLoading(true); try { await sendEmailOTP(); toast.success('تم إرسال رمز تحقق جديد'); } catch (err) { toast.error('فشل في إرسال الرمز'); } finally { setIsLoading(false); } }; const fadeInUp = { initial: { opacity: 0, y: 20 }, animate: { opacity: 1, y: 0 }, transition: { duration: 0.5 } }; const staggerContainer = { animate: { transition: { staggerChildren: 0.1 } } }; const backgroundElements = useMemo(() => { const circles = [ { style: { top: '20%', right: '20%', width: '256px', height: '256px' }, className: 'bg-purple-500/5' }, { style: { bottom: '20%', left: '20%', width: '320px', height: '320px' }, className: 'bg-amber-500/5' }, { style: { top: '50%', left: '50%', width: '384px', height: '384px', transform: 'translate(-50%, -50%)' }, className: 'bg-purple-500/5' } ]; const dots = Array.from({ length: 10 }).map((_, i) => ({ left: `${5 + i * 10}%`, top: `${10 + ((i * 7) % 80)}%`, size: `${80 + (i % 5) * 15}px` })); return ( <> {circles.map((circle, i) => (
))} {dots.map((dot, i) => ( ))} > ); }, []); const stepTitles = [ 'المعلومات الأساسية', 'معلومات الهوية', 'معلومات الوكالة', 'تأكيد التسجيل' ]; const stepDescriptions = [ 'أدخل معلوماتك الأساسية', 'يرجى رفع صور الهوية للتحقق', 'أدخل بيانات الوكالة والترخيص', 'راجع معلوماتك قبل الإرسال' ]; const renderStepFields = () => { switch (step) { case 1: return ( <>{errors.firstName}
)}{errors.lastName}
)}{errors.email}
}{errors.phone}
}{errors.password}
}{errors.confirmPassword}
)}{errors.nationalNumber}
)}{errors.whatsapp}
}اضغط لرفع الصورة
JPEG, PNG, JPG • حتى 5MB
> )}{errors.front}
}اضغط لرفع الصورة
JPEG, PNG, JPG • حتى 5MB
> )}{errors.back}
}{errors.agencyAddress}
)}{errors.licenseNumber}
)}اضغط لرفع صورة الترخيص
JPEG, PNG, JPG • حتى 5MB
> )}{errors.license}
}الوجه الأمامي
الوجه الخلفي
صورة الترخيص
{stepDescriptions[step - 1]}
تم إرسال رمز التحقق إلى
{formData.email}