All checks were successful
Build frontend / build (push) Successful in 46s
- Desktop: hover dropdown with Android APK download + iOS coming soon - Mobile: download links in hamburger menu
797 lines
37 KiB
JavaScript
797 lines
37 KiB
JavaScript
"use client";
|
|
|
|
import { usePathname } from "next/navigation";
|
|
import { useTranslation } from "react-i18next";
|
|
import Link from "next/link";
|
|
import Image from "next/image";
|
|
import { NavLink, MobileNavLink } from "./components/NavLinks";
|
|
import {
|
|
Globe,
|
|
LogIn,
|
|
UserPlus,
|
|
UserCircle,
|
|
LogOut,
|
|
Calendar,
|
|
Building,
|
|
PlusCircle,
|
|
Heart,
|
|
MessageCircle,
|
|
Settings,
|
|
Edit,
|
|
Phone,
|
|
Mail,
|
|
MapPin,
|
|
Camera,
|
|
Shield,
|
|
Bell,
|
|
Home,
|
|
ChevronDown,
|
|
Menu,
|
|
X,
|
|
TrendingUp,
|
|
CalendarDays,
|
|
Clock,
|
|
Users,
|
|
DollarSign,
|
|
} from "lucide-react";
|
|
import { useState, useEffect, useRef } from "react";
|
|
import { motion, AnimatePresence } from "framer-motion";
|
|
import AuthService from "./services/AuthService";
|
|
import { UserRole, UserRoleLabels } from "./enums/UserRole";
|
|
import "./i18n/config";
|
|
|
|
export default function ClientLayout({ children }) {
|
|
const { t, i18n } = useTranslation();
|
|
const pathname = usePathname();
|
|
const [currentLanguage, setCurrentLanguage] = useState("en");
|
|
const [isMobileMenuOpen, setIsMobileMenuOpen] = useState(false);
|
|
const [showUserMenu, setShowUserMenu] = useState(false);
|
|
const [user, setUser] = useState(null);
|
|
const [isMounted, setIsMounted] = useState(false);
|
|
const currentYear = new Date().getFullYear();
|
|
const menuRef = useRef(null);
|
|
|
|
useEffect(() => {
|
|
setIsMounted(true);
|
|
const savedLanguage = localStorage.getItem("language") || "ar";
|
|
setCurrentLanguage(savedLanguage);
|
|
i18n.changeLanguage(savedLanguage);
|
|
|
|
// Load user from JWT via AuthService
|
|
const authUser = AuthService.getUser();
|
|
if (authUser) {
|
|
setUser({
|
|
name: authUser.name || authUser.email,
|
|
email: authUser.email,
|
|
phone: authUser.phone,
|
|
role: AuthService.isAdmin() ? UserRole.ADMIN
|
|
: AuthService.isOwner() ? UserRole.OWNER
|
|
: UserRole.CUSTOMER,
|
|
});
|
|
} else {
|
|
setUser(null);
|
|
}
|
|
|
|
if (savedLanguage === "ar") {
|
|
document.documentElement.dir = "rtl";
|
|
document.documentElement.lang = "ar";
|
|
} else {
|
|
document.documentElement.dir = "ltr";
|
|
document.documentElement.lang = "en";
|
|
}
|
|
}, [i18n]);
|
|
|
|
useEffect(() => {
|
|
const handleClickOutside = (event) => {
|
|
if (menuRef.current && !menuRef.current.contains(event.target)) {
|
|
setShowUserMenu(false);
|
|
}
|
|
};
|
|
document.addEventListener("mousedown", handleClickOutside);
|
|
return () => document.removeEventListener("mousedown", handleClickOutside);
|
|
}, []);
|
|
|
|
const changeLanguage = (lng) => {
|
|
i18n.changeLanguage(lng);
|
|
setCurrentLanguage(lng);
|
|
localStorage.setItem("language", lng);
|
|
|
|
if (lng === "ar") {
|
|
document.documentElement.dir = "rtl";
|
|
document.documentElement.lang = "ar";
|
|
} else {
|
|
document.documentElement.dir = "ltr";
|
|
document.documentElement.lang = "en";
|
|
}
|
|
};
|
|
|
|
const toggleMobileMenu = () => {
|
|
setIsMobileMenuOpen(!isMobileMenuOpen);
|
|
};
|
|
|
|
const closeMobileMenu = () => {
|
|
setIsMobileMenuOpen(false);
|
|
};
|
|
|
|
const logout = () => {
|
|
AuthService.deleteToken();
|
|
setUser(null);
|
|
setShowUserMenu(false);
|
|
window.location.href = "/";
|
|
};
|
|
|
|
const isAuthPage = [
|
|
"/login",
|
|
"/register",
|
|
"/forgot-password",
|
|
"/auth/choose-role",
|
|
].includes(pathname);
|
|
|
|
const isProfilePage = pathname === "/profile";
|
|
|
|
const isOwner = user?.role === UserRole.OWNER;
|
|
const isAdmin = user?.role === UserRole.ADMIN;
|
|
const isCustomer = user?.role === UserRole.CUSTOMER;
|
|
const isAuthenticated = !!user;
|
|
|
|
const getUserInitial = () => {
|
|
if (user?.name) {
|
|
return user.name.charAt(0).toUpperCase();
|
|
}
|
|
return <UserCircle className="w-5 h-5" />;
|
|
};
|
|
|
|
if (!isMounted) {
|
|
return (
|
|
<div className="min-h-screen flex items-center justify-center">
|
|
<div className="text-center">
|
|
<div className="w-16 h-16 border-4 border-amber-500 border-t-transparent rounded-full animate-spin mx-auto mb-4"></div>
|
|
<p className="text-gray-600">جاري التحميل...</p>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{!isAuthPage && (
|
|
<nav className="fixed top-0 left-0 right-0 bg-white/95 backdrop-blur-sm border-b border-gray-200 z-50 transition-all duration-300 shadow-sm">
|
|
<div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
|
|
<div
|
|
className={`flex justify-between items-center h-20 ${currentLanguage === "ar" ? "flex-row-reverse" : ""}`}
|
|
>
|
|
<div className="flex items-center">
|
|
<Link href="/" className="flex items-center space-x-3 group">
|
|
<div className="relative w-[150px] h-[60px]">
|
|
<Image
|
|
src="/logo.png"
|
|
alt={t("logoAlt")}
|
|
fill
|
|
className="object-contain"
|
|
priority
|
|
/>
|
|
</div>
|
|
<span className="text-3xl font-bold text-gray-800 hidden md:block">
|
|
{t("brandNamePart1")}
|
|
<span className="text-amber-600">
|
|
{t("brandNamePart2")}
|
|
</span>
|
|
</span>
|
|
</Link>
|
|
</div>
|
|
|
|
<div className="hidden md:flex items-center space-x-4">
|
|
<div
|
|
className={`flex items-center space-x-1 ${currentLanguage === "ar" ? "flex-row-reverse space-x-reverse" : ""}`}
|
|
>
|
|
{/* Download App Dropdown */}
|
|
<div className="relative group">
|
|
<button className="flex items-center gap-2 px-3 py-2 text-gray-700 hover:text-amber-600 hover:bg-amber-50 rounded-lg transition-all">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="currentColor" viewBox="0 0 16 16">
|
|
<path d="M11 2a3 3 0 0 0-3 3v6.5a.5.5 0 0 0 1 0V5a2 2 0 1 1 4 0v6.5a.5.5 0 0 0 1 0V5a3 3 0 0 0-3-3z"/>
|
|
<path d="M1.5 12.5A1.5 1.5 0 0 0 3 14h10a1.5 1.5 0 0 0 0-3H3a1.5 1.5 0 0 0-1.5 1.5z"/>
|
|
</svg>
|
|
<span className="text-sm font-semibold">تحميل التطبيق</span>
|
|
<svg className="w-4 h-4 transition-transform group-hover:rotate-180" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path strokeLinecap="round" strokeLinejoin="round" strokeWidth="2" d="M19 9l-7 7-7-7"/></svg>
|
|
</button>
|
|
<div className="absolute right-0 mt-2 w-64 bg-white rounded-xl shadow-xl border border-gray-200 overflow-hidden z-50 opacity-0 invisible group-hover:opacity-100 group-hover:visible transition-all duration-200 translate-y-2 group-hover:translate-y-0">
|
|
<div className="p-2">
|
|
<a href="/files/SweetHome.apk" download
|
|
className="flex items-center gap-3 px-4 py-3 rounded-lg hover:bg-green-50 transition-colors">
|
|
<div className="w-10 h-10 bg-green-100 rounded-lg flex items-center justify-center">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="22" height="22" fill="#16a34a" viewBox="0 0 16 16">
|
|
<path d="M2.76 3.061a.5.5 0 0 1 .679.2l1.283 2.352A8.9 8.9 0 0 1 8 5a8.9 8.9 0 0 1 3.278.613l1.283-2.352a.5.5 0 1 1 .878.478l-1.252 2.295C14.475 7.266 16 9.477 16 12H0c0-2.523 1.525-4.734 3.813-5.966L2.56 3.74a.5.5 0 0 1 .2-.678ZM5 10a1 1 0 1 0 0-2 1 1 0 0 0 0 2m6 0a1 1 0 1 0 0-2 1 1 0 0 0 0 2"/>
|
|
</svg>
|
|
</div>
|
|
<div>
|
|
<p className="font-semibold text-gray-900 text-sm">Android</p>
|
|
<p className="text-xs text-green-600">تحميل APK</p>
|
|
</div>
|
|
</a>
|
|
<div className="flex items-center gap-3 px-4 py-3 rounded-lg opacity-50 cursor-not-allowed">
|
|
<div className="w-10 h-10 bg-gray-100 rounded-lg flex items-center justify-center">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" fill="#6b7280" viewBox="0 0 16 16">
|
|
<path d="M11.182.008C11.148-.03 9.923.023 8.857 1.18c-1.066 1.156-.902 2.482-.878 2.516.024.034 1.52.087 2.475-1.258.955-1.345.762-2.391.728-2.43Zm3.314 11.733c-.048-.096-2.325-1.234-2.113-3.422.212-2.189 1.675-2.789 1.698-2.854.023-.065-.597-.79-1.254-1.157a3.692 3.692 0 0 0-1.563-.434c-.108-.003-.483-.095-1.254.116-.508.139-1.653.589-1.968.607-.316.018-1.256-.522-2.267-.665-.647-.125-1.333.131-1.824.328-.49.196-1.422.754-2.074 2.237-.652 1.482-.311 3.83-.067 4.56.244.729.625 1.924 1.273 2.796.576.984 1.34 1.667 1.659 1.899.319.232 1.219.385 1.843.067.502-.308 1.408-.485 1.766-.472.357.013 1.061.154 1.782.539.571.197 1.111.115 1.652-.105.541-.221 1.324-1.059 2.238-2.758.347-.79.505-1.217.473-1.282Z"/>
|
|
</svg>
|
|
</div>
|
|
<div>
|
|
<p className="font-semibold text-gray-400 text-sm">iOS</p>
|
|
<p className="text-xs text-gray-400">قريباً</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<NavLink href="/">الرئيسية</NavLink>
|
|
<NavLink href="/properties">عقاراتنا</NavLink>
|
|
|
|
{isAdmin && (
|
|
<NavLink href="/admin">
|
|
<span className="flex items-center gap-2">
|
|
<Shield className="w-4 h-4" />
|
|
الإدارة
|
|
</span>
|
|
</NavLink>
|
|
)}
|
|
|
|
{isOwner && (
|
|
<>
|
|
<NavLink href="/owner/properties">
|
|
<span className="flex items-center gap-2">
|
|
<Building className="w-4 h-4" />
|
|
عقاراتي
|
|
</span>
|
|
</NavLink>
|
|
<NavLink href="/owner/bookings">
|
|
<span className="flex items-center gap-2">
|
|
<Calendar className="w-4 h-4" />
|
|
الحجوزات
|
|
</span>
|
|
</NavLink>
|
|
<NavLink href="/owner/calendar">
|
|
<span className="flex items-center gap-2">
|
|
<CalendarDays className="w-4 h-4" />
|
|
التقويم
|
|
</span>
|
|
</NavLink>
|
|
<NavLink href="/owner/profits">
|
|
<span className="flex items-center gap-2">
|
|
<TrendingUp className="w-4 h-4" />
|
|
الأرباح
|
|
</span>
|
|
</NavLink>
|
|
</>
|
|
)}
|
|
|
|
{!user && !isAuthPage && (
|
|
<>
|
|
<NavLink href="/login">
|
|
<span className="flex items-center gap-2">
|
|
<LogIn className="w-4 h-4" />
|
|
تسجيل الدخول
|
|
</span>
|
|
</NavLink>
|
|
<NavLink href="/auth/choose-role">
|
|
<span className="flex items-center gap-2">
|
|
<UserPlus className="w-4 h-4" />
|
|
إنشاء حساب
|
|
</span>
|
|
</NavLink>
|
|
</>
|
|
)}
|
|
</div>
|
|
|
|
{/* <motion.button
|
|
whileHover={{ scale: 1.1, rotate: 360 }}
|
|
whileTap={{ scale: 0.9 }}
|
|
onClick={() =>
|
|
changeLanguage(currentLanguage === "en" ? "ar" : "en")
|
|
}
|
|
className="flex items-center justify-center w-10 h-10 bg-gray-100 hover:bg-gray-200 rounded-full transition-all duration-200 ml-4"
|
|
>
|
|
<Globe className="w-5 h-5 text-gray-700" />
|
|
</motion.button> */}
|
|
|
|
{user && (
|
|
<div className="relative" ref={menuRef}>
|
|
<motion.button
|
|
onClick={() => setShowUserMenu(!showUserMenu)}
|
|
className="w-10 h-10 bg-gradient-to-br from-amber-500 to-amber-600 rounded-full flex items-center justify-center text-white font-bold text-lg shadow-lg hover:shadow-xl transition-all hover:scale-105 focus:outline-none"
|
|
whileHover={{ scale: 1.1 }}
|
|
whileTap={{ scale: 0.95 }}
|
|
>
|
|
{getUserInitial()}
|
|
</motion.button>
|
|
|
|
<AnimatePresence>
|
|
{showUserMenu && (
|
|
<motion.div
|
|
initial={{ opacity: 0, y: -10 }}
|
|
animate={{ opacity: 1, y: 0 }}
|
|
exit={{ opacity: 0, y: -10 }}
|
|
className="absolute left-0 mt-2 w-72 bg-white rounded-xl shadow-xl border border-gray-200 overflow-hidden z-50"
|
|
>
|
|
<div className="bg-gradient-to-r from-amber-500 to-amber-600 p-4 text-white">
|
|
<div className="flex items-center gap-3">
|
|
<div className="w-12 h-12 bg-white/20 rounded-full flex items-center justify-center text-xl font-bold">
|
|
{getUserInitial()}
|
|
</div>
|
|
<div>
|
|
<p className="font-bold">
|
|
{user?.name || "مستخدم"}
|
|
</p>
|
|
<p className="text-xs text-amber-100">
|
|
{user?.email || ""}
|
|
</p>
|
|
<p className="text-xs text-amber-100 mt-1">
|
|
{UserRoleLabels[user?.role] || 'زائر'}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="p-2">
|
|
<Link
|
|
href="/profile"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<UserCircle className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">الملف الشخصي</p>
|
|
<p className="text-xs text-gray-500">
|
|
عرض وتعديل معلوماتك
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/favorites"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Heart className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">المفضلة</p>
|
|
<p className="text-xs text-gray-500">
|
|
العقارات المحفوظة
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
{isOwner && (
|
|
<>
|
|
<div className="border-t border-gray-100 my-2"></div>
|
|
|
|
<Link
|
|
href="/owner/properties"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Building className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">عقاراتي</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة عقاراتك
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/owner/properties/add"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<PlusCircle className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">إضافة عقار</p>
|
|
<p className="text-xs text-gray-500">
|
|
أضف عقاراً جديداً
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/owner/bookings"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Calendar className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">الحجوزات</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة حجوزاتك
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/owner/calendar"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<CalendarDays className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">التقويم</p>
|
|
<p className="text-xs text-gray-500">
|
|
جدول توفر العقارات
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/owner/profits"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<TrendingUp className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">الأرباح</p>
|
|
<p className="text-xs text-gray-500">
|
|
إحصائيات وأرباح
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
</>
|
|
)}
|
|
|
|
{isAdmin && (
|
|
<>
|
|
<div className="border-t border-gray-100 my-2"></div>
|
|
|
|
<Link
|
|
href="/admin"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Shield className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">لوحة التحكم</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة المنصة
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/admin/users"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Users className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">المستخدمين</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة المستخدمين
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/admin/properties"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Building className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">العقارات</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة جميع العقارات
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/admin/bookings"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Calendar className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">الحجوزات</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة الحجوزات
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
|
|
<Link
|
|
href="/admin/ledger"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<DollarSign className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">دفتر الحسابات</p>
|
|
<p className="text-xs text-gray-500">
|
|
إدارة المعاملات المالية
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
</>
|
|
)}
|
|
|
|
{isCustomer && (
|
|
<>
|
|
<div className="border-t border-gray-100 my-2"></div>
|
|
|
|
<Link
|
|
href="/tenant/bookings"
|
|
className="flex items-center gap-3 px-4 py-3 text-gray-700 hover:bg-amber-50 rounded-lg transition-colors"
|
|
onClick={() => setShowUserMenu(false)}
|
|
>
|
|
<Calendar className="w-5 h-5 text-amber-500" />
|
|
<div>
|
|
<p className="font-medium">حجوزاتي</p>
|
|
<p className="text-xs text-gray-500">
|
|
عرض حجوزاتك
|
|
</p>
|
|
</div>
|
|
</Link>
|
|
</>
|
|
)}
|
|
|
|
<div className="border-t border-gray-100 my-2"></div>
|
|
|
|
<button
|
|
onClick={logout}
|
|
className="w-full flex items-center gap-3 px-4 py-3 text-red-600 hover:bg-red-50 rounded-lg transition-colors"
|
|
>
|
|
<LogOut className="w-5 h-5" />
|
|
<div>
|
|
<p className="font-medium">تسجيل الخروج</p>
|
|
<p className="text-xs text-red-400">
|
|
إنهاء الجلسة الحالية
|
|
</p>
|
|
</div>
|
|
</button>
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
</AnimatePresence>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
<div className="md:hidden flex items-center gap-3">
|
|
<motion.button
|
|
whileHover={{ scale: 1.1, rotate: 360 }}
|
|
whileTap={{ scale: 0.9 }}
|
|
onClick={() =>
|
|
changeLanguage(currentLanguage === "en" ? "ar" : "en")
|
|
}
|
|
className="flex items-center justify-center w-10 h-10 bg-gray-100 hover:bg-gray-200 rounded-full transition-colors"
|
|
>
|
|
<Globe className="w-5 h-5 text-gray-700" />
|
|
</motion.button>
|
|
|
|
{user && (
|
|
<button
|
|
onClick={() => setShowUserMenu(!showUserMenu)}
|
|
className="w-10 h-10 bg-gradient-to-br from-amber-500 to-amber-600 rounded-full flex items-center justify-center text-white font-bold shadow-lg"
|
|
>
|
|
{getUserInitial()}
|
|
</button>
|
|
)}
|
|
|
|
<button
|
|
type="button"
|
|
onClick={toggleMobileMenu}
|
|
className="inline-flex items-center justify-center p-2 rounded-md text-gray-700 hover:text-amber-600 hover:bg-gray-100 focus:outline-none focus:ring-2 focus:ring-inset focus:ring-amber-500"
|
|
>
|
|
<span className="sr-only">{t("openMainMenu")}</span>
|
|
{isMobileMenuOpen ? (
|
|
<X className="w-6 h-6" />
|
|
) : (
|
|
<Menu className="w-6 h-6" />
|
|
)}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{isMobileMenuOpen && (
|
|
<motion.div
|
|
initial={{ opacity: 0, y: -10 }}
|
|
animate={{ opacity: 1, y: 0 }}
|
|
exit={{ opacity: 0, y: -10 }}
|
|
className="md:hidden bg-white border-t border-gray-200 shadow-lg"
|
|
>
|
|
<div className="px-2 pt-2 pb-3 space-y-1">
|
|
<MobileNavLink href="/" onClick={closeMobileMenu}>
|
|
{t("home")}
|
|
</MobileNavLink>
|
|
<MobileNavLink href="/properties" onClick={closeMobileMenu}>
|
|
{t("ourProducts")}
|
|
</MobileNavLink>
|
|
|
|
{/* Download App - Mobile */}
|
|
<div className="border-t border-gray-200 my-2"></div>
|
|
<p className="px-3 py-1 text-xs text-gray-400 font-medium">تحميل التطبيق</p>
|
|
<a href="/files/SweetHome.apk" download onClick={closeMobileMenu}
|
|
className="flex items-center gap-2 px-3 py-2 rounded-md text-green-600 hover:bg-green-50 transition-colors">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" fill="currentColor" viewBox="0 0 16 16">
|
|
<path d="M2.76 3.061a.5.5 0 0 1 .679.2l1.283 2.352A8.9 8.9 0 0 1 8 5a8.9 8.9 0 0 1 3.278.613l1.283-2.352a.5.5 0 1 1 .878.478l-1.252 2.295C14.475 7.266 16 9.477 16 12H0c0-2.523 1.525-4.734 3.813-5.966L2.56 3.74a.5.5 0 0 1 .2-.678ZM5 10a1 1 0 1 0 0-2 1 1 0 0 0 0 2m6 0a1 1 0 1 0 0-2 1 1 0 0 0 0 2"/>
|
|
</svg>
|
|
<span className="text-sm font-medium">Android - تحميل APK</span>
|
|
</a>
|
|
<div className="flex items-center gap-2 px-3 py-2 rounded-md text-gray-400 cursor-not-allowed opacity-50">
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" viewBox="0 0 16 16">
|
|
<path d="M11.182.008C11.148-.03 9.923.023 8.857 1.18c-1.066 1.156-.902 2.482-.878 2.516.024.034 1.52.087 2.475-1.258.955-1.345.762-2.391.728-2.43Zm3.314 11.733c-.048-.096-2.325-1.234-2.113-3.422.212-2.189 1.675-2.789 1.698-2.854.023-.065-.597-.79-1.254-1.157a3.692 3.692 0 0 0-1.563-.434c-.108-.003-.483-.095-1.254.116-.508.139-1.653.589-1.968.607-.316.018-1.256-.522-2.267-.665-.647-.125-1.333.131-1.824.328-.49.196-1.422.754-2.074 2.237-.652 1.482-.311 3.83-.067 4.56.244.729.625 1.924 1.273 2.796.576.984 1.34 1.667 1.659 1.899.319.232 1.219.385 1.843.067.502-.308 1.408-.485 1.766-.472.357.013 1.061.154 1.782.539.571.197 1.111.115 1.652-.105.541-.221 1.324-1.059 2.238-2.758.347-.79.505-1.217.473-1.282Z"/>
|
|
</svg>
|
|
<span className="text-sm">iOS - قريباً</span>
|
|
</div>
|
|
<div className="border-t border-gray-200 my-2"></div>
|
|
|
|
{isAdmin && (
|
|
<MobileNavLink href="/admin" onClick={closeMobileMenu}>
|
|
<span className="flex items-center gap-2">
|
|
<Shield className="w-4 h-4" />
|
|
الإدارة
|
|
</span>
|
|
</MobileNavLink>
|
|
)}
|
|
|
|
{isOwner && (
|
|
<>
|
|
<MobileNavLink
|
|
href="/owner/properties"
|
|
onClick={closeMobileMenu}
|
|
>
|
|
<span className="flex items-center gap-2">
|
|
<Building className="w-4 h-4" />
|
|
عقاراتي
|
|
</span>
|
|
</MobileNavLink>
|
|
<MobileNavLink
|
|
href="/owner/bookings"
|
|
onClick={closeMobileMenu}
|
|
>
|
|
<span className="flex items-center gap-2">
|
|
<Calendar className="w-4 h-4" />
|
|
الحجوزات
|
|
</span>
|
|
</MobileNavLink>
|
|
<MobileNavLink
|
|
href="/owner/calendar"
|
|
onClick={closeMobileMenu}
|
|
>
|
|
<span className="flex items-center gap-2">
|
|
<CalendarDays className="w-4 h-4" />
|
|
التقويم
|
|
</span>
|
|
</MobileNavLink>
|
|
<MobileNavLink
|
|
href="/owner/profits"
|
|
onClick={closeMobileMenu}
|
|
>
|
|
<span className="flex items-center gap-2">
|
|
<TrendingUp className="w-4 h-4" />
|
|
الأرباح
|
|
</span>
|
|
</MobileNavLink>
|
|
</>
|
|
)}
|
|
|
|
{!user && (
|
|
<>
|
|
<div className="border-t border-gray-200 my-2"></div>
|
|
<MobileNavLink href="/login" onClick={closeMobileMenu}>
|
|
<span className="flex items-center gap-2">
|
|
<LogIn className="w-4 h-4" />
|
|
تسجيل الدخول
|
|
</span>
|
|
</MobileNavLink>
|
|
<MobileNavLink
|
|
href="/auth/choose-role"
|
|
onClick={closeMobileMenu}
|
|
>
|
|
<span className="flex items-center gap-2">
|
|
<UserPlus className="w-4 h-4" />
|
|
إنشاء حساب
|
|
</span>
|
|
</MobileNavLink>
|
|
</>
|
|
)}
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
</nav>
|
|
)}
|
|
|
|
<main
|
|
className={`${!isAuthPage && !isProfilePage ? "pt-20" : ""} min-h-screen bg-gradient-to-b from-gray-50 to-white ${currentLanguage === "ar" ? "text-right" : "text-left"}`}
|
|
>
|
|
{children}
|
|
</main>
|
|
|
|
{!isAuthPage && !isProfilePage && (
|
|
<footer className="bg-gray-900 text-white py-12">
|
|
<div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
|
|
<div
|
|
className={`grid grid-cols-1 md:grid-cols-4 gap-8 ${currentLanguage === "ar" ? "text-right" : "text-left"}`}
|
|
>
|
|
<div className="space-y-4">
|
|
<div
|
|
className={`flex items-center ${currentLanguage === "ar" ? "flex-row-reverse" : "space-x-3"}`}
|
|
>
|
|
<div className="relative w-10 h-10">
|
|
<Image
|
|
src="/logo.png"
|
|
alt={t("logoAlt")}
|
|
fill
|
|
className="object-contain"
|
|
sizes="40px"
|
|
/>
|
|
</div>
|
|
<span className="text-3xl font-bold">
|
|
{t("brandNamePart1")}
|
|
<span className="text-amber-400">
|
|
{t("brandNamePart2")}
|
|
</span>
|
|
</span>
|
|
</div>
|
|
<p className="text-gray-400 text-sm">
|
|
{t("footerDescription")}
|
|
</p>
|
|
</div>
|
|
<div>
|
|
<h3 className="text-lg font-semibold mb-4">
|
|
{t("quickLinks")}
|
|
</h3>
|
|
<ul className="space-y-2">
|
|
<li>
|
|
<Link
|
|
href="/"
|
|
className="text-gray-400 hover:text-white transition-colors block py-1"
|
|
>
|
|
{t("home")}
|
|
</Link>
|
|
</li>
|
|
<li>
|
|
<Link
|
|
href="/properties"
|
|
className="text-gray-400 hover:text-white transition-colors block py-1"
|
|
>
|
|
{t("ourProducts")}
|
|
</Link>
|
|
</li>
|
|
{isAdmin && (
|
|
<li>
|
|
<Link
|
|
href="/admin"
|
|
className="text-gray-400 hover:text-white transition-colors block py-1"
|
|
>
|
|
الإدارة
|
|
</Link>
|
|
</li>
|
|
)}
|
|
</ul>
|
|
</div>
|
|
<div>
|
|
<h3 className="text-lg font-semibold mb-4">{t("contactUs")}</h3>
|
|
<ul className="space-y-3 text-gray-400">
|
|
<li className="flex items-center gap-2">
|
|
<Phone className="w-5 h-5" />
|
|
<span>{t("phone")}</span>
|
|
</li>
|
|
<li className="flex items-center gap-2">
|
|
<Mail className="w-5 h-5" />
|
|
<span>{t("footerEmail")}</span>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
<div className="mt-8 pt-8 border-t border-gray-800 text-center text-gray-400 text-sm">
|
|
<p>
|
|
© {currentYear} {t("copyright")}. {t("allRightsReserved")}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</footer>
|
|
)}
|
|
</>
|
|
);
|
|
}
|