search bar now working

This commit is contained in:
Mann Patel
2025-03-29 17:28:09 -06:00
parent 91ec43627a
commit 2e77ef49f4
12 changed files with 687 additions and 71 deletions

View File

@@ -39,7 +39,6 @@ const Home = () => {
setError(error.message);
}
};
fetchProducts();
}, []);
@@ -112,65 +111,186 @@ const Home = () => {
</div> */}
{/* Recent Listings */}
<div>
<div className="relative py-4">
<h2 className="text-xl font-semibold text-gray-800 mb-4">
Recent Listings
Recommendation
</h2>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-4 gap-6">
{listings.map((listing) => (
<Link
key={listing.id}
to={`/product/${listing.id}`}
className="bg-white border border-gray-200 hover:shadow-md transition-shadow"
>
<div className="relative">
<img
src={listing.image}
alt={listing.title}
className="w-full h-48 object-cover"
/>
<button
onClick={(e) => toggleFavorite(listing.id, e)}
className="absolute top-2 right-2 p-1 bg-white rounded-full shadow-sm"
>
<Heart
className={`h-5 w-5 ${
listing.isFavorite
? "text-red-500 fill-red-500"
: "text-gray-400"
}`}
/>
</button>
</div>
<div className="p-4">
<div className="flex justify-between items-start mb-2">
<div className="relative">
{/* Left Button - Overlaid on products */}
<button
onClick={() =>
document
.getElementById("RecomContainer")
.scrollBy({ left: -400, behavior: "smooth" })
}
className="absolute left-0 top-1/2 transform -translate-y-1/2 bg-gray-800 bg-opacity-70 text-white p-4 rounded-full z-20 hidden md:flex items-center justify-center w-12 h-12"
>
</button>
{/* Scrollable Listings Container */}
<div
id="RecomContainer"
className="overflow-x-auto whitespace-nowrap flex space-x-6 scroll-smooth scrollbar-hide px-10 pl-0"
>
{listings.map((listing) => (
<Link
key={listing.id}
to={`/product/${listing.id}`}
className="bg-white border border-gray-200 hover:shadow-md transition-shadow w-70 flex-shrink-0 relative"
>
<div className="relative">
<img
src={listing.image}
alt={listing.title}
className="w-full h-48 object-cover"
/>
<button
onClick={(e) => toggleFavorite(listing.id, e)}
className="absolute top-2 right-2 p-2 bg-white rounded-full shadow-sm"
>
<Heart
className={`h-6 w-6 ${
listing.isFavorite
? "text-red-500 fill-red-500"
: "text-gray-400"
}`}
/>
</button>
</div>
<div className="p-4">
<h3 className="text-lg font-medium text-gray-800 leading-tight">
{listing.title}
</h3>
<span className="font-semibold text-green-600">
<span className="font-semibold text-green-600 block mt-1">
${listing.price}
</span>
<div className="flex items-center text-sm text-gray-500 mt-2">
<Tag className="h-4 w-4 mr-1" />
<span>{listing.category}</span>
<span className="mx-2"></span>
<span>{listing.condition}</span>
</div>
<div className="flex justify-between items-center pt-2 border-t border-gray-100 mt-3">
<span className="text-xs text-gray-500">
{listing.datePosted}
</span>
<span className="text-sm font-medium text-gray-700">
{listing.seller}
</span>
</div>
</div>
</Link>
))}
</div>
{/* Right Button - Overlaid on products */}
<button
onClick={() =>
document
.getElementById("RecomContainer")
.scrollBy({ left: 400, behavior: "smooth" })
}
className="absolute right-0 top-1/2 transform -translate-y-1/2 bg-gray-800 bg-opacity-70 text-white p-4 rounded-full z-20 hidden md:flex items-center justify-center w-12 h-12"
>
</button>
</div>
</div>
{/* Recent Listings */}
<div className="relative py-4">
<h2 className="text-xl font-semibold text-gray-800 mb-4">
Recent Listings
</h2>
<div className="relative">
{/* Left Button - Overlaid on products */}
<button
onClick={() =>
document
.getElementById("listingsContainer")
.scrollBy({ left: -400, behavior: "smooth" })
}
className="absolute left-0 top-1/2 transform -translate-y-1/2 bg-gray-800 bg-opacity-70 text-white p-4 rounded-full z-20 hidden md:flex items-center justify-center w-12 h-12"
>
</button>
{/* Scrollable Listings Container */}
<div
id="listingsContainer"
className="overflow-x-auto whitespace-nowrap flex space-x-6 scroll-smooth scrollbar-hide px-10 pl-0"
>
{listings.map((listing) => (
<Link
key={listing.id}
to={`/product/${listing.id}`}
className="bg-white border border-gray-200 hover:shadow-md transition-shadow w-70 flex-shrink-0 relative"
>
<div className="relative">
<img
src={listing.image}
alt={listing.title}
className="w-full h-48 object-cover"
/>
<button
onClick={(e) => toggleFavorite(listing.id, e)}
className="absolute top-2 right-2 p-2 bg-white rounded-full shadow-sm"
>
<Heart
className={`h-6 w-6 ${
listing.isFavorite
? "text-red-500 fill-red-500"
: "text-gray-400"
}`}
/>
</button>
</div>
<div className="flex items-center text-sm text-gray-500 mb-3">
<Tag className="h-4 w-4 mr-1" />
<span>{listing.category}</span>
<span className="mx-2"></span>
<span>{listing.condition}</span>
</div>
<div className="p-4">
<h3 className="text-lg font-medium text-gray-800 leading-tight">
{listing.title}
</h3>
<span className="font-semibold text-green-600 block mt-1">
${listing.price}
</span>
<div className="flex justify-between items-center pt-2 border-t border-gray-100">
<span className="text-xs text-gray-500">
{listing.datePosted}
</span>
<span className="text-sm font-medium text-gray-700">
{listing.seller}
</span>
<div className="flex items-center text-sm text-gray-500 mt-2">
<Tag className="h-4 w-4 mr-1" />
<span>{listing.category}</span>
<span className="mx-2"></span>
<span>{listing.condition}</span>
</div>
<div className="flex justify-between items-center pt-2 border-t border-gray-100 mt-3">
<span className="text-xs text-gray-500">
{listing.datePosted}
</span>
<span className="text-sm font-medium text-gray-700">
{listing.seller}
</span>
</div>
</div>
</div>
</Link>
))}
</Link>
))}
</div>
{/* Right Button - Overlaid on products */}
<button
onClick={() =>
document
.getElementById("listingsContainer")
.scrollBy({ left: 400, behavior: "smooth" })
}
className="absolute right-0 top-1/2 transform -translate-y-1/2 bg-gray-800 bg-opacity-70 text-white p-4 rounded-full z-20 hidden md:flex items-center justify-center w-12 h-12"
>
</button>
</div>
</div>
</div>

View File

@@ -133,11 +133,11 @@ const ProductDetail = () => {
<div className="max-w-6xl mx-auto px-4 py-8">
<div className="mb-6">
<Link
to="/"
to="/search"
className="flex items-center text-green-600 hover:text-green-700"
>
<ArrowLeft className="h-4 w-4 mr-1" />
<span>Back to listings</span>
<span>Back</span>
</Link>
</div>

View File

@@ -0,0 +1,203 @@
import React, { useState, useEffect } from "react";
import { Filter, Grid, Heart, Tag, X } from "lucide-react";
import { useLocation, Link } from "react-router-dom";
import axios from "axios";
const SearchPage = () => {
const location = useLocation();
const queryParams = new URLSearchParams(location.search);
const nameParam = queryParams.get("name") || "";
const initialSearchQuery = location.state?.query || nameParam || "";
const [products, setProducts] = useState([]);
const [filteredProducts, setFilteredProducts] = useState([]);
const [searchQuery, setSearchQuery] = useState(initialSearchQuery);
const [loading, setLoading] = useState(false);
const [error, setError] = useState(null);
const [priceRange, setPriceRange] = useState({ min: 0, max: 1000 });
const [isFilterOpen, setIsFilterOpen] = useState(false);
useEffect(() => {
fetchProducts(initialSearchQuery);
}, [initialSearchQuery]);
const fetchProducts = async (query) => {
setLoading(true);
setError(null);
try {
const response = await axios.get(
`http://localhost:3030/api/search_products/search`,
{
params: { name: query },
},
);
if (response.data.success) {
const transformedProducts = response.data.data.map((product) => ({
id: product.ProductID,
title: product.Name,
description: product.Description || "",
price: product.Price || 0,
category: product.Category || "Uncategorized",
condition: product.Condition || "Used",
image: product.images,
seller: product.SellerName || "Unknown Seller",
isFavorite: false,
}));
setProducts(transformedProducts);
setFilteredProducts(transformedProducts);
} else {
setError(response.data.message || "Failed to fetch products");
setProducts([]);
setFilteredProducts([]);
}
} catch (err) {
console.error("Error fetching products:", err);
setError(err.response?.data?.message || "Error connecting to the server");
setProducts([]);
setFilteredProducts([]);
} finally {
setLoading(false);
}
};
const toggleFavorite = (id, e) => {
e.preventDefault();
setProducts((prev) =>
prev.map((product) =>
product.id === id
? { ...product, isFavorite: !product.isFavorite }
: product,
),
);
};
const filterProducts = () => {
let result = products;
result = result.filter(
(product) =>
product.price >= priceRange.min && product.price <= priceRange.max,
);
setFilteredProducts(result);
};
const applyFilters = () => {
filterProducts();
setIsFilterOpen(false);
};
const resetFilters = () => {
setPriceRange({ min: 0, max: 1000 });
setFilteredProducts(products);
};
return (
<div className="container mx-auto px-4 py-8">
<div className="flex flex-col md:flex-row gap-6">
<div
className={`
fixed inset-0 z-50 bg-white transform transition-transform duration-300
${isFilterOpen ? "translate-x-0" : "translate-x-full"}
md:translate-x-0 md:relative md:block md:w-72
overflow-y-auto shadow-lg rounded-lg
`}
>
<div className="md:hidden flex justify-between items-center p-4 border-b">
<h3 className="text-lg font-semibold">Filters</h3>
<button onClick={() => setIsFilterOpen(false)}>
<X className="text-gray-600" />
</button>
</div>
<div className="p-4 space-y-4">
<div className="bg-gray-50 rounded-lg p-3">
<h3 className="font-semibold text-gray-700 mb-3">Price Range</h3>
<div className="space-y-2">
<div className="flex space-x-2">
<input
type="number"
placeholder="Min"
value={priceRange.min}
onChange={(e) =>
setPriceRange((prev) => ({
...prev,
min: Number(e.target.value),
}))
}
className="w-full p-2 border rounded text-gray-700"
/>
<input
type="number"
placeholder="Max"
value={priceRange.max}
onChange={(e) =>
setPriceRange((prev) => ({
...prev,
max: Number(e.target.value),
}))
}
className="w-full p-2 border rounded text-gray-700"
/>
</div>
</div>
</div>
<div className="flex space-x-2">
<button
onClick={applyFilters}
className="w-full bg-green-500 text-white p-3 rounded-lg hover:bg-green-600 transition-colors"
>
Apply Filters
</button>
<button
onClick={resetFilters}
className="w-full bg-gray-200 text-gray-700 p-3 rounded-lg hover:bg-gray-300 transition-colors"
>
Reset
</button>
</div>
</div>
</div>
<div className="flex-1 mt-4 md:mt-0">
<h2 className="text-2xl font-bold text-gray-800">
{filteredProducts.length} Results
{searchQuery && (
<span className="text-lg font-normal text-gray-600">
{" "}
for "{searchQuery}"
</span>
)}
</h2>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4 mt-4">
{filteredProducts.map((listing) => (
<Link
key={listing.id}
to={`/product/${listing.id}`}
className="bg-white border border-gray-200 rounded-lg hover:shadow-md transition-shadow block"
>
<img
src={listing.image}
alt={listing.title}
className="w-full h-48 object-cover rounded-t-lg"
/>
<div className="p-4">
<h3 className="text-lg font-medium text-gray-800">
{listing.title}
</h3>
<p className="text-green-600 font-semibold">
${Number(listing.price).toFixed(2)}
</p>
</div>
</Link>
))}
</div>
</div>
</div>
</div>
);
};
export default SearchPage;