diff --git a/CHANGELOG.md b/CHANGELOG.md index ead8b93bb3..ac7775d99a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,7 @@ ## Added - Add "Learn More" button to "Disappearing Messages" dialog #4330 - new icon for Mac users +- smooth-scroll to newly arriving messages instead of jumping instantly #4125 ## Changed - enable Telegram-style Ctrl + ArrowUp to reply by default #4333 diff --git a/packages/frontend/src/components/message/MessageList.tsx b/packages/frontend/src/components/message/MessageList.tsx index a949363341..91c4c2db41 100644 --- a/packages/frontend/src/components/message/MessageList.tsx +++ b/packages/frontend/src/components/message/MessageList.tsx @@ -209,6 +209,9 @@ export default function MessageList({ accountId, chat, refComposer }: Props) { } }, [jumpToMessage]) + const pendingProgrammaticSmoothScrollTo = useRef(null) + const pendingProgrammaticSmoothScrollTimeout = useRef(-1) + const onScroll = useCallback( (ev: React.UIEvent | null) => { if (!messageListRef.current) { @@ -277,6 +280,10 @@ export default function MessageList({ accountId, chat, refComposer }: Props) { showJumpDownButton, ] ) + const onScrollEnd = useCallback((_ev: Event) => { + clearTimeout(pendingProgrammaticSmoothScrollTimeout.current) + pendingProgrammaticSmoothScrollTo.current = null + }, []) // This `useLayoutEffect` is made to run whenever `viewState` changes. // `viewState` controls the desired scroll position of `messageListRef`. @@ -293,6 +300,33 @@ export default function MessageList({ accountId, chat, refComposer }: Props) { return } + if (pendingProgrammaticSmoothScrollTo.current != null) { + // Let's finish the pending scroll immediately + // so that our further calculations that are based on `scrollTop` + // (e.g. whether we're close to the bottom (`ifClose`)) are correct. + // + // FYI instead of interrupting the pending scroll, we could + // postpone calling `unlockScroll` when initiating a smooth scroll + // until the said scroll finishes (see `scheduler.lockedQueuedEffect()`). + // This would queue new scrollTo "events" until after + // the smooth scroll finishes. + log.debug( + 'New viewState received, but a previous programmatic smooth scroll ' + + "is pending. Let's finish the pending one immediately. " + + `Scrolling to ${pendingProgrammaticSmoothScrollTo.current}` + ) + messageListRef.current.scrollTop = + pendingProgrammaticSmoothScrollTo.current + clearTimeout(pendingProgrammaticSmoothScrollTimeout.current) + pendingProgrammaticSmoothScrollTo.current = null + + // But keep in mind that we record `lastKnownScrollTop` + // in `chat_view_reducer`, and that recording could happen during + // a pending smooth scroll. + // This does not appear to matter though. We don't use + // `lastKnownScrollTop` too much. + } + const { scrollTo, lastKnownScrollHeight } = viewState log.debug( @@ -407,7 +441,44 @@ export default function MessageList({ accountId, chat, refComposer }: Props) { ) if (shouldScrollToBottom) { - messageListRef.current.scrollTop = messageListRef.current.scrollHeight + const scrollTo = messageListRef.current.scrollHeight + // Smooth scroll for newly arrived messages. + // TODO also add this for self-sent messages. + // In that case 'scrollToMessage' is used though... + messageListRef.current.scrollTo({ + top: scrollTo, + behavior: 'smooth', + }) + pendingProgrammaticSmoothScrollTo.current = scrollTo + + // Smooth scroll duration is not defined by the spec: + // https://drafts.csswg.org/cssom-view/#scrolling: + // > in a user-agent-defined fashion + // > over a user-agent-defined amount of time + // As of 2024-09, on Firefox it appears to range from + // 300 to 1000 ms, depending on scroll amount. + // On Chromium: 50-700 + const smoothScrollMaxDuration = 1000 + + // Why is 'scrollend' event not enough? + // - Because the user might interrup such a scroll and start scrolling + // wherever they like, and 'scrollend' won't fire + // until they finish scrolling. + // - Because 'scrollend' is not supported by WebKit yet + // https://webkit.org/b/201556 + // and we'll be running on WebKit when we switch to Tauri. + clearTimeout(pendingProgrammaticSmoothScrollTimeout.current) + pendingProgrammaticSmoothScrollTimeout.current = window.setTimeout( + () => { + pendingProgrammaticSmoothScrollTo.current = null + + console.warn( + 'Smooth scroll: scrollend did not fire before timeout.\n' + + 'Did the user scroll, or did the smooth scroll take so long?' + ) + }, + smoothScrollMaxDuration + ) } } else { log.debug( @@ -541,6 +612,7 @@ export default function MessageList({ accountId, chat, refComposer }: Props) { > ) => void + onScrollEnd: (event: Event) => void oldestFetchedMessageIndex: number messageListItems: T.MessageListItem[] activeView: T.MessageListItem[] @@ -587,6 +660,7 @@ export const MessageListInner = React.memo( }) => { const { onScroll, + onScrollEnd, messageListItems, messageCache, activeView, @@ -696,6 +770,20 @@ export const MessageListInner = React.memo( } }, [hasChatChanged]) + // onScrollend is not defined in React, let's attach manually... + useEffect(() => { + const el = messageListRef.current + if (!el) { + return + } + + el.addEventListener('scrollend', onScrollEnd) + return () => el.removeEventListener('scrollend', onScrollEnd) + + // Yes, re-run on every re-render, because `messageListRef` might change + // over the lifetime of this component. + }) + if (!loaded) { return (