@hunterchen/canvas 0.8.0 → 0.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/dist/components/canvas/backgrounds.js +67 -38
  2. package/dist/components/canvas/backgrounds.js.map +1 -1
  3. package/dist/components/canvas/canvas.d.ts +2 -0
  4. package/dist/components/canvas/canvas.d.ts.map +1 -1
  5. package/dist/components/canvas/canvas.js +459 -387
  6. package/dist/components/canvas/canvas.js.map +1 -1
  7. package/dist/components/canvas/component.js +108 -174
  8. package/dist/components/canvas/component.js.map +1 -1
  9. package/dist/components/canvas/draggable.js +168 -151
  10. package/dist/components/canvas/draggable.js.map +1 -1
  11. package/dist/components/canvas/navbar/index.js +164 -142
  12. package/dist/components/canvas/navbar/index.js.map +1 -1
  13. package/dist/components/canvas/navbar/single-button.js +176 -149
  14. package/dist/components/canvas/navbar/single-button.js.map +1 -1
  15. package/dist/components/canvas/toolbar.js +121 -82
  16. package/dist/components/canvas/toolbar.js.map +1 -1
  17. package/dist/components/canvas/wrapper.js +127 -99
  18. package/dist/components/canvas/wrapper.js.map +1 -1
  19. package/dist/contexts/CanvasContext.js +25 -17
  20. package/dist/contexts/CanvasContext.js.map +1 -1
  21. package/dist/contexts/PerformanceContext.js +51 -50
  22. package/dist/contexts/PerformanceContext.js.map +1 -1
  23. package/dist/hooks/usePerformanceMode.js +36 -37
  24. package/dist/hooks/usePerformanceMode.js.map +1 -1
  25. package/dist/hooks/useWindowDimensions.js +22 -18
  26. package/dist/hooks/useWindowDimensions.js.map +1 -1
  27. package/dist/index.d.ts +1 -1
  28. package/dist/index.d.ts.map +1 -1
  29. package/dist/index.js +17 -21
  30. package/dist/lib/canvas.js +65 -72
  31. package/dist/lib/canvas.js.map +1 -1
  32. package/dist/lib/constants.js +78 -92
  33. package/dist/lib/constants.js.map +1 -1
  34. package/dist/lib/utils.js +10 -5
  35. package/dist/lib/utils.js.map +1 -1
  36. package/dist/types/index.d.ts +22 -0
  37. package/dist/types/index.d.ts.map +1 -1
  38. package/dist/utils/performance.js +18 -23
  39. package/dist/utils/performance.js.map +1 -1
  40. package/package.json +7 -21
  41. package/src/components/canvas/canvas.tsx +16 -4
  42. package/src/index.ts +1 -0
  43. package/src/types/index.ts +24 -0
  44. package/dist/components/canvas/offest.js +0 -12
  45. package/dist/components/canvas/offest.js.map +0 -1
  46. package/dist/index.js.map +0 -1
  47. package/dist/types/index.js +0 -6
  48. package/dist/types/index.js.map +0 -1
@@ -1,163 +1,180 @@
1
- import { jsx as _jsx } from "react/jsx-runtime";
2
- import { useRef, useEffect, forwardRef, useState, useCallback, } from "react";
3
- import { animate, motion, useAnimationControls, useMotionValue, } from "framer-motion";
4
- import { useCanvasContext } from "../../contexts/CanvasContext";
5
- const defaultPos = { x: 0, y: 0 };
6
- export const Draggable = forwardRef((props, ref) => {
7
- const { initialPos: passedPos, children, style, shouldStopPropagation = () => true, ...restProps } = props;
8
- const { scale: parentZoom, isResetting, maxZIndex, setMaxZIndex, } = useCanvasContext();
9
- const initialPos = passedPos ?? defaultPos;
10
- const x = useMotionValue(initialPos.x);
11
- const y = useMotionValue(initialPos.y);
12
- const logicalPositionRef = useRef({ ...initialPos });
13
- const controls = useAnimationControls();
14
- const [zIndex, setZIndex] = useState(1);
15
- useEffect(() => {
16
- if (isResetting) {
17
- logicalPositionRef.current = { ...initialPos };
18
- void animate(x, initialPos.x, {
19
- duration: 0.3,
20
- type: "spring",
21
- damping: 14,
22
- stiffness: 120,
23
- mass: 1,
24
- });
25
- void animate(y, initialPos.y, {
26
- duration: 0.3,
27
- type: "spring",
28
- damping: 14,
29
- stiffness: 120,
30
- mass: 1,
31
- });
32
- }
33
- }, [initialPos, controls, isResetting, x, y]);
34
- const handleDrag = (_event, info) => {
35
- controls.stop();
36
- const deltaParentX = info.delta.x / parentZoom.get();
37
- const deltaParentY = info.delta.y / parentZoom.get();
38
- logicalPositionRef.current.x += deltaParentX;
39
- logicalPositionRef.current.y += deltaParentY;
40
- x.set(logicalPositionRef.current.x);
41
- y.set(logicalPositionRef.current.y);
42
- if (zIndex < maxZIndex) {
43
- setZIndex(maxZIndex + 1);
44
- setMaxZIndex(maxZIndex + 1);
45
- }
46
- };
47
- return (_jsx(motion.div, { ref: ref, dragMomentum: false, drag: true, animate: controls, onDrag: handleDrag, style: {
48
- ...style,
49
- x,
50
- y,
51
- zIndex,
52
- }, initial: {
53
- scale: 1,
54
- filter: "drop-shadow(0 0px 0px rgba(0, 0, 0, 0)) brightness(1)",
55
- position: "relative",
56
- }, onPointerDown: (e) => {
57
- if (shouldStopPropagation?.(e)) {
58
- e.stopPropagation();
59
- }
60
- }, transition: {
61
- duration: 0.1,
62
- ease: "easeOut",
63
- }, ...restProps, children: children }));
1
+ import { useCanvasContext } from "../../contexts/CanvasContext.js";
2
+ import { animate, motion, useAnimationControls, useMotionValue } from "framer-motion";
3
+ import React, { forwardRef, useCallback, useEffect, useRef, useState } from "react";
4
+ import { jsx } from "react/jsx-runtime";
5
+
6
+ //#region src/components/canvas/draggable.tsx
7
+ const defaultPos = {
8
+ x: 0,
9
+ y: 0
10
+ };
11
+ const Draggable = forwardRef((props, ref) => {
12
+ const { initialPos: passedPos, children, style, shouldStopPropagation = () => true, ...restProps } = props;
13
+ const { scale: parentZoom, isResetting, maxZIndex, setMaxZIndex } = useCanvasContext();
14
+ const initialPos = passedPos ?? defaultPos;
15
+ const x = useMotionValue(initialPos.x);
16
+ const y = useMotionValue(initialPos.y);
17
+ const logicalPositionRef = useRef({ ...initialPos });
18
+ const controls = useAnimationControls();
19
+ const [zIndex, setZIndex] = useState(1);
20
+ useEffect(() => {
21
+ if (isResetting) {
22
+ logicalPositionRef.current = { ...initialPos };
23
+ animate(x, initialPos.x, {
24
+ duration: .3,
25
+ type: "spring",
26
+ damping: 14,
27
+ stiffness: 120,
28
+ mass: 1
29
+ });
30
+ animate(y, initialPos.y, {
31
+ duration: .3,
32
+ type: "spring",
33
+ damping: 14,
34
+ stiffness: 120,
35
+ mass: 1
36
+ });
37
+ }
38
+ }, [
39
+ initialPos,
40
+ controls,
41
+ isResetting,
42
+ x,
43
+ y
44
+ ]);
45
+ const handleDrag = (_event, info) => {
46
+ controls.stop();
47
+ const deltaParentX = info.delta.x / parentZoom.get();
48
+ const deltaParentY = info.delta.y / parentZoom.get();
49
+ logicalPositionRef.current.x += deltaParentX;
50
+ logicalPositionRef.current.y += deltaParentY;
51
+ x.set(logicalPositionRef.current.x);
52
+ y.set(logicalPositionRef.current.y);
53
+ if (zIndex < maxZIndex) {
54
+ setZIndex(maxZIndex + 1);
55
+ setMaxZIndex(maxZIndex + 1);
56
+ }
57
+ };
58
+ return /* @__PURE__ */ jsx(motion.div, {
59
+ ref,
60
+ dragMomentum: false,
61
+ drag: true,
62
+ animate: controls,
63
+ onDrag: handleDrag,
64
+ style: {
65
+ ...style,
66
+ x,
67
+ y,
68
+ zIndex
69
+ },
70
+ initial: {
71
+ scale: 1,
72
+ filter: "drop-shadow(0 0px 0px rgba(0, 0, 0, 0)) brightness(1)",
73
+ position: "relative"
74
+ },
75
+ onPointerDown: (e) => {
76
+ if (shouldStopPropagation?.(e)) e.stopPropagation();
77
+ },
78
+ transition: {
79
+ duration: .1,
80
+ ease: "easeOut"
81
+ },
82
+ ...restProps,
83
+ children
84
+ });
64
85
  });
65
86
  Draggable.displayName = "Draggable";
66
87
  function drawImageToCanvas(img, canvas) {
67
- const ctx = canvas.getContext("2d", { willReadFrequently: true });
68
- if (!ctx)
69
- return;
70
- ctx.clearRect(0, 0, canvas.width, canvas.height);
71
- ctx.drawImage(img, 0, 0);
88
+ const ctx = canvas.getContext("2d", { willReadFrequently: true });
89
+ if (!ctx) return;
90
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
91
+ ctx.drawImage(img, 0, 0);
72
92
  }
73
93
  function getAlphaAtCoords(clientX, clientY, canvas, img) {
74
- if (!canvas || !img)
75
- return 0;
76
- const ctx = canvas.getContext("2d");
77
- if (!ctx)
78
- return 0;
79
- const rect = img.getBoundingClientRect();
80
- const x = ((clientX - rect.left) / rect.width) * img.naturalWidth;
81
- const y = ((clientY - rect.top) / rect.height) * img.naturalHeight;
82
- const alpha = ctx.getImageData(x, y, 1, 1).data[3] ?? 0;
83
- return alpha;
94
+ if (!canvas || !img) return 0;
95
+ const ctx = canvas.getContext("2d");
96
+ if (!ctx) return 0;
97
+ const rect = img.getBoundingClientRect();
98
+ const x = (clientX - rect.left) / rect.width * img.naturalWidth;
99
+ const y = (clientY - rect.top) / rect.height * img.naturalHeight;
100
+ return ctx.getImageData(x, y, 1, 1).data[3] ?? 0;
84
101
  }
85
102
  function isMouseOverImage(clientX, clientY, img) {
86
- if (!img)
87
- return false;
88
- const rect = img.getBoundingClientRect();
89
- return (clientX >= rect.left &&
90
- clientX <= rect.right &&
91
- clientY >= rect.top &&
92
- clientY <= rect.bottom);
103
+ if (!img) return false;
104
+ const rect = img.getBoundingClientRect();
105
+ return clientX >= rect.left && clientX <= rect.right && clientY >= rect.top && clientY <= rect.bottom;
93
106
  }
94
107
  function updateCursor(opaque, isMouseDown, img) {
95
- let cursor = "url('customcursor.svg'), auto"; // default
96
- if (opaque)
97
- cursor = "grab";
98
- if (isMouseDown)
99
- cursor = "grabbing";
100
- if (img)
101
- img.style.cursor = cursor;
108
+ let cursor = "url('customcursor.svg'), auto";
109
+ if (opaque) cursor = "grab";
110
+ if (isMouseDown) cursor = "grabbing";
111
+ if (img) img.style.cursor = cursor;
102
112
  }
103
- export function DraggableImage(props) {
104
- const { src, alt, width, height, initialPos, animate, className, scale, hoverScale, ...restProps } = props;
105
- const imgRef = useRef(null);
106
- const [isOpaque, setIsOpaque] = useState(true); // default to true for better UX
107
- const canvasRef = useRef(null);
108
- const isMouseDown = useRef(false);
109
- // create a invisible canvas element to check the alpha value of the image
110
- useEffect(() => {
111
- if (typeof window !== "undefined" && !canvasRef.current) {
112
- canvasRef.current = document.createElement("canvas");
113
- }
114
- const img = imgRef.current;
115
- const canvas = canvasRef.current;
116
- if (!img || !canvas)
117
- return;
118
- if (!img.complete) {
119
- img.onload = () => drawImageToCanvas(img, canvas);
120
- }
121
- else {
122
- drawImageToCanvas(img, canvas);
123
- }
124
- return () => {
125
- if (img)
126
- img.onload = null;
127
- };
128
- }, []);
129
- // handle global mouse move to update cursor and opacity
130
- useEffect(() => {
131
- const handleGlobalMouseMove = (e) => {
132
- if (!isMouseDown.current &&
133
- isMouseOverImage(e.clientX, e.clientY, imgRef.current)) {
134
- const alpha = getAlphaAtCoords(e.clientX, e.clientY, canvasRef.current, imgRef.current);
135
- // checking alpha > n rather than 0 to not trigger on shadows and such
136
- const opaque = alpha > 128;
137
- setIsOpaque(opaque);
138
- updateCursor(opaque, false, imgRef.current);
139
- }
140
- };
141
- window.addEventListener("mousemove", handleGlobalMouseMove);
142
- return () => {
143
- window.removeEventListener("mousemove", handleGlobalMouseMove);
144
- };
145
- }, []);
146
- const handlePointerDown = useCallback((e) => {
147
- isMouseDown.current = true;
148
- e.stopPropagation(); // Prevents the event from bubbling up
149
- updateCursor(true, true, imgRef.current);
150
- }, []);
151
- const handlePointerUp = () => {
152
- isMouseDown.current = false;
153
- updateCursor(isOpaque, false, imgRef.current);
154
- };
155
- const hoverScaleValue = isOpaque ? (hoverScale ?? (scale ?? 1)) : (scale ?? 1);
156
- return (_jsx(Draggable, { initialPos: initialPos, className: className, drag: isOpaque, style: {
157
- height: 0,
158
- }, ...restProps, children: _jsx(motion.img, { ref: imgRef, src: src, alt: alt, width: width, height: height, animate: animate, draggable: "false", whileHover: { scale: hoverScaleValue }, style: {
159
- scale: scale ?? 1,
160
- pointerEvents: isOpaque ? "auto" : "none",
161
- }, onPointerDown: handlePointerDown, onPointerUp: handlePointerUp }) }));
113
+ function DraggableImage(props) {
114
+ const { src, alt, width, height, initialPos, animate, className, scale, hoverScale, ...restProps } = props;
115
+ const imgRef = useRef(null);
116
+ const [isOpaque, setIsOpaque] = useState(true);
117
+ const canvasRef = useRef(null);
118
+ const isMouseDown = useRef(false);
119
+ useEffect(() => {
120
+ if (typeof window !== "undefined" && !canvasRef.current) canvasRef.current = document.createElement("canvas");
121
+ const img = imgRef.current;
122
+ const canvas = canvasRef.current;
123
+ if (!img || !canvas) return;
124
+ if (!img.complete) img.onload = () => drawImageToCanvas(img, canvas);
125
+ else drawImageToCanvas(img, canvas);
126
+ return () => {
127
+ if (img) img.onload = null;
128
+ };
129
+ }, []);
130
+ useEffect(() => {
131
+ const handleGlobalMouseMove = (e) => {
132
+ if (!isMouseDown.current && isMouseOverImage(e.clientX, e.clientY, imgRef.current)) {
133
+ const opaque = getAlphaAtCoords(e.clientX, e.clientY, canvasRef.current, imgRef.current) > 128;
134
+ setIsOpaque(opaque);
135
+ updateCursor(opaque, false, imgRef.current);
136
+ }
137
+ };
138
+ window.addEventListener("mousemove", handleGlobalMouseMove);
139
+ return () => {
140
+ window.removeEventListener("mousemove", handleGlobalMouseMove);
141
+ };
142
+ }, []);
143
+ const handlePointerDown = useCallback((e) => {
144
+ isMouseDown.current = true;
145
+ e.stopPropagation();
146
+ updateCursor(true, true, imgRef.current);
147
+ }, []);
148
+ const handlePointerUp = () => {
149
+ isMouseDown.current = false;
150
+ updateCursor(isOpaque, false, imgRef.current);
151
+ };
152
+ const hoverScaleValue = isOpaque ? hoverScale ?? scale ?? 1 : scale ?? 1;
153
+ return /* @__PURE__ */ jsx(Draggable, {
154
+ initialPos,
155
+ className,
156
+ drag: isOpaque,
157
+ style: { height: 0 },
158
+ ...restProps,
159
+ children: /* @__PURE__ */ jsx(motion.img, {
160
+ ref: imgRef,
161
+ src,
162
+ alt,
163
+ width,
164
+ height,
165
+ animate,
166
+ draggable: "false",
167
+ whileHover: { scale: hoverScaleValue },
168
+ style: {
169
+ scale: scale ?? 1,
170
+ pointerEvents: isOpaque ? "auto" : "none"
171
+ },
172
+ onPointerDown: handlePointerDown,
173
+ onPointerUp: handlePointerUp
174
+ })
175
+ });
162
176
  }
177
+
178
+ //#endregion
179
+ export { Draggable, DraggableImage };
163
180
  //# sourceMappingURL=draggable.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"draggable.js","sourceRoot":"","sources":["../../../src/components/canvas/draggable.tsx"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"draggable.js","names":[],"sources":["../../../src/components/canvas/draggable.tsx"],"sourcesContent":["import React, {\n useRef,\n useEffect,\n forwardRef,\n useState,\n useCallback,\n} from \"react\";\nimport {\n animate,\n motion,\n useAnimationControls,\n useMotionValue,\n type HTMLMotionProps,\n type PanInfo,\n} from \"framer-motion\";\nimport { useCanvasContext } from \"../../contexts/CanvasContext\";\n\ninterface Point {\n x: number;\n y: number;\n}\n\nexport interface DraggableProps extends HTMLMotionProps<\"div\"> {\n initialPos?: Point;\n shouldStopPropagation?: (e: React.PointerEvent) => boolean;\n}\n\nconst defaultPos = { x: 0, y: 0 };\n\nexport const Draggable = forwardRef<HTMLDivElement, DraggableProps>(\n (props, ref) => {\n const {\n initialPos: passedPos,\n children,\n style,\n shouldStopPropagation = () => true,\n ...restProps\n } = props;\n\n const {\n scale: parentZoom,\n isResetting,\n maxZIndex,\n setMaxZIndex,\n } = useCanvasContext();\n\n const initialPos = passedPos ?? defaultPos;\n\n const x = useMotionValue(initialPos.x);\n const y = useMotionValue(initialPos.y);\n\n const logicalPositionRef = useRef<Point>({ ...initialPos });\n const controls = useAnimationControls();\n\n const [zIndex, setZIndex] = useState(1);\n\n useEffect(() => {\n if (isResetting) {\n logicalPositionRef.current = { ...initialPos };\n void animate(x, initialPos.x, {\n duration: 0.3,\n type: \"spring\",\n damping: 14,\n stiffness: 120,\n mass: 1,\n });\n void animate(y, initialPos.y, {\n duration: 0.3,\n type: \"spring\",\n damping: 14,\n stiffness: 120,\n mass: 1,\n });\n }\n }, [initialPos, controls, isResetting, x, y]);\n\n const handleDrag = (\n _event: MouseEvent | TouchEvent | PointerEvent,\n info: PanInfo,\n ) => {\n controls.stop();\n const deltaParentX = info.delta.x / parentZoom.get();\n const deltaParentY = info.delta.y / parentZoom.get();\n\n logicalPositionRef.current.x += deltaParentX;\n logicalPositionRef.current.y += deltaParentY;\n\n x.set(logicalPositionRef.current.x);\n y.set(logicalPositionRef.current.y);\n\n if (zIndex < maxZIndex) {\n setZIndex(maxZIndex + 1);\n setMaxZIndex(maxZIndex + 1);\n }\n };\n\n return (\n <motion.div\n ref={ref}\n dragMomentum={false}\n drag\n animate={controls}\n onDrag={handleDrag}\n style={{\n ...style,\n x,\n y,\n zIndex,\n }}\n initial={{\n scale: 1,\n filter: \"drop-shadow(0 0px 0px rgba(0, 0, 0, 0)) brightness(1)\",\n position: \"relative\",\n }}\n onPointerDown={(e: React.PointerEvent) => {\n if (shouldStopPropagation?.(e)) {\n e.stopPropagation();\n }\n }}\n transition={{\n duration: 0.1,\n ease: \"easeOut\",\n }}\n {...restProps}\n >\n {children}\n </motion.div>\n );\n },\n);\n\nDraggable.displayName = \"Draggable\";\n\nexport interface DraggableImageProps extends DraggableProps {\n src: string;\n alt?: string;\n width?: string | number;\n height?: string | number;\n scale?: number;\n hoverScale?: number;\n}\n\nfunction drawImageToCanvas(img: HTMLImageElement, canvas: HTMLCanvasElement) {\n const ctx = canvas.getContext(\"2d\", { willReadFrequently: true });\n if (!ctx) return;\n ctx.clearRect(0, 0, canvas.width, canvas.height);\n ctx.drawImage(img, 0, 0);\n}\n\nfunction getAlphaAtCoords(\n clientX: number,\n clientY: number,\n canvas: HTMLCanvasElement | null,\n img: HTMLImageElement | null,\n): number {\n if (!canvas || !img) return 0;\n\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) return 0;\n\n const rect = img.getBoundingClientRect();\n\n const x = ((clientX - rect.left) / rect.width) * img.naturalWidth;\n const y = ((clientY - rect.top) / rect.height) * img.naturalHeight;\n\n const alpha = ctx.getImageData(x, y, 1, 1).data[3] ?? 0;\n return alpha;\n}\n\nfunction isMouseOverImage(\n clientX: number,\n clientY: number,\n img: HTMLImageElement | null,\n) {\n if (!img) return false;\n const rect = img.getBoundingClientRect();\n return (\n clientX >= rect.left &&\n clientX <= rect.right &&\n clientY >= rect.top &&\n clientY <= rect.bottom\n );\n}\n\nfunction updateCursor(\n opaque: boolean,\n isMouseDown: boolean,\n img: HTMLImageElement | null,\n) {\n let cursor = \"url('customcursor.svg'), auto\"; // default\n if (opaque) cursor = \"grab\";\n if (isMouseDown) cursor = \"grabbing\";\n if (img) img.style.cursor = cursor;\n}\n\nexport function DraggableImage(props: DraggableImageProps) {\n const {\n src,\n alt,\n width,\n height,\n initialPos,\n animate,\n className,\n scale,\n hoverScale,\n ...restProps\n } = props;\n const imgRef = useRef<HTMLImageElement>(null);\n const [isOpaque, setIsOpaque] = useState(true); // default to true for better UX\n const canvasRef = useRef<HTMLCanvasElement | null>(null);\n const isMouseDown = useRef(false);\n\n // create a invisible canvas element to check the alpha value of the image\n useEffect(() => {\n if (typeof window !== \"undefined\" && !canvasRef.current) {\n canvasRef.current = document.createElement(\"canvas\");\n }\n const img = imgRef.current;\n const canvas = canvasRef.current;\n if (!img || !canvas) return;\n if (!img.complete) {\n img.onload = () => drawImageToCanvas(img, canvas);\n } else {\n drawImageToCanvas(img, canvas);\n }\n return () => {\n if (img) img.onload = null;\n };\n }, []);\n\n // handle global mouse move to update cursor and opacity\n useEffect(() => {\n const handleGlobalMouseMove = (e: MouseEvent) => {\n if (\n !isMouseDown.current &&\n isMouseOverImage(e.clientX, e.clientY, imgRef.current)\n ) {\n const alpha = getAlphaAtCoords(\n e.clientX,\n e.clientY,\n canvasRef.current,\n imgRef.current,\n );\n\n // checking alpha > n rather than 0 to not trigger on shadows and such\n const opaque = alpha > 128;\n\n setIsOpaque(opaque);\n updateCursor(opaque, false, imgRef.current);\n }\n };\n window.addEventListener(\"mousemove\", handleGlobalMouseMove);\n return () => {\n window.removeEventListener(\"mousemove\", handleGlobalMouseMove);\n };\n }, []);\n\n const handlePointerDown = useCallback((e: React.PointerEvent) => {\n isMouseDown.current = true;\n e.stopPropagation(); // Prevents the event from bubbling up\n updateCursor(true, true, imgRef.current);\n }, []);\n\n const handlePointerUp = () => {\n isMouseDown.current = false;\n updateCursor(isOpaque, false, imgRef.current);\n };\n\n const hoverScaleValue = isOpaque ? (hoverScale ?? (scale ?? 1)) : (scale ?? 1);\n\n return (\n <Draggable\n initialPos={initialPos}\n className={className}\n drag={isOpaque}\n style={{\n height: 0,\n }}\n {...restProps}\n >\n <motion.img\n ref={imgRef}\n src={src}\n alt={alt}\n width={width}\n height={height}\n animate={animate}\n draggable=\"false\"\n whileHover={{ scale: hoverScaleValue }}\n style={{\n scale: scale ?? 1,\n pointerEvents: isOpaque ? \"auto\" : \"none\",\n }}\n onPointerDown={handlePointerDown}\n onPointerUp={handlePointerUp}\n />\n </Draggable>\n );\n}\n"],"mappings":"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"}