Skip to content

Commit

Permalink
Merge pull request #189 from inokawa/revert-overscan
Browse files Browse the repository at this point in the history
Revert "Prerender items longer for scroll direction"
  • Loading branch information
inokawa authored Sep 28, 2023
2 parents 3ac54ad + 9ed3bc4 commit d011b3f
Show file tree
Hide file tree
Showing 7 changed files with 45 additions and 129 deletions.
2 changes: 1 addition & 1 deletion .size-limit.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
"name": "Total",
"path": "lib/index.mjs",
"import": "*",
"limit": "5.25 kB"
"limit": "5.20 kB"
},
{
"name": "VList",
Expand Down
3 changes: 1 addition & 2 deletions src/core/scroller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@ import {
ACTION_SCROLL_END,
UPDATE_SIZE,
ACTION_MANUAL_SCROLL,
SCROLL_IDLE,
} from "./store";
import { ScrollToIndexAlign } from "./types";
import { debounce, throttle, timeout, clamp } from "./utils";
Expand All @@ -20,7 +19,7 @@ const createOnWheel = (
onScrollStopped: () => void
) => {
return throttle((e: WheelEvent) => {
if (store._getScrollDirection() === SCROLL_IDLE) {
if (!store._getIsScrolling()) {
// Scroll start should be detected with scroll event
return;
}
Expand Down
20 changes: 12 additions & 8 deletions src/core/store.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ const SUBPIXEL_THRESHOLD = 1.5; // 0.5 * 3
export const SCROLL_IDLE = 0;
export const SCROLL_DOWN = 1;
export const SCROLL_UP = 2;
export type ScrollDirection =
type ScrollDirection =
| typeof SCROLL_IDLE
| typeof SCROLL_DOWN
| typeof SCROLL_UP;
Expand Down Expand Up @@ -69,7 +69,7 @@ type Subscriber = () => void;
export const UPDATE_SCROLL = 0b00001;
export const UPDATE_SIZE = 0b00010;
export const UPDATE_JUMP = 0b00100;
export const UPDATE_SCROLL_DIRECTION = 0b01000;
export const UPDATE_IS_SCROLLING = 0b01000;
export const UPDATE_SCROLL_WITH_EVENT = 0b10000;

export type VirtualStore = {
Expand All @@ -82,7 +82,7 @@ export type VirtualStore = {
_getItemsLength(): number;
_getScrollOffset(): number;
_getScrollOffsetMax(): number;
_getScrollDirection(): ScrollDirection;
_getIsScrolling(): boolean;
_getViewportSize(): number;
_getCorrectedScrollSize(): number;
_getJumpCount(): number;
Expand Down Expand Up @@ -122,8 +122,12 @@ export const createVirtualStore = (
const updateScrollDirection = (dir: ScrollDirection): boolean => {
const prev = _scrollDirection;
_scrollDirection = dir;

// Return true if scrolling is just started or stopped
return _scrollDirection !== prev;
return (
_scrollDirection !== prev &&
(_scrollDirection === SCROLL_IDLE || prev === SCROLL_IDLE)
);
};

return {
Expand Down Expand Up @@ -179,8 +183,8 @@ export const createVirtualStore = (
return scrollOffset;
},
_getScrollOffsetMax: getScrollOffsetMax,
_getScrollDirection() {
return _scrollDirection;
_getIsScrolling() {
return _scrollDirection !== SCROLL_IDLE;
},
_getViewportSize() {
return viewportSize;
Expand Down Expand Up @@ -321,7 +325,7 @@ export const createVirtualStore = (
!_isManualScrolling
) {
if (updateScrollDirection(delta < 0 ? SCROLL_UP : SCROLL_DOWN)) {
mutated += UPDATE_SCROLL_DIRECTION;
mutated += UPDATE_IS_SCROLLING;
}
}

Expand All @@ -344,7 +348,7 @@ export const createVirtualStore = (
}
case ACTION_SCROLL_END: {
if (updateScrollDirection(SCROLL_IDLE)) {
mutated = UPDATE_SCROLL_DIRECTION;
mutated = UPDATE_IS_SCROLLING;
}
_isShifting = _isManualScrolling = false;
break;
Expand Down
55 changes: 14 additions & 41 deletions src/react/VGrid.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,24 +10,18 @@ import {
} from "react";
import {
ACTION_ITEMS_LENGTH_CHANGE,
UPDATE_SCROLL_DIRECTION,
UPDATE_IS_SCROLLING,
UPDATE_JUMP,
UPDATE_SCROLL,
UPDATE_SIZE,
VirtualStore,
createVirtualStore,
SCROLL_IDLE,
} from "../core/store";
import { useIsomorphicLayoutEffect } from "./useIsomorphicLayoutEffect";
import { useSelector } from "./useSelector";
import { values } from "../core/utils";
import { max, min, values } from "../core/utils";
import { createScroller } from "../core/scroller";
import {
clampEndIndex,
clampStartIndex,
emptyComponents,
refKey,
} from "./utils";
import { emptyComponents, refKey } from "./utils";
import { useStatic } from "./useStatic";
import {
CustomViewportComponent,
Expand Down Expand Up @@ -321,15 +315,15 @@ export const VGrid = forwardRef<VGridHandle, VGridProps>(
hStore._getRange,
UPDATE_SCROLL + UPDATE_SIZE
);
const vScrollDirection = useSelector(
const verticalScrolling = useSelector(
vStore,
vStore._getScrollDirection,
UPDATE_SCROLL_DIRECTION
vStore._getIsScrolling,
UPDATE_IS_SCROLLING
);
const hScrollDirection = useSelector(
const horizontalScrolling = useSelector(
hStore,
hStore._getScrollDirection,
UPDATE_SCROLL_DIRECTION
hStore._getIsScrolling,
UPDATE_IS_SCROLLING
);
const vJumpCount = useSelector(vStore, vStore._getJumpCount, UPDATE_JUMP);
const hJumpCount = useSelector(hStore, hStore._getJumpCount, UPDATE_JUMP);
Expand All @@ -346,8 +340,6 @@ export const VGrid = forwardRef<VGridHandle, VGridProps>(
true
);
const rootRef = useRef<HTMLDivElement>(null);
const vScrolling = vScrollDirection !== SCROLL_IDLE;
const hScrolling = hScrollDirection !== SCROLL_IDLE;

useIsomorphicLayoutEffect(() => {
const root = rootRef[refKey]!;
Expand Down Expand Up @@ -421,29 +413,10 @@ export const VGrid = forwardRef<VGridHandle, VGridProps>(
};
}, [children]);

const overscanedStartRowIndex = clampStartIndex(
startRowIndex,
overscan,
vScrollDirection
);
const overscanedEndRowIndex = clampEndIndex(
endRowIndex,
overscan,
vScrollDirection,
rowCount
);
const overscanedStartColIndex = clampStartIndex(
startColIndex,
overscan,
hScrollDirection
);
const overscanedEndColIndex = clampEndIndex(
endColIndex,
overscan,
hScrollDirection,
colCount
);

const overscanedStartRowIndex = max(startRowIndex - overscan, 0);
const overscanedEndRowIndex = min(endRowIndex + overscan, rowCount - 1);
const overscanedStartColIndex = max(startColIndex - overscan, 0);
const overscanedEndColIndex = min(endColIndex + overscan, colCount - 1);
const items = useMemo(() => {
const res: ReactElement[] = [];
for (let i = overscanedStartRowIndex; i <= overscanedEndRowIndex; i++) {
Expand Down Expand Up @@ -478,7 +451,7 @@ export const VGrid = forwardRef<VGridHandle, VGridProps>(
ref={rootRef}
width={width}
height={height}
scrolling={vScrolling || hScrolling}
scrolling={verticalScrolling || horizontalScrolling}
attrs={useMemo(
() => ({
...viewportAttrs,
Expand Down
34 changes: 8 additions & 26 deletions src/react/VList.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,22 +13,14 @@ import {
createVirtualStore,
UPDATE_SIZE,
UPDATE_JUMP,
UPDATE_SCROLL_DIRECTION,
UPDATE_IS_SCROLLING,
UPDATE_SCROLL,
SCROLL_IDLE,
} from "../core/store";
import { useIsomorphicLayoutEffect } from "./useIsomorphicLayoutEffect";
import { useSelector } from "./useSelector";
import { exists, values } from "../core/utils";
import { exists, max, min, values } from "../core/utils";
import { createScroller } from "../core/scroller";
import {
MayHaveKey,
clampEndIndex,
clampStartIndex,
emptyComponents,
flattenChildren,
refKey,
} from "./utils";
import { MayHaveKey, emptyComponents, flattenChildren, refKey } from "./utils";
import { useStatic } from "./useStatic";
import { useLatestRef } from "./useLatestRef";
import { createResizer } from "../core/resizer";
Expand Down Expand Up @@ -235,10 +227,10 @@ export const VList = forwardRef<VListHandle, VListProps>(
store._getRange,
UPDATE_SCROLL + UPDATE_SIZE
);
const scrollDirection = useSelector(
const scrolling = useSelector(
store,
store._getScrollDirection,
UPDATE_SCROLL_DIRECTION
store._getIsScrolling,
UPDATE_IS_SCROLLING
);
const jumpCount = useSelector(store, store._getJumpCount, UPDATE_JUMP);
const scrollSize = useSelector(
Expand All @@ -248,7 +240,6 @@ export const VList = forwardRef<VListHandle, VListProps>(
true
);
const rootRef = useRef<HTMLDivElement>(null);
const scrolling = scrollDirection !== SCROLL_IDLE;

useIsomorphicLayoutEffect(() => {
const root = rootRef[refKey]!;
Expand Down Expand Up @@ -307,17 +298,8 @@ export const VList = forwardRef<VListHandle, VListProps>(
[]

Check warning on line 298 in src/react/VList.tsx

View workflow job for this annotation

GitHub Actions / check

React Hook useImperativeHandle has missing dependencies: 'scroller._scrollBy', 'scroller._scrollTo', 'scroller._scrollToIndex', and 'store'. Either include them or remove the dependency array
);

const overscanedStartIndex = clampStartIndex(
startIndex,
overscan,
scrollDirection
);
const overscanedEndIndex = clampEndIndex(
endIndex,
overscan,
scrollDirection,
count
);
const overscanedStartIndex = max(startIndex - overscan, 0);
const overscanedEndIndex = min(endIndex + overscan, count - 1);
const items = useMemo(() => {
const res: ReactElement[] = [];
for (let i = overscanedStartIndex; i <= overscanedEndIndex; i++) {
Expand Down
34 changes: 8 additions & 26 deletions src/react/WVList.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,25 +9,17 @@ import {
} from "react";
import {
ACTION_ITEMS_LENGTH_CHANGE,
UPDATE_SCROLL_DIRECTION,
UPDATE_IS_SCROLLING,
UPDATE_JUMP,
UPDATE_SCROLL,
UPDATE_SIZE,
createVirtualStore,
SCROLL_IDLE,
} from "../core/store";
import { useIsomorphicLayoutEffect } from "./useIsomorphicLayoutEffect";
import { useSelector } from "./useSelector";
import { exists, values } from "../core/utils";
import { exists, max, min, values } from "../core/utils";
import { createWindowScroller } from "../core/scroller";
import {
MayHaveKey,
clampEndIndex,
clampStartIndex,
emptyComponents,
flattenChildren,
refKey,
} from "./utils";
import { MayHaveKey, emptyComponents, flattenChildren, refKey } from "./utils";
import { useStatic } from "./useStatic";
import { useLatestRef } from "./useLatestRef";
import { createWindowResizer } from "../core/resizer";
Expand Down Expand Up @@ -181,10 +173,10 @@ export const WVList = forwardRef<WVListHandle, WVListProps>(
store._getRange,
UPDATE_SCROLL + UPDATE_SIZE
);
const scrollDirection = useSelector(
const scrolling = useSelector(
store,
store._getScrollDirection,
UPDATE_SCROLL_DIRECTION
store._getIsScrolling,
UPDATE_IS_SCROLLING
);
const jumpCount = useSelector(store, store._getJumpCount, UPDATE_JUMP);
const scrollSize = useSelector(
Expand All @@ -194,7 +186,6 @@ export const WVList = forwardRef<WVListHandle, WVListProps>(
true
);
const rootRef = useRef<HTMLDivElement>(null);
const scrolling = scrollDirection !== SCROLL_IDLE;

useIsomorphicLayoutEffect(() => {
const root = rootRef[refKey]!;
Expand Down Expand Up @@ -237,17 +228,8 @@ export const WVList = forwardRef<WVListHandle, WVListProps>(
[]
);

const overscanedStartIndex = clampStartIndex(
startIndex,
overscan,
scrollDirection
);
const overscanedEndIndex = clampEndIndex(
endIndex,
overscan,
scrollDirection,
count
);
const overscanedStartIndex = max(startIndex - overscan, 0);
const overscanedEndIndex = min(endIndex + overscan, count - 1);
const items = useMemo(() => {
const res: ReactElement[] = [];
for (let i = overscanedStartIndex; i <= overscanedEndIndex; i++) {
Expand Down
26 changes: 1 addition & 25 deletions src/react/utils.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { ReactElement, ReactFragment, ReactNode } from "react";
import { exists, max, min, isArray } from "../core/utils";
import { SCROLL_DOWN, SCROLL_UP, ScrollDirection } from "../core/store";
import { exists, isArray } from "../core/utils";

export const refKey = "current";

Expand Down Expand Up @@ -36,26 +35,3 @@ export const flattenChildren = (children: ReactNode): ItemElement[] => {
};

export type MayHaveKey = { key?: React.Key };

export const clampStartIndex = (
startIndex: number,
overscan: number,
scrollDirection: ScrollDirection
): number => {
return max(
startIndex - (scrollDirection === SCROLL_DOWN ? 1 : max(1, overscan)),
0
);
};

export const clampEndIndex = (
endIndex: number,
overscan: number,
scrollDirection: ScrollDirection,
count: number
): number => {
return min(
endIndex + (scrollDirection === SCROLL_UP ? 1 : max(1, overscan)),
count - 1
);
};

0 comments on commit d011b3f

Please sign in to comment.