mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-02 12:47:41 +02:00
104 lines
4.1 KiB
TypeScript
104 lines
4.1 KiB
TypeScript
import * as React from 'react';
|
|
import shim from '@joplin/lib/shim';
|
|
import { Size } from '@joplin/utils/types';
|
|
import { useCallback, useState, useRef, useEffect, useMemo } from 'react';
|
|
|
|
const useScroll = (itemsPerLine: number, noteCount: number, itemSize: Size, listSize: Size, listRef: React.MutableRefObject<HTMLDivElement>) => {
|
|
const [scrollTop, setScrollTop] = useState(0);
|
|
const lastScrollSetTime = useRef(0);
|
|
|
|
const maxScrollTop = useMemo(() => {
|
|
return Math.max(0, itemSize.height * noteCount - listSize.height);
|
|
}, [itemSize.height, noteCount, listSize.height]);
|
|
|
|
// This ugly hack is necessary because setting scrollTop at a high
|
|
// frequency, while scrolling with the keyboard, is unreliable - the
|
|
// property will appear to be set (reading it back gives the correct value),
|
|
// but the scrollbar will not be at the expected position. That can be
|
|
// verified by moving the scrollbar a little and reading the event value -
|
|
// it will be different from what was set, and what was read.
|
|
//
|
|
// As a result, since we can't rely on setting or reading that value (to
|
|
// check if it's correct), we forcefully set it multiple times over the next
|
|
// few milliseconds, hoping that maybe one of these attempts will stick.
|
|
//
|
|
// This is most likely a race condition in either Chromium or Electron
|
|
// although I couldn't find an upstream issue.
|
|
//
|
|
// Setting the value only once after a short time, for example 10ms, helps
|
|
// but still fails now and then. Setting it after 500ms would probably work
|
|
// reliably but it's too slow so it makes sense to do it in an interval.
|
|
|
|
const setScrollTopLikeYouMeanItTimer = useRef(null);
|
|
const setScrollTopLikeYouMeanItStartTime = useRef(0);
|
|
const setScrollTopLikeYouMeanIt = useCallback((newScrollTop: number) => {
|
|
if (setScrollTopLikeYouMeanItTimer.current) shim.clearInterval(setScrollTopLikeYouMeanItTimer.current);
|
|
setScrollTopLikeYouMeanItStartTime.current = Date.now();
|
|
|
|
listRef.current.scrollTop = newScrollTop;
|
|
lastScrollSetTime.current = Date.now();
|
|
|
|
setScrollTopLikeYouMeanItTimer.current = shim.setInterval(() => {
|
|
if (!listRef.current) {
|
|
shim.clearInterval(setScrollTopLikeYouMeanItTimer.current);
|
|
setScrollTopLikeYouMeanItTimer.current = null;
|
|
return;
|
|
}
|
|
|
|
listRef.current.scrollTop = newScrollTop;
|
|
lastScrollSetTime.current = Date.now();
|
|
|
|
if (Date.now() - setScrollTopLikeYouMeanItStartTime.current > 1000) {
|
|
shim.clearInterval(setScrollTopLikeYouMeanItTimer.current);
|
|
setScrollTopLikeYouMeanItTimer.current = null;
|
|
}
|
|
}, 10);
|
|
}, [listRef]);
|
|
|
|
useEffect(() => {
|
|
if (setScrollTopLikeYouMeanItTimer.current) shim.clearInterval(setScrollTopLikeYouMeanItTimer.current);
|
|
setScrollTopLikeYouMeanItTimer.current = null;
|
|
}, []);
|
|
|
|
const makeItemIndexVisible = useCallback((itemIndex: number) => {
|
|
const lineTopFloat = scrollTop / itemSize.height;
|
|
const topFloat = lineTopFloat * itemsPerLine;
|
|
const lineBottomFloat = (scrollTop + listSize.height - itemSize.height) / itemSize.height;
|
|
const bottomFloat = lineBottomFloat * itemsPerLine;
|
|
const top = Math.min(noteCount - 1, Math.floor(topFloat) + 1);
|
|
const bottom = Math.max(0, Math.floor(bottomFloat));
|
|
|
|
if (itemIndex >= top && itemIndex <= bottom) return;
|
|
|
|
const lineIndex = Math.floor(itemIndex / itemsPerLine);
|
|
|
|
let newScrollTop = 0;
|
|
if (itemIndex < top) {
|
|
newScrollTop = itemSize.height * lineIndex;
|
|
} else {
|
|
newScrollTop = itemSize.height * (lineIndex + 1) - listSize.height;
|
|
}
|
|
|
|
if (newScrollTop < 0) newScrollTop = 0;
|
|
if (newScrollTop > maxScrollTop) newScrollTop = maxScrollTop;
|
|
|
|
setScrollTop(newScrollTop);
|
|
setScrollTopLikeYouMeanIt(newScrollTop);
|
|
}, [itemsPerLine, noteCount, itemSize.height, scrollTop, listSize.height, maxScrollTop, setScrollTopLikeYouMeanIt]);
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const onScroll = useCallback((event: any) => {
|
|
// Ignore the scroll event if it has just been set programmatically.
|
|
if (Date.now() - lastScrollSetTime.current < 500) return;
|
|
setScrollTop(event.target.scrollTop);
|
|
}, []);
|
|
|
|
return {
|
|
scrollTop,
|
|
onScroll,
|
|
makeItemIndexVisible,
|
|
};
|
|
};
|
|
|
|
export default useScroll;
|