UNPKG

@amaui/ui-react

Version:
153 lines (152 loc) 9.58 kB
import _objectWithoutProperties from "@babel/runtime/helpers/objectWithoutProperties"; import _defineProperty from "@babel/runtime/helpers/defineProperty"; const _excluded = ["onChange", "items", "image", "delay", "precise", "draggedIsElement", "isEqual", "getDraggingElement", "onDraggedElement", "onDragStart", "className", "children"]; function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; } import React from 'react'; import { debounce, equalDeep, is, isEnvironment } from '@amaui/utils'; import { style as styleMethod, useAmauiTheme } from '@amaui/style-react'; const useStyle = styleMethod(theme => ({ root: {} }), { name: 'amaui-DragAndDropList' }); const DragAndDropList = /*#__PURE__*/React.forwardRef((props_, ref) => { const theme = useAmauiTheme(); const props = React.useMemo(() => _objectSpread(_objectSpread(_objectSpread({}, theme?.ui?.elements?.all?.props?.default), theme?.ui?.elements?.amauiDragAndDropList?.props?.default), props_), [props_]); const { onChange: onChange_, items, image, delay = 0, precise = true, draggedIsElement = true, isEqual, getDraggingElement, onDraggedElement, onDragStart: onDragStart_, className, children } = props, other = _objectWithoutProperties(props, _excluded); const { classes } = useStyle(); const refs = { root: React.useRef(undefined), dragging: React.useRef(undefined), rectDragged: React.useRef(undefined), isDragging: React.useRef(), previous: React.useRef(undefined) }; const onChange = React.useMemo(() => { return debounce((indexPrevious, indexNew) => onChange_(indexPrevious, indexNew), delay); }, [onChange_, delay]); React.useEffect(() => { const onMouseUp = event => { // in use case // where onDragEnd is never emited // due to original element having it // has been removed from the dom // prior to onDragEnd event is to be emited // alternative way to provide a callback // to dragging has ended if (refs.isDragging.current) { if (is('function', onDraggedElement)) onDraggedElement(null); refs.isDragging.current = false; } }; const rootDocument = isEnvironment('browser') ? refs.root.current?.ownerDocument || window.document : undefined; rootDocument.body.addEventListener('mouseup', onMouseUp); rootDocument.body.addEventListener('touchend', onMouseUp); return () => { rootDocument.body.removeEventListener('mouseup', onMouseUp); rootDocument.body.removeEventListener('touchend', onMouseUp); }; }, [onDraggedElement]); const img = React.useMemo(() => { const rootDocument = isEnvironment('browser') ? refs.root.current?.ownerDocument || window.document : undefined; const element = rootDocument.createElement('img'); element.src = is('string', image) ? image : ``; return element; }, []); const onDragStart = item => async event => { event.dataTransfer.setData('text', item?.value !== undefined ? item.value : item); if (image) event.dataTransfer.setDragImage(img, 0, 0); refs.isDragging.current = true; const dragging = is('function', getDraggingElement) ? getDraggingElement(event) : event.target; if (onDragStart_) onDragStart_(item, event); refs.dragging.current = draggedIsElement ? dragging : dragging?.dataset?.amauiDragAndDropListValue; refs.rectDragged.current = { height: dragging.clientHeight, x: event.clientX - dragging.offsetLeft, y: event.clientY - dragging.offsetTop }; setTimeout(() => { if (is('function', onDraggedElement)) onDraggedElement?.(item); }); }; const onDragEnd = () => event => { // clean up refs.isDragging.current = false; refs.rectDragged.current = null; refs.dragging.current = null; if (is('function', onDraggedElement)) onDraggedElement(null); }; const onDragOver = () => event => { event.preventDefault(); const rootDocument = isEnvironment('browser') ? refs.root.current?.ownerDocument || window.document : undefined; const over = event.currentTarget; const dragging = draggedIsElement ? refs.dragging.current : rootDocument.body.querySelector(`[data-amaui-drag-and-drop-list-value="${refs.dragging.current}"]`); if (!(over && dragging)) return; if (precise) { if (over !== dragging) { const rectOver = { height: over.clientHeight, x: over.offsetLeft, y: over.offsetTop }; const mousePosition = { x: event.clientX - rectOver.x, y: event.clientY - rectOver.y }; const partBottom = Math.abs(refs.rectDragged.current.height - refs.rectDragged.current.y); const partTop = refs.rectDragged.current.y; const half = rectOver.height / 2; const positionTopBottom = partBottom + mousePosition.y >= half ? 'bottom' : 'top'; const positionBottomTop = mousePosition.y - partTop <= half ? 'top' : 'bottom'; const overIndex = items.findIndex(item => isEqual ? isEqual(item, over.dataset.amauiDragAndDropListValue) : item === over.dataset.amauiDragAndDropListValue); const draggedIndex = items.findIndex(item => isEqual ? isEqual(item, dragging.dataset.amauiDragAndDropListValue) : item === dragging.dataset.amauiDragAndDropListValue); // if dragged is above over & bottom swap their indexes // if dragged is below over && top swap their indexes if (draggedIndex < overIndex && positionTopBottom === 'bottom' || draggedIndex > overIndex && positionBottomTop === 'top') { if (!refs.previous.current || !equalDeep(refs.previous.current, [draggedIndex, overIndex])) { onChange(draggedIndex, overIndex); refs.previous.current = [draggedIndex, overIndex]; } } } } else { const overIndex = items.findIndex(item => isEqual ? isEqual(item, over.dataset.amauiDragAndDropListValue) : item === over.dataset.amauiDragAndDropListValue); const draggedIndex = items.findIndex(item => isEqual ? isEqual(item, dragging.dataset.amauiDragAndDropListValue) : item === dragging.dataset.amauiDragAndDropListValue); if (!refs.previous.current || !equalDeep(refs.previous.current, [draggedIndex, overIndex])) { if (is('function', onChange)) onChange(draggedIndex, overIndex); refs.previous.current = [draggedIndex, overIndex]; } } }; if (is('function', children)) return children({ ref: item => { if (ref) { if (is('function', ref)) ref(item);else ref.current = item; } refs.root.current = item; }, onDragStart, onDragOver, onDragEnd }); return /*#__PURE__*/React.createElement(React.Fragment, null, children); }); DragAndDropList.displayName = 'amaui-DragAndDropList'; export default DragAndDropList;