/*! * Draggable 3.12.2 * https://greensock.com * * @license Copyright 2008-2023, GreenSock. All rights reserved. * Subject to the terms at https://greensock.com/standard-license or for * Club GreenSock members, the agreement issued with that membership. * @author: Jack Doyle, jack@greensock.com */ /* eslint-disable */ import { getGlobalMatrix, Matrix2D } from "./utils/matrix.js"; let gsap, _win, _doc, _docElement, _body, _tempDiv, _placeholderDiv, _coreInitted, _checkPrefix, _toArray, _supportsPassive, _isTouchDevice, _touchEventLookup, _isMultiTouching, _isAndroid, InertiaPlugin, _defaultCursor, _supportsPointer, _context, _getStyleSaver, _dragCount = 0, _windowExists = () => typeof(window) !== "undefined", _getGSAP = () => gsap || (_windowExists() && (gsap = window.gsap) && gsap.registerPlugin && gsap), _isFunction = value => typeof(value) === "function", _isObject = value => typeof(value) === "object", _isUndefined = value => typeof(value) === "undefined", _emptyFunc = () => false, _transformProp = "transform", _transformOriginProp = "transformOrigin", _round = value => Math.round(value * 10000) / 10000, _isArray = Array.isArray, _createElement = (type, ns) => { let e = _doc.createElementNS ? _doc.createElementNS((ns || "http://www.w3.org/1999/xhtml").replace(/^https/, "http"), type) : _doc.createElement(type); //some servers swap in https for http in the namespace which can break things, making "style" inaccessible. return e.style ? e : _doc.createElement(type); //some environments won't allow access to the element's style when created with a namespace in which case we default to the standard createElement() to work around the issue. Also note that when GSAP is embedded directly inside an SVG file, createElement() won't allow access to the style object in Firefox (see https://greensock.com/forums/topic/20215-problem-using-tweenmax-in-standalone-self-containing-svg-file-err-cannot-set-property-csstext-of-undefined/). }, _RAD2DEG = 180 / Math.PI, _bigNum = 1e20, _identityMatrix = new Matrix2D(), _getTime = Date.now || (() => new Date().getTime()), _renderQueue = [], _lookup = {}, //when a Draggable is created, the target gets a unique _gsDragID property that allows gets associated with the Draggable instance for quick lookups in Draggable.get(). This avoids circular references that could cause gc problems. _lookupCount = 0, _clickableTagExp = /^(?:a|input|textarea|button|select)$/i, _lastDragTime = 0, _temp1 = {}, // a simple object we reuse and populate (usually x/y properties) to conserve memory and improve performance. _windowProxy = {}, //memory/performance optimization - we reuse this object during autoScroll to store window-related bounds/offsets. _copy = (obj, factor) => { let copy = {}, p; for (p in obj) { copy[p] = factor ? obj[p] * factor : obj[p]; } return copy; }, _extend = (obj, defaults) => { for (let p in defaults) { if (!(p in obj)) { obj[p] = defaults[p]; } } return obj; }, _setTouchActionForAllDescendants = (elements, value) => { let i = elements.length, children; while (i--) { value ? (elements[i].style.touchAction = value) : elements[i].style.removeProperty("touch-action"); children = elements[i].children; children && children.length && _setTouchActionForAllDescendants(children, value); } }, _renderQueueTick = () => _renderQueue.forEach(func => func()), _addToRenderQueue = func => { _renderQueue.push(func); if (_renderQueue.length === 1) { gsap.ticker.add(_renderQueueTick); } }, _renderQueueTimeout = () => !_renderQueue.length && gsap.ticker.remove(_renderQueueTick), _removeFromRenderQueue = func => { let i = _renderQueue.length; while (i--) { if (_renderQueue[i] === func) { _renderQueue.splice(i, 1); } } gsap.to(_renderQueueTimeout, {overwrite:true, delay:15, duration:0, onComplete:_renderQueueTimeout, data:"_draggable"}); //remove the "tick" listener only after the render queue is empty for 15 seconds (to improve performance). Adding/removing it constantly for every click/touch wouldn't deliver optimal speed, and we also don't want the ticker to keep calling the render method when things are idle for long periods of time (we want to improve battery life on mobile devices). }, _setDefaults = (obj, defaults) => { for (let p in defaults) { if (!(p in obj)) { obj[p] = defaults[p]; } } return obj; }, _addListener = (element, type, func, capture) => { if (element.addEventListener) { let touchType = _touchEventLookup[type]; capture = capture || (_supportsPassive ? {passive: false} : null); element.addEventListener(touchType || type, func, capture); (touchType && type !== touchType) && element.addEventListener(type, func, capture);//some browsers actually support both, so must we. But pointer events cover all. } }, _removeListener = (element, type, func, capture) => { if (element.removeEventListener) { let touchType = _touchEventLookup[type]; element.removeEventListener(touchType || type, func, capture); (touchType && type !== touchType) && element.removeEventListener(type, func, capture); } }, _preventDefault = event => { event.preventDefault && event.preventDefault(); event.preventManipulation && event.preventManipulation(); //for some Microsoft browsers }, _hasTouchID = (list, ID) => { let i = list.length; while (i--) { if (list[i].identifier === ID) { return true; } } }, _onMultiTouchDocumentEnd = event => { _isMultiTouching = (event.touches && _dragCount < event.touches.length); _removeListener(event.target, "touchend", _onMultiTouchDocumentEnd); }, _onMultiTouchDocument = event => { _isMultiTouching = (event.touches && _dragCount < event.touches.length); _addListener(event.target, "touchend", _onMultiTouchDocumentEnd); }, _getDocScrollTop = doc => _win.pageYOffset || doc.scrollTop || doc.documentElement.scrollTop || doc.body.scrollTop || 0, _getDocScrollLeft = doc => _win.pageXOffset || doc.scrollLeft || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0, _addScrollListener = (e, callback) => { _addListener(e, "scroll", callback); if (!_isRoot(e.parentNode)) { _addScrollListener(e.parentNode, callback); } }, _removeScrollListener = (e, callback) => { _removeListener(e, "scroll", callback); if (!_isRoot(e.parentNode)) { _removeScrollListener(e.parentNode, callback); } }, _isRoot = e => !!(!e || e === _docElement || e.nodeType === 9 || e === _doc.body || e === _win || !e.nodeType || !e.parentNode), _getMaxScroll = (element, axis) => { let dim = (axis === "x") ? "Width" : "Height", scroll = "scroll" + dim, client = "client" + dim; return Math.max(0, _isRoot(element) ? Math.max(_docElement[scroll], _body[scroll]) - (_win["inner" + dim] || _docElement[client] || _body[client]) : element[scroll] - element[client]); }, _recordMaxScrolls = (e, skipCurrent) => { //records _gsMaxScrollX and _gsMaxScrollY properties for the element and all ancestors up the chain so that we can cap it, otherwise dragging beyond the edges with autoScroll on can endlessly scroll. let x = _getMaxScroll(e, "x"), y = _getMaxScroll(e, "y"); if (_isRoot(e)) { e = _windowProxy; } else { _recordMaxScrolls(e.parentNode, skipCurrent); } e._gsMaxScrollX = x; e._gsMaxScrollY = y; if (!skipCurrent) { e._gsScrollX = e.scrollLeft || 0; e._gsScrollY = e.scrollTop || 0; } }, _setStyle = (element, property, value) => { let style = element.style; if (!style) { return; } if (_isUndefined(style[property])) { property = _checkPrefix(property, element) || property; } if (value == null) { style.removeProperty && style.removeProperty(property.replace(/([A-Z])/g, "-$1").toLowerCase()); } else { style[property] = value; } }, _getComputedStyle = element => _win.getComputedStyle((element instanceof Element) ? element : element.host || (element.parentNode || {}).host || element), //the "host" stuff helps to accommodate ShadowDom objects. _tempRect = {}, //reuse to reduce garbage collection tasks _parseRect = e => { //accepts a DOM element, a mouse event, or a rectangle object and returns the corresponding rectangle with left, right, width, height, top, and bottom properties if (e === _win) { _tempRect.left = _tempRect.top = 0; _tempRect.width = _tempRect.right = _docElement.clientWidth || e.innerWidth || _body.clientWidth || 0; _tempRect.height = _tempRect.bottom = ((e.innerHeight || 0) - 20 < _docElement.clientHeight) ? _docElement.clientHeight : e.innerHeight || _body.clientHeight || 0; return _tempRect; } let doc = e.ownerDocument || _doc, r = !_isUndefined(e.pageX) ? {left: e.pageX - _getDocScrollLeft(doc), top: e.pageY - _getDocScrollTop(doc), right: e.pageX - _getDocScrollLeft(doc) + 1, bottom: e.pageY - _getDocScrollTop(doc) + 1} : (!e.nodeType && !_isUndefined(e.left) && !_isUndefined(e.top)) ? e : _toArray(e)[0].getBoundingClientRect(); if (_isUndefined(r.right) && !_isUndefined(r.width)) { r.right = r.left + r.width; r.bottom = r.top + r.height; } else if (_isUndefined(r.width)) { //some browsers don't include width and height properties. We can't just set them directly on r because some browsers throw errors, so create a new generic object. r = {width: r.right - r.left, height: r.bottom - r.top, right: r.right, left: r.left, bottom: r.bottom, top: r.top}; } return r; }, _dispatchEvent = (target, type, callbackName) => { let vars = target.vars, callback = vars[callbackName], listeners = target._listeners[type], result; if (_isFunction(callback)) { result = callback.apply(vars.callbackScope || target, vars[callbackName + "Params"] || [target.pointerEvent]); } if (listeners && target.dispatchEvent(type) === false) { result = false; } return result; }, _getBounds = (target, context) => { //accepts any of the following: a DOM element, jQuery object, selector text, or an object defining bounds as {top, left, width, height} or {minX, maxX, minY, maxY}. Returns an object with left, top, width, and height properties. let e = _toArray(target)[0], top, left, offset; if (!e.nodeType && e !== _win) { if (!_isUndefined(target.left)) { offset = {x:0, y:0}; //_getOffsetTransformOrigin(context); //the bounds should be relative to the origin return {left: target.left - offset.x, top: target.top - offset.y, width: target.width, height: target.height}; } left = target.min || target.minX || target.minRotation || 0; top = target.min || target.minY || 0; return {left:left, top:top, width:(target.max || target.maxX || target.maxRotation || 0) - left, height:(target.max || target.maxY || 0) - top}; } return _getElementBounds(e, context); }, _point1 = {}, //we reuse to minimize garbage collection tasks. _getElementBounds = (element, context) => { context = _toArray(context)[0]; let isSVG = (element.getBBox && element.ownerSVGElement), doc = element.ownerDocument || _doc, left, right, top, bottom, matrix, p1, p2, p3, p4, bbox, width, height, cs; if (element === _win) { top = _getDocScrollTop(doc); left = _getDocScrollLeft(doc); right = left + (doc.documentElement.clientWidth || element.innerWidth || doc.body.clientWidth || 0); bottom = top + (((element.innerHeight || 0) - 20 < doc.documentElement.clientHeight) ? doc.documentElement.clientHeight : element.innerHeight || doc.body.clientHeight || 0); //some browsers (like Firefox) ignore absolutely positioned elements, and collapse the height of the documentElement, so it could be 8px, for example, if you have just an absolutely positioned div. In that case, we use the innerHeight to resolve this. } else if (context === _win || _isUndefined(context)) { return element.getBoundingClientRect(); } else { left = top = 0; if (isSVG) { bbox = element.getBBox(); width = bbox.width; height = bbox.height; } else { if (element.viewBox && (bbox = element.viewBox.baseVal)) { left = bbox.x || 0; top = bbox.y || 0; width = bbox.width; height = bbox.height; } if (!width) { cs = _getComputedStyle(element); bbox = cs.boxSizing === "border-box"; width = (parseFloat(cs.width) || element.clientWidth || 0) + (bbox ? 0 : parseFloat(cs.borderLeftWidth) + parseFloat(cs.borderRightWidth)); height = (parseFloat(cs.height) || element.clientHeight || 0) + (bbox ? 0 : parseFloat(cs.borderTopWidth) + parseFloat(cs.borderBottomWidth)); } } right = width; bottom = height; } if (element === context) { return {left:left, top:top, width: right - left, height: bottom - top}; } matrix = getGlobalMatrix(context, true).multiply(getGlobalMatrix(element)); p1 = matrix.apply({x:left, y:top}); p2 = matrix.apply({x:right, y:top}); p3 = matrix.apply({x:right, y:bottom}); p4 = matrix.apply({x:left, y:bottom}); left = Math.min(p1.x, p2.x, p3.x, p4.x); top = Math.min(p1.y, p2.y, p3.y, p4.y); return {left: left, top: top, width: Math.max(p1.x, p2.x, p3.x, p4.x) - left, height: Math.max(p1.y, p2.y, p3.y, p4.y) - top}; }, _parseInertia = (draggable, snap, max, min, factor, forceZeroVelocity) => { let vars = {}, a, i, l; if (snap) { if (factor !== 1 && snap instanceof Array) { //some data must be altered to make sense, like if the user passes in an array of rotational values in degrees, we must convert it to radians. Or for scrollLeft and scrollTop, we invert the values. vars.end = a = []; l = snap.length; if (_isObject(snap[0])) { //if the array is populated with objects, like points ({x:100, y:200}), make copies before multiplying by the factor, otherwise we'll mess up the originals and the user may reuse it elsewhere. for (i = 0; i < l; i++) { a[i] = _copy(snap[i], factor); } } else { for (i = 0; i < l; i++) { a[i] = snap[i] * factor; } } max += 1.1; //allow 1.1 pixels of wiggle room when snapping in order to work around some browser inconsistencies in the way bounds are reported which can make them roughly a pixel off. For example, if "snap:[-$('#menu').width(), 0]" was defined and #menu had a wrapper that was used as the bounds, some browsers would be one pixel off, making the minimum -752 for example when snap was [-753,0], thus instead of snapping to -753, it would snap to 0 since -753 was below the minimum. min -= 1.1; } else if (_isFunction(snap)) { vars.end = value => { let result = snap.call(draggable, value), copy, p; if (factor !== 1) { if (_isObject(result)) { copy = {}; for (p in result) { copy[p] = result[p] * factor; } result = copy; } else { result *= factor; } } return result; //we need to ensure that we can scope the function call to the Draggable instance itself so that users can access important values like maxX, minX, maxY, minY, x, and y from within that function. }; } else { vars.end = snap; } } if (max || max === 0) { vars.max = max; } if (min || min === 0) { vars.min = min; } if (forceZeroVelocity) { vars.velocity = 0; } return vars; }, _isClickable = element => { //sometimes it's convenient to mark an element as clickable by adding a data-clickable="true" attribute (in which case we won't preventDefault() the mouse/touch event). This method checks if the element is an , , or