Skip to content

Instantly share code, notes, and snippets.

@tripolskypetr
Last active February 4, 2023 19:00
Show Gist options
  • Save tripolskypetr/ed87a9ba73c89c97ae942e42a895de57 to your computer and use it in GitHub Desktop.
Save tripolskypetr/ed87a9ba73c89c97ae942e42a895de57 to your computer and use it in GitHub Desktop.
Virtualized view for React
import * as React from "react";
import { useCallback, useState, useEffect, useLayoutEffect, useRef, useMemo } from "react";
import { makeStyles } from "../../styles";
import Box, { BoxProps } from "@mui/material/Box";
import useActualCallback from "../../hooks/useActualCallback";
import useSingleton from "../../hooks/useSingleton";
import throttle from "../../utils/hof/throttle";
import classNames from "../../utils/classNames";
const DEFAULT_MIN_HEIGHT = 60;
const DEFAULT_BUFFER_SIZE = 5;
const ROOT_ELEMENT = "virtual-view-root";
const CHILD_ELEMENT = "virtual-view-child";
const DATASET_ID = "list_item_idx";
interface IVirtualViewProps
extends Omit<
BoxProps,
keyof {
ref: never;
}
> {
loading?: boolean;
hasMore?: boolean;
minHeight?: number;
bufferSize?: number;
children: React.ReactNode;
onDataRequest?: () => void;
}
const useStyles = makeStyles()({
root: {
position: "relative",
overflowY: 'auto',
width: "100%",
minHeight: '50px',
},
adjust: {
position: 'absolute',
visibility: 'hidden',
height: 1,
width: 1,
left: 0,
},
});
export const VirtualView = ({
className,
minHeight = DEFAULT_MIN_HEIGHT,
bufferSize = DEFAULT_BUFFER_SIZE,
children: upperChildren,
hasMore = true,
loading = false,
onDataRequest = () => undefined,
...otherProps
}: IVirtualViewProps) => {
const { classes } = useStyles();
const isMounted = useRef(true);
const [rowHeightMap, setRowHeightMap] = useState(
() => new Map<number, number>()
);
const onDataRequest$ = useActualCallback(onDataRequest);
const elementRefMap = useSingleton(() => new Map<number, HTMLDivElement>());
const [scrollPosition, setScrollPosition] = useState(0);
const [containerHeight, setContainerHeight] = useState(0);
const resizeObserver = useSingleton(
() =>
new ResizeObserver(([record]) => {
const element = record.target as HTMLDivElement;
if (!element) {
return;
}
if (element.classList.contains(ROOT_ELEMENT)) {
setContainerHeight(element.offsetHeight);
}
if (element.classList.contains(CHILD_ELEMENT)) {
const elementId = Number(element.dataset[DATASET_ID]);
const { offsetHeight } = element;
if (!Number.isNaN(elementId) && offsetHeight > minHeight) {
setRowHeightMap((rowHeightMap) => {
rowHeightMap.set(elementId, element.offsetHeight);
return new Map(rowHeightMap);
});
}
}
})
);
const getStartIndex = useCallback(
(scrollPosition: number) => {
let startScrollPos = scrollPosition;
let idx = 0;
while (startScrollPos >= 0) {
startScrollPos -= rowHeightMap.get(idx) || minHeight;
idx++;
}
return Math.max(idx - bufferSize, 0);
},
[rowHeightMap, bufferSize]
);
const getEndIndex = useCallback(
(scrollPosition: number, totalLength: number) => {
let endScrollPos = scrollPosition + containerHeight;
let idx = 0;
while (endScrollPos >= 0) {
endScrollPos -= rowHeightMap.get(idx) || minHeight;
idx++;
}
return Math.min(idx - 1 + bufferSize, totalLength - 1);
},
[rowHeightMap, bufferSize, containerHeight]
);
const getTopPos = useCallback(
(elementIndex: number) => {
let totalTop = 0;
for (let idx = 0; idx !== elementIndex; idx++) {
totalTop += rowHeightMap.get(idx) || minHeight;
}
return totalTop;
},
[rowHeightMap, minHeight]
);
const scrollAdjust = useMemo(() => {
const children = React.Children.toArray(upperChildren);
let totalHeight = 0;
for (let idx = 0; idx !== children.length; idx++) {
totalHeight += rowHeightMap.get(idx) || minHeight;
}
return totalHeight;
}, [
rowHeightMap,
upperChildren,
minHeight,
]);
const visibleChildren = React.useMemo(() => {
const children = React.Children.toArray(upperChildren);
const startIndex = getStartIndex(scrollPosition);
const endIndex = getEndIndex(scrollPosition, children.length);
for (const [index, element] of elementRefMap.entries()) {
if (index < startIndex || index > endIndex) {
resizeObserver.unobserve(element);
elementRefMap.delete(index);
}
}
let isBottomReached = true;
isBottomReached = isBottomReached && hasMore;
isBottomReached = isBottomReached && !loading;
isBottomReached = isBottomReached && scrollPosition > containerHeight - 10;
isBottomReached = isBottomReached && children.length === endIndex + 1;
if (isBottomReached) {
onDataRequest$();
}
return children.slice(startIndex, endIndex + 1).map((child, index) =>
React.cloneElement(child as React.ReactElement, {
ref: (element: HTMLDivElement | null) => {
if (!element) {
return;
}
const elementIdx = startIndex + index;
const prevElement = elementRefMap.get(elementIdx);
if (element === prevElement) {
return;
}
prevElement && resizeObserver.unobserve(prevElement);
element.classList.add(CHILD_ELEMENT);
element.dataset[DATASET_ID] = String(elementIdx);
resizeObserver.observe(element);
elementRefMap.set(elementIdx, element);
setRowHeightMap((rowHeightMap) => {
let isChanged = true;
isChanged = isChanged && element.offsetHeight > minHeight;
isChanged = isChanged && rowHeightMap.get(elementIdx) !== element.offsetHeight;
if (isChanged) {
rowHeightMap.set(elementIdx, element.offsetHeight);
return new Map(rowHeightMap);
} else {
return rowHeightMap;
}
});
},
style: {
position: "absolute",
top: getTopPos(startIndex + index),
minHeight: minHeight,
minWidth: '100%',
left: 0,
},
})
);
}, [
hasMore,
loading,
upperChildren,
containerHeight,
minHeight,
scrollPosition,
rowHeightMap,
getStartIndex,
getEndIndex,
getTopPos,
onDataRequest$
]);
const handleRef = useCallback((element: HTMLDivElement | null) => {
if (element) {
element.addEventListener("scroll", throttle((e: any) => {
setScrollPosition(e.target.scrollTop)
}, 50));
setContainerHeight(element.offsetHeight);
}
}, []);
useEffect(
() => () => {
resizeObserver.disconnect();
},
[]
);
useLayoutEffect(
() => () => {
isMounted.current = false;
},
[]
);
return (
<Box
className={classNames(className, classes.root, ROOT_ELEMENT)}
{...otherProps}
ref={handleRef}
>
{visibleChildren}
<div
className={classes.adjust}
style={{
top: scrollAdjust,
}}
/>
</Box>
);
};
export default VirtualView;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment