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.

SideCart.jsx 7.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. import { useEffect, useState } from "react";
  2. import { Drawer, Box, Typography, IconButton, Button } from "@mui/material";
  3. import Grid from '@mui/material/Grid2';
  4. import CloseIcon from "@mui/icons-material/Close";
  5. import { updateItemQuantity } from "../../redux/slices/cartSlice";
  6. import AddIcon from '@mui/icons-material/Add';
  7. import RemoveIcon from '@mui/icons-material/Remove';
  8. import { useSelector, useDispatch } from 'react-redux';
  9. const SideCart = ({ open, onClose }) => {
  10. const dispatch = useDispatch()
  11. const cart = useSelector((state) => state.cart.cart)
  12. const [cartProducts, setCartProducts] = useState([]) // this is being used as view only, will not used as a way to manipulate cart in any way
  13. useEffect(() => {
  14. setCartProducts(cart?.lines?.nodes || [])
  15. }, [cart])
  16. const handleUpdateCart = ({ quantity, lineId }) => {
  17. let cartHistory = localStorage.getItem('amber-cart');
  18. cartHistory = cartHistory ? JSON.parse(cartHistory) : {};
  19. dispatch(updateItemQuantity({
  20. cartId: cartHistory.id,
  21. lineId,
  22. quantity,
  23. }))
  24. }
  25. return (
  26. <Drawer
  27. open={open}
  28. onClose={onClose}
  29. anchor="right"
  30. sx={{
  31. "& .MuiDrawer-paper": {
  32. backgroundColor: "white",
  33. color: "black",
  34. width: {
  35. xs: "90%",
  36. sm: "90%",
  37. md: "30%",
  38. },
  39. height: "97%",
  40. padding: 2,
  41. },
  42. }}
  43. >
  44. <Box sx={{ display: "flex", flexDirection: "column", height: "100%" }}>
  45. {/* Header */}
  46. <Box
  47. sx={{
  48. display: "flex",
  49. justifyContent: "space-between",
  50. alignItems: "center",
  51. marginBottom: 2,
  52. }}
  53. >
  54. <Typography variant="h6">Your Cart</Typography>
  55. <IconButton onClick={onClose}>
  56. <CloseIcon />
  57. </IconButton>
  58. </Box>
  59. <Box sx={{ flexGrow: 1 }}>
  60. {/* Product List */}
  61. {cartProducts && cartProducts?.length > 0 ? (
  62. cartProducts.map(({ id, cost, merchandise, quantity }, index) => {
  63. let { amount, currencyCode } = cost.totalAmount
  64. let { image, product, title } = merchandise
  65. // ID
  66. const parts = product?.id?.split('/');
  67. let prodID = parts[parts.length - 1];
  68. return (
  69. <>
  70. <Box
  71. key={index}
  72. sx={{
  73. display: "flex",
  74. alignItems: "center",
  75. padding: 1,
  76. borderBottom: "1px solid #eee",
  77. }}
  78. >
  79. <Grid container>
  80. {/* Product Image */}
  81. <Grid item size={2}>
  82. <Box
  83. component="img"
  84. src={merchandise?.image?.src}
  85. alt={product.name}
  86. sx={{
  87. width: "100%",
  88. height: "auto",
  89. borderRadius: 2,
  90. }}
  91. />
  92. </Grid>
  93. {/* Product Details */}
  94. <Grid item size={10} sx={{ paddingLeft: 1 }}>
  95. <Typography variant="body2" sx={{
  96. fontWeight: "400", fontSize: {
  97. xs: "0.875rem",
  98. sm: "0.875rem",
  99. md: "1.1rem",
  100. }
  101. }}>
  102. {product?.title}
  103. </Typography>
  104. <Typography variant="body2" sx={{
  105. fontWeight: "100", fontSize: {
  106. xs: "0.875rem",
  107. sm: "0.875rem",
  108. md: "1.1rem",
  109. }
  110. }}>{`VARIANT: ${title}`}</Typography>
  111. <Box sx={{ display: "flex", alignItems: "center" }}>
  112. <Typography variant="body2" sx={{
  113. fontWeight: "100", fontSize: {
  114. xs: "0.875rem",
  115. sm: "0.875rem",
  116. md: "1.1rem",
  117. }
  118. }}>{`${currencyCode} ${parseFloat(amount).toFixed(2)}`}</Typography>
  119. <Box sx={{ display: "flex", ml: "auto" }}>
  120. <IconButton onClick={() => {
  121. handleUpdateCart({ quantity: quantity - 1, lineId: id })
  122. }}>
  123. <RemoveIcon sx={{ fontSize: 16, margin: "0 15px" }} />
  124. </IconButton>
  125. <p style={{ fontSize: 20, fontWeight: "bold" }}>{quantity}</p>
  126. <IconButton onClick={() => {
  127. handleUpdateCart({ quantity: quantity + 1, lineId: id })
  128. }}>
  129. <AddIcon sx={{ fontSize: 16, margin: "0 15px" }} />
  130. </IconButton>
  131. </Box>
  132. </Box>
  133. </Grid>
  134. </Grid>
  135. </Box>
  136. </>
  137. )
  138. })
  139. ) : (
  140. <Typography variant="body2" color="text.secondary">
  141. Your cart is empty.
  142. </Typography>
  143. )}
  144. </Box>
  145. <Box sx={{ mt: "auto", display:(cartProducts.length > 0) ? "block" : "none" }}>
  146. {/* INVOICES */}
  147. <Grid container spacing={2}>
  148. <Grid item size={12}>
  149. <Box
  150. display="flex"
  151. justifyContent="space-between"
  152. alignItems="center"
  153. >
  154. <Typography variant="body1" sx={{ fontWeight: "400" }}>Subtotal</Typography>
  155. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  156. {`
  157. ${cart?.cost?.subtotalAmount?.currencyCode}
  158. ${parseFloat(cart?.cost?.subtotalAmount?.amount).toFixed(2)}
  159. `}
  160. </Typography>
  161. </Box>
  162. <Box
  163. display="flex"
  164. justifyContent="space-between"
  165. alignItems="center"
  166. >
  167. <Typography variant="body1" sx={{ fontWeight: "400" }}>Taxes</Typography>
  168. <Typography variant="body1" sx={{ fontWeight: "400" }}>
  169. {`
  170. ${cart?.cost?.totalTaxAmount?.currencyCode}
  171. ${parseFloat(cart?.cost?.totalTaxAmount?.amount).toFixed(2)}
  172. `}
  173. </Typography>
  174. </Box>
  175. <Box
  176. display="flex"
  177. justifyContent="space-between"
  178. alignItems="center"
  179. >
  180. <Typography variant="body1" sx={{ fontWeight: "400" }}>Total</Typography>
  181. <Typography variant="body1" sx={{ fontWeight: "bold" }}>
  182. {`
  183. ${cart?.cost?.totalAmount?.currencyCode}
  184. ${parseFloat(cart?.cost?.totalAmount?.amount).toFixed(2)}
  185. `}
  186. </Typography>
  187. </Box>
  188. </Grid>
  189. <Grid item size={12} sx={{ textAlign: "center" }}>
  190. <Button
  191. variant="contained"
  192. color="primary"
  193. onClick={() => {
  194. window.location.href = '/cart'
  195. }}>
  196. CHECKOUT NOW
  197. </Button>
  198. </Grid>
  199. </Grid>
  200. </Box>
  201. </Box>
  202. </Drawer>
  203. );
  204. };
  205. export default SideCart;