UNPKG

@onesy/ui-react

Version:
163 lines (160 loc) 11.2 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _utils = require("@onesy/utils"); var _styleReact = require("@onesy/style-react"); var _jsxRuntime = require("react/jsx-runtime"); const _excluded = ["ref", "onChange", "items", "image", "delay", "precise", "draggedIsElement", "isEqual", "getDraggingElement", "onDraggedElement", "onDragStart", "className", "children"]; function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } const DragAndDropList = props_ => { var _theme$ui, _theme$ui2; const theme = (0, _styleReact.useOnesyTheme)(); const props = _objectSpread(_objectSpread(_objectSpread({}, theme === null || theme === void 0 || (_theme$ui = theme.ui) === null || _theme$ui === void 0 || (_theme$ui = _theme$ui.elements) === null || _theme$ui === void 0 || (_theme$ui = _theme$ui.all) === null || _theme$ui === void 0 || (_theme$ui = _theme$ui.props) === null || _theme$ui === void 0 ? void 0 : _theme$ui.default), theme === null || theme === void 0 || (_theme$ui2 = theme.ui) === null || _theme$ui2 === void 0 || (_theme$ui2 = _theme$ui2.elements) === null || _theme$ui2 === void 0 || (_theme$ui2 = _theme$ui2.onesyDragAndDropList) === null || _theme$ui2 === void 0 || (_theme$ui2 = _theme$ui2.props) === null || _theme$ui2 === void 0 ? void 0 : _theme$ui2.default), props_); const { ref, onChange: onChange_, items, image, delay = 0, precise = true, draggedIsElement = true, isEqual, getDraggingElement, onDraggedElement, onDragStart: onDragStart_, className, children // @ts-ignore } = props, other = (0, _objectWithoutProperties2.default)(props, _excluded); const refs = { root: _react.default.useRef(null), dragging: _react.default.useRef(null), rectDragged: _react.default.useRef(null), isDragging: _react.default.useRef(null), previous: _react.default.useRef(null) }; const onChange = _react.default.useMemo(() => { return (0, _utils.debounce)((indexPrevious, indexNew) => onChange_(indexPrevious, indexNew), delay); }, [onChange_, delay]); _react.default.useEffect(() => { var _refs$root$current; 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 ((0, _utils.is)('function', onDraggedElement)) onDraggedElement(null); refs.isDragging.current = false; } }; const rootDocument = (0, _utils.isEnvironment)('browser') ? ((_refs$root$current = refs.root.current) === null || _refs$root$current === void 0 ? void 0 : _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); }; }, []); const img = _react.default.useMemo(() => { var _refs$root$current2; const rootDocument_0 = (0, _utils.isEnvironment)('browser') ? ((_refs$root$current2 = refs.root.current) === null || _refs$root$current2 === void 0 ? void 0 : _refs$root$current2.ownerDocument) || window.document : undefined; const element = rootDocument_0.createElement('img'); element.src = (0, _utils.is)('string', image) ? image : ``; return element; }, []); const onDragStart = item => async event_0 => { var _dragging$dataset; event_0.dataTransfer.setData('text', (item === null || item === void 0 ? void 0 : item.value) !== undefined ? item.value : item); if (image) event_0.dataTransfer.setDragImage(img, 0, 0); refs.isDragging.current = true; const dragging = (0, _utils.is)('function', getDraggingElement) ? getDraggingElement(event_0) : event_0.target; if (onDragStart_) onDragStart_(item, event_0); refs.dragging.current = draggedIsElement ? dragging : dragging === null || dragging === void 0 || (_dragging$dataset = dragging.dataset) === null || _dragging$dataset === void 0 ? void 0 : _dragging$dataset.onesyDragAndDropListValue; refs.rectDragged.current = { height: dragging.clientHeight, x: event_0.clientX - dragging.offsetLeft, y: event_0.clientY - dragging.offsetTop }; setTimeout(() => { if ((0, _utils.is)('function', onDraggedElement)) onDraggedElement === null || onDraggedElement === void 0 || onDraggedElement(item); }); }; const onDragEnd = () => event_1 => { // clean up refs.isDragging.current = false; refs.rectDragged.current = null; refs.dragging.current = null; if ((0, _utils.is)('function', onDraggedElement)) onDraggedElement(null); }; const onDragOver = () => event_2 => { var _refs$root$current3; event_2.preventDefault(); const rootDocument_1 = (0, _utils.isEnvironment)('browser') ? ((_refs$root$current3 = refs.root.current) === null || _refs$root$current3 === void 0 ? void 0 : _refs$root$current3.ownerDocument) || window.document : undefined; const over = event_2.currentTarget; const dragging_0 = draggedIsElement ? refs.dragging.current : rootDocument_1.body.querySelector(`[data-onesy-drag-and-drop-list-value="${refs.dragging.current}"]`); if (!(over && dragging_0)) return; if (precise) { if (over !== dragging_0) { const rectOver = { height: over.clientHeight, x: over.offsetLeft, y: over.offsetTop }; const mousePosition = { x: event_2.clientX - rectOver.x, y: event_2.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_0 => isEqual ? isEqual(item_0, over.dataset.onesyDragAndDropListValue) : item_0 === over.dataset.onesyDragAndDropListValue); const draggedIndex = items.findIndex(item_1 => isEqual ? isEqual(item_1, dragging_0.dataset.onesyDragAndDropListValue) : item_1 === dragging_0.dataset.onesyDragAndDropListValue); // 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 || !(0, _utils.equalDeep)(refs.previous.current, [draggedIndex, overIndex])) { onChange(draggedIndex, overIndex); refs.previous.current = [draggedIndex, overIndex]; } } } } else { const overIndex_0 = items.findIndex(item_2 => isEqual ? isEqual(item_2, over.dataset.onesyDragAndDropListValue) : item_2 === over.dataset.onesyDragAndDropListValue); const draggedIndex_0 = items.findIndex(item_3 => isEqual ? isEqual(item_3, dragging_0.dataset.onesyDragAndDropListValue) : item_3 === dragging_0.dataset.onesyDragAndDropListValue); if (!refs.previous.current || !(0, _utils.equalDeep)(refs.previous.current, [draggedIndex_0, overIndex_0])) { if ((0, _utils.is)('function', onChange)) onChange(draggedIndex_0, overIndex_0); refs.previous.current = [draggedIndex_0, overIndex_0]; } } }; if ((0, _utils.is)('function', children)) return children({ ref: item_4 => { if (ref) { if ((0, _utils.is)('function', ref)) ref(item_4);else ref.current = item_4; } refs.root.current = item_4; }, onDragStart, onDragOver, onDragEnd }); return /*#__PURE__*/(0, _jsxRuntime.jsx)(_jsxRuntime.Fragment, { children: children }); }; DragAndDropList.displayName = 'onesy-DragAndDropList'; var _default = exports.default = DragAndDropList;