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.

ProductSelected.jsx 5.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. import { useEffect, useState, useRef } from 'react';
  2. import { Box, Typography, Button, useMediaQuery } from '@mui/material';
  3. import Grid from '@mui/material/Grid2';
  4. import { useSelector, useDispatch } from 'react-redux';
  5. import { fetchProducts } from '../../redux/slices/productSlice';
  6. import { Swiper, SwiperSlide } from 'swiper/react';
  7. import { Scrollbar, A11y } from 'swiper/modules';
  8. import ChevronLeftIcon from '@mui/icons-material/ChevronLeft';
  9. import KeyboardArrowRightIcon from '@mui/icons-material/KeyboardArrowRight';
  10. const ProductSelected = () => {
  11. const swiperRef = useRef(null); // Create a ref for the Swiper instance
  12. const products = useSelector((state) => state.products.products.data)
  13. const [filterProducts, setFilterProducts] = useState([])
  14. useEffect(() => {
  15. if (products.length > 0) {
  16. let selectedProducts = products.filter(({ selected }) => selected) || []
  17. //sort
  18. selectedProducts = selectedProducts.sort((a, b) => new Date(b.createdAt) - new Date(a.createdAt));
  19. setFilterProducts(selectedProducts)
  20. }
  21. }, [products])
  22. const swipeToLeft = () => {
  23. if (swiperRef.current) {
  24. swiperRef.current.slidePrev(); // Move to the previous slide
  25. }
  26. };
  27. const swipeToRight = () => {
  28. if (swiperRef.current) {
  29. swiperRef.current.slideNext(); // Move to the previous slide
  30. }
  31. };
  32. const renderProduct = (handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, extra_desc) => {
  33. return (
  34. <Grid item size={{ xs: 6, sm: 6, md: 3 }}>
  35. <a
  36. href={`/products/${handle}`}
  37. style={{
  38. color: 'black',
  39. textDecoration: 'none'
  40. }}>
  41. <Box
  42. sx={{
  43. overflow: 'hidden',
  44. position: 'relative',
  45. cursor: 'pointer',
  46. }}
  47. >
  48. <img
  49. src={img_url}
  50. alt={title}
  51. style={{
  52. width: '100%',
  53. aspectRatio: '3 / 4',
  54. objectFit: 'cover',
  55. objectPosition: 'top center'
  56. }}
  57. />
  58. <Button sx={{
  59. position: "absolute",
  60. top:{
  61. xs:0,
  62. sm:0,
  63. md:20
  64. },
  65. left: {
  66. xs:0,
  67. sm:0,
  68. md:20
  69. },
  70. boxShadow: 0,
  71. fontSize:10
  72. }} variant="contained">
  73. NEW
  74. </Button>
  75. <Box sx={{ pt: 3 }}>
  76. <Typography variant="body1" sx={{ fontWeight: "400", mb: 1 }}>
  77. {collection_name}
  78. </Typography>
  79. <Typography variant="h6" sx={{ fontWeight: "bolder", mb: 1 }}>
  80. {title}
  81. </Typography>
  82. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  83. {`${minPriceCurrency} ${parseFloat(minPrice).toFixed(2)}`}
  84. </Typography>
  85. <Typography variant="body1" sx={{ mt: 2 }}>
  86. {extra_desc}
  87. </Typography>
  88. </Box>
  89. </Box>
  90. </a>
  91. </Grid>
  92. )
  93. }
  94. return (
  95. <Box sx={{
  96. mb: 5,
  97. }}>
  98. {/* OFFSET PURPOSE */}
  99. <Grid size={{ xs: 12, md: 4 }} sx={{ mx: "auto", display: "flex", alignItems: "center", justifyContent: "center", my: 4 }}>
  100. <ChevronLeftIcon
  101. style={{
  102. cursor: "pointer",
  103. color: "#B7B7B7"
  104. }}
  105. onClick={swipeToLeft}
  106. />
  107. <Typography variant="h4" sx={{ mx: 2 }}>
  108. NEW IN
  109. </Typography>
  110. <KeyboardArrowRightIcon
  111. style={{
  112. cursor: "pointer",
  113. color: "#B7B7B7"
  114. }}
  115. onClick={swipeToRight}
  116. />
  117. </Grid>
  118. <Swiper
  119. modules={[Scrollbar, A11y]}
  120. spaceBetween={20}
  121. breakpoints={{
  122. 0: { slidesPerView: 2 }, // For very small screens
  123. 640: { slidesPerView: 2 }, // For screens >= 640px
  124. 1024: { slidesPerView: 4 }, // For screens >= 1024px
  125. }}
  126. pagination={{ clickable: true }}
  127. scrollbar={{ draggable: true }}
  128. onSwiper={(swiper) => (swiperRef.current = swiper)}
  129. >
  130. {filterProducts.map((product, index) => {
  131. let { handle, title, images, collections, minVariantPrice, maxVariantPrice, productType, variants } = product
  132. let minPrice = minVariantPrice.amount
  133. let minPriceCurrency = minVariantPrice.currencyCode
  134. let maxPrice = maxVariantPrice.amount
  135. let maxPriceCurrency = maxVariantPrice.currencyCode
  136. let img_url = images[0]?.url
  137. let collection_name = collections[0]?.title
  138. return (
  139. <SwiperSlide>
  140. {renderProduct(handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, "")}
  141. </SwiperSlide>
  142. )
  143. })}
  144. </Swiper>
  145. </Box>
  146. );
  147. };
  148. export default ProductSelected;