2022-05-19 14:09:22 -07:00
|
|
|
import React, { useEffect, useRef } from 'react';
|
2022-05-19 18:03:52 -07:00
|
|
|
import { Virtuoso, Components, VirtuosoProps, VirtuosoHandle, ListRange } from 'react-virtuoso';
|
2022-04-21 19:50:12 -07:00
|
|
|
|
|
|
|
import PullToRefresh from 'soapbox/components/pull-to-refresh';
|
2022-05-01 12:36:29 -07:00
|
|
|
import { useSettings } from 'soapbox/hooks';
|
2022-04-21 19:50:12 -07:00
|
|
|
|
2022-05-01 12:28:31 -07:00
|
|
|
import LoadMore from './load_more';
|
2022-04-21 19:50:12 -07:00
|
|
|
import { Spinner, Text } from './ui';
|
|
|
|
|
2022-04-22 10:24:09 -07:00
|
|
|
type Context = {
|
|
|
|
itemClassName?: string,
|
|
|
|
listClassName?: string,
|
|
|
|
}
|
|
|
|
|
2022-06-01 16:47:07 -07:00
|
|
|
type SavedScrollPosition = {
|
|
|
|
index: number,
|
|
|
|
offset: number,
|
|
|
|
}
|
|
|
|
|
2022-04-22 10:24:09 -07:00
|
|
|
// NOTE: It's crucial to space lists with **padding** instead of margin!
|
|
|
|
// Pass an `itemClassName` like `pb-3`, NOT a `space-y-3` className
|
|
|
|
// https://virtuoso.dev/troubleshooting#list-does-not-scroll-to-the-bottom--items-jump-around
|
2022-04-23 09:28:28 -07:00
|
|
|
const Item: Components<Context>['Item'] = ({ context, ...rest }) => (
|
2022-04-22 10:24:09 -07:00
|
|
|
<div className={context?.itemClassName} {...rest} />
|
|
|
|
);
|
|
|
|
|
2022-04-22 08:23:53 -07:00
|
|
|
// Ensure the className winds up here
|
2022-04-23 09:28:28 -07:00
|
|
|
const List: Components<Context>['List'] = React.forwardRef((props, ref) => {
|
2022-04-21 19:50:12 -07:00
|
|
|
const { context, ...rest } = props;
|
2022-04-22 10:24:09 -07:00
|
|
|
return <div ref={ref} className={context?.listClassName} {...rest} />;
|
2022-04-21 19:50:12 -07:00
|
|
|
});
|
|
|
|
|
2022-05-13 18:23:03 -07:00
|
|
|
interface IScrollableList extends VirtuosoProps<any, any> {
|
2022-04-21 19:50:12 -07:00
|
|
|
scrollKey?: string,
|
|
|
|
onLoadMore?: () => void,
|
|
|
|
isLoading?: boolean,
|
|
|
|
showLoading?: boolean,
|
|
|
|
hasMore?: boolean,
|
2022-06-02 11:32:08 -07:00
|
|
|
prepend?: React.ReactNode,
|
2022-04-21 19:50:12 -07:00
|
|
|
alwaysPrepend?: boolean,
|
|
|
|
emptyMessage?: React.ReactNode,
|
|
|
|
children: Iterable<React.ReactNode>,
|
|
|
|
onScrollToTop?: () => void,
|
|
|
|
onScroll?: () => void,
|
2022-06-02 11:32:08 -07:00
|
|
|
placeholderComponent?: React.ComponentType | React.NamedExoticComponent,
|
2022-04-21 19:50:12 -07:00
|
|
|
placeholderCount?: number,
|
|
|
|
onRefresh?: () => Promise<any>,
|
|
|
|
className?: string,
|
2022-04-22 10:24:09 -07:00
|
|
|
itemClassName?: string,
|
2022-05-17 06:09:53 -07:00
|
|
|
id?: string,
|
2022-05-19 09:29:28 -07:00
|
|
|
style?: React.CSSProperties,
|
|
|
|
useWindowScroll?: boolean
|
2022-04-21 19:50:12 -07:00
|
|
|
}
|
|
|
|
|
2022-04-22 08:23:53 -07:00
|
|
|
/** Legacy ScrollableList with Virtuoso for backwards-compatibility */
|
2022-05-13 18:23:03 -07:00
|
|
|
const ScrollableList = React.forwardRef<VirtuosoHandle, IScrollableList>(({
|
2022-04-21 19:50:12 -07:00
|
|
|
prepend = null,
|
2022-04-22 08:23:53 -07:00
|
|
|
alwaysPrepend,
|
2022-04-21 19:50:12 -07:00
|
|
|
children,
|
|
|
|
isLoading,
|
|
|
|
emptyMessage,
|
|
|
|
showLoading,
|
|
|
|
onRefresh,
|
|
|
|
onScroll,
|
|
|
|
onScrollToTop,
|
|
|
|
onLoadMore,
|
|
|
|
className,
|
2022-04-22 10:24:09 -07:00
|
|
|
itemClassName,
|
2022-05-17 06:09:53 -07:00
|
|
|
id,
|
2022-04-22 08:23:53 -07:00
|
|
|
hasMore,
|
2022-04-21 19:50:12 -07:00
|
|
|
placeholderComponent: Placeholder,
|
|
|
|
placeholderCount = 0,
|
2022-05-13 18:23:03 -07:00
|
|
|
initialTopMostItemIndex = 0,
|
2022-05-19 09:29:28 -07:00
|
|
|
style = {},
|
|
|
|
useWindowScroll = true,
|
2022-05-13 18:23:03 -07:00
|
|
|
}, ref) => {
|
2022-05-01 12:36:29 -07:00
|
|
|
const settings = useSettings();
|
2022-05-01 12:28:31 -07:00
|
|
|
const autoloadMore = settings.get('autoloadMore');
|
|
|
|
|
2022-06-01 16:47:07 -07:00
|
|
|
// Preserve scroll position
|
|
|
|
const scrollDataKey = `soapbox:scrollData:${location.pathname}`;
|
|
|
|
const scrollData: SavedScrollPosition | null = JSON.parse(sessionStorage.getItem(scrollDataKey)!);
|
|
|
|
const topIndex = useRef<number>(scrollData ? scrollData.index : 0);
|
|
|
|
const topOffset = useRef<number>(scrollData ? scrollData.offset : 0);
|
2022-05-19 14:09:22 -07:00
|
|
|
|
2022-04-22 10:56:06 -07:00
|
|
|
/** Normalized children */
|
|
|
|
const elements = Array.from(children || []);
|
|
|
|
|
2022-04-21 19:50:12 -07:00
|
|
|
const showPlaceholder = showLoading && Placeholder && placeholderCount > 0;
|
2022-04-22 10:56:06 -07:00
|
|
|
|
|
|
|
// NOTE: We are doing some trickery to load a feed of placeholders
|
|
|
|
// Virtuoso's `EmptyPlaceholder` unfortunately doesn't work for our use-case
|
|
|
|
const data = showPlaceholder ? Array(placeholderCount).fill('') : elements;
|
|
|
|
const isEmpty = data.length === 0; // Yes, if it has placeholders it isn't "empty"
|
|
|
|
|
|
|
|
// Add a placeholder at the bottom for loading
|
|
|
|
// (Don't use Virtuoso's `Footer` component because it doesn't preserve its height)
|
2022-05-01 12:28:31 -07:00
|
|
|
if (hasMore && (autoloadMore || isLoading) && Placeholder) {
|
2022-04-22 10:56:06 -07:00
|
|
|
data.push(<Placeholder />);
|
2022-05-01 12:28:31 -07:00
|
|
|
} else if (hasMore && (autoloadMore || isLoading)) {
|
2022-04-22 10:56:06 -07:00
|
|
|
data.push(<Spinner />);
|
|
|
|
}
|
2022-04-21 19:50:12 -07:00
|
|
|
|
2022-06-01 16:47:07 -07:00
|
|
|
const handleScroll = () => {
|
|
|
|
const node = document.querySelector(`[data-virtuoso-scroller] [data-item-index="${topIndex.current}"]`);
|
|
|
|
if (node) {
|
|
|
|
topOffset.current = node.getBoundingClientRect().top * -1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-05-19 14:09:22 -07:00
|
|
|
useEffect(() => {
|
2022-06-01 16:47:07 -07:00
|
|
|
document.addEventListener('scroll', handleScroll);
|
|
|
|
sessionStorage.removeItem(scrollDataKey);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
const data = { index: topIndex.current, offset: topOffset.current };
|
|
|
|
sessionStorage.setItem(scrollDataKey, JSON.stringify(data));
|
|
|
|
document.removeEventListener('scroll', handleScroll);
|
|
|
|
};
|
2022-05-19 14:09:22 -07:00
|
|
|
}, []);
|
|
|
|
|
2022-04-22 08:23:53 -07:00
|
|
|
/* Render an empty state instead of the scrollable list */
|
2022-04-22 10:56:06 -07:00
|
|
|
const renderEmpty = (): JSX.Element => {
|
2022-04-22 08:23:53 -07:00
|
|
|
return (
|
|
|
|
<div className='mt-2'>
|
|
|
|
{alwaysPrepend && prepend}
|
|
|
|
|
|
|
|
<div className='bg-primary-50 dark:bg-slate-700 mt-2 rounded-lg text-center p-8'>
|
|
|
|
{isLoading ? (
|
|
|
|
<Spinner />
|
|
|
|
) : (
|
|
|
|
<Text>{emptyMessage}</Text>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-04-22 10:56:06 -07:00
|
|
|
/** Render a single item */
|
|
|
|
const renderItem = (_i: number, element: JSX.Element): JSX.Element => {
|
2022-04-21 19:50:12 -07:00
|
|
|
if (showPlaceholder) {
|
|
|
|
return <Placeholder />;
|
|
|
|
} else {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-23 14:43:37 -07:00
|
|
|
const handleEndReached = () => {
|
2022-05-01 12:28:31 -07:00
|
|
|
if (autoloadMore && hasMore && onLoadMore) {
|
2022-04-23 14:43:37 -07:00
|
|
|
onLoadMore();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-05-01 12:28:31 -07:00
|
|
|
const loadMore = () => {
|
|
|
|
if (autoloadMore || !hasMore || !onLoadMore) {
|
|
|
|
return null;
|
|
|
|
} else {
|
|
|
|
return <LoadMore visible={!isLoading} onClick={onLoadMore} />;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-06-01 16:47:07 -07:00
|
|
|
const handleRangeChange = (range: ListRange) => {
|
|
|
|
topIndex.current = range.startIndex;
|
2022-06-01 17:36:55 -07:00
|
|
|
handleScroll();
|
2022-05-19 18:03:52 -07:00
|
|
|
};
|
|
|
|
|
2022-04-22 10:56:06 -07:00
|
|
|
/** Render the actual Virtuoso list */
|
|
|
|
const renderFeed = (): JSX.Element => (
|
|
|
|
<Virtuoso
|
2022-05-13 18:23:03 -07:00
|
|
|
ref={ref}
|
2022-05-17 06:09:53 -07:00
|
|
|
id={id}
|
2022-05-19 09:29:28 -07:00
|
|
|
useWindowScroll={useWindowScroll}
|
2022-04-22 10:56:06 -07:00
|
|
|
className={className}
|
|
|
|
data={data}
|
|
|
|
startReached={onScrollToTop}
|
2022-04-23 14:43:37 -07:00
|
|
|
endReached={handleEndReached}
|
2022-04-22 10:56:06 -07:00
|
|
|
isScrolling={isScrolling => isScrolling && onScroll && onScroll()}
|
|
|
|
itemContent={renderItem}
|
2022-06-01 16:47:07 -07:00
|
|
|
initialTopMostItemIndex={showLoading ? 0 : initialTopMostItemIndex || (scrollData ? { align: 'start', index: scrollData.index, offset: scrollData.offset } : 0)}
|
|
|
|
rangeChanged={handleRangeChange}
|
2022-05-19 09:29:28 -07:00
|
|
|
style={style}
|
2022-04-22 10:56:06 -07:00
|
|
|
context={{
|
|
|
|
listClassName: className,
|
|
|
|
itemClassName,
|
|
|
|
}}
|
|
|
|
components={{
|
2022-06-02 11:32:08 -07:00
|
|
|
Header: () => <>{prepend}</>,
|
2022-04-22 10:56:06 -07:00
|
|
|
ScrollSeekPlaceholder: Placeholder as any,
|
|
|
|
EmptyPlaceholder: () => renderEmpty(),
|
|
|
|
List,
|
|
|
|
Item,
|
2022-05-01 12:28:31 -07:00
|
|
|
Footer: loadMore,
|
2022-04-22 10:56:06 -07:00
|
|
|
}}
|
2022-06-01 17:36:55 -07:00
|
|
|
overscan={{ main: 200, reverse: 200 }}
|
2022-04-22 10:56:06 -07:00
|
|
|
/>
|
|
|
|
);
|
2022-04-22 08:23:53 -07:00
|
|
|
|
2022-04-22 10:56:06 -07:00
|
|
|
/** Conditionally render inner elements */
|
|
|
|
const renderBody = (): JSX.Element => {
|
|
|
|
if (isEmpty) {
|
|
|
|
return renderEmpty();
|
|
|
|
} else {
|
|
|
|
return renderFeed();
|
|
|
|
}
|
|
|
|
};
|
2022-04-22 08:23:53 -07:00
|
|
|
|
2022-04-21 19:50:12 -07:00
|
|
|
return (
|
|
|
|
<PullToRefresh onRefresh={onRefresh}>
|
2022-04-22 10:56:06 -07:00
|
|
|
{renderBody()}
|
2022-04-21 19:50:12 -07:00
|
|
|
</PullToRefresh>
|
|
|
|
);
|
2022-05-13 18:23:03 -07:00
|
|
|
});
|
2022-04-21 19:50:12 -07:00
|
|
|
|
|
|
|
export default ScrollableList;
|