From d484233cef022c50a2b70d856408c526a2d76e4a Mon Sep 17 00:00:00 2001 From: Alex Reardon Date: Fri, 18 May 2018 16:24:35 +1000 Subject: [PATCH] almost done with fluid scroller --- .../state/auto-scroll/fluid-scroller.spec.js | 484 ++++++------------ 1 file changed, 169 insertions(+), 315 deletions(-) diff --git a/test/unit/state/auto-scroll/fluid-scroller.spec.js b/test/unit/state/auto-scroll/fluid-scroller.spec.js index 88f409f03c..1f7af22519 100644 --- a/test/unit/state/auto-scroll/fluid-scroller.spec.js +++ b/test/unit/state/auto-scroll/fluid-scroller.spec.js @@ -35,7 +35,6 @@ import { getDraggableDimension, getDroppableDimension, getPreset, - withImpact, addDraggable, addDroppable, } from '../../../utils/dimension'; @@ -150,7 +149,7 @@ describe('fluid auto scrolling', () => { }; }; - describe.only('window scrolling', () => { + describe('window scrolling', () => { const thresholds: PixelThresholds = getPixelThresholds(scrollableViewport.subject, axis); const crossAxisThresholds: PixelThresholds = getPixelThresholds( scrollableViewport.subject, @@ -681,11 +680,11 @@ describe('fluid auto scrolling', () => { requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalled(); // moving forwards - const { id, scroll } = mocks.scrollDroppable.mock.calls[0]; + const { id, offset } = mocks.scrollDroppable.mock.calls[0][0]; expect(id).toBe(scrollable.descriptor.id); - expect(scroll[axis.line]).toBeGreaterThan(0); - expect(scroll[axis.crossAxisLine]).toBe(0); + expect(offset[axis.line]).toBeGreaterThan(0); + expect(offset[axis.crossAxisLine]).toBe(0); }); it('should throttle multiple scrolls into a single animation frame', () => { @@ -724,7 +723,7 @@ describe('fluid auto scrolling', () => { }), scrollable)); requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalledTimes(1); - const scroll1: Position = (mocks.scrollDroppable.mock.calls[0][1] : any); + const { offset: scroll1 } = mocks.scrollDroppable.mock.calls[0][0]; fluidScroll(addDroppable(dragTo({ selection: target2, @@ -732,7 +731,7 @@ describe('fluid auto scrolling', () => { }), scrollable)); requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalledTimes(2); - const scroll2: Position = (mocks.scrollDroppable.mock.calls[1][1] : any); + const { offset: scroll2 } = mocks.scrollDroppable.mock.calls[1][0]; expect(scroll1[axis.line]).toBeLessThan(scroll2[axis.line]); @@ -744,38 +743,32 @@ describe('fluid auto scrolling', () => { it('should have the top speed at the max speed point', () => { const expected: Position = patch(axis.line, config.maxScrollSpeed); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: onMaxBoundary, - viewport: unscrollableViewport, - }), scrollable), - ); + fluidScroll(addDroppable(dragTo({ + selection: onMaxBoundary, + viewport: unscrollableViewport, + }), scrollable)); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, - expected - ); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, + offset: expected, + }); }); it('should have the top speed when moving beyond the max speed point', () => { const target: Position = add(onMaxBoundary, patch(axis.line, 1)); const expected: Position = patch(axis.line, config.maxScrollSpeed); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrollable), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrollable)); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, - expected - ); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, + offset: expected, + }); }); it('should allow scrolling to the end of the droppable', () => { @@ -784,13 +777,10 @@ describe('fluid auto scrolling', () => { const maxChange: Position = getClosestScrollable(scrollable).scroll.max; const scrolled: DroppableDimension = scrollDroppable(scrollable, maxChange); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.flush(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -810,46 +800,30 @@ describe('fluid auto scrolling', () => { patch(axis.line, 1) ); const tooBigOnMainAxis: DraggableDimension = getDraggableDimension({ - descriptor: { - id: 'too big', - droppableId: preset.home.descriptor.id, - // after the last item - index: preset.inHomeList.length, - }, + descriptor: preset.inHome1.descriptor, borderBox: expanded, }); const selection: Position = onMaxBoundaryOfBoth; - const custom: State = (() => { - const base: State = state.dragging( - preset.inHome1.descriptor.id, - selection, - unscrollableViewport, + const custom: DraggingState = + addDroppable( + addDraggable( + state.dragging( + preset.inHome1.descriptor.id, + selection, + unscrollableViewport, + ), tooBigOnMainAxis + ), scrollable ); - const updated: State = { - ...base, - drag: { - ...base.drag, - initial: { - // $ExpectError - ...base.drag.initial, - descriptor: tooBigOnMainAxis.descriptor, - }, - }, - }; - - return addDroppable(addDraggable(updated, tooBigOnMainAxis), scrollable); - })(); - fluidScroll(custom); requestAnimationFrame.flush(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, // scroll ocurred on the cross axis, but not on the main axis - patch(axis.crossAxisLine, config.maxScrollSpeed) - ); + offset: patch(axis.crossAxisLine, config.maxScrollSpeed), + }); }); }); @@ -860,46 +834,25 @@ describe('fluid auto scrolling', () => { patch(axis.crossAxisLine, 1) ); const tooBigOnCrossAxis: DraggableDimension = getDraggableDimension({ - descriptor: { - id: 'too big', - droppableId: preset.home.descriptor.id, - // after the last item - index: preset.inHomeList.length, - }, + descriptor: preset.inHome1.descriptor, borderBox: expanded, }); const selection: Position = onMaxBoundaryOfBoth; - const custom: State = (() => { - const base: State = state.dragging( - preset.inHome1.descriptor.id, - selection, - unscrollableViewport, - ); - - const updated: State = { - ...base, - drag: { - ...base.drag, - initial: { - // $ExpectError - ...base.drag.initial, - descriptor: tooBigOnCrossAxis.descriptor, - }, - }, - }; - - return addDroppable(addDraggable(updated, tooBigOnCrossAxis), scrollable); - })(); + const custom: DraggingState = addDroppable(addDraggable(state.dragging( + preset.inHome1.descriptor.id, + selection, + unscrollableViewport, + ), tooBigOnCrossAxis), scrollable); fluidScroll(custom); requestAnimationFrame.flush(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, // scroll ocurred on the main axis, but not on the cross axis - patch(axis.line, config.maxScrollSpeed) - ); + offset: patch(axis.line, config.maxScrollSpeed), + }); }); }); @@ -910,37 +863,16 @@ describe('fluid auto scrolling', () => { patch(axis.line, 1, 1) ); const tooBig: DraggableDimension = getDraggableDimension({ - descriptor: { - id: 'too big', - droppableId: preset.home.descriptor.id, - // after the last item - index: preset.inHomeList.length, - }, + descriptor: preset.inHome1.descriptor, borderBox: expanded, }); const selection: Position = onMaxBoundaryOfBoth; - const custom: State = (() => { - const base: State = state.dragging( - preset.inHome1.descriptor.id, - selection, - unscrollableViewport, - ); - - const updated: State = { - ...base, - drag: { - ...base.drag, - initial: { - // $ExpectError - ...base.drag.initial, - descriptor: tooBig.descriptor, - }, - }, - }; - - return addDroppable(addDraggable(updated, tooBig), scrollable); - })(); + const custom: DraggingState = addDroppable(addDraggable(state.dragging( + preset.inHome1.descriptor.id, + selection, + unscrollableViewport, + ), tooBig), scrollable); fluidScroll(custom); @@ -957,13 +889,10 @@ describe('fluid auto scrolling', () => { const maxChange: Position = getClosestScrollable(scrollable).scroll.max; const scrolled: DroppableDimension = scrollDroppable(scrollable, maxChange); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.flush(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1001,17 +930,17 @@ describe('fluid auto scrolling', () => { const target: Position = add(onEndOfFrame, placeholder); const expected: Position = patch(axis.line, config.maxScrollSpeed); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - impact: overForeign, - viewport: unscrollableViewport, - }), scrolledForeign), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + impact: overForeign, + viewport: unscrollableViewport, + }), scrolledForeign)); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith(foreign.descriptor.id, expected); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: foreign.descriptor.id, + offset: expected, + }); }); it('should not allow scrolling past the placeholder buffer', () => { @@ -1029,14 +958,11 @@ describe('fluid auto scrolling', () => { patch(axis.line, 1), ); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - impact: overForeign, - viewport: unscrollableViewport, - }), scrolledForeign), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + impact: overForeign, + viewport: unscrollableViewport, + }), scrolledForeign)); requestAnimationFrame.flush(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1061,13 +987,10 @@ describe('fluid auto scrolling', () => { ); it('should not scroll if not past the start threshold', () => { - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: onStartBoundary, - viewport: unscrollableViewport, - }), scrolled) - ); + fluidScroll(addDroppable(dragTo({ + selection: onStartBoundary, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.flush(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1077,49 +1000,37 @@ describe('fluid auto scrolling', () => { // going backwards const target: Position = subtract(onStartBoundary, patch(axis.line, 1)); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); // only called after a frame requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalled(); - const [id, scroll] = mocks.scrollDroppable.mock.calls[0]; + const { id, offset } = mocks.scrollDroppable.mock.calls[0][0]; // validation expect(id).toBe(scrollable.descriptor.id); // moving backwards - expect(scroll[axis.line]).toBeLessThan(0); - expect(scroll[axis.crossAxisLine]).toBe(0); + expect(offset[axis.line]).toBeLessThan(0); + expect(offset[axis.crossAxisLine]).toBe(0); }); it('should throttle multiple scrolls into a single animation frame', () => { const target1: Position = subtract(onStartBoundary, patch(axis.line, 1)); const target2: Position = subtract(onStartBoundary, patch(axis.line, 2)); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target1, - viewport: unscrollableViewport, - }), scrolled), - ); - autoScroller.onStateChange( - addDroppable(dragTo({ - selection: target1, - viewport: unscrollableViewport, - }), scrolled), - addDroppable(dragTo({ - selection: target2, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target1, + viewport: unscrollableViewport, + }), scrolled)); + fluidScroll(addDroppable(dragTo({ + selection: target2, + viewport: unscrollableViewport, + }), scrolled)); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1138,30 +1049,21 @@ describe('fluid auto scrolling', () => { const target1: Position = subtract(onStartBoundary, patch(axis.line, 1)); const target2: Position = subtract(onStartBoundary, patch(axis.line, 2)); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target1, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target1, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalledTimes(1); - const scroll1: Position = (mocks.scrollDroppable.mock.calls[0][1] : any); + const { offset: scroll1 } = mocks.scrollDroppable.mock.calls[0][0]; - autoScroller.onStateChange( - addDroppable(dragTo({ - selection: target1, - viewport: unscrollableViewport, - }), scrolled), - addDroppable(dragTo({ - selection: target2, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target2, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalledTimes(2); - const scroll2: Position = (mocks.scrollDroppable.mock.calls[1][1] : any); + const { offset: scroll2 } = mocks.scrollDroppable.mock.calls[1][0]; // moving backwards expect(scroll1[axis.line]).toBeGreaterThan(scroll2[axis.line]); @@ -1174,73 +1076,46 @@ describe('fluid auto scrolling', () => { it('should have the top speed at the max speed point', () => { const expected: Position = patch(axis.line, -config.maxScrollSpeed); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: onMaxBoundary, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: onMaxBoundary, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, - expected - ); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, + offset: expected, + }); }); it('should have the top speed when moving beyond the max speed point', () => { const target: Position = subtract(onMaxBoundary, patch(axis.line, 1)); const expected: Position = patch(axis.line, -config.maxScrollSpeed); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrollable.descriptor.id, - expected - ); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrollable.descriptor.id, + offset: expected, + }); }); it('should not scroll if the item is too big', () => { const expanded: Spacing = expandByPosition(frameClient.borderBox, { x: 1, y: 1 }); const tooBig: DraggableDimension = getDraggableDimension({ - descriptor: { - id: 'too big', - droppableId: preset.home.descriptor.id, - // after the last item - index: preset.inHomeList.length, - }, + descriptor: preset.inHome1.descriptor, borderBox: expanded, }); const selection: Position = onMaxBoundary; - const custom: State = (() => { - const base: State = state.dragging( - preset.inHome1.descriptor.id, - selection, - unscrollableViewport, - ); - - const updated: State = { - ...base, - drag: { - ...base.drag, - initial: { - // $ExpectError - ...base.drag.initial, - descriptor: tooBig.descriptor, - }, - }, - }; - - return addDroppable(addDraggable(updated, tooBig), scrolled); - })(); + const custom: DraggingState = addDroppable(addDraggable(state.dragging( + preset.inHome1.descriptor.id, + selection, + unscrollableViewport, + ), tooBig), scrolled); fluidScroll(custom); @@ -1255,8 +1130,7 @@ describe('fluid auto scrolling', () => { } // scrolling to max scroll point - autoScroller.onStateChange( - state.idle, + fluidScroll( // scrollable cannot be scrolled backwards addDroppable(dragTo({ selection: target, @@ -1294,13 +1168,10 @@ describe('fluid auto scrolling', () => { it('should scroll if moving beyond the start threshold', () => { const target: Position = add(onStartBoundary, patch(axis.crossAxisLine, 1)); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1308,7 +1179,7 @@ describe('fluid auto scrolling', () => { requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalled(); // moving forwards - const [id, scroll] = mocks.scrollDroppable.mock.calls[0]; + const { id, offset: scroll } = mocks.scrollDroppable.mock.calls[0][0]; expect(id).toBe(scrolled.descriptor.id); expect(scroll[axis.crossAxisLine]).toBeGreaterThan(0); @@ -1328,13 +1199,10 @@ describe('fluid auto scrolling', () => { ); it('should not scroll if not past the start threshold', () => { - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: onStartBoundary, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: onStartBoundary, + viewport: unscrollableViewport, + }), scrolled)); requestAnimationFrame.flush(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1343,13 +1211,10 @@ describe('fluid auto scrolling', () => { it('should scroll if moving beyond the start threshold', () => { const target: Position = subtract(onStartBoundary, patch(axis.crossAxisLine, 1)); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - ); + fluidScroll(addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled)); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1357,8 +1222,8 @@ describe('fluid auto scrolling', () => { requestAnimationFrame.step(); expect(mocks.scrollDroppable).toHaveBeenCalled(); // moving backwards - const request: Position = mocks.scrollDroppable.mock.calls[0][1]; - expect(request[axis.crossAxisLine]).toBeLessThan(0); + const { offset } = mocks.scrollDroppable.mock.calls[0][0]; + expect(offset[axis.crossAxisLine]).toBeLessThan(0); }); }); @@ -1408,25 +1273,22 @@ describe('fluid auto scrolling', () => { ); // subject no longer visible expect(scrolled.viewport.clippedPageMarginBox).toBe(null); - // const target: Position = add(endOfFrame, patch(axis.line, 1)); + const custom: DraggingState = { + ...addDroppable(dragTo({ + selection: endOfFrame, + viewport: unscrollableViewport, + }), scrolled), + // being super clear that we are not currently over any droppable + impact: noImpact, + }; - autoScroller.onStateChange( - state.idle, - withImpact( - addDroppable(dragTo({ - selection: endOfFrame, - viewport: unscrollableViewport, - }), scrolled), - // being super clear that we are not currently over any droppable - noImpact, - ) - ); + fluidScroll(custom); requestAnimationFrame.step(); - expect(mocks.scrollDroppable).toHaveBeenCalledWith( - scrolled.descriptor.id, - maxScrollSpeed, - ); + expect(mocks.scrollDroppable).toHaveBeenCalledWith({ + id: scrolled.descriptor.id, + offset: maxScrollSpeed, + }); }); it('should not scroll the frame if not over the frame', () => { @@ -1438,18 +1300,16 @@ describe('fluid auto scrolling', () => { // subject no longer visible expect(scrolled.viewport.clippedPageMarginBox).toBe(null); const target: Position = add(endOfFrame, patch(axis.line, 1)); + const custom: DraggingState = { + ...addDroppable(dragTo({ + selection: target, + viewport: unscrollableViewport, + }), scrolled), + // being super clear that we are not currently over any droppable + impact: noImpact, + }; - autoScroller.onStateChange( - state.idle, - withImpact( - addDroppable(dragTo({ - selection: target, - viewport: unscrollableViewport, - }), scrolled), - // being super clear that we are not currently over any droppable - noImpact, - ) - ); + fluidScroll(custom); requestAnimationFrame.step(); expect(mocks.scrollDroppable).not.toHaveBeenCalled(); @@ -1487,13 +1347,10 @@ describe('fluid auto scrolling', () => { scrollableViewport.subject.center[axis.crossAxisLine], ); - autoScroller.onStateChange( - state.idle, - addDroppable(dragTo({ - selection: onMaxBoundary, - viewport: scrollableViewport, - }), custom), - ); + fluidScroll(addDroppable(dragTo({ + selection: onMaxBoundary, + viewport: scrollableViewport, + }), custom)); requestAnimationFrame.step(); expect(mocks.scrollWindow).toHaveBeenCalled(); @@ -1501,7 +1358,7 @@ describe('fluid auto scrolling', () => { }); }); - describe('on drag end', () => { + describe.skip('on drag end', () => { const endDragStates = [ state.idle, state.dropAnimating(), @@ -1529,10 +1386,7 @@ describe('fluid auto scrolling', () => { expect(mocks.scrollWindow).not.toHaveBeenCalled(); // should cancel the next frame - autoScroller.onStateChange(dragTo({ - selection: onMaxBoundary, - viewport: scrollableViewport, - }), end); + fluidScroll(end); requestAnimationFrame.flush(); expect(mocks.scrollWindow).not.toHaveBeenCalled();