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

ProductList.jsx 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  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. setTimeout(() => {
  68. setFilteredProducts(newFilteredProducts)
  69. }, 100)
  70. const tagList = getAllTags(newFilteredProducts);
  71. setTagFilterOption(tagList);
  72. // Filter will only exist if the user haven't click on collection
  73. if (!sessionStorage.getItem('amber-select-collection')) {
  74. const collectionList = getAllCollection(newFilteredProducts);
  75. setCollectionFilterOption(collectionList);
  76. } else {
  77. const selectedColletion = JSON.parse(sessionStorage.getItem('amber-select-collection'))
  78. setCollection(selectedColletion?.title)
  79. }
  80. }
  81. }, [products])
  82. useEffect(() => {
  83. if (products?.length > 0) {
  84. let productType = sessionStorage.getItem('amber-select-product-type')
  85. let newFilteredProducts = products.filter(
  86. (product) => product.productType === productType
  87. );
  88. // Tags
  89. newFilteredProducts = newFilteredProducts.filter(
  90. (product) => {
  91. if (tags == 'all') {
  92. return product.productType === productType
  93. } else {
  94. return product.productType === productType && product.tags.includes(tags)
  95. }
  96. }
  97. );
  98. // Collection
  99. newFilteredProducts = newFilteredProducts.filter(
  100. (product) => {
  101. if (collection == 'all') {
  102. return product.productType === productType
  103. } else {
  104. return product.productType === productType && product.collections.some(data => data?.title === collection)
  105. }
  106. }
  107. );
  108. if (sort === "title") {
  109. newFilteredProducts = newFilteredProducts.sort((a, b) => a.title.localeCompare(b.title));
  110. } else if (sort === "new") {
  111. newFilteredProducts = newFilteredProducts.sort((a, b) => new Date(b.createdAt) - new Date(a.createdAt));
  112. } else if (sort === "old") {
  113. newFilteredProducts = newFilteredProducts.sort((a, b) => new Date(a.createdAt) - new Date(b.createdAt));
  114. }
  115. setFilteredProducts([]) //cause I want those fadeIn animation
  116. setTimeout(() => {
  117. setFilteredProducts(newFilteredProducts)
  118. }, 100)
  119. }
  120. }, [tags, collection, sort])
  121. const handleChange = (event) => {
  122. //setInput({ ...input, [event.target.name]: event.target.value });
  123. };
  124. const renderProduct = (handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, extra_desc, selected = false) => {
  125. return (
  126. <Grid className="animate__animated animate__fadeIn" item size={{ xs: 6, sm: 6, md: 3 }}>
  127. <a href={`/products/${handle}`} style={{ textDecoration: "none", color: "#000" }}>
  128. <Box
  129. sx={{
  130. overflow: 'hidden',
  131. position: 'relative',
  132. cursor: 'pointer'
  133. }}
  134. >
  135. <img
  136. src={img_url}
  137. alt={title}
  138. style={{
  139. width: '100%',
  140. aspectRatio: '3 / 4',
  141. objectFit: 'cover',
  142. objectPosition: 'top center'
  143. }}
  144. />
  145. {(selected) && <Button
  146. sx={{
  147. position: "absolute",
  148. top: {
  149. xs: 0,
  150. sm: 0,
  151. md: 20
  152. },
  153. left: {
  154. xs: 0,
  155. sm: 0,
  156. md: 20
  157. },
  158. boxShadow: 0,
  159. fontSize: 10
  160. }}
  161. variant="contained">
  162. NEW
  163. </Button>}
  164. <Box sx={{ pb: 5, pt: 3, width: "80%" }}>
  165. <Typography variant="body1" sx={{ fontWeight: "400", mb: 1 }}>
  166. {collection_name}
  167. </Typography>
  168. <Typography variant="h6" sx={{ fontWeight: "bolder", mb: 1 }}>
  169. {title}
  170. </Typography>
  171. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  172. {`${minPriceCurrency} ${parseFloat(minPrice).toFixed(2)}`}
  173. </Typography>
  174. <Typography variant="body1" sx={{ mt: 2 }}>
  175. {extra_desc}
  176. </Typography>
  177. </Box>
  178. </Box>
  179. </a>
  180. </Grid>
  181. )
  182. }
  183. return (
  184. <>
  185. {/* FILTER */}
  186. <Box
  187. sx={{
  188. display: "flex",
  189. justifyContent: "space-between",
  190. py: 0,
  191. flexDirection: {
  192. xs: "column",
  193. sm: "row",
  194. md: "row",
  195. lg: "row"
  196. },
  197. alignItems: "center",
  198. backgroundColor: "background.black",
  199. color: "white",
  200. px: 2, // Add padding around the box
  201. my: 4
  202. }}
  203. >
  204. {/* Left Side: Page Title */}
  205. <Typography variant="body2" sx={{ fontSize: 10, mt: { xs: 1, sm: 1, md: 0 } }}>
  206. {`${filteredProducts.length} Item`}
  207. </Typography>
  208. {/* Right Side: Option Inputs */}
  209. <Box sx={{
  210. display: "flex", gap: 2, flexDirection: {
  211. xs: "column",
  212. sm: "row",
  213. md: "row",
  214. lg: "row"
  215. },
  216. }}>
  217. {(tagFilterOption.length > 0) && <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  218. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Tag</Typography>
  219. <Select
  220. value={tags}
  221. onChange={(event) => {
  222. setTags(event.target.value);
  223. }}
  224. sx={{
  225. '& .MuiSelect-select': {
  226. border: "none"
  227. }
  228. }}
  229. input={<BootstrapInput />}
  230. name="type"
  231. >
  232. <MenuItem value={'all'}>All</MenuItem>
  233. {tagFilterOption?.map((data) => (<MenuItem value={data}>{data}</MenuItem>))}
  234. </Select>
  235. </FormControl>}
  236. {(collectionFilterOption.length > 0) && <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  237. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Collection</Typography>
  238. <Select
  239. value={collection}
  240. onChange={(event) => {
  241. setCollection(event.target.value);
  242. }}
  243. sx={{
  244. '& .MuiSelect-select': {
  245. border: "none"
  246. }
  247. }}
  248. input={<BootstrapInput />}
  249. name="type"
  250. >
  251. <MenuItem value={'all'}>All</MenuItem>
  252. {collectionFilterOption?.map(({ title }) => (<MenuItem value={title}>{title}</MenuItem>))}
  253. </Select>
  254. </FormControl>}
  255. <FormControl sx={{ m: 1, display: "flex", flexDirection: "row" }} variant="standard">
  256. <Typography variant="body2" sx={{ mr: 1, my: "auto" }}>Sort</Typography>
  257. <Select
  258. value={sort}
  259. onChange={(event) => {
  260. setSort(event.target.value);
  261. }}
  262. sx={{
  263. '& .MuiSelect-select': {
  264. border: "none"
  265. }
  266. }}
  267. input={<BootstrapInput />}
  268. name="sort"
  269. >
  270. <MenuItem defaultValue value={'title'}>Title</MenuItem>
  271. <MenuItem defaultValue value={'new'}>Newest</MenuItem>
  272. <MenuItem defaultValue value={'old'}>Oldest</MenuItem>
  273. </Select>
  274. </FormControl>
  275. </Box>
  276. </Box>
  277. {/* LIST */}
  278. <Box sx={{ mb: 5 }}>
  279. <Grid container spacing={0.5} columns={12}>
  280. {filteredProducts.map((product, index) => {
  281. let { handle, title, images, collections, minVariantPrice, maxVariantPrice, productType, variants, selected } = product
  282. let minPrice = minVariantPrice.amount
  283. let minPriceCurrency = minVariantPrice.currencyCode
  284. let maxPrice = maxVariantPrice.amount
  285. let maxPriceCurrency = maxVariantPrice.currencyCode
  286. let img_url = images[0]?.url
  287. let collection_name = collections[0]?.title
  288. if (index < size) {
  289. return renderProduct(handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, "", selected)
  290. }
  291. })}
  292. </Grid>
  293. </Box>
  294. </>
  295. );
  296. };
  297. export default ProductList;