UNPKG

react-beautiful-dnd

Version:

Beautiful, accessible drag and drop for lists with React.js

150 lines (125 loc) 3.35 kB
// @flow import { add, apply, isEqual } from '../position'; import type { ClosestScrollable, DroppableDimension, Position, Viewport, } from '../../types'; type CanScrollArgs = {| max: Position, current: Position, change: Position, |} const origin: Position = { x: 0, y: 0 }; const smallestSigned = apply((value: number) => { if (value === 0) { return 0; } return value > 0 ? 1 : -1; }); type GetRemainderArgs = {| current: Position, max: Position, change: Position, |} // We need to figure out how much of the movement // cannot be done with a scroll export const getOverlap = (() => { const getRemainder = (target: number, max: number): number => { if (target < 0) { return target; } if (target > max) { return target - max; } return 0; }; return ({ current, max, change, }: GetRemainderArgs): ?Position => { const targetScroll: Position = add(current, change); const overlap: Position = { x: getRemainder(targetScroll.x, max.x), y: getRemainder(targetScroll.y, max.y), }; if (isEqual(overlap, origin)) { return null; } return overlap; }; })(); export const canPartiallyScroll = ({ max, current, change, }: CanScrollArgs): boolean => { // Only need to be able to move the smallest amount in the desired direction const smallestChange: Position = smallestSigned(change); const overlap: ?Position = getOverlap({ max, current, change: smallestChange, }); // no overlap at all - we can move there! if (!overlap) { return true; } // if there was an x value, but there is no x overlap - then we can scroll on the x! if (smallestChange.x !== 0 && overlap.x === 0) { return true; } // if there was an y value, but there is no y overlap - then we can scroll on the y! if (smallestChange.y !== 0 && overlap.y === 0) { return true; } return false; }; export const canScrollWindow = (viewport: Viewport, change: Position): boolean => canPartiallyScroll({ current: viewport.scroll, max: viewport.maxScroll, change, }); export const canScrollDroppable = ( droppable: DroppableDimension, change: Position, ): boolean => { const closestScrollable: ?ClosestScrollable = droppable.viewport.closestScrollable; // Cannot scroll when there is no scroll container! if (!closestScrollable) { return false; } return canPartiallyScroll({ current: closestScrollable.scroll.current, max: closestScrollable.scroll.max, change, }); }; export const getWindowOverlap = (viewport: Viewport, change: Position): ?Position => { if (!canScrollWindow(viewport, change)) { return null; } const max: Position = viewport.maxScroll; const current: Position = viewport.scroll; return getOverlap({ current, max, change, }); }; export const getDroppableOverlap = (droppable: DroppableDimension, change: Position): ?Position => { if (!canScrollDroppable(droppable, change)) { return null; } const closestScrollable: ?ClosestScrollable = droppable.viewport.closestScrollable; // Cannot scroll when there is no scroll container! if (!closestScrollable) { return null; } return getOverlap({ current: closestScrollable.scroll.current, max: closestScrollable.scroll.max, change, }); };