Web Dev Solutions

Catalin Mititiuc

aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorCatalin Mititiuc <webdevcat@proton.me>2024-04-12 10:57:15 -0700
committerCatalin Mititiuc <webdevcat@proton.me>2024-04-15 09:48:00 -0700
commit263201d869956b94660d4efa8297e89dadbe36a8 (patch)
tree8fa47922163548dda0739aaa5ff0612de96c9fd1 /src/modules/pan.js
Initial commit
Diffstat (limited to 'src/modules/pan.js')
-rw-r--r--src/modules/pan.js77
1 files changed, 77 insertions, 0 deletions
diff --git a/src/modules/pan.js b/src/modules/pan.js
new file mode 100644
index 0000000..9faae0a
--- /dev/null
+++ b/src/modules/pan.js
@@ -0,0 +1,77 @@
+const minDistanceThreshold = 5;
+
+function distanceBetween({ x: x1, y: y1 }, { x: x2, y: y2 }) {
+ return Math.sqrt((x2 - x1) ** 2 + (y2 - y1) ** 2);
+}
+
+function minDistanceThresholdIsMet(startPt, endPt) {
+ return distanceBetween(startPt, endPt) >= minDistanceThreshold;
+}
+
+function getPositionChangeInLocalCoords(svg, startPt, endPt) {
+ const matrix = svg.getScreenCTM().inverse(),
+ localStartPt = startPt.matrixTransform(matrix),
+ localEndPt = endPt.matrixTransform(matrix);
+
+ return {
+ x: localStartPt.x - localEndPt.x,
+ y: localStartPt.y - localEndPt.y
+ };
+}
+
+function stopEventPropagationToChildren(svg, type) {
+ svg.addEventListener(type, e => e.stopPropagation(), { capture: true, once: true });
+}
+
+function setToCurrentPointerCoords(point, e) {
+ point.x = e.clientX;
+ point.y = e.clientY;
+
+ return point;
+}
+
+function getPanCoords(svg, startPt, movePt, initialPos) {
+ const posChange = getPositionChangeInLocalCoords(svg, startPt, movePt);
+
+ return {
+ x: initialPos.x + posChange.x,
+ y: initialPos.y + posChange.y
+ };
+}
+
+function setViewBoxPosition(svg, { x, y }) {
+ const { width, height } = svg.viewBox.baseVal;
+
+ svg.setAttributeNS(null, 'viewBox', `${x} ${y} ${width} ${height}`);
+}
+
+export default function (svg, e) {
+ const { x, y } = svg.viewBox.baseVal,
+ startPt = setToCurrentPointerCoords(new DOMPoint(), e),
+ movePt = new DOMPoint();
+
+ let isPanning = false;
+
+ function pointerMove(e) {
+ setToCurrentPointerCoords(movePt, e);
+
+ if (!isPanning && minDistanceThresholdIsMet(startPt, movePt)) {
+ isPanning = true;
+ e.target.setPointerCapture(e.pointerId);
+ setToCurrentPointerCoords(startPt, e);
+ stopEventPropagationToChildren(svg, 'click');
+ }
+
+ if (isPanning) {
+ setViewBoxPosition(svg, getPanCoords(svg, startPt, movePt, { x, y }));
+ }
+ }
+
+ svg.addEventListener('pointermove', pointerMove);
+
+ svg.addEventListener(
+ 'pointerup',
+ () => svg.removeEventListener('pointermove', pointerMove),
+ { once: true }
+ );
+}