Amber Shopify Project created using ReactJS+React-Redux with GraphQL API integration. Storefront Shopify API: https://github.com/Shopify/shopify-app-js/tree/main/packages/api-clients/storefront-api-client#readme
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ProductList.jsx 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. import { useEffect, useState } from "react";
  2. import {
  3. Box,
  4. Typography,
  5. Button,
  6. FormControl,
  7. Select,
  8. MenuItem,
  9. InputBase,
  10. } from "@mui/material";
  11. import Grid from "@mui/material/Grid2";
  12. import { styled } from "@mui/material";
  13. //REDUX
  14. import { useSelector, useDispatch } from "react-redux";
  15. import { fetchProducts } from "../../redux/slices/productSlice";
  16. import { useNavigate } from "react-router-dom";
  17. //UTIL FUNCTION
  18. function getAllTags(data) {
  19. const products = data || [];
  20. const allTags = products.flatMap((product) => product.tags);
  21. const uniqueTags = [...new Set(allTags)];
  22. return uniqueTags;
  23. }
  24. function getAllCollection(data) {
  25. const products = data || [];
  26. const allCollection = products.flatMap((product) => product.collections);
  27. const uniqueCollection = Array.from(
  28. new Map(allCollection.map((item) => [item?.title, item])).values()
  29. );
  30. return uniqueCollection;
  31. }
  32. const BootstrapInput = styled(InputBase)(({ theme }) => ({
  33. "label + &": {
  34. marginTop: theme.spacing(3),
  35. },
  36. "& .MuiInputBase-input": {
  37. position: "relative",
  38. backgroundColor: "#2E2E2E",
  39. border: "1px solid #ced4da",
  40. color: "#FFF",
  41. fontSize: 13,
  42. padding: "5px 0",
  43. paddingRight: "50px !important",
  44. paddingLeft: "10px",
  45. transition: theme.transitions.create(["border-color", "box-shadow"]),
  46. "&:focus": {
  47. borderRadius: 4,
  48. borderColor: "#80bdff",
  49. boxShadow: "0 0 0 0.2rem rgba(0,123,255,.25)",
  50. },
  51. },
  52. "& .MuiSvgIcon-root": {
  53. color: "#FFF !important",
  54. },
  55. }));
  56. const ProductList = ({ size = 99999 }) => {
  57. const products = useSelector((state) => state.products.products.data); // only used as referenced
  58. const [filteredProducts, setFilteredProducts] = useState([]); // this one is the actual data to be rendered
  59. const [tagFilterOption, setTagFilterOption] = useState([]);
  60. const [collectionFilterOption, setCollectionFilterOption] = useState([]);
  61. const dispatch = useDispatch();
  62. const navigate = useNavigate();
  63. //filter
  64. const [tags, setTags] = useState("all");
  65. const [collection, setCollection] = useState("all");
  66. const [sort, setSort] = useState("new");
  67. useEffect(() => {
  68. dispatch(fetchProducts());
  69. }, []);
  70. useEffect(() => {
  71. console.log("Products: ", products);
  72. if (products.length > 0) {
  73. let newFilteredProducts = filterProducts();
  74. setFilteredProducts([]);
  75. setTimeout(() => {
  76. setFilteredProducts(newFilteredProducts);
  77. }, 100);
  78. const tagList = getAllTags(newFilteredProducts);
  79. setTagFilterOption(tagList);
  80. // Filter will only exist if the user haven't click on collection
  81. if (!sessionStorage.getItem("amber-select-collection")) {
  82. const collectionList = getAllCollection(newFilteredProducts);
  83. setCollectionFilterOption(collectionList);
  84. } else {
  85. const selectedColletion = JSON.parse(
  86. sessionStorage.getItem("amber-select-collection")
  87. );
  88. setCollection(selectedColletion?.title);
  89. }
  90. }
  91. }, [products]);
  92. useEffect(() => {
  93. let newFilteredProducts = filterProducts();
  94. setFilteredProducts([]);
  95. setTimeout(() => {
  96. setFilteredProducts(newFilteredProducts);
  97. }, 100);
  98. }, [tags, collection, sort]);
  99. const filterProducts = () => {
  100. if (products?.length > 0) {
  101. let productType = sessionStorage.getItem("amber-select-product-type");
  102. let newFilteredProducts = products.filter(
  103. (product) => product.productType === productType
  104. );
  105. // Tags
  106. newFilteredProducts = newFilteredProducts.filter((product) => {
  107. if (tags == "all") {
  108. return product.productType === productType;
  109. } else {
  110. return (
  111. product.productType === productType && product.tags.includes(tags)
  112. );
  113. }
  114. });
  115. // Collection
  116. newFilteredProducts = newFilteredProducts.filter((product) => {
  117. if (collection == "all") {
  118. return product.productType === productType;
  119. } else {
  120. return (
  121. product.productType === productType &&
  122. product.collections.some((data) => data?.title === collection)
  123. );
  124. }
  125. });
  126. if (sort === "title") {
  127. newFilteredProducts = newFilteredProducts.sort((a, b) =>
  128. a.title.localeCompare(b.title)
  129. );
  130. } else if (sort === "new") {
  131. newFilteredProducts = newFilteredProducts.sort(
  132. (a, b) => new Date(b.createdAt) - new Date(a.createdAt)
  133. );
  134. } else if (sort === "old") {
  135. newFilteredProducts = newFilteredProducts.sort(
  136. (a, b) => new Date(a.createdAt) - new Date(b.createdAt)
  137. );
  138. }
  139. return newFilteredProducts;
  140. }
  141. return [];
  142. };
  143. const handleChange = (event) => {
  144. //setInput({ ...input, [event.target.name]: event.target.value });
  145. };
  146. const renderProduct = (
  147. handle,
  148. img_url,
  149. title,
  150. collection_name,
  151. minPrice,
  152. minPriceCurrency,
  153. maxPrice,
  154. maxPriceCurrency,
  155. extra_desc,
  156. selected = false
  157. ) => {
  158. return (
  159. <Grid
  160. className="animate__animated animate__fadeIn"
  161. item
  162. size={{ xs: 6, sm: 6, md: 3 }}
  163. >
  164. <Box
  165. sx={{
  166. overflow: "hidden",
  167. position: "relative",
  168. cursor: "pointer",
  169. }}
  170. onClick={()=>{
  171. navigate(`/products/${handle}`)
  172. }}
  173. >
  174. <img
  175. src={img_url}
  176. alt={title}
  177. style={{
  178. width: "100%",
  179. aspectRatio: "3 / 4",
  180. objectFit: "cover",
  181. objectPosition: "top center",
  182. }}
  183. />
  184. {selected && (
  185. <Button
  186. sx={{
  187. position: "absolute",
  188. top: {
  189. xs: 0,
  190. sm: 0,
  191. md: 10,
  192. lg: 20,
  193. },
  194. left: {
  195. xs: 0,
  196. sm: 0,
  197. md: 10,
  198. lg: 20,
  199. },
  200. boxShadow: 0,
  201. fontSize: 10,
  202. }}
  203. variant="contained"
  204. >
  205. NEW
  206. </Button>
  207. )}
  208. <Box sx={{ pb: 3, pt: 1, width: "90%" }}>
  209. <Typography
  210. variant="body2"
  211. sx={{
  212. fontWeight: "100",
  213. fontSize: {
  214. xs: "0.875rem",
  215. sm: "0.875rem",
  216. md: "1.1rem",
  217. },
  218. }}
  219. >
  220. {collection_name}
  221. </Typography>
  222. <Typography
  223. variant="body2"
  224. sx={{
  225. fontWeight: "400",
  226. fontSize: {
  227. xs: "0.875rem",
  228. sm: "0.875rem",
  229. md: "1.1rem",
  230. },
  231. }}
  232. >
  233. {title}
  234. </Typography>
  235. <Typography
  236. variant="body2"
  237. sx={{
  238. fontWeight: "100",
  239. fontSize: {
  240. xs: "0.875rem",
  241. sm: "0.875rem",
  242. md: "1.1rem",
  243. },
  244. }}
  245. >
  246. {`${minPriceCurrency} ${parseFloat(minPrice).toFixed(2)}`}
  247. </Typography>
  248. </Box>
  249. </Box>
  250. </Grid>
  251. );
  252. };
  253. return (
  254. <>
  255. {/* FILTER */}
  256. <Box
  257. sx={{
  258. display: "flex",
  259. justifyContent: "space-between",
  260. py: 0,
  261. flexDirection: {
  262. xs: "column",
  263. sm: "row",
  264. md: "row",
  265. lg: "row",
  266. },
  267. alignItems: "center",
  268. backgroundColor: "background.black",
  269. color: "white",
  270. px: 2, // Add padding around the box
  271. my: 4,
  272. }}
  273. >
  274. {/* Left Side: Page Title */}
  275. <Typography
  276. variant="body2"
  277. sx={{ fontSize: 10, mt: { xs: 1, sm: 1, md: 0 } }}
  278. >
  279. {`${filteredProducts.length} Item`}
  280. </Typography>
  281. {/* Right Side: Option Inputs */}
  282. <Box
  283. sx={{
  284. display: "flex",
  285. gap: 2,
  286. flexDirection: "row",
  287. flexWrap: "wrap",
  288. justifyContent: "space-between",
  289. py: {
  290. xs: 2,
  291. sm: 2,
  292. md: 0,
  293. },
  294. }}
  295. >
  296. {tagFilterOption.length > 0 && (
  297. <FormControl
  298. sx={{ m: 1, display: "flex", flexDirection: "row" }}
  299. variant="standard"
  300. >
  301. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>
  302. Tag
  303. </Typography>
  304. <Select
  305. value={tags}
  306. onChange={(event) => {
  307. setTags(event.target.value);
  308. }}
  309. sx={{
  310. "& .MuiSelect-select": {
  311. border: "none",
  312. },
  313. }}
  314. input={<BootstrapInput />}
  315. name="type"
  316. >
  317. <MenuItem value={"all"}>All</MenuItem>
  318. {tagFilterOption?.map((data) => (
  319. <MenuItem value={data}>{data}</MenuItem>
  320. ))}
  321. </Select>
  322. </FormControl>
  323. )}
  324. {collectionFilterOption.length > 0 && (
  325. <FormControl
  326. sx={{ m: 1, display: "flex", flexDirection: "row" }}
  327. variant="standard"
  328. >
  329. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>
  330. Collection
  331. </Typography>
  332. <Select
  333. value={collection}
  334. onChange={(event) => {
  335. setCollection(event.target.value);
  336. }}
  337. sx={{
  338. "& .MuiSelect-select": {
  339. border: "none",
  340. },
  341. }}
  342. input={<BootstrapInput />}
  343. name="type"
  344. >
  345. <MenuItem value={"all"}>All</MenuItem>
  346. {collectionFilterOption?.map(({ title }) => (
  347. <MenuItem value={title}>{title}</MenuItem>
  348. ))}
  349. </Select>
  350. </FormControl>
  351. )}
  352. <FormControl
  353. sx={{ m: 1, display: "flex", flexDirection: "row" }}
  354. variant="standard"
  355. >
  356. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>
  357. Sort
  358. </Typography>
  359. <Select
  360. value={sort}
  361. onChange={(event) => {
  362. setSort(event.target.value);
  363. }}
  364. sx={{
  365. "& .MuiSelect-select": {
  366. border: "none",
  367. },
  368. }}
  369. input={<BootstrapInput />}
  370. name="sort"
  371. >
  372. <MenuItem defaultValue value={"title"}>
  373. Title
  374. </MenuItem>
  375. <MenuItem defaultValue value={"new"}>
  376. Newest
  377. </MenuItem>
  378. <MenuItem defaultValue value={"old"}>
  379. Oldest
  380. </MenuItem>
  381. </Select>
  382. </FormControl>
  383. </Box>
  384. </Box>
  385. {/* LIST */}
  386. <Box sx={{ mb: 5 }}>
  387. <Grid container spacing={0.5} columns={12}>
  388. {filteredProducts.map((product, index) => {
  389. let {
  390. handle,
  391. title,
  392. images,
  393. collections,
  394. minVariantPrice,
  395. maxVariantPrice,
  396. productType,
  397. variants,
  398. selected,
  399. } = product;
  400. let minPrice = minVariantPrice.amount;
  401. let minPriceCurrency = minVariantPrice.currencyCode;
  402. let maxPrice = maxVariantPrice.amount;
  403. let maxPriceCurrency = maxVariantPrice.currencyCode;
  404. let img_url = images[0]?.url;
  405. let collection_name = collections[0]?.title;
  406. if (index < size) {
  407. return renderProduct(
  408. handle,
  409. img_url,
  410. title,
  411. collection_name,
  412. minPrice,
  413. minPriceCurrency,
  414. maxPrice,
  415. maxPriceCurrency,
  416. "",
  417. selected
  418. );
  419. }
  420. })}
  421. </Grid>
  422. </Box>
  423. </>
  424. );
  425. };
  426. export default ProductList;