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
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

ImageView.jsx 6.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import { useState, useEffect } from "react";
  2. import { useSelector } from "react-redux";
  3. import {
  4. Box,
  5. Typography,
  6. IconButton,
  7. Dialog,
  8. DialogContent,
  9. } from "@mui/material";
  10. import Grid from '@mui/material/Grid2';
  11. import CloseIcon from "@mui/icons-material/Close";
  12. import ZoomInIcon from '@mui/icons-material/ZoomIn';
  13. import ZoomOutIcon from '@mui/icons-material/ZoomOut';
  14. import defaultImage from "../../assets/images/default.png";
  15. import { Carousel } from 'react-responsive-carousel';
  16. // Utility function to check if an object is empty
  17. const isEmptyObject = (obj) => Object.keys(obj).length === 0 && obj.constructor === Object;
  18. const ImageView = () => {
  19. const product = useSelector((state) => state.products.product.data)
  20. const [previewImage, setPreviewImage] = useState("");
  21. const [isZoomed, setIsZoomed] = useState(false);
  22. const [zoomSize, setZoomSize] = useState(100)
  23. const [currentIndex, setCurrentIndex] = useState(0);
  24. useEffect(() => {
  25. if (!isEmptyObject(product)) {
  26. setPreviewImage(product?.images[0]?.url)
  27. }
  28. }, [product])
  29. const handleThumbnailClick = (image) => {
  30. setPreviewImage(image);
  31. };
  32. const handleZoomToggle = () => {
  33. setIsZoomed(!isZoomed);
  34. };
  35. const handleSlideChange = (index) => {
  36. setPreviewImage(product?.images[index]?.url)
  37. };
  38. const onZoomIn = () => {
  39. if (zoomSize > 310) return
  40. setZoomSize(zoomSize + 20)
  41. }
  42. const onZoomOut = () => {
  43. if (zoomSize < 40) return
  44. setZoomSize(zoomSize - 20)
  45. }
  46. return (
  47. <Box sx={{ flexGrow: 1 }}>
  48. <Grid container spacing={2}>
  49. {/* Image Preview Section */}
  50. <Grid item size={12}>
  51. <Box
  52. sx={{
  53. overflow: "hidden",
  54. cursor: "pointer",
  55. px: {
  56. xs: 0,
  57. sm: 0,
  58. md: 15
  59. }
  60. }}
  61. >
  62. <Carousel
  63. showStatus={false}
  64. showIndicators={false}
  65. onClickItem={() => {
  66. handleZoomToggle()
  67. setZoomSize(100)
  68. }}
  69. onChange={(index) => handleSlideChange(index)}
  70. selectedItem={currentIndex}>
  71. {product?.images?.map(({ src, url }, index) => (
  72. <Box
  73. key={index}
  74. component="img"
  75. src={url}
  76. alt={`img_${index + 1}`}
  77. sx={{
  78. width: "100%",
  79. zIndex: 0,
  80. position: "relative",
  81. top: "50%",
  82. transform: "translateY(-50%)"
  83. }}
  84. />
  85. ))}
  86. </Carousel>
  87. </Box>
  88. </Grid>
  89. {/* Image Control Section */}
  90. <Grid item size={12}>
  91. <Box
  92. sx={{
  93. display: "flex",
  94. flexDirection: "row",
  95. justifyContent: {
  96. xs: "left",
  97. sm: "left",
  98. md: "center"
  99. },
  100. overflowX: "scroll",
  101. overflowY: "none",
  102. padding: 2,
  103. gap: 2
  104. }}
  105. >
  106. {product?.images?.map(({ src, url }, index) => (
  107. <IconButton
  108. key={index}
  109. onClick={() => {
  110. handleThumbnailClick(url)
  111. setCurrentIndex(index)
  112. }}
  113. sx={{
  114. p: 0.4,
  115. borderRadius: 1,
  116. backgroundColor: (previewImage == url) ? "#95AAC5" : "none"
  117. }}
  118. >
  119. <img
  120. src={url}
  121. alt={`img_${index + 1}`}
  122. style={{
  123. width: "100%",
  124. height: "auto",
  125. maxWidth: "75px",
  126. }}
  127. />
  128. </IconButton>
  129. ))}
  130. </Box>
  131. </Grid>
  132. </Grid>
  133. {/* Zoom Dialog */}
  134. <Dialog fullScreen open={isZoomed} onClose={handleZoomToggle} maxWidth="lg">
  135. <DialogContent
  136. sx={{
  137. position: "relative",
  138. overflow: "hidden", // Prevent scrolling on DialogContent
  139. height: "90vh", // Adjust height as needed
  140. p: 0
  141. }}
  142. >
  143. {/* Fixed position close button */}
  144. <Box
  145. sx={{
  146. width: 50,
  147. height: 50,
  148. position: "absolute",
  149. top: 10,
  150. right: {
  151. xs: 1.5,
  152. sm: 1.5,
  153. md: 10
  154. },
  155. zIndex: 2, // Ensure it stays above other content
  156. }}
  157. >
  158. <IconButton
  159. onClick={handleZoomToggle}
  160. sx={{
  161. backgroundColor: "white",
  162. color: "black",
  163. "&:hover": {
  164. backgroundColor: "lightgray",
  165. },
  166. }}
  167. >
  168. <CloseIcon />
  169. </IconButton>
  170. <IconButton
  171. onClick={onZoomIn}
  172. sx={{
  173. backgroundColor: "white",
  174. color: "black",
  175. mt: 1,
  176. "&:hover": {
  177. backgroundColor: "lightgray",
  178. },
  179. }}
  180. >
  181. <ZoomInIcon />
  182. </IconButton>
  183. <IconButton
  184. onClick={onZoomOut}
  185. sx={{
  186. backgroundColor: "white",
  187. color: "black",
  188. mt: 1,
  189. "&:hover": {
  190. backgroundColor: "lightgray",
  191. },
  192. }}
  193. >
  194. <ZoomOutIcon />
  195. </IconButton>
  196. </Box>
  197. {/* Scrollable content */}
  198. <Box
  199. sx={{
  200. overflow: "auto", // Allow scrolling within this Box
  201. height: "100%", // Fill available height of DialogContent
  202. display: "flex"
  203. }}
  204. >
  205. <img
  206. src={previewImage || defaultImage}
  207. alt="Zoomed Preview"
  208. style={{
  209. width: `${zoomSize}%`,
  210. margin: "auto auto",
  211. height: "auto",
  212. }}
  213. />
  214. </Box>
  215. </DialogContent>
  216. </Dialog>
  217. </Box>
  218. );
  219. };
  220. export default ImageView;