// Copyright 2019-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { first, get, isNumber, last, pick, throttle } from 'lodash'; import classNames from 'classnames'; import type { ReactChild, ReactNode, RefObject } from 'react'; import React from 'react'; import { createSelector } from 'reselect'; import Measure from 'react-measure'; import { ScrollDownButton } from './ScrollDownButton'; import type { AssertProps, LocalizerType, ThemeType } from '../../types/Util'; import type { ConversationType } from '../../state/ducks/conversations'; import type { PreferredBadgeSelectorType } from '../../state/selectors/badges'; import { assertDev, strictAssert } from '../../util/assert'; import { missingCaseError } from '../../util/missingCaseError'; import { clearTimeoutIfNecessary } from '../../util/clearTimeoutIfNecessary'; import { WidthBreakpoint } from '../_util'; import type { PropsActions as MessageActionsType } from './Message'; import type { PropsActions as UnsupportedMessageActionsType } from './UnsupportedMessage'; import type { PropsActionsType as ChatSessionRefreshedNotificationActionsType } from './ChatSessionRefreshedNotification'; import type { PropsActionsType as GroupV2ChangeActionsType } from './GroupV2Change'; import { ErrorBoundary } from './ErrorBoundary'; import type { PropsActions as SafetyNumberActionsType } from './SafetyNumberNotification'; import { Intl } from '../Intl'; import { TimelineWarning } from './TimelineWarning'; import { TimelineWarnings } from './TimelineWarnings'; import { NewlyCreatedGroupInvitedContactsDialog } from '../NewlyCreatedGroupInvitedContactsDialog'; import { ContactSpoofingType } from '../../util/contactSpoofing'; import type { PropsType as SmartContactSpoofingReviewDialogPropsType } from '../../state/smart/ContactSpoofingReviewDialog'; import type { GroupNameCollisionsWithIdsByTitle } from '../../util/groupMemberNameCollisions'; import { hasUnacknowledgedCollisions } from '../../util/groupMemberNameCollisions'; import { TimelineFloatingHeader } from './TimelineFloatingHeader'; import { getScrollAnchorBeforeUpdate, getWidthBreakpoint, ScrollAnchor, TimelineMessageLoadingState, UnreadIndicatorPlacement, } from '../../util/timelineUtil'; import { getScrollBottom, scrollToBottom, setScrollBottom, } from '../../util/scrollUtil'; import { LastSeenIndicator } from './LastSeenIndicator'; import { MINUTE } from '../../util/durations'; const AT_BOTTOM_THRESHOLD = 15; const AT_BOTTOM_DETECTOR_STYLE = { height: AT_BOTTOM_THRESHOLD }; const MIN_ROW_HEIGHT = 18; const SCROLL_DOWN_BUTTON_THRESHOLD = 8; const LOAD_NEWER_THRESHOLD = 5; export type WarningType = | { type: ContactSpoofingType.DirectConversationWithSameTitle; safeConversation: ConversationType; } | { type: ContactSpoofingType.MultipleGroupMembersWithSameTitle; acknowledgedGroupNameCollisions: GroupNameCollisionsWithIdsByTitle; groupNameCollisions: GroupNameCollisionsWithIdsByTitle; }; export type ContactSpoofingReviewPropType = | { type: ContactSpoofingType.DirectConversationWithSameTitle; possiblyUnsafeConversation: ConversationType; safeConversation: ConversationType; } | { type: ContactSpoofingType.MultipleGroupMembersWithSameTitle; collisionInfoByTitle: Record< string, Array<{ oldName?: string; conversation: ConversationType; }> >; }; export type PropsDataType = { haveNewest: boolean; haveOldest: boolean; messageChangeCounter: number; messageLoadingState?: TimelineMessageLoadingState; isNearBottom?: boolean; items: ReadonlyArray; oldestUnseenIndex?: number; scrollToIndex?: number; scrollToIndexCounter: number; totalUnseen: number; }; type PropsHousekeepingType = { id: string; isConversationSelected: boolean; isGroupV1AndDisabled?: boolean; isIncomingMessageRequest: boolean; isSomeoneTyping: boolean; unreadCount?: number; selectedMessageId?: string; invitedContactsForNewlyCreatedGroup: Array; warning?: WarningType; contactSpoofingReview?: ContactSpoofingReviewPropType; discardMessages: ( _: Readonly< | { conversationId: string; numberToKeepAtBottom: number; } | { conversationId: string; numberToKeepAtTop: number } > ) => void; getTimestampForMessage: (messageId: string) => undefined | number; getPreferredBadge: PreferredBadgeSelectorType; i18n: LocalizerType; theme: ThemeType; renderItem: (props: { actionProps: PropsActionsType; containerElementRef: RefObject; containerWidthBreakpoint: WidthBreakpoint; conversationId: string; isOldestTimelineItem: boolean; messageId: string; nextMessageId: undefined | string; previousMessageId: undefined | string; unreadIndicatorPlacement: undefined | UnreadIndicatorPlacement; }) => JSX.Element; renderHeroRow: ( id: string, unblurAvatar: () => void, updateSharedGroups: () => unknown ) => JSX.Element; renderTypingBubble: (id: string) => JSX.Element; renderContactSpoofingReviewDialog: ( props: SmartContactSpoofingReviewDialogPropsType ) => JSX.Element; }; export type PropsActionsType = { acknowledgeGroupMemberNameCollisions: ( groupNameCollisions: Readonly ) => void; clearInvitedUuidsForNewlyCreatedGroup: () => void; closeContactSpoofingReview: () => void; setIsNearBottom: (conversationId: string, isNearBottom: boolean) => unknown; reviewGroupMemberNameCollision: (groupConversationId: string) => void; reviewMessageRequestNameCollision: ( _: Readonly<{ safeConversationId: string; }> ) => void; learnMoreAboutDeliveryIssue: () => unknown; loadOlderMessages: (messageId: string) => unknown; loadNewerMessages: (messageId: string) => unknown; loadNewestMessages: (messageId: string, setFocus?: boolean) => unknown; markMessageRead: (messageId: string) => unknown; onBlock: (conversationId: string) => unknown; onBlockAndReportSpam: (conversationId: string) => unknown; onDelete: (conversationId: string) => unknown; onUnblock: (conversationId: string) => unknown; peekGroupCallForTheFirstTime: (conversationId: string) => unknown; peekGroupCallIfItHasMembers: (conversationId: string) => unknown; removeMember: (conversationId: string) => unknown; selectMessage: (messageId: string, conversationId: string) => unknown; clearSelectedMessage: () => unknown; unblurAvatar: () => void; updateSharedGroups: () => unknown; } & MessageActionsType & SafetyNumberActionsType & UnsupportedMessageActionsType & GroupV2ChangeActionsType & ChatSessionRefreshedNotificationActionsType; export type PropsType = PropsDataType & PropsHousekeepingType & PropsActionsType; type StateType = { hasDismissedDirectContactSpoofingWarning: boolean; hasRecentlyScrolled: boolean; lastMeasuredWarningHeight: number; newestBottomVisibleMessageId?: string; oldestPartiallyVisibleMessageId?: string; widthBreakpoint: WidthBreakpoint; }; const scrollToUnreadIndicator = Symbol('scrollToUnreadIndicator'); type SnapshotType = | null | typeof scrollToUnreadIndicator | { scrollToIndex: number } | { scrollTop: number } | { scrollBottom: number }; const getActions = createSelector( // It is expensive to pick so many properties out of the `props` object so we // use `createSelector` to memoize them by the last seen `props` object. (props: PropsType) => props, (props: PropsType): PropsActionsType => { const unsafe = pick(props, [ 'acknowledgeGroupMemberNameCollisions', 'blockGroupLinkRequests', 'clearInvitedUuidsForNewlyCreatedGroup', 'closeContactSpoofingReview', 'setIsNearBottom', 'reviewGroupMemberNameCollision', 'reviewMessageRequestNameCollision', 'learnMoreAboutDeliveryIssue', 'loadOlderMessages', 'loadNewerMessages', 'loadNewestMessages', 'markMessageRead', 'markViewed', 'onBlock', 'onBlockAndReportSpam', 'onDelete', 'onUnblock', 'peekGroupCallForTheFirstTime', 'peekGroupCallIfItHasMembers', 'removeMember', 'selectMessage', 'clearSelectedMessage', 'unblurAvatar', 'updateSharedGroups', 'doubleCheckMissingQuoteReference', 'checkForAccount', 'reactToMessage', 'replyToMessage', 'retryDeleteForEveryone', 'retrySend', 'showForwardMessageModal', 'deleteMessage', 'deleteMessageForEveryone', 'showMessageDetail', 'openConversation', 'openGiftBadge', 'showContactDetail', 'showContactModal', 'kickOffAttachmentDownload', 'markAttachmentAsCorrupted', 'messageExpanded', 'showVisualAttachment', 'downloadAttachment', 'displayTapToViewMessage', 'openLink', 'scrollToQuotedMessage', 'showExpiredIncomingTapToViewToast', 'showExpiredOutgoingTapToViewToast', 'startConversation', 'showIdentity', 'downloadNewVersion', 'contactSupport', 'viewStory', ]); const safe: AssertProps = unsafe; return safe; } ); export class Timeline extends React.Component< PropsType, StateType, SnapshotType > { private readonly containerRef = React.createRef(); private readonly messagesRef = React.createRef(); private readonly atBottomDetectorRef = React.createRef(); private readonly lastSeenIndicatorRef = React.createRef(); private intersectionObserver?: IntersectionObserver; private intersectionObserverCallbackFrame?: number; // This is a best guess. It will likely be overridden when the timeline is measured. private maxVisibleRows = Math.ceil(window.innerHeight / MIN_ROW_HEIGHT); private hasRecentlyScrolledTimeout?: NodeJS.Timeout; private delayedPeekTimeout?: NodeJS.Timeout; private peekInterval?: NodeJS.Timeout; override state: StateType = { hasRecentlyScrolled: true, hasDismissedDirectContactSpoofingWarning: false, // These may be swiftly overridden. lastMeasuredWarningHeight: 0, widthBreakpoint: WidthBreakpoint.Wide, }; private onScroll = (): void => { this.setState(oldState => // `onScroll` is called frequently, so it's performance-sensitive. We try our best // to return `null` from this updater because [that won't cause a re-render][0]. // // [0]: https://github.com/facebook/react/blob/29b7b775f2ecf878eaf605be959d959030598b07/packages/react-reconciler/src/ReactUpdateQueue.js#L401-L404 oldState.hasRecentlyScrolled ? null : { hasRecentlyScrolled: true } ); clearTimeoutIfNecessary(this.hasRecentlyScrolledTimeout); this.hasRecentlyScrolledTimeout = setTimeout(() => { this.setState({ hasRecentlyScrolled: false }); }, 3000); }; private scrollToItemIndex(itemIndex: number): void { this.messagesRef.current ?.querySelector(`[data-item-index="${itemIndex}"]`) ?.scrollIntoViewIfNeeded(); } private scrollToBottom = (setFocus?: boolean): void => { const { selectMessage, id, items } = this.props; if (setFocus && items && items.length > 0) { const lastIndex = items.length - 1; const lastMessageId = items[lastIndex]; selectMessage(lastMessageId, id); } else { const containerEl = this.containerRef.current; if (containerEl) { scrollToBottom(containerEl); } } }; private onClickScrollDownButton = (): void => { this.scrollDown(false); }; private scrollDown = (setFocus?: boolean): void => { const { haveNewest, id, items, loadNewestMessages, messageLoadingState, oldestUnseenIndex, selectMessage, } = this.props; const { newestBottomVisibleMessageId } = this.state; if (!items || items.length < 1) { return; } if (messageLoadingState) { this.scrollToBottom(setFocus); return; } if ( newestBottomVisibleMessageId && isNumber(oldestUnseenIndex) && items.findIndex(item => item === newestBottomVisibleMessageId) < oldestUnseenIndex ) { if (setFocus) { const messageId = items[oldestUnseenIndex]; selectMessage(messageId, id); } else { this.scrollToItemIndex(oldestUnseenIndex); } } else if (haveNewest) { this.scrollToBottom(setFocus); } else { const lastId = last(items); if (lastId) { loadNewestMessages(lastId, setFocus); } } }; private isAtBottom(): boolean { const containerEl = this.containerRef.current; if (!containerEl) { return false; } const isScrolledNearBottom = getScrollBottom(containerEl) <= AT_BOTTOM_THRESHOLD; const hasScrollbars = containerEl.clientHeight < containerEl.scrollHeight; return isScrolledNearBottom || !hasScrollbars; } private updateIntersectionObserver(): void { const containerEl = this.containerRef.current; const messagesEl = this.messagesRef.current; const atBottomDetectorEl = this.atBottomDetectorRef.current; if (!containerEl || !messagesEl || !atBottomDetectorEl) { return; } const { haveNewest, haveOldest, id, items, loadNewerMessages, loadOlderMessages, messageLoadingState, setIsNearBottom, } = this.props; // We re-initialize the `IntersectionObserver`. We don't want stale references to old // props, and we care about the order of `IntersectionObserverEntry`s. (We could do // this another way, but this approach works.) this.intersectionObserver?.disconnect(); if (this.intersectionObserverCallbackFrame !== undefined) { window.cancelAnimationFrame(this.intersectionObserverCallbackFrame); } const intersectionRatios = new Map(); const intersectionObserverCallback: IntersectionObserverCallback = entries => { // The first time this callback is called, we'll get entries in observation order // (which should match DOM order). We don't want to delete anything from our map // because we don't want the order to change at all. entries.forEach(entry => { intersectionRatios.set(entry.target, entry.intersectionRatio); }); let newIsNearBottom = false; let oldestPartiallyVisible: undefined | Element; let newestPartiallyVisible: undefined | Element; let newestFullyVisible: undefined | Element; for (const [element, intersectionRatio] of intersectionRatios) { if (intersectionRatio === 0) { continue; } // We use this "at bottom detector" for two reasons, both for performance. It's // usually faster to use an `IntersectionObserver` instead of a scroll event, // and we want to do that here. // // 1. We can determine whether we're near the bottom without `onScroll` // 2. We need this information when deciding whether the bottom of the last // message is visible. We want to get an intersection observer event when the // bottom of the container comes into view. if (element === atBottomDetectorEl) { newIsNearBottom = true; } else { oldestPartiallyVisible = oldestPartiallyVisible || element; newestPartiallyVisible = element; if (intersectionRatio === 1) { newestFullyVisible = element; } } } // If a message is fully visible, then you can see its bottom. If not, there's a // very tall message around. We assume you can see the bottom of a message if // (1) another message is partly visible right below it, or (2) you're near the // bottom of the scrollable container. let newestBottomVisible: undefined | Element; if (newestFullyVisible) { newestBottomVisible = newestFullyVisible; } else if ( newIsNearBottom || newestPartiallyVisible !== oldestPartiallyVisible ) { newestBottomVisible = oldestPartiallyVisible; } const oldestPartiallyVisibleMessageId = getMessageIdFromElement( oldestPartiallyVisible ); const newestBottomVisibleMessageId = getMessageIdFromElement(newestBottomVisible); this.setState({ oldestPartiallyVisibleMessageId, newestBottomVisibleMessageId, }); setIsNearBottom(id, newIsNearBottom); if (newestBottomVisibleMessageId) { this.markNewestBottomVisibleMessageRead(); const rowIndex = getRowIndexFromElement(newestBottomVisible); const maxRowIndex = items.length - 1; if ( !messageLoadingState && !haveNewest && isNumber(rowIndex) && maxRowIndex >= 0 && rowIndex >= maxRowIndex - LOAD_NEWER_THRESHOLD ) { loadNewerMessages(newestBottomVisibleMessageId); } } if ( !messageLoadingState && !haveOldest && oldestPartiallyVisibleMessageId && oldestPartiallyVisibleMessageId === items[0] ) { loadOlderMessages(oldestPartiallyVisibleMessageId); } }; this.intersectionObserver = new IntersectionObserver( (entries, observer) => { assertDev( this.intersectionObserver === observer, 'observer.disconnect() should prevent callbacks from firing' ); // `react-measure` schedules the callbacks on the next tick and so // should we because we want other parts of this component to respond // to resize events before we recalculate what is visible. this.intersectionObserverCallbackFrame = window.requestAnimationFrame( () => { // Observer was updated from under us if (this.intersectionObserver !== observer) { return; } intersectionObserverCallback(entries, observer); } ); }, { root: containerEl, threshold: [0, 1], } ); for (const child of messagesEl.children) { if ((child as HTMLElement).dataset.messageId) { this.intersectionObserver.observe(child); } } this.intersectionObserver.observe(atBottomDetectorEl); } private markNewestBottomVisibleMessageRead = throttle((): void => { const { markMessageRead } = this.props; const { newestBottomVisibleMessageId } = this.state; if (newestBottomVisibleMessageId) { markMessageRead(newestBottomVisibleMessageId); } }, 500); public override componentDidMount(): void { const containerEl = this.containerRef.current; const messagesEl = this.messagesRef.current; const { isConversationSelected } = this.props; strictAssert( // We don't render anything unless the conversation is selected (containerEl && messagesEl) || !isConversationSelected, ' mounted without some refs' ); this.updateIntersectionObserver(); window.SignalContext.activeWindowService.registerForActive( this.markNewestBottomVisibleMessageRead ); this.delayedPeekTimeout = setTimeout(() => { const { id, peekGroupCallForTheFirstTime } = this.props; this.delayedPeekTimeout = undefined; peekGroupCallForTheFirstTime(id); }, 500); this.peekInterval = setInterval(() => { const { id, peekGroupCallIfItHasMembers } = this.props; peekGroupCallIfItHasMembers(id); }, MINUTE); } public override componentWillUnmount(): void { const { delayedPeekTimeout, peekInterval } = this; window.SignalContext.activeWindowService.unregisterForActive( this.markNewestBottomVisibleMessageRead ); this.intersectionObserver?.disconnect(); clearTimeoutIfNecessary(delayedPeekTimeout); if (peekInterval) { clearInterval(peekInterval); } } public override getSnapshotBeforeUpdate( prevProps: Readonly ): SnapshotType { const containerEl = this.containerRef.current; if (!containerEl) { return null; } const { props } = this; const { scrollToIndex } = props; const scrollAnchor = getScrollAnchorBeforeUpdate( prevProps, props, this.isAtBottom() ); switch (scrollAnchor) { case ScrollAnchor.ChangeNothing: return null; case ScrollAnchor.ScrollToBottom: return { scrollBottom: 0 }; case ScrollAnchor.ScrollToIndex: if (scrollToIndex === undefined) { assertDev( false, ' got "scroll to index" scroll anchor, but no index' ); return null; } return { scrollToIndex }; case ScrollAnchor.ScrollToUnreadIndicator: return scrollToUnreadIndicator; case ScrollAnchor.Top: return { scrollTop: containerEl.scrollTop }; case ScrollAnchor.Bottom: return { scrollBottom: getScrollBottom(containerEl) }; default: throw missingCaseError(scrollAnchor); } } public override componentDidUpdate( prevProps: Readonly, _prevState: Readonly, snapshot: Readonly ): void { const { items: oldItems, messageChangeCounter: previousMessageChangeCounter, messageLoadingState: previousMessageLoadingState, } = prevProps; const { discardMessages, id, items: newItems, messageChangeCounter, messageLoadingState, } = this.props; const containerEl = this.containerRef.current; if (containerEl && snapshot) { if (snapshot === scrollToUnreadIndicator) { const lastSeenIndicatorEl = this.lastSeenIndicatorRef.current; if (lastSeenIndicatorEl) { lastSeenIndicatorEl.scrollIntoView(); } else { scrollToBottom(containerEl); assertDev( false, ' expected a last seen indicator but it was not found' ); } } else if ('scrollToIndex' in snapshot) { this.scrollToItemIndex(snapshot.scrollToIndex); } else if ('scrollTop' in snapshot) { containerEl.scrollTop = snapshot.scrollTop; } else { setScrollBottom(containerEl, snapshot.scrollBottom); } } if (oldItems.length !== newItems.length) { this.updateIntersectionObserver(); // This condition is somewhat arbitrary. const numberToKeepAtBottom = this.maxVisibleRows * 2; const shouldDiscardOlderMessages: boolean = this.isAtBottom() && newItems.length > numberToKeepAtBottom; if (shouldDiscardOlderMessages) { discardMessages({ conversationId: id, numberToKeepAtBottom, }); } const loadingStateThatJustFinished: | undefined | TimelineMessageLoadingState = !messageLoadingState && previousMessageLoadingState ? previousMessageLoadingState : undefined; const numberToKeepAtTop = this.maxVisibleRows * 5; const shouldDiscardNewerMessages: boolean = !this.isAtBottom() && loadingStateThatJustFinished === TimelineMessageLoadingState.LoadingOlderMessages && newItems.length > numberToKeepAtTop; if (shouldDiscardNewerMessages) { discardMessages({ conversationId: id, numberToKeepAtTop, }); } } if (previousMessageChangeCounter !== messageChangeCounter) { this.markNewestBottomVisibleMessageRead(); } } private handleBlur = (event: React.FocusEvent): void => { const { clearSelectedMessage } = this.props; const { currentTarget } = event; // Thanks to https://gist.github.com/pstoica/4323d3e6e37e8a23dd59 setTimeout(() => { // If focus moved to one of our portals, we do not clear the selected // message so that focus stays inside the portal. We need to be careful // to not create colliding keyboard shortcuts between selected messages // and our portals! const portals = Array.from( document.querySelectorAll('body > div:not(.inbox)') ); if (portals.some(el => el.contains(document.activeElement))) { return; } if (!currentTarget.contains(document.activeElement)) { clearSelectedMessage(); } }, 0); }; private handleKeyDown = ( event: React.KeyboardEvent ): void => { const { selectMessage, selectedMessageId, items, id } = this.props; const commandKey = get(window, 'platform') === 'darwin' && event.metaKey; const controlKey = get(window, 'platform') !== 'darwin' && event.ctrlKey; const commandOrCtrl = commandKey || controlKey; if (!items || items.length < 1) { return; } if (selectedMessageId && !commandOrCtrl && event.key === 'ArrowUp') { const selectedMessageIndex = items.findIndex( item => item === selectedMessageId ); if (selectedMessageIndex < 0) { return; } const targetIndex = selectedMessageIndex - 1; if (targetIndex < 0) { return; } const messageId = items[targetIndex]; selectMessage(messageId, id); event.preventDefault(); event.stopPropagation(); return; } if (selectedMessageId && !commandOrCtrl && event.key === 'ArrowDown') { const selectedMessageIndex = items.findIndex( item => item === selectedMessageId ); if (selectedMessageIndex < 0) { return; } const targetIndex = selectedMessageIndex + 1; if (targetIndex >= items.length) { return; } const messageId = items[targetIndex]; selectMessage(messageId, id); event.preventDefault(); event.stopPropagation(); return; } if (commandOrCtrl && event.key === 'ArrowUp') { const firstMessageId = first(items); if (firstMessageId) { selectMessage(firstMessageId, id); event.preventDefault(); event.stopPropagation(); } return; } if (commandOrCtrl && event.key === 'ArrowDown') { this.scrollDown(true); event.preventDefault(); event.stopPropagation(); } }; public override render(): JSX.Element | null { const { acknowledgeGroupMemberNameCollisions, clearInvitedUuidsForNewlyCreatedGroup, closeContactSpoofingReview, contactSpoofingReview, getPreferredBadge, getTimestampForMessage, haveNewest, haveOldest, i18n, id, invitedContactsForNewlyCreatedGroup, isConversationSelected, isGroupV1AndDisabled, isSomeoneTyping, items, messageLoadingState, oldestUnseenIndex, onBlock, onBlockAndReportSpam, onDelete, onUnblock, removeMember, renderHeroRow, renderItem, renderTypingBubble, renderContactSpoofingReviewDialog, reviewGroupMemberNameCollision, reviewMessageRequestNameCollision, showContactModal, theme, totalUnseen, unblurAvatar, unreadCount, updateSharedGroups, } = this.props; const { hasRecentlyScrolled, lastMeasuredWarningHeight, newestBottomVisibleMessageId, oldestPartiallyVisibleMessageId, widthBreakpoint, } = this.state; // As a performance optimization, we don't need to render anything if this // conversation isn't the active one. if (!isConversationSelected) { return null; } const areThereAnyMessages = items.length > 0; const areAnyMessagesUnread = Boolean(unreadCount); const areAnyMessagesBelowCurrentPosition = !haveNewest || Boolean( newestBottomVisibleMessageId && newestBottomVisibleMessageId !== last(items) ); const areSomeMessagesBelowCurrentPosition = !haveNewest || (newestBottomVisibleMessageId && !items .slice(-SCROLL_DOWN_BUTTON_THRESHOLD) .includes(newestBottomVisibleMessageId)); const areUnreadBelowCurrentPosition = Boolean( areThereAnyMessages && areAnyMessagesUnread && areAnyMessagesBelowCurrentPosition ); const shouldShowScrollDownButton = Boolean( areThereAnyMessages && (areUnreadBelowCurrentPosition || areSomeMessagesBelowCurrentPosition) ); const actionProps = getActions(this.props); let floatingHeader: ReactNode; // It's possible that a message was removed from `items` but we still have its ID in // state. `getTimestampForMessage` might return undefined in that case. const oldestPartiallyVisibleMessageTimestamp = oldestPartiallyVisibleMessageId ? getTimestampForMessage(oldestPartiallyVisibleMessageId) : undefined; if ( oldestPartiallyVisibleMessageId && oldestPartiallyVisibleMessageTimestamp ) { const isLoadingMessages = Boolean(messageLoadingState); floatingHeader = ( ); } const messageNodes: Array = []; for (let itemIndex = 0; itemIndex < items.length; itemIndex += 1) { const previousItemIndex = itemIndex - 1; const nextItemIndex = itemIndex + 1; const previousMessageId: undefined | string = items[previousItemIndex]; const nextMessageId: undefined | string = items[nextItemIndex]; const messageId = items[itemIndex]; if (!messageId) { assertDev( false, ' iterated through items and got an empty message ID' ); continue; } let unreadIndicatorPlacement: undefined | UnreadIndicatorPlacement; if (oldestUnseenIndex === itemIndex) { unreadIndicatorPlacement = UnreadIndicatorPlacement.JustAbove; messageNodes.push( ); } else if (oldestUnseenIndex === nextItemIndex) { unreadIndicatorPlacement = UnreadIndicatorPlacement.JustBelow; } messageNodes.push(
{renderItem({ actionProps, containerElementRef: this.containerRef, containerWidthBreakpoint: widthBreakpoint, conversationId: id, isOldestTimelineItem: haveOldest && itemIndex === 0, messageId, nextMessageId, previousMessageId, unreadIndicatorPlacement, })}
); } const warning = Timeline.getWarning(this.props, this.state); let timelineWarning: ReactNode; if (warning) { let text: ReactChild; let onClose: () => void; switch (warning.type) { case ContactSpoofingType.DirectConversationWithSameTitle: text = ( { reviewMessageRequestNameCollision({ safeConversationId: warning.safeConversation.id, }); }} > {i18n('ContactSpoofing__same-name__link')} ), }} /> ); onClose = () => { this.setState({ hasDismissedDirectContactSpoofingWarning: true, }); }; break; case ContactSpoofingType.MultipleGroupMembersWithSameTitle: { const { groupNameCollisions } = warning; text = ( result + conversations.length, 0 ) .toString(), link: ( { reviewGroupMemberNameCollision(id); }} > {i18n('ContactSpoofing__same-name-in-group__link')} ), }} /> ); onClose = () => { acknowledgeGroupMemberNameCollisions(groupNameCollisions); }; break; } default: throw missingCaseError(warning); } timelineWarning = ( { if (!bounds) { assertDev(false, 'We should be measuring the bounds'); return; } this.setState({ lastMeasuredWarningHeight: bounds.height }); }} > {({ measureRef }) => ( {text} )} ); } let contactSpoofingReviewDialog: ReactNode; if (contactSpoofingReview) { const commonProps = { getPreferredBadge, i18n, onBlock, onBlockAndReportSpam, onClose: closeContactSpoofingReview, onDelete, onShowContactModal: showContactModal, onUnblock, removeMember, theme, }; switch (contactSpoofingReview.type) { case ContactSpoofingType.DirectConversationWithSameTitle: contactSpoofingReviewDialog = renderContactSpoofingReviewDialog({ ...commonProps, type: ContactSpoofingType.DirectConversationWithSameTitle, possiblyUnsafeConversation: contactSpoofingReview.possiblyUnsafeConversation, safeConversation: contactSpoofingReview.safeConversation, }); break; case ContactSpoofingType.MultipleGroupMembersWithSameTitle: contactSpoofingReviewDialog = renderContactSpoofingReviewDialog({ ...commonProps, type: ContactSpoofingType.MultipleGroupMembersWithSameTitle, groupConversationId: id, collisionInfoByTitle: contactSpoofingReview.collisionInfoByTitle, }); break; default: throw missingCaseError(contactSpoofingReview); } } return ( <> { const { isNearBottom } = this.props; strictAssert(bounds, 'We should be measuring the bounds'); this.setState({ widthBreakpoint: getWidthBreakpoint(bounds.width), }); this.maxVisibleRows = Math.ceil(bounds.height / MIN_ROW_HEIGHT); const containerEl = this.containerRef.current; if (containerEl && isNearBottom) { scrollToBottom(containerEl); } }} > {({ measureRef }) => (
{timelineWarning} {floatingHeader}
{haveOldest && ( <> {Timeline.getWarning(this.props, this.state) && (
)} {renderHeroRow(id, unblurAvatar, updateSharedGroups)} )} {messageNodes} {isSomeoneTyping && haveNewest && renderTypingBubble(id)}
{shouldShowScrollDownButton ? ( ) : null}
)} {Boolean(invitedContactsForNewlyCreatedGroup.length) && ( )} {contactSpoofingReviewDialog} ); } private static getWarning( { warning }: PropsType, state: StateType ): undefined | WarningType { if (!warning) { return undefined; } switch (warning.type) { case ContactSpoofingType.DirectConversationWithSameTitle: { const { hasDismissedDirectContactSpoofingWarning } = state; return hasDismissedDirectContactSpoofingWarning ? undefined : warning; } case ContactSpoofingType.MultipleGroupMembersWithSameTitle: return hasUnacknowledgedCollisions( warning.acknowledgedGroupNameCollisions, warning.groupNameCollisions ) ? warning : undefined; default: throw missingCaseError(warning); } } } function getMessageIdFromElement( element: undefined | Element ): undefined | string { return element instanceof HTMLElement ? element.dataset.messageId : undefined; } function getRowIndexFromElement( element: undefined | Element ): undefined | number { return element instanceof HTMLElement && element.dataset.itemIndex ? parseInt(element.dataset.itemIndex, 10) : undefined; } function showDebugLog() { window.showDebugLog(); }