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>
|
|
|
|
|
);
|
|
|
|
|
}
|