fake_shop/app/good/[id]/page.jsx

310 lines
9.7 KiB
React
Raw Normal View History

2025-02-10 19:24:19 +08:00
"use client";
import { useParams } from "next/navigation";
import { useEffect, useState } from "react";
import Image from "next/image";
import { useRouter } from "next/navigation";
import toast from "react-hot-toast";
import AddToCartModal from "@/components/AddToCartModal";
2025-02-14 19:24:31 +08:00
import Link from "next/link";
import { AiOutlineHome, AiOutlineShoppingCart } from "react-icons/ai";
import { FiUser } from "react-icons/fi";
2025-02-10 19:24:19 +08:00
export default function ProductDetail() {
const params = useParams();
const [product, setProduct] = useState(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState(null);
const [quantity, setQuantity] = useState(1);
const router = useRouter();
const [showModal, setShowModal] = useState(false);
2025-02-14 19:24:31 +08:00
const [recommendedProducts, setRecommendedProducts] = useState([]);
2025-02-10 19:24:19 +08:00
useEffect(() => {
const fetchProduct = async () => {
try {
const res = await fetch(`/api/products/${params.id}`);
if (!res.ok) {
throw new Error("获取商品详情失败");
}
const data = await res.json();
setProduct(data);
2025-02-14 19:24:31 +08:00
fetchRecommendedProducts(); // 直接获取推荐商品
2025-02-10 19:24:19 +08:00
} catch (err) {
setError(err.message);
} finally {
setLoading(false);
}
};
2025-02-14 19:24:31 +08:00
const fetchRecommendedProducts = async () => {
try {
const res = await fetch(`/api/products`); // 获取所有商品
if (!res.ok) {
throw new Error("获取推荐商品失败");
}
const data = await res.json();
setRecommendedProducts(data.slice(0, 4)); // 取前四个商品
} catch (err) {
console.error(err.message);
}
};
2025-02-10 19:24:19 +08:00
fetchProduct();
}, [params.id]);
const handleAddToCart = async () => {
try {
const token = localStorage.getItem("token");
if (!token) {
router.push("/login");
return;
}
const res = await fetch("/api/cart", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({
productId: params.id,
quantity: quantity,
}),
});
if (!res.ok) {
throw new Error("添加购物车失败");
}
toast.success("添加成功");
handleModalClose();
} catch (error) {
toast.error(error.message);
}
};
const handleBuyNow = () => {
const token = localStorage.getItem("token");
if (!token) {
router.push("/login");
return;
}
router.push(`/checkout?products=${params.id}&quantity=${quantity}`);
};
const handleAddToCartClick = () => {
const token = localStorage.getItem("token");
if (!token) {
router.push("/login");
return;
}
setShowModal(true);
};
const handleModalClose = () => {
setShowModal(false);
};
if (loading) {
return <div className="p-4">加载中...</div>;
}
if (error) {
return <div className="p-4 text-red-500">错误: {error}</div>;
}
if (!product) {
return <div className="p-4">商品不存在</div>;
}
return (
<div className="flex flex-col min-h-screen pb-20 bg-gray-50">
{/* 商品图片 */}
<div className="relative aspect-square">
<Image
src={product.imageUrl}
alt={product.title}
fill
className="object-cover"
/>
</div>
{/* 商品信息 */}
2025-02-25 19:44:39 +08:00
<div className="flex-1 p-4 bg-white shadow-lg">
2025-02-10 19:24:19 +08:00
<div className="flex items-center">
<span className="text-2xl text-red-500 font-bold">
¥{product.price}
</span>
{product.isVip && (
<span className="ml-2 px-2 py-0.5 text-xs text-yellow-600 border border-yellow-600 rounded">
VIP
</span>
)}
</div>
<h1 className="mt-2 text-lg font-medium">{product.title}</h1>
<div className="mt-2 text-sm text-gray-500">
<span>库存: {product.stock}</span>
<span className="ml-4">销量: {product.sales}</span>
</div>
{/* 配送信息 */}
<div className="mt-4 p-3 bg-gray-50 rounded-lg">
<div className="flex justify-between items-center">
<span className="text-gray-500">配送</span>
<span>{product.delivery}</span>
</div>
<div className="mt-2 flex justify-between items-center">
<span className="text-gray-500">服务</span>
<div className="flex gap-2">
{product.services.map((service, index) => (
<span key={index} className="text-sm text-gray-600">
{service}
</span>
))}
</div>
</div>
</div>
{/* 商品描述 */}
<div className="mt-6">
<h2 className="text-lg font-bold mb-2">商品详情</h2>
<p className="text-gray-600">{product.description}</p>
<h3 className="mt-4 font-bold">产品特点</h3>
<ul className="mt-2 space-y-2">
{product.features.map((feature, index) => (
<li key={index} className="flex items-center text-gray-600">
<span className="mr-2"></span>
{feature}
</li>
))}
</ul>
</div>
</div>
2025-02-25 19:44:39 +08:00
{/* 店铺模块 */}
<div className="p-4 bg-white mt-4 rounded-lg shadow-lg flex flex-col">
<Link href="/shop" className="flex flex-row items-center">
<Image
src="https://s2.loli.net/2025/02/25/G1yo27THWhNC5uf.jpg" // 替换为实际的店铺头像路径
alt="店铺头像"
width={80}
height={80}
className="rounded-lg border-2 border-yellow-400 mr-4"
/>
<div className="flex flex-col flex-1 justify-around items-start">
<span className="text-lg font-semibold text-gray-800">
宠物用品专卖店
</span>
<div className="flex items-center">
<span className="mr-2 text-gray-600">评分:</span>
<div className="flex">
{[...Array(5)].map((_, index) => (
<svg
key={index}
className="w-5 h-5 text-yellow-500"
fill="currentColor"
viewBox="0 0 20 20"
>
<path d="M10 15l-5.878 3.09 1.121-6.535L0 6.545l6.545-.955L10 0l2.455 5.59L20 6.545l-5.243 4.005 1.121 6.535z" />
</svg>
))}
</div>
</div>
<p className="text-yellow-700 bg-yellow-400 px-1 rounded">
官方正品旗舰店{" >"}
</p>
</div>
<button className="bg-red-500 h-12 px-2 rounded-lg text-white">
进店看看
</button>
</Link>
<div className="flex flex-row mt-2 justify-between">
<div className="flex">
<span className="text-gray-500">宝贝质量:</span>
<span className="font-bold">4.9</span>
</div>
<div className="flex">
<span className="text-gray-500">物流速度:</span>
<span className="font-bold">5.0</span>
</div>
<div className="flex">
<span className="text-gray-500">服务保障:</span>
<span className="font-bold">4.8</span>
</div>
</div>
</div>
2025-02-14 19:24:31 +08:00
{/* 推荐商品模块 */}
<div className="p-4 bg-white mt-4">
<h2 className="text-lg font-bold mb-2">猜你喜欢</h2>
<div className="grid grid-cols-2 gap-4">
{recommendedProducts.map((item) => (
<Link key={item._id} href={`/good/${item._id}`}>
<div className="bg-gray-100 rounded-lg p-2 overflow-hidden">
<Image
src={item.imageUrl}
alt={item.title}
width={200}
height={200}
className="object-cover rounded h-32 w-full"
/>
<h3 className="text-sm font-medium mt-2">{item.title}</h3>
<p className="text-red-500">¥{item.price}</p>
</div>
</Link>
))}
</div>
</div>
2025-02-10 19:24:19 +08:00
{/* 底部购买栏 */}
2025-02-14 19:24:31 +08:00
<div className="fixed gap-4 bottom-0 left-0 right-0 flex items-center p-4 bg-white border-t">
<Link href="/tab">
<button className="flex-1 text-gray-500 flex flex-col items-center">
<AiOutlineHome className="mb-1 w-6 h-6" />
<span className="text-xs">首页</span>
</button>
</Link>
<Link href="/contact">
<button className="flex-1 text-gray-500 flex flex-col items-center">
<FiUser className="mb-1 w-6 h-6" />
<span className="text-xs">客服</span>
</button>
</Link>
<Link href="/tab/cart">
<button className="flex-1 text-gray-500 flex flex-col items-center">
<AiOutlineShoppingCart className="mb-1 w-6 h-6" />
<span className="text-xs">购物车</span>
</button>
</Link>
2025-02-10 19:24:19 +08:00
<button
className="flex-1 bg-yellow-400 text-white py-2 rounded-full"
onClick={handleAddToCartClick}
>
加入购物车
</button>
<button
2025-02-14 19:24:31 +08:00
className="flex-1 bg-red-500 text-white py-2 rounded-full"
2025-02-10 19:24:19 +08:00
onClick={handleBuyNow}
>
立即购买
</button>
</div>
<AddToCartModal
isOpen={showModal}
onClose={handleModalClose}
product={product}
quantity={quantity}
setQuantity={setQuantity}
onAddToCart={handleAddToCart}
/>
</div>
);
}