function simulateEvent(el, type, options = {}) { let event; if (!el) return null; if (/^mouse/.test(type)) { event = el.ownerDocument.createEvent('MouseEvents'); event.initMouseEvent(type, true, true, el.ownerDocument.defaultView, options.button, options.screenX, options.screenY, options.clientX, options.clientY, options.ctrlKey, options.altKey, options.shiftKey, options.metaKey, options.button, el); } else { event = el.ownerDocument.createEvent('CustomEvent'); event.initCustomEvent(type, true, true, el.ownerDocument.defaultView, options.button, options.screenX, options.screenY, options.clientX, options.clientY, options.ctrlKey, options.altKey, options.shiftKey, options.metaKey, options.button, el); event.dataTransfer = { data: {}, setData(key, val) { this.data[key] = val; }, getData(key) { return this.data[key]; }, }; } if (el.dispatchEvent) { el.dispatchEvent(event); } else if (el.fireEvent) { el.fireEvent(`on${type}`, event); } return event; } function isLast(target) { const el = typeof target.el === 'string' ? document.getElementById(target.el.substr(1)) : target.el; const { children } = el; return children.length - 1 === target.index; } function getTarget(target) { const el = typeof target.el === 'string' ? document.getElementById(target.el.substr(1)) : target.el; const { children } = el; return ( children[target.index] || children[target.index === 'first' ? 0 : -1] || children[target.index === 'last' ? children.length - 1 : -1] || el ); } function getRect(el) { const rect = el.getBoundingClientRect(); const width = rect.right - rect.left; const height = (rect.bottom - rect.top) + 10; return { x: rect.left, y: rect.top, cx: rect.left + (width / 2), cy: rect.top + (height / 2), w: width, h: height, hw: width / 2, wh: height / 2, }; } export default function simulateDrag(options) { const { to, from } = options; to.el = to.el || from.el; const fromEl = getTarget(from); const toEl = getTarget(to); const firstEl = getTarget({ el: to.el, index: 'first', }); const lastEl = getTarget({ el: options.to.el, index: 'last', }); const fromRect = getRect(fromEl); const toRect = getRect(toEl); const firstRect = getRect(firstEl); const lastRect = getRect(lastEl); const startTime = new Date().getTime(); const duration = options.duration || 1000; simulateEvent(fromEl, 'mousedown', { button: 0, clientX: fromRect.cx, clientY: fromRect.cy, }); if (options.ontap) options.ontap(); window.SIMULATE_DRAG_ACTIVE = 1; if (options.to.index === 0) { toRect.cy = firstRect.y; } else if (isLast(options.to)) { toRect.cy = lastRect.y + lastRect.h + 50; } const dragInterval = setInterval(() => { const progress = (new Date().getTime() - startTime) / duration; const x = (fromRect.cx + ((toRect.cx - fromRect.cx) * progress)); const y = (fromRect.cy + ((toRect.cy - fromRect.cy) * progress)); const overEl = fromEl.ownerDocument.elementFromPoint(x, y); simulateEvent(overEl, 'mousemove', { clientX: x, clientY: y, }); if (progress >= 1) { if (options.ondragend) options.ondragend(); simulateEvent(toEl, 'mouseup'); clearInterval(dragInterval); window.SIMULATE_DRAG_ACTIVE = 0; } }, 100); return { target: fromEl, fromList: fromEl.parentNode, toList: toEl.parentNode, }; }