我目前正在使用react-router
制作一个产品过滤器网站。我可以得到searchParams
,例如“color:black”,但只要我选择一个不同的颜色,它就会替换当前选择的值,但我想设置多个值示例:黑色、灰色等。
我尝试了searcgParams.getAll
,但无法让它工作。
import { useState } from "react";
import { Link, useSearchParams, useLoaderData } from "react-router-dom";
import FilterNav from "../../components/FilterNav";
import { getProducts } from "../../api";
export function loader() {
return getProducts();
}
export default function Products() {
const [searchParams, setSearchParams] = useSearchParams();
// const [selectedCategory, setSelectedCategory] = useState(
// searchParams.get("category")
// );
// const [selectedBrand, setSelectedBrand] = useState(searchParams.get("brand"));
// const [selectedColor, setSelectedColor] = useState(searchParams.get("color"));
const [error, setError] = useState(null);
const products = useLoaderData();
const uniqueCategories = [
...new Set(products.map((product) => product.category)),
];
const uniqueBrands = [...new Set(products.map((product) => product.brand))];
const uniqueColors = [...new Set(products.map((product) => product.color))];
const uniquePrices = [...new Set(products.map((product) => product.price))];
const selectedCategory = searchParams.get("category");
const selectedBrand = searchParams.get("brand");
const selectedColor = searchParams.get("color");
function handleFilterChange(key, value) {
setSearchParams((prevParams) => {
let values = prevParams.get(key)?.split(",");
if (values) {
// existing values, add/remove specific values
if (values.includes(value)) {
// remove value from array
values = values.filter((currentValue) => currentValue !== value);
} else {
// append value to array
values.push(value);
}
if (!!values.length) {
// set new key-value if values is still populated
prevParams.set(key, values);
} else {
// delete key if values array is empty
prevParams.delete(key);
}
} else {
// no values for key, create new array with value
prevParams.set(key, [value]);
}
if (value === null) {
prevParams.delete(key);
}
return prevParams;
});
}
// function handleFilterChange(key, value) {
// setSearchParams((prevParams) => {
// if (value === null) {
// prevParams.delete(key);
// } else {
// prevParams.set(key, value);
// }
// return prevParams;
// });
// }
const filteredProducts = products.filter((product) => {
const filteredBrand =
!selectedBrand || selectedBrand.includes(product.brand);
const filteredCategory =
!selectedCategory || selectedCategory.includes(product.category);
const filteredColor =
!selectedColor || selectedColor.includes(product.color);
return filteredBrand && filteredCategory && filteredColor;
});
const allProducts = filteredProducts.map((product) => (
<div key={product.id} className="product-tile">
<Link
to={product.id}
state={{
search: `?${searchParams.toString()}`,
category: selectedCategory,
}}
>
<h2>
{product.brand} {product.name}
</h2>
<img src={product.image} />
<div className="product-info">
<p>
{product.category} - {product.color} - ${product.price}{" "}
</p>
</div>
</Link>
</div>
));
if (error) {
return <h1>There was an error: {error.message}</h1>;
}
return (
<div className="product-list-container">
<FilterNav
categoryOptions={uniqueCategories}
brandOptions={uniqueBrands}
colorOptions={uniqueColors}
selectedCategory={selectedCategory}
selectedBrand={selectedBrand}
selectedColor={selectedColor}
handleFilterChange={handleFilterChange}
/>
<div className="product-list">{allProducts}</div>
</div>
);
}
import { useState } from "react";
export default function FilterNav(props) {
// const [selectedCategories, setSelectedCategories] = useState([]);
// const handleCategoryClick = (category) => {
// if (selectedCategories.includes(category)) {
// setSelectedCategories(selectedCategories.filter((c) => c !== category));
// props.handleFilterChange("category", null); // Remove the category filter
// } else {
// setSelectedCategories([...selectedCategories, category]);
// props.handleFilterChange("category", category); // Apply the category filter
// }
// };
const renderFilterList = (options, key) => {
return options.map((value, id) => {
return (
<a onClick={() => props.handleFilterChange(key, value)} key={id}>
{value}
</a>
);
});
};
return (
<>
<div className="product-list-filter-buttons">
<div className="dropdown">
<button className="dropbtn">Brands</button>
<div className="dropdown-content">
{renderFilterList(props.brandOptions, "brand")}
</div>
</div>
<div className="dropdown">
<button className="dropbtn">Categories</button>
<div className="dropdown-content">
{renderFilterList(props.categoryOptions, "category")}
</div>
</div>
<div className="dropdown">
<button className="dropbtn">Colors</button>
<div className="dropdown-content">
{renderFilterList(props.colorOptions, "color")}
</div>
</div>
{props.selectedCategory ||
props.selectedBrand ||
props.selectedColor ? (
<button
onClick={() => {
props.handleFilterChange("category", null);
props.handleFilterChange("brand", null);
props.handleFilterChange("color", null);
}}
className="product-type clear-filters"
>
Clear all filters
</button>
) : null}
</div>
<div>
{props.selectedCategory
? props.categoryOptions.map((category, index) => (
<button
key={index}
onClick={() => props.handleFilterChange("category", category)}
>
<span aria-hidden="true">× {category}</span>
</button>
))
: null}
{props.selectedBrand ? (
<button onClick={() => props.handleFilterChange("brand", null)}>
<span aria-hidden="true">× {props.selectedBrand}</span>
</button>
) : null}
{props.selectedColor ? (
<button onClick={() => props.handleFilterChange("color", null)}>
<span aria-hidden="true">× {props.selectedColor}</span>
</button>
) : null}
</div>
</>
);
}
1条答案
按热度按时间elcex8rz1#
你可能不想使用
searchParams.set
方法来***替换***现有的key-value条目,而是想使用searchParams.append
,它将为值添加多个键。示例:
然后使用
searchParams.getAll
将返回与特定键关联的值数组。例如,如果搜索字符串类似于“...?颜色=黑色+颜色=灰色”
上面的方法是一个全有或全无的解决方案,例如。使用
searchParams.delete("color")
将删除***所有***color
queryString参数。如果你想更细粒度地控制单个颜色选择,那么你需要自己手动管理。下面的示例应该接近您可能正在寻找的单个参数: