244 lines
7.9 KiB
TypeScript
244 lines
7.9 KiB
TypeScript
"use client";
|
|
|
|
import { useState, useEffect } from "react";
|
|
import Link from "next/link";
|
|
import { Menu, Heart, Truck, ShoppingCart, User } from "lucide-react";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Badge } from "@/components/ui/badge";
|
|
import {
|
|
Sheet,
|
|
SheetContent,
|
|
SheetHeader,
|
|
SheetTitle,
|
|
} from "@/components/ui/sheet";
|
|
import { ScrollArea } from "@/components/ui/scroll-area";
|
|
import { useCategories, useFavorites, useOrders } from "@/lib/hooks";
|
|
import { useCartCount } from "@/features/cart/hooks/useCart";
|
|
import { useRouter } from "next/navigation";
|
|
import { useAuthStatus } from "@/lib/hooks/useAuth";
|
|
import { useTranslations } from "next-intl";
|
|
import AuthDialog from "./ui/AuthDialog";
|
|
|
|
interface MobileBottomNavProps {
|
|
locale?: string;
|
|
translations?: {
|
|
catalog: string;
|
|
favorites: string;
|
|
orders: string;
|
|
cart: string;
|
|
login: string;
|
|
profile: string;
|
|
};
|
|
onLoginClick?: () => void;
|
|
}
|
|
|
|
export default function MobileBottomNav({
|
|
locale = "ru",
|
|
translations,
|
|
onLoginClick,
|
|
}: MobileBottomNavProps) {
|
|
const [isClient, setIsClient] = useState(false);
|
|
const [isCategoryOpen, setIsCategoryOpen] = useState(false);
|
|
const [isLoginOpen, setIsLoginOpen] = useState(false);
|
|
const t = useTranslations();
|
|
|
|
const { isAuthenticated, isLoading: authLoading } = useAuthStatus();
|
|
|
|
const { data: categories = [] } = useCategories();
|
|
|
|
// OPTIMIZED: Use event-driven cart count instead of full cart data
|
|
const cartCount = useCartCount();
|
|
|
|
const { data: favoritesData } = useFavorites();
|
|
const { data: ordersData } = useOrders();
|
|
const router = useRouter();
|
|
|
|
useEffect(() => {
|
|
setIsClient(true);
|
|
}, []);
|
|
|
|
const handleProfileClick = (e: React.MouseEvent) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
|
|
console.log("[MobileBottomNav] Profile clicked", {
|
|
authLoading,
|
|
isAuthenticated,
|
|
hasOnLoginClick: !!onLoginClick,
|
|
});
|
|
|
|
if (authLoading) {
|
|
return;
|
|
}
|
|
|
|
if (isAuthenticated) {
|
|
router.push(`/${locale}/me`);
|
|
} else {
|
|
if (onLoginClick) {
|
|
console.log("[MobileBottomNav] Calling parent onLoginClick");
|
|
onLoginClick();
|
|
} else {
|
|
console.log("[MobileBottomNav] Using local login dialog");
|
|
setIsLoginOpen(true);
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleNavigation = (path: string) => (e: React.MouseEvent) => {
|
|
e.preventDefault();
|
|
console.log("[MobileBottomNav] Navigating to:", path);
|
|
router.push(path);
|
|
};
|
|
|
|
if (!isClient) return null;
|
|
|
|
return (
|
|
<>
|
|
{/* Mobile Bottom Navigation */}
|
|
<div className="fixed bottom-0 left-0 right-0 z-50 bg-white border-t shadow-lg lg:hidden">
|
|
<div className="flex items-center justify-around h-16 px-2">
|
|
{/* Catalog Button */}
|
|
<Button
|
|
variant="ghost"
|
|
size="sm"
|
|
className="flex-col gap-0.5 h-auto px-2 py-2"
|
|
onClick={() => {
|
|
console.log("[MobileBottomNav] Catalog clicked");
|
|
setIsCategoryOpen(true);
|
|
}}
|
|
>
|
|
<Menu className="h-5 w-5 text-gray-600" />
|
|
<span className="text-xs text-gray-700">{t("common.catalog")}</span>
|
|
</Button>
|
|
|
|
{/* Favorites Button */}
|
|
<Button
|
|
variant="ghost"
|
|
size="sm"
|
|
className="relative flex-col gap-0.5 h-auto px-2 py-2"
|
|
onClick={handleNavigation("/favorites")}
|
|
>
|
|
<div className="relative">
|
|
<Heart className="h-5 w-5 text-gray-600" />
|
|
{(favoritesData?.length || 0) > 0 && (
|
|
<Badge
|
|
variant="destructive"
|
|
className="absolute -right-2 -top-2 h-4 w-4 flex items-center justify-center p-0 text-[10px]"
|
|
>
|
|
{favoritesData?.length}
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
<span className="text-xs text-gray-700">
|
|
{t("common.favorites")}
|
|
</span>
|
|
</Button>
|
|
|
|
{/* Orders Button */}
|
|
<Button
|
|
variant="ghost"
|
|
size="sm"
|
|
className="relative flex-col gap-0.5 h-auto px-2 py-2"
|
|
onClick={handleNavigation("/orders")}
|
|
>
|
|
<div className="relative">
|
|
<Truck className="h-5 w-5 text-gray-600" />
|
|
{(ordersData?.length || 0) > 0 && (
|
|
<Badge
|
|
variant="destructive"
|
|
className="absolute -right-2 -top-2 h-4 w-4 flex items-center justify-center p-0 text-[10px]"
|
|
>
|
|
{ordersData?.length}
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
<span className="text-xs text-gray-700">{t("common.orders")}</span>
|
|
</Button>
|
|
|
|
{/* Cart Button - OPTIMIZED */}
|
|
<Button
|
|
variant="ghost"
|
|
size="sm"
|
|
className="relative flex-col gap-0.5 h-auto px-2 py-2"
|
|
onClick={handleNavigation("/cart")}
|
|
>
|
|
<div className="relative">
|
|
<ShoppingCart className="h-5 w-5 text-gray-600" />
|
|
{cartCount > 0 && (
|
|
<Badge
|
|
variant="destructive"
|
|
className="absolute -right-2 -top-2 h-4 w-4 flex items-center justify-center p-0 text-[10px]"
|
|
>
|
|
{cartCount}
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
<span className="text-xs text-gray-700">{t("common.cart")}</span>
|
|
</Button>
|
|
|
|
{/* Profile/Login Button */}
|
|
<Button
|
|
variant="ghost"
|
|
size="sm"
|
|
className="flex-col gap-0.5 h-auto px-2 py-2"
|
|
onClick={handleProfileClick}
|
|
disabled={authLoading}
|
|
>
|
|
<User className="h-5 w-5 text-gray-600" />
|
|
<span className="text-xs text-gray-700">
|
|
{authLoading
|
|
? "..."
|
|
: isAuthenticated
|
|
? t("common.profile")
|
|
: t("common.login")}
|
|
</span>
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Category Sheet/Drawer */}
|
|
<Sheet open={isCategoryOpen} onOpenChange={setIsCategoryOpen}>
|
|
<SheetContent side="left" className="w-[300px] p-0">
|
|
<SheetHeader className="p-4 border-b">
|
|
<SheetTitle>{t("common.catalog")}</SheetTitle>
|
|
</SheetHeader>
|
|
<ScrollArea className="h-[calc(100vh-80px)]">
|
|
<div className="p-4">
|
|
{categories.map((category) => (
|
|
<div key={category.id} className="mb-4">
|
|
<Link
|
|
href={`/category/${category.slug}?category_id=${category.id}`}
|
|
onClick={() => setIsCategoryOpen(false)}
|
|
className="flex items-center gap-3 px-3 py-2 rounded-lg hover:bg-gray-100 transition-colors font-semibold"
|
|
>
|
|
<span>{category.name}</span>
|
|
</Link>
|
|
|
|
{/* Subcategories */}
|
|
{category.children && category.children.length > 0 && (
|
|
<div className="ml-8 mt-2 space-y-1">
|
|
{category.children.map((child: any) => (
|
|
<Link
|
|
key={child.id}
|
|
href={`/category/${child.slug}?category_id=${child.id}`}
|
|
onClick={() => setIsCategoryOpen(false)}
|
|
className="block px-3 py-2 text-sm text-gray-600 hover:text-primary hover:bg-gray-50 rounded-lg transition-colors"
|
|
>
|
|
{child.name}
|
|
</Link>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</ScrollArea>
|
|
</SheetContent>
|
|
</Sheet>
|
|
|
|
{/* Local Auth Dialog */}
|
|
<AuthDialog isOpen={isLoginOpen} onClose={() => setIsLoginOpen(false)} />
|
|
</>
|
|
);
|
|
}
|