first commit
This commit is contained in:
292
app/[locale]/product/[slug]/ProductPageContent.tsx
Normal file
292
app/[locale]/product/[slug]/ProductPageContent.tsx
Normal file
@@ -0,0 +1,292 @@
|
||||
"use client"
|
||||
|
||||
import { useState } from "react"
|
||||
import Image from "next/image"
|
||||
import Link from "next/link"
|
||||
import { Minus, Plus, Heart, ShoppingCart, Store } from "lucide-react"
|
||||
import { Button } from "@/components/ui/button"
|
||||
import { Card } from "@/components/ui/card"
|
||||
import { Separator } from "@/components/ui/separator"
|
||||
import { Badge } from "@/components/ui/badge"
|
||||
import { Avatar, AvatarFallback } from "@/components/ui/avatar"
|
||||
import placeholder from "@/public/jb.webp"
|
||||
import { useProduct, useCategories } from "@/lib/hooks"
|
||||
import { Skeleton } from "@/components/ui/skeleton"
|
||||
|
||||
interface ProductDetailProps {
|
||||
slug: string
|
||||
}
|
||||
|
||||
const ProductPageContent = ({ slug }: ProductDetailProps) => {
|
||||
const [isClient, setIsClient] = useState(false)
|
||||
const [selectedImage, setSelectedImage] = useState(0)
|
||||
const [quantity, setQuantity] = useState(1)
|
||||
const [isFavorite, setIsFavorite] = useState(false)
|
||||
const [isInCart, setIsInCart] = useState(false)
|
||||
const [isLoading, setIsLoading] = useState(false)
|
||||
|
||||
const { data: product, isLoading: productLoading, error } = useProduct(slug)
|
||||
const { data: categoriesData } = useCategories()
|
||||
|
||||
if (!isClient) {
|
||||
typeof window !== "undefined" && setIsClient(true)
|
||||
}
|
||||
|
||||
const t = {
|
||||
addToCart: "Add to Cart",
|
||||
goToCart: "Go to Cart",
|
||||
price: "Price:",
|
||||
aboutProduct: "About Product",
|
||||
brand: "Brand",
|
||||
model: "Model",
|
||||
description: "Product Description",
|
||||
recommended: "Recommended Products",
|
||||
store: "Store",
|
||||
writeToStore: "Write to Store",
|
||||
color: "Color:",
|
||||
}
|
||||
|
||||
const handleAddToCart = async () => {
|
||||
setIsLoading(true)
|
||||
try {
|
||||
// TODO: implement cart API call
|
||||
await new Promise((resolve) => setTimeout(resolve, 500))
|
||||
setIsInCart(true)
|
||||
} finally {
|
||||
setIsLoading(false)
|
||||
}
|
||||
}
|
||||
|
||||
const handleQuantityChange = async (newQuantity: number) => {
|
||||
if (newQuantity < 1) return
|
||||
setIsLoading(true)
|
||||
try {
|
||||
setQuantity(newQuantity)
|
||||
// TODO: implement cart quantity update API call
|
||||
await new Promise((resolve) => setTimeout(resolve, 300))
|
||||
} finally {
|
||||
setIsLoading(false)
|
||||
}
|
||||
}
|
||||
|
||||
const handleToggleFavorite = () => {
|
||||
setIsFavorite(!isFavorite)
|
||||
// TODO: implement favorites API call
|
||||
}
|
||||
|
||||
if (productLoading) {
|
||||
return (
|
||||
<div className="container mx-auto px-4 py-8">
|
||||
<div className="flex flex-col lg:flex-row gap-8">
|
||||
<div className="flex-1 max-w-2xl">
|
||||
<Skeleton className="aspect-square w-full rounded-2xl" />
|
||||
<div className="mt-4 flex gap-2">
|
||||
{[1, 2, 3].map((i) => (
|
||||
<Skeleton key={i} className="w-16 h-16 rounded" />
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex-1 space-y-6">
|
||||
<Skeleton className="h-10 w-64" />
|
||||
<Skeleton className="h-20 w-full" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
if (error || !product) {
|
||||
return (
|
||||
<div className="container mx-auto px-4 py-8 text-center">
|
||||
<h2 className="text-2xl font-bold text-red-600">Product not found</h2>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="container mx-auto px-4 py-8">
|
||||
<div className="flex flex-col lg:flex-row gap-8">
|
||||
{/* Product Images */}
|
||||
<div className="flex-1 max-w-2xl">
|
||||
<div className="relative">
|
||||
<div className="relative aspect-square w-full rounded-2xl overflow-hidden bg-gray-50">
|
||||
{product.labels && product.labels.length > 0 && (
|
||||
<div className="absolute top-0 right-0 z-10 flex flex-col gap-1">
|
||||
{product.labels.map((label) => (
|
||||
<Badge
|
||||
key={label.text}
|
||||
className="rounded-l-md rounded-r-none text-white text-xs font-bold uppercase"
|
||||
style={{ backgroundColor: label.bg_color }}
|
||||
>
|
||||
{label.text}
|
||||
</Badge>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
<Image
|
||||
src={product.images?.[selectedImage] || product.image || placeholder}
|
||||
alt={product.name}
|
||||
fill
|
||||
className="object-contain"
|
||||
priority
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* Thumbnail Images */}
|
||||
{product.images && product.images.length > 1 && (
|
||||
<div className="mt-4 flex gap-2 overflow-x-auto pb-2">
|
||||
{product.images.map((image, index) => (
|
||||
<button
|
||||
key={index}
|
||||
onClick={() => setSelectedImage(index)}
|
||||
className={`relative w-16 h-16 rounded overflow-hidden border ${
|
||||
selectedImage === index ? "border-black" : "border-transparent"
|
||||
}`}
|
||||
>
|
||||
<Image
|
||||
src={image || "/placeholder.svg"}
|
||||
alt={`${product.name} thumbnail ${index + 1}`}
|
||||
fill
|
||||
className="object-cover"
|
||||
/>
|
||||
</button>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Product Info */}
|
||||
<div className="flex-1 space-y-6">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold mb-2">{product.name}</h1>
|
||||
{product.category && (
|
||||
<div className="flex gap-2 flex-wrap">
|
||||
<span className="text-sm text-gray-500">Category: {product.category}</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Product Info Table */}
|
||||
<Card className="p-4 rounded-xl">
|
||||
<h3 className="text-xl font-semibold mb-4">{t.aboutProduct}</h3>
|
||||
<div className="space-y-3">
|
||||
{product.brand && (
|
||||
<>
|
||||
<div className="flex justify-between items-center py-2">
|
||||
<span className="text-gray-500">{t.brand}</span>
|
||||
<span className="font-medium">{product.brand}</span>
|
||||
</div>
|
||||
<Separator />
|
||||
</>
|
||||
)}
|
||||
{product.stock !== undefined && (
|
||||
<>
|
||||
<div className="flex justify-between items-center py-2">
|
||||
<span className="text-gray-500">Stock</span>
|
||||
<span className="font-medium">{product.stock}</span>
|
||||
</div>
|
||||
<Separator />
|
||||
</>
|
||||
)}
|
||||
</div>
|
||||
</Card>
|
||||
|
||||
{/* Description */}
|
||||
{product.description && (
|
||||
<div>
|
||||
<h3 className="text-xl font-semibold mb-3">{t.description}</h3>
|
||||
<p className="text-gray-700 leading-relaxed">{product.description}</p>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Price & Actions Sidebar */}
|
||||
<div className="lg:w-[420px] space-y-4">
|
||||
<Card className="p-6 rounded-xl shadow-lg">
|
||||
<div className="flex justify-between items-center mb-6">
|
||||
<span className="text-lg text-gray-500">{t.price}</span>
|
||||
<span className="text-3xl font-bold">${product.price}</span>
|
||||
</div>
|
||||
|
||||
<div className="space-y-4">
|
||||
{isInCart ? (
|
||||
<div className="space-y-3">
|
||||
<Link href="/cart">
|
||||
<Button size="lg" className="w-full rounded-xl text-lg font-bold bg-green-600 hover:bg-green-700">
|
||||
<ShoppingCart className="mr-2 h-5 w-5" />
|
||||
{t.goToCart}
|
||||
</Button>
|
||||
</Link>
|
||||
|
||||
<div className="flex items-center gap-2">
|
||||
<Button
|
||||
variant="outline"
|
||||
size="icon"
|
||||
onClick={() => handleQuantityChange(quantity - 1)}
|
||||
disabled={quantity === 1 || isLoading}
|
||||
className="rounded-xl bg-blue-50 flex-shrink-0"
|
||||
>
|
||||
<Minus className="h-5 w-5" />
|
||||
</Button>
|
||||
<div className="flex-1 text-center font-semibold text-lg">{quantity}</div>
|
||||
<Button
|
||||
variant="outline"
|
||||
size="icon"
|
||||
onClick={() => handleQuantityChange(quantity + 1)}
|
||||
disabled={isLoading}
|
||||
className="rounded-xl bg-blue-50 flex-shrink-0"
|
||||
>
|
||||
<Plus className="h-5 w-5" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
) : (
|
||||
<Button
|
||||
size="lg"
|
||||
onClick={handleAddToCart}
|
||||
disabled={isLoading}
|
||||
className="w-full rounded-xl text-lg font-bold"
|
||||
>
|
||||
<ShoppingCart className="mr-2 h-5 w-5" />
|
||||
{t.addToCart}
|
||||
</Button>
|
||||
)}
|
||||
|
||||
<Button
|
||||
variant="outline"
|
||||
size="lg"
|
||||
onClick={handleToggleFavorite}
|
||||
className={`w-full rounded-xl ${
|
||||
isFavorite ? "bg-red-50 border-red-200 hover:bg-red-100" : "bg-blue-50"
|
||||
}`}
|
||||
>
|
||||
<Heart className={`h-6 w-6 ${isFavorite ? "fill-red-500 text-red-500" : ""}`} />
|
||||
</Button>
|
||||
</div>
|
||||
</Card>
|
||||
|
||||
{/* Seller Card */}
|
||||
<Card className="p-6 rounded-xl">
|
||||
<div className="flex items-center gap-4 mb-4">
|
||||
<Avatar className="w-14 h-14">
|
||||
<AvatarFallback>
|
||||
<Store className="h-6 w-6" />
|
||||
</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<p className="text-sm text-gray-500">{t.store}</p>
|
||||
<h4 className="text-xl font-bold hover:text-primary cursor-pointer">Official Store</h4>
|
||||
</div>
|
||||
</div>
|
||||
<Button variant="outline" size="lg" disabled className="w-full rounded-xl bg-transparent">
|
||||
{t.writeToStore}
|
||||
</Button>
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
export default ProductPageContent
|
||||
39
app/[locale]/product/[slug]/page.tsx
Normal file
39
app/[locale]/product/[slug]/page.tsx
Normal file
@@ -0,0 +1,39 @@
|
||||
import type { Metadata } from "next"
|
||||
import { notFound } from "next/navigation"
|
||||
import ProductPageContent from "./ProductPageContent"
|
||||
|
||||
type Props = {
|
||||
params: Promise<{ locale: string; slug: string }>
|
||||
}
|
||||
|
||||
export const revalidate = 3600 // ISR: Revalidate every hour
|
||||
|
||||
export async function generateMetadata({ params }: Props): Promise<Metadata> {
|
||||
const { locale, slug } = await params
|
||||
|
||||
return {
|
||||
title: `Product ${slug} | E-Commerce`,
|
||||
description: `View details for product ${slug}`,
|
||||
openGraph: {
|
||||
locale,
|
||||
type: "website",
|
||||
title: `Product ${slug} | E-Commerce`,
|
||||
description: `View details for product ${slug}`,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
export async function generateStaticParams() {
|
||||
// Generate static params for popular products
|
||||
return [{ slug: "nike-air-max" }, { slug: "adidas-ultraboost" }]
|
||||
}
|
||||
|
||||
export default async function ProductPage(props: Props) {
|
||||
const params = await props.params
|
||||
|
||||
if (!params.slug) {
|
||||
notFound()
|
||||
}
|
||||
|
||||
return <ProductPageContent slug={params.slug} />
|
||||
}
|
||||
9
app/[locale]/product/[slug]/product-content.tsx
Normal file
9
app/[locale]/product/[slug]/product-content.tsx
Normal file
@@ -0,0 +1,9 @@
|
||||
"use client"
|
||||
export default function ProductPageContent({ slug }: { slug: string }) {
|
||||
return (
|
||||
<div className="container mx-auto px-4 py-8">
|
||||
<h1 className="text-3xl font-bold">Product: {slug}</h1>
|
||||
{/* Product content will go here */}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
Reference in New Issue
Block a user