ecom / client /src /components /cart /cart-sidebar.tsx
shashwatIDR's picture
Upload 147 files
b89a86e verified
import { useLocation } from "wouter";
import { useCart } from "@/hooks/use-cart";
import { Button } from "@/components/ui/button";
import { Separator } from "@/components/ui/separator";
import { Sheet, SheetContent, SheetHeader, SheetTitle } from "@/components/ui/sheet";
import { ShoppingCart, Plus, Minus, Trash2, X } from "lucide-react";
import { useToast } from "@/hooks/use-toast";
interface CartSidebarProps {
isOpen: boolean;
onClose: () => void;
}
export default function CartSidebar({ isOpen, onClose }: CartSidebarProps) {
const [, setLocation] = useLocation();
const { items, itemCount, subtotal, tax, total, updateQuantity, removeItem } = useCart();
const { toast } = useToast();
const handleUpdateQuantity = async (id: string, newQuantity: number) => {
if (newQuantity < 1) return;
try {
await updateQuantity(id, newQuantity);
} catch (error) {
toast({
variant: "destructive",
title: "Error",
description: "Failed to update quantity.",
});
}
};
const handleRemoveItem = async (id: string) => {
try {
await removeItem(id);
toast({
title: "Item removed",
description: "Item has been removed from your cart.",
});
} catch (error) {
toast({
variant: "destructive",
title: "Error",
description: "Failed to remove item.",
});
}
};
const handleCheckout = () => {
onClose();
setLocation('/checkout');
};
return (
<Sheet open={isOpen} onOpenChange={onClose}>
<SheetContent side="right" className="w-96 flex flex-col">
<SheetHeader>
<SheetTitle className="flex items-center justify-between" data-testid="cart-title">
Shopping Cart
<Button
variant="ghost"
size="icon"
onClick={onClose}
data-testid="cart-close"
>
<X className="h-4 w-4" />
</Button>
</SheetTitle>
</SheetHeader>
{/* Cart Items */}
<div className="flex-1 overflow-y-auto py-4" data-testid="cart-items">
{items.length === 0 ? (
<div className="text-center py-12">
<ShoppingCart className="h-12 w-12 text-gray-400 mx-auto mb-4" />
<p className="text-gray-500" data-testid="empty-cart">Your cart is empty</p>
</div>
) : (
<div className="space-y-4">
{items.map((item) => (
<div
key={item.id}
className="flex items-center space-x-4 p-3 border border-gray-200 rounded-lg"
data-testid={`cart-item-${item.id}`}
>
{/* Product Image */}
<div className="w-16 h-16 bg-gray-100 rounded-lg overflow-hidden flex-shrink-0">
{item.product?.images && item.product.images.length > 0 ? (
<img
src={item.product.images[0]}
alt={item.product.title}
className="w-full h-full object-cover"
data-testid={`cart-item-image-${item.id}`}
/>
) : (
<div className="w-full h-full bg-gray-200 flex items-center justify-center">
<span className="text-xs text-gray-400">No Image</span>
</div>
)}
</div>
<div className="flex-1">
<h3 className="font-medium text-gray-900 text-sm line-clamp-2" data-testid={`cart-item-title-${item.id}`}>
{item.product?.title || 'Unknown Product'}
</h3>
<p className="text-xs text-gray-500" data-testid={`cart-item-seller-${item.id}`}>
{item.product?.store?.name || item.product?.seller?.username || 'Unknown Seller'}
</p>
<div className="flex items-center justify-between mt-1">
<span className="font-semibold text-blue-600" data-testid={`cart-item-price-${item.id}`}>
₹{item.product ? parseFloat(item.product.price).toFixed(2) : '0.00'}
</span>
<div className="flex items-center space-x-2">
<Button
size="icon"
variant="outline"
className="h-6 w-6"
onClick={() => handleUpdateQuantity(item.id, item.quantity - 1)}
disabled={item.quantity <= 1}
data-testid={`cart-decrease-${item.id}`}
>
<Minus className="h-3 w-3" />
</Button>
<span className="text-sm w-8 text-center" data-testid={`cart-quantity-${item.id}`}>
{item.quantity}
</span>
<Button
size="icon"
variant="outline"
className="h-6 w-6"
onClick={() => handleUpdateQuantity(item.id, item.quantity + 1)}
data-testid={`cart-increase-${item.id}`}
>
<Plus className="h-3 w-3" />
</Button>
</div>
</div>
</div>
<Button
size="icon"
variant="ghost"
className="text-red-500 hover:text-red-700 h-8 w-8"
onClick={() => handleRemoveItem(item.id)}
data-testid={`cart-remove-${item.id}`}
>
<Trash2 className="h-4 w-4" />
</Button>
</div>
))}
</div>
)}
</div>
{/* Cart Summary */}
{items.length > 0 && (
<div className="border-t pt-4">
<div className="space-y-2 mb-4" data-testid="cart-summary">
<div className="flex justify-between text-sm">
<span className="text-gray-600">Subtotal:</span>
<span className="text-gray-900" data-testid="cart-subtotal">
₹{subtotal.toFixed(2)}
</span>
</div>
<div className="flex justify-between text-sm">
<span className="text-gray-600">Shipping:</span>
<span className="text-green-600" data-testid="cart-shipping">Free</span>
</div>
<div className="flex justify-between text-sm">
<span className="text-gray-600">Tax:</span>
<span className="text-gray-900" data-testid="cart-tax">
₹{tax.toFixed(2)}
</span>
</div>
<Separator />
<div className="flex justify-between font-semibold">
<span className="text-gray-900">Total:</span>
<span className="text-gray-900" data-testid="cart-total">
₹{total.toFixed(2)}
</span>
</div>
</div>
<div className="space-y-2">
<Button
className="w-full bg-blue-600 hover:bg-blue-700"
onClick={handleCheckout}
data-testid="checkout-button"
>
Proceed to Checkout
</Button>
<Button
variant="outline"
className="w-full"
onClick={onClose}
data-testid="continue-shopping"
>
Continue Shopping
</Button>
</div>
</div>
)}
</SheetContent>
</Sheet>
);
}