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 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. import { useEffect, useState } from 'react';
  2. import { Box, Typography, Button, FormControl, Select, MenuItem, InputBase } from '@mui/material';
  3. import Grid from '@mui/material/Grid2';
  4. import { styled } from "@mui/material";
  5. //REDUX
  6. import { useSelector, useDispatch } from 'react-redux';
  7. import { fetchProducts } from '../../redux/slices/productSlice';
  8. //UTIL FUNCTION
  9. function getAllTags(data) {
  10. const products = data || [];
  11. const allTags = products.flatMap(product => product.tags);
  12. const uniqueTags = [...new Set(allTags)];
  13. return uniqueTags;
  14. }
  15. function getAllCollection(data) {
  16. const products = data || [];
  17. const allCollection = products.flatMap(product => product.collections);
  18. const uniqueCollection = Array.from(
  19. new Map(allCollection.map(item => [item.title, item])).values()
  20. );
  21. return uniqueCollection;
  22. }
  23. const BootstrapInput = styled(InputBase)(({ theme }) => ({
  24. 'label + &': {
  25. marginTop: theme.spacing(3),
  26. },
  27. '& .MuiInputBase-input': {
  28. position: 'relative',
  29. backgroundColor: "#2E2E2E",
  30. border: '1px solid #ced4da',
  31. color: "#FFF",
  32. fontSize: 13,
  33. padding: '5px 0',
  34. paddingRight: '50px !important',
  35. paddingLeft: "10px",
  36. transition: theme.transitions.create(['border-color', 'box-shadow']),
  37. '&:focus': {
  38. borderRadius: 4,
  39. borderColor: '#80bdff',
  40. boxShadow: '0 0 0 0.2rem rgba(0,123,255,.25)',
  41. },
  42. },
  43. '& .MuiSvgIcon-root': {
  44. color: "#FFF !important"
  45. },
  46. }));
  47. const ProductList = ({ size = 99999 }) => {
  48. const products = useSelector((state) => state.products.products.data) // only used as referenced
  49. const [filteredProducts, setFilteredProducts] = useState([]) // this one is the actual data to be rendered
  50. const [tagFilterOption, setTagFilterOption] = useState([])
  51. const [collectionFilterOption, setCollectionFilterOption] = useState([])
  52. const dispatch = useDispatch();
  53. //filter
  54. const [tags, setTags] = useState('all');
  55. const [collection, setCollection] = useState('all');
  56. const [sort, setSort] = useState('title')
  57. useEffect(() => {
  58. dispatch(fetchProducts())
  59. }, [])
  60. useEffect(() => {
  61. console.log("Products: ", products)
  62. if (products.length > 0) {
  63. let productType = sessionStorage.getItem('amber-select-product-type')
  64. let newFilteredProducts = products.filter(
  65. (product) => product.productType === productType
  66. );
  67. setFilteredProducts(newFilteredProducts)
  68. const tagList = getAllTags(newFilteredProducts);
  69. setTagFilterOption(tagList);
  70. // Filter will only exist if the user haven't click on collection
  71. if (!sessionStorage.getItem('amber-select-collection')) {
  72. const collectionList = getAllCollection(newFilteredProducts);
  73. setCollectionFilterOption(collectionList);
  74. } else {
  75. setCollection(sessionStorage.getItem('amber-select-collection'))
  76. }
  77. }
  78. }, [products])
  79. useEffect(() => {
  80. if (products?.length > 0) {
  81. let productType = sessionStorage.getItem('amber-select-product-type')
  82. let newFilteredProducts = products.filter(
  83. (product) => product.productType === productType
  84. );
  85. // Tags
  86. newFilteredProducts = newFilteredProducts.filter(
  87. (product) => {
  88. if (tags == 'all') {
  89. return product.productType === productType
  90. } else {
  91. return product.productType === productType && product.tags.includes(tags)
  92. }
  93. }
  94. );
  95. // Collection
  96. newFilteredProducts = newFilteredProducts.filter(
  97. (product) => {
  98. if (collection == 'all') {
  99. return product.productType === productType
  100. } else {
  101. return product.productType === productType && product.collections.some(data => data.title === collection)
  102. }
  103. }
  104. );
  105. if (sort === "title") {
  106. newFilteredProducts = newFilteredProducts.sort((a, b) => a.title.localeCompare(b.title));
  107. } else if (sort === "new") {
  108. newFilteredProducts = newFilteredProducts.sort((a, b) => new Date(b.createdAt) - new Date(a.createdAt));
  109. } else if (sort === "old") {
  110. newFilteredProducts = newFilteredProducts.sort((a, b) => new Date(a.createdAt) - new Date(b.createdAt));
  111. }
  112. setFilteredProducts(newFilteredProducts)
  113. }
  114. }, [tags, collection, sort])
  115. const handleChange = (event) => {
  116. //setInput({ ...input, [event.target.name]: event.target.value });
  117. };
  118. const renderProduct = (handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, extra_desc) => {
  119. return (
  120. <Grid className="animate__animated animate__fadeIn" item size={{ xs: 6, sm: 6, md: 3 }}>
  121. <a href={`/products/${handle}`} style={{textDecoration:"none",color:"#000"}}>
  122. <Box
  123. sx={{
  124. overflow: 'hidden',
  125. position: 'relative',
  126. cursor: 'pointer'
  127. }}
  128. >
  129. <img
  130. src={img_url}
  131. alt={title}
  132. style={{
  133. width: '100%',
  134. aspectRatio: '3 / 4',
  135. objectFit: 'cover',
  136. objectPosition: 'top center'
  137. }}
  138. />
  139. {/* <Button sx={{ position: "absolute", top: 20, left: 20, boxShadow: 0 }} variant="contained">
  140. NEW
  141. </Button> */}
  142. <Box sx={{ pb: 5, pt: 3, width: "80%" }}>
  143. <Typography variant="body1" sx={{ fontWeight: "400", mb: 1 }}>
  144. {collection_name}
  145. </Typography>
  146. <Typography variant="h5" sx={{ fontWeight: "bolder", mb: 1 }}>
  147. {title}
  148. </Typography>
  149. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  150. {`${minPriceCurrency} ${parseFloat(minPrice).toFixed(2)}`}
  151. </Typography>
  152. <Typography variant="body1" sx={{ mt: 2 }}>
  153. {extra_desc}
  154. </Typography>
  155. </Box>
  156. </Box>
  157. </a>
  158. </Grid>
  159. )
  160. }
  161. return (
  162. <>
  163. {/* FILTER */}
  164. <Box
  165. sx={{
  166. display: "flex",
  167. justifyContent: "space-between",
  168. py: 0,
  169. flexDirection: {
  170. xs: "column",
  171. sm: "row",
  172. md: "row",
  173. lg: "row"
  174. },
  175. alignItems: "center",
  176. backgroundColor: "background.black",
  177. color: "white",
  178. px: 2, // Add padding around the box
  179. my: 4
  180. }}
  181. >
  182. {/* Left Side: Page Title */}
  183. <Typography variant="body2" sx={{ fontSize: 10, mt: { xs: 1, sm: 1, md: 0 } }}>
  184. {`${filteredProducts.length} Item`}
  185. </Typography>
  186. {/* Right Side: Option Inputs */}
  187. <Box sx={{
  188. display: "flex", gap: 2, flexDirection: {
  189. xs: "column",
  190. sm: "row",
  191. md: "row",
  192. lg: "row"
  193. },
  194. }}>
  195. {(tagFilterOption.length > 0) && <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  196. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Tag</Typography>
  197. <Select
  198. value={tags}
  199. onChange={(event) => {
  200. setTags(event.target.value);
  201. }}
  202. sx={{
  203. '& .MuiSelect-select': {
  204. border: "none"
  205. }
  206. }}
  207. input={<BootstrapInput />}
  208. name="type"
  209. >
  210. <MenuItem value={'all'}>All</MenuItem>
  211. {tagFilterOption?.map((data) => (<MenuItem value={data}>{data}</MenuItem>))}
  212. </Select>
  213. </FormControl>}
  214. {(collectionFilterOption.length > 0) && <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  215. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Collection</Typography>
  216. <Select
  217. value={collection}
  218. onChange={(event) => {
  219. setCollection(event.target.value);
  220. }}
  221. sx={{
  222. '& .MuiSelect-select': {
  223. border: "none"
  224. }
  225. }}
  226. input={<BootstrapInput />}
  227. name="type"
  228. >
  229. <MenuItem value={'all'}>All</MenuItem>
  230. {collectionFilterOption?.map(({ title }) => (<MenuItem value={title}>{title}</MenuItem>))}
  231. </Select>
  232. </FormControl>}
  233. <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  234. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Sort</Typography>
  235. <Select
  236. value={sort}
  237. onChange={(event) => {
  238. setSort(event.target.value);
  239. }}
  240. sx={{
  241. '& .MuiSelect-select': {
  242. border: "none"
  243. }
  244. }}
  245. input={<BootstrapInput />}
  246. name="sort"
  247. >
  248. <MenuItem defaultValue value={'title'}>Title</MenuItem>
  249. <MenuItem defaultValue value={'new'}>Newest</MenuItem>
  250. <MenuItem defaultValue value={'old'}>Oldest</MenuItem>
  251. </Select>
  252. </FormControl>
  253. </Box>
  254. </Box>
  255. {/* LIST */}
  256. <Box sx={{ mb: 5 }}>
  257. <Grid container spacing={0.5} columns={12}>
  258. {filteredProducts.map((product, index) => {
  259. let { handle, title, images, collections, minVariantPrice, maxVariantPrice, productType, variants } = product
  260. let minPrice = minVariantPrice.amount
  261. let minPriceCurrency = minVariantPrice.currencyCode
  262. let maxPrice = maxVariantPrice.amount
  263. let maxPriceCurrency = maxVariantPrice.currencyCode
  264. let img_url = images[0].url
  265. let collection_name = collections[0].title
  266. if (index < size) {
  267. return renderProduct(handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, "")
  268. }
  269. })}
  270. </Grid>
  271. </Box>
  272. </>
  273. );
  274. };
  275. export default ProductList;