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.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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
  59. sx={{
  60. position: "absolute",
  61. top:{
  62. xs:0,
  63. sm:0,
  64. md:10,
  65. lg: 20
  66. },
  67. left: {
  68. xs:0,
  69. sm:0,
  70. md:10,
  71. lg: 20
  72. },
  73. boxShadow: 0,
  74. fontSize: 10
  75. }}
  76. variant="contained"
  77. >
  78. NEW
  79. </Button>
  80. <Box sx={{ pt: 3, width: "80%" }}>
  81. <Typography variant="body1" sx={{ fontWeight: "400", mb: 1 }}>
  82. {collection_name}
  83. </Typography>
  84. <Typography variant="h6" sx={{ fontWeight: "bolder", mb: 1 }}>
  85. {title}
  86. </Typography>
  87. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  88. {`${minPriceCurrency} ${parseFloat(minPrice).toFixed(2)}`}
  89. </Typography>
  90. <Typography variant="body1" sx={{ mt: 2 }}>
  91. {extra_desc}
  92. </Typography>
  93. </Box>
  94. </Box>
  95. </a>
  96. </Grid>
  97. )
  98. }
  99. return (
  100. <Box sx={{
  101. mb: 5,
  102. }}>
  103. {/* OFFSET PURPOSE */}
  104. <Grid size={{ xs: 12, md: 4 }} sx={{ mx: "auto", display: "flex", alignItems: "center", justifyContent: "center", my: 4 }}>
  105. <ChevronLeftIcon
  106. style={{
  107. cursor: "pointer",
  108. color: "#B7B7B7"
  109. }}
  110. onClick={swipeToLeft}
  111. />
  112. <Typography variant="h4" sx={{ mx: 2 }}>
  113. NEW IN
  114. </Typography>
  115. <KeyboardArrowRightIcon
  116. style={{
  117. cursor: "pointer",
  118. color: "#B7B7B7"
  119. }}
  120. onClick={swipeToRight}
  121. />
  122. </Grid>
  123. <Swiper
  124. modules={[Scrollbar, A11y]}
  125. spaceBetween={20}
  126. breakpoints={{
  127. 0: { slidesPerView: 2 }, // For very small screens
  128. 640: { slidesPerView: 2 }, // For screens >= 640px
  129. 1024: { slidesPerView: 4 }, // For screens >= 1024px
  130. }}
  131. pagination={{ clickable: true }}
  132. scrollbar={{ draggable: true }}
  133. onSwiper={(swiper) => (swiperRef.current = swiper)}
  134. >
  135. {filterProducts.map((product, index) => {
  136. let { handle, title, images, collections, minVariantPrice, maxVariantPrice, productType, variants } = product
  137. let minPrice = minVariantPrice.amount
  138. let minPriceCurrency = minVariantPrice.currencyCode
  139. let maxPrice = maxVariantPrice.amount
  140. let maxPriceCurrency = maxVariantPrice.currencyCode
  141. let img_url = images[0]?.url
  142. let collection_name = collections[0]?.title
  143. return (
  144. <SwiperSlide>
  145. {renderProduct(handle, img_url, title, collection_name, minPrice, minPriceCurrency, maxPrice, maxPriceCurrency, "")}
  146. </SwiperSlide>
  147. )
  148. })}
  149. </Swiper>
  150. </Box>
  151. );
  152. };
  153. export default ProductSelected;