1590 lines
46 KiB
TypeScript
1590 lines
46 KiB
TypeScript
// Copyright 2019-2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import { debounce, get, isEqual, isNumber, pick } from 'lodash';
|
|
import classNames from 'classnames';
|
|
import type { ReactChild, ReactNode, RefObject } from 'react';
|
|
import React from 'react';
|
|
import { createSelector } from 'reselect';
|
|
import type { Grid, ListRowProps } from 'react-virtualized';
|
|
import { AutoSizer, CellMeasurer, List } from 'react-virtualized';
|
|
import Measure from 'react-measure';
|
|
|
|
import * as log from '../../logging/log';
|
|
|
|
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 { assert } from '../../util/assert';
|
|
import { missingCaseError } from '../../util/missingCaseError';
|
|
import { createRefMerger } from '../../util/refMerger';
|
|
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 { 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 { ContactSpoofingReviewDialog } from './ContactSpoofingReviewDialog';
|
|
import type { GroupNameCollisionsWithIdsByTitle } from '../../util/groupMemberNameCollisions';
|
|
import { hasUnacknowledgedCollisions } from '../../util/groupMemberNameCollisions';
|
|
import { TimelineFloatingHeader } from './TimelineFloatingHeader';
|
|
import {
|
|
RowHeightCache,
|
|
fromItemIndexToRow,
|
|
fromRowToItemIndex,
|
|
getEphemeralRows,
|
|
getHeroRow,
|
|
getLastSeenIndicatorRow,
|
|
getRowCount,
|
|
getTypingBubbleRow,
|
|
getWidthBreakpoint,
|
|
} from '../../util/timelineUtil';
|
|
|
|
const ESTIMATED_ROW_HEIGHT = 64;
|
|
const AT_BOTTOM_THRESHOLD = 15;
|
|
const NEAR_BOTTOM_THRESHOLD = 15;
|
|
const AT_TOP_THRESHOLD = 10;
|
|
const LOAD_MORE_THRESHOLD = 30;
|
|
const SCROLL_DOWN_BUTTON_THRESHOLD = 8;
|
|
export const LOAD_COUNTDOWN = 1;
|
|
|
|
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;
|
|
isLoadingMessages: boolean;
|
|
isNearBottom?: boolean;
|
|
items: ReadonlyArray<string>;
|
|
loadCountdownStart?: number;
|
|
messageHeightChangeBaton?: unknown;
|
|
messageHeightChangeIndex?: number;
|
|
oldestUnreadIndex?: number;
|
|
resetCounter: number;
|
|
scrollToIndex?: number;
|
|
scrollToIndexCounter: number;
|
|
totalUnread: number;
|
|
};
|
|
|
|
type PropsHousekeepingType = {
|
|
id: string;
|
|
areWeAdmin?: boolean;
|
|
isGroupV1AndDisabled?: boolean;
|
|
isIncomingMessageRequest: boolean;
|
|
typingContactId?: string;
|
|
unreadCount?: number;
|
|
|
|
selectedMessageId?: string;
|
|
invitedContactsForNewlyCreatedGroup: Array<ConversationType>;
|
|
|
|
warning?: WarningType;
|
|
contactSpoofingReview?: ContactSpoofingReviewPropType;
|
|
|
|
getTimestampForMessage: (messageId: string) => undefined | number;
|
|
getPreferredBadge: PreferredBadgeSelectorType;
|
|
i18n: LocalizerType;
|
|
theme: ThemeType;
|
|
|
|
renderItem: (props: {
|
|
actionProps: PropsActionsType;
|
|
containerElementRef: RefObject<HTMLElement>;
|
|
containerWidthBreakpoint: WidthBreakpoint;
|
|
conversationId: string;
|
|
isOldestTimelineItem: boolean;
|
|
messageId: string;
|
|
nextMessageId: undefined | string;
|
|
onHeightChange: (messageId: string) => unknown;
|
|
previousMessageId: undefined | string;
|
|
}) => JSX.Element;
|
|
renderLastSeenIndicator: (id: string) => JSX.Element;
|
|
renderHeroRow: (
|
|
id: string,
|
|
resizeHeroRow: () => unknown,
|
|
unblurAvatar: () => void,
|
|
updateSharedGroups: () => unknown
|
|
) => JSX.Element;
|
|
renderTypingBubble: (id: string) => JSX.Element;
|
|
};
|
|
|
|
export type PropsActionsType = {
|
|
acknowledgeGroupMemberNameCollisions: (
|
|
groupNameCollisions: Readonly<GroupNameCollisionsWithIdsByTitle>
|
|
) => void;
|
|
clearChangedMessages: (conversationId: string, baton: unknown) => unknown;
|
|
clearInvitedUuidsForNewlyCreatedGroup: () => void;
|
|
closeContactSpoofingReview: () => void;
|
|
setLoadCountdownStart: (
|
|
conversationId: string,
|
|
loadCountdownStart?: number
|
|
) => unknown;
|
|
setIsNearBottom: (conversationId: string, isNearBottom: boolean) => unknown;
|
|
reviewGroupMemberNameCollision: (groupConversationId: string) => void;
|
|
reviewMessageRequestNameCollision: (
|
|
_: Readonly<{
|
|
safeConversationId: string;
|
|
}>
|
|
) => void;
|
|
|
|
learnMoreAboutDeliveryIssue: () => unknown;
|
|
loadAndScroll: (messageId: string) => 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;
|
|
removeMember: (conversationId: string) => unknown;
|
|
selectMessage: (messageId: string, conversationId: string) => unknown;
|
|
clearSelectedMessage: () => unknown;
|
|
unblurAvatar: () => void;
|
|
updateSharedGroups: () => unknown;
|
|
} & Omit<MessageActionsType, 'onHeightChange'> &
|
|
SafetyNumberActionsType &
|
|
UnsupportedMessageActionsType &
|
|
ChatSessionRefreshedNotificationActionsType;
|
|
|
|
export type PropsType = PropsDataType &
|
|
PropsHousekeepingType &
|
|
PropsActionsType;
|
|
|
|
type OnScrollParamsType = {
|
|
scrollTop: number;
|
|
clientHeight: number;
|
|
scrollHeight: number;
|
|
|
|
clientWidth: number;
|
|
scrollWidth?: number;
|
|
scrollLeft?: number;
|
|
scrollToColumn?: number;
|
|
_hasScrolledToColumnTarget?: boolean;
|
|
scrollToRow?: number;
|
|
_hasScrolledToRowTarget?: boolean;
|
|
};
|
|
|
|
type VisibleRowType = {
|
|
id: string;
|
|
offsetTop: number;
|
|
row: number;
|
|
};
|
|
|
|
type StateType = {
|
|
atBottom: boolean;
|
|
atTop: boolean;
|
|
hasRecentlyScrolled: boolean;
|
|
oneTimeScrollRow?: number;
|
|
visibleRows?: {
|
|
newestFullyVisible?: VisibleRowType;
|
|
oldestPartiallyVisibleMessageId?: string;
|
|
oldestFullyVisible?: VisibleRowType;
|
|
};
|
|
|
|
widthBreakpoint: WidthBreakpoint;
|
|
|
|
prevPropScrollToIndex?: number;
|
|
prevPropScrollToIndexCounter?: number;
|
|
propScrollToIndex?: number;
|
|
|
|
shouldShowScrollDownButton: boolean;
|
|
areUnreadBelowCurrentPosition: boolean;
|
|
|
|
hasDismissedDirectContactSpoofingWarning: boolean;
|
|
lastMeasuredWarningHeight: 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',
|
|
'clearChangedMessages',
|
|
'clearInvitedUuidsForNewlyCreatedGroup',
|
|
'closeContactSpoofingReview',
|
|
'setLoadCountdownStart',
|
|
'setIsNearBottom',
|
|
'reviewGroupMemberNameCollision',
|
|
'reviewMessageRequestNameCollision',
|
|
'learnMoreAboutDeliveryIssue',
|
|
'loadAndScroll',
|
|
'loadOlderMessages',
|
|
'loadNewerMessages',
|
|
'loadNewestMessages',
|
|
'markMessageRead',
|
|
'markViewed',
|
|
'onBlock',
|
|
'onBlockAndReportSpam',
|
|
'onDelete',
|
|
'onUnblock',
|
|
'peekGroupCallForTheFirstTime',
|
|
'removeMember',
|
|
'selectMessage',
|
|
'clearSelectedMessage',
|
|
'unblurAvatar',
|
|
'updateSharedGroups',
|
|
|
|
'doubleCheckMissingQuoteReference',
|
|
'checkForAccount',
|
|
'reactToMessage',
|
|
'replyToMessage',
|
|
'retrySend',
|
|
'showForwardMessageModal',
|
|
'deleteMessage',
|
|
'deleteMessageForEveryone',
|
|
'showMessageDetail',
|
|
'openConversation',
|
|
'showContactDetail',
|
|
'showContactModal',
|
|
'kickOffAttachmentDownload',
|
|
'markAttachmentAsCorrupted',
|
|
'messageExpanded',
|
|
'showVisualAttachment',
|
|
'downloadAttachment',
|
|
'displayTapToViewMessage',
|
|
'openLink',
|
|
'scrollToQuotedMessage',
|
|
'showExpiredIncomingTapToViewToast',
|
|
'showExpiredOutgoingTapToViewToast',
|
|
|
|
'showIdentity',
|
|
|
|
'downloadNewVersion',
|
|
|
|
'contactSupport',
|
|
]);
|
|
|
|
const safe: AssertProps<PropsActionsType, typeof unsafe> = unsafe;
|
|
|
|
return safe;
|
|
}
|
|
);
|
|
|
|
export class Timeline extends React.PureComponent<PropsType, StateType> {
|
|
private cellSizeCache = new RowHeightCache(ESTIMATED_ROW_HEIGHT);
|
|
|
|
private mostRecentWidth = 0;
|
|
|
|
private mostRecentHeight = 0;
|
|
|
|
private offsetFromBottom: number | undefined = 0;
|
|
|
|
private resizeFlag = false;
|
|
|
|
private readonly containerRef = React.createRef<HTMLDivElement>();
|
|
|
|
private readonly listRef = React.createRef<List>();
|
|
|
|
private loadCountdownTimeout: NodeJS.Timeout | null = null;
|
|
|
|
private hasRecentlyScrolledTimeout?: NodeJS.Timeout;
|
|
|
|
private delayedPeekTimeout?: NodeJS.Timeout;
|
|
|
|
private containerRefMerger = createRefMerger();
|
|
|
|
constructor(props: PropsType) {
|
|
super(props);
|
|
|
|
const { scrollToIndex, isIncomingMessageRequest } = this.props;
|
|
const oneTimeScrollRow = isIncomingMessageRequest
|
|
? undefined
|
|
: getLastSeenIndicatorRow(props);
|
|
|
|
// We only stick to the bottom if this is not an incoming message request.
|
|
const atBottom = !isIncomingMessageRequest;
|
|
|
|
this.state = {
|
|
atBottom,
|
|
atTop: false,
|
|
hasRecentlyScrolled: true,
|
|
oneTimeScrollRow,
|
|
propScrollToIndex: scrollToIndex,
|
|
prevPropScrollToIndex: scrollToIndex,
|
|
shouldShowScrollDownButton: false,
|
|
areUnreadBelowCurrentPosition: false,
|
|
hasDismissedDirectContactSpoofingWarning: false,
|
|
lastMeasuredWarningHeight: 0,
|
|
// This may be swiftly overridden.
|
|
widthBreakpoint: WidthBreakpoint.Wide,
|
|
};
|
|
}
|
|
|
|
public static getDerivedStateFromProps(
|
|
props: PropsType,
|
|
state: StateType
|
|
): StateType {
|
|
if (
|
|
isNumber(props.scrollToIndex) &&
|
|
(props.scrollToIndex !== state.prevPropScrollToIndex ||
|
|
props.scrollToIndexCounter !== state.prevPropScrollToIndexCounter)
|
|
) {
|
|
return {
|
|
...state,
|
|
propScrollToIndex: props.scrollToIndex,
|
|
prevPropScrollToIndex: props.scrollToIndex,
|
|
prevPropScrollToIndexCounter: props.scrollToIndexCounter,
|
|
};
|
|
}
|
|
|
|
return state;
|
|
}
|
|
|
|
private getList = (): List | null => {
|
|
if (!this.listRef) {
|
|
return null;
|
|
}
|
|
|
|
const { current } = this.listRef;
|
|
|
|
return current;
|
|
};
|
|
|
|
private getGrid = (): Grid | undefined => {
|
|
const list = this.getList();
|
|
if (!list) {
|
|
return;
|
|
}
|
|
|
|
return list.Grid;
|
|
};
|
|
|
|
private getScrollContainer = (): HTMLDivElement | undefined => {
|
|
// We're using an internal variable (_scrollingContainer)) here,
|
|
// so cannot rely on the public type.
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
const grid: any = this.getGrid();
|
|
if (!grid) {
|
|
return;
|
|
}
|
|
|
|
return grid._scrollingContainer as HTMLDivElement;
|
|
};
|
|
|
|
private recomputeRowHeights = (row?: number): void => {
|
|
const list = this.getList();
|
|
if (!list) {
|
|
return;
|
|
}
|
|
|
|
list.recomputeRowHeights(row);
|
|
};
|
|
|
|
private onHeightOnlyChange = (): void => {
|
|
const grid = this.getGrid();
|
|
const scrollContainer = this.getScrollContainer();
|
|
if (!grid || !scrollContainer) {
|
|
return;
|
|
}
|
|
|
|
if (!isNumber(this.offsetFromBottom)) {
|
|
return;
|
|
}
|
|
|
|
const { clientHeight, scrollHeight, scrollTop } = scrollContainer;
|
|
const newOffsetFromBottom = Math.max(
|
|
0,
|
|
scrollHeight - clientHeight - scrollTop
|
|
);
|
|
const delta = newOffsetFromBottom - this.offsetFromBottom;
|
|
|
|
// TODO: DESKTOP-687
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
(grid as any).scrollToPosition({
|
|
scrollTop: scrollContainer.scrollTop + delta,
|
|
});
|
|
};
|
|
|
|
private resize = (row?: number): void => {
|
|
this.offsetFromBottom = undefined;
|
|
this.resizeFlag = false;
|
|
if (isNumber(row) && row > 0) {
|
|
this.cellSizeCache.clearPlus(row);
|
|
} else {
|
|
this.cellSizeCache.clearAll();
|
|
}
|
|
|
|
this.recomputeRowHeights(row || 0);
|
|
};
|
|
|
|
private resizeHeroRow = (): void => {
|
|
this.resize(0);
|
|
};
|
|
|
|
private resizeMessage = (messageId: string): void => {
|
|
const { items } = this.props;
|
|
|
|
if (!items || !items.length) {
|
|
return;
|
|
}
|
|
|
|
const index = items.findIndex(item => item === messageId);
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
|
|
const row = fromItemIndexToRow(index, this.props);
|
|
this.resize(row);
|
|
};
|
|
|
|
private onScroll = (data: OnScrollParamsType): void => {
|
|
// Ignore scroll events generated as react-virtualized recursively scrolls and
|
|
// re-measures to get us where we want to go.
|
|
if (
|
|
isNumber(data.scrollToRow) &&
|
|
data.scrollToRow >= 0 &&
|
|
!data._hasScrolledToRowTarget
|
|
) {
|
|
return;
|
|
}
|
|
|
|
// Sometimes react-virtualized ends up with some incorrect math - we've scrolled below
|
|
// what should be possible. In this case, we leave everything the same and ask
|
|
// react-virtualized to try again. Without this, we'll set atBottom to true and
|
|
// pop the user back down to the bottom.
|
|
const { clientHeight, scrollHeight, scrollTop } = data;
|
|
if (scrollTop + clientHeight > scrollHeight) {
|
|
return;
|
|
}
|
|
|
|
this.setState({ hasRecentlyScrolled: true });
|
|
if (this.hasRecentlyScrolledTimeout) {
|
|
clearTimeout(this.hasRecentlyScrolledTimeout);
|
|
}
|
|
this.hasRecentlyScrolledTimeout = setTimeout(() => {
|
|
this.setState({ hasRecentlyScrolled: false });
|
|
}, 3000);
|
|
|
|
this.updateScrollMetrics(data);
|
|
this.updateWithVisibleRows();
|
|
};
|
|
|
|
private onRowsRendered = (): void => {
|
|
// React Virtualized doesn't respect `scrollToIndex` in some cases, likely
|
|
// because it hasn't rendered that row yet.
|
|
const { oneTimeScrollRow } = this.state;
|
|
if (isNumber(oneTimeScrollRow)) {
|
|
this.getList()?.scrollToRow(oneTimeScrollRow);
|
|
}
|
|
};
|
|
|
|
private updateScrollMetrics = debounce(
|
|
(data: OnScrollParamsType) => {
|
|
const { clientHeight, clientWidth, scrollHeight, scrollTop } = data;
|
|
|
|
if (clientHeight <= 0 || scrollHeight <= 0) {
|
|
return;
|
|
}
|
|
|
|
const {
|
|
haveNewest,
|
|
haveOldest,
|
|
id,
|
|
isIncomingMessageRequest,
|
|
setIsNearBottom,
|
|
setLoadCountdownStart,
|
|
} = this.props;
|
|
|
|
if (
|
|
this.mostRecentHeight &&
|
|
clientHeight !== this.mostRecentHeight &&
|
|
this.mostRecentWidth &&
|
|
clientWidth === this.mostRecentWidth
|
|
) {
|
|
this.onHeightOnlyChange();
|
|
}
|
|
|
|
// If we've scrolled, we want to reset these
|
|
const oneTimeScrollRow = undefined;
|
|
const propScrollToIndex = undefined;
|
|
|
|
this.offsetFromBottom = Math.max(
|
|
0,
|
|
scrollHeight - clientHeight - scrollTop
|
|
);
|
|
|
|
// If there's an active message request, we won't stick to the bottom of the
|
|
// conversation as new messages come in.
|
|
const atBottom = isIncomingMessageRequest
|
|
? false
|
|
: haveNewest && this.offsetFromBottom <= AT_BOTTOM_THRESHOLD;
|
|
|
|
const isNearBottom =
|
|
haveNewest && this.offsetFromBottom <= NEAR_BOTTOM_THRESHOLD;
|
|
const atTop = scrollTop <= AT_TOP_THRESHOLD;
|
|
const loadCountdownStart = atTop && !haveOldest ? Date.now() : undefined;
|
|
|
|
if (this.loadCountdownTimeout) {
|
|
clearTimeout(this.loadCountdownTimeout);
|
|
this.loadCountdownTimeout = null;
|
|
}
|
|
if (isNumber(loadCountdownStart)) {
|
|
this.loadCountdownTimeout = setTimeout(
|
|
this.loadOlderMessages,
|
|
LOAD_COUNTDOWN
|
|
);
|
|
}
|
|
|
|
// Variable collision
|
|
// eslint-disable-next-line react/destructuring-assignment
|
|
if (loadCountdownStart !== this.props.loadCountdownStart) {
|
|
setLoadCountdownStart(id, loadCountdownStart);
|
|
}
|
|
|
|
// Variable collision
|
|
// eslint-disable-next-line react/destructuring-assignment
|
|
if (isNearBottom !== this.props.isNearBottom) {
|
|
setIsNearBottom(id, isNearBottom);
|
|
}
|
|
|
|
this.setState({
|
|
atBottom,
|
|
atTop,
|
|
oneTimeScrollRow,
|
|
propScrollToIndex,
|
|
});
|
|
},
|
|
50,
|
|
{ maxWait: 50 }
|
|
);
|
|
|
|
private updateVisibleRows = (): void => {
|
|
const scrollContainer = this.getScrollContainer();
|
|
if (!scrollContainer) {
|
|
return;
|
|
}
|
|
|
|
if (scrollContainer.clientHeight === 0) {
|
|
return;
|
|
}
|
|
|
|
const innerScrollContainer = scrollContainer.children[0];
|
|
if (!innerScrollContainer) {
|
|
return;
|
|
}
|
|
|
|
let newestFullyVisible: undefined | VisibleRowType;
|
|
let oldestPartiallyVisibleMessageId: undefined | string;
|
|
let oldestFullyVisible: undefined | VisibleRowType;
|
|
|
|
const { children } = innerScrollContainer;
|
|
const visibleTop = scrollContainer.scrollTop;
|
|
const visibleBottom = visibleTop + scrollContainer.clientHeight;
|
|
|
|
for (let i = children.length - 1; i >= 0; i -= 1) {
|
|
const child = children[i] as HTMLDivElement;
|
|
const { id, offsetTop, offsetHeight } = child;
|
|
|
|
if (!id) {
|
|
continue;
|
|
}
|
|
|
|
const bottom = offsetTop + offsetHeight;
|
|
|
|
if (bottom - AT_BOTTOM_THRESHOLD <= visibleBottom) {
|
|
const row = parseInt(child.getAttribute('data-row') || '-1', 10);
|
|
newestFullyVisible = { offsetTop, row, id };
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
const max = children.length;
|
|
for (let i = 0; i < max; i += 1) {
|
|
const child = children[i] as HTMLDivElement;
|
|
const { id, offsetTop, offsetHeight } = child;
|
|
|
|
if (!id) {
|
|
continue;
|
|
}
|
|
|
|
const bottom = offsetTop + offsetHeight;
|
|
|
|
if (bottom >= visibleTop && !oldestPartiallyVisibleMessageId) {
|
|
oldestPartiallyVisibleMessageId = id;
|
|
}
|
|
|
|
if (offsetTop + AT_TOP_THRESHOLD >= visibleTop) {
|
|
oldestFullyVisible = {
|
|
offsetTop,
|
|
row: parseInt(child.getAttribute('data-row') || '-1', 10),
|
|
id,
|
|
};
|
|
break;
|
|
}
|
|
}
|
|
|
|
this.setState(oldState => {
|
|
const visibleRows = {
|
|
newestFullyVisible,
|
|
oldestPartiallyVisibleMessageId,
|
|
oldestFullyVisible,
|
|
};
|
|
|
|
// This avoids a render loop.
|
|
return isEqual(oldState.visibleRows, visibleRows)
|
|
? null
|
|
: { visibleRows };
|
|
});
|
|
};
|
|
|
|
private updateWithVisibleRows = debounce(
|
|
() => {
|
|
const {
|
|
unreadCount,
|
|
haveNewest,
|
|
haveOldest,
|
|
isLoadingMessages,
|
|
items,
|
|
loadNewerMessages,
|
|
markMessageRead,
|
|
} = this.props;
|
|
|
|
if (!items || items.length < 1) {
|
|
return;
|
|
}
|
|
|
|
this.updateVisibleRows();
|
|
const { visibleRows } = this.state;
|
|
if (!visibleRows) {
|
|
return;
|
|
}
|
|
|
|
const { newestFullyVisible, oldestFullyVisible } = visibleRows;
|
|
if (!newestFullyVisible) {
|
|
return;
|
|
}
|
|
|
|
markMessageRead(newestFullyVisible.id);
|
|
|
|
const newestRow = getRowCount(this.props) - 1;
|
|
const oldestRow = fromItemIndexToRow(0, this.props);
|
|
|
|
// Loading newer messages (that go below current messages) is pain-free and quick
|
|
// we'll just kick these off immediately.
|
|
if (
|
|
!isLoadingMessages &&
|
|
!haveNewest &&
|
|
newestFullyVisible.row > newestRow - LOAD_MORE_THRESHOLD
|
|
) {
|
|
const lastId = items[items.length - 1];
|
|
loadNewerMessages(lastId);
|
|
}
|
|
|
|
// Loading older messages is more destructive, as they requires a recalculation of
|
|
// all locations of things below. So we need to be careful with these loads.
|
|
// Generally we hid this behind a countdown spinner at the top of the window, but
|
|
// this is a special-case for the situation where the window is so large and that
|
|
// all the messages are visible.
|
|
const oldestVisible = Boolean(
|
|
oldestFullyVisible && oldestRow === oldestFullyVisible.row
|
|
);
|
|
const newestVisible = newestRow === newestFullyVisible.row;
|
|
if (oldestVisible && newestVisible && !haveOldest) {
|
|
this.loadOlderMessages();
|
|
}
|
|
|
|
const lastIndex = items.length - 1;
|
|
const lastItemRow = fromItemIndexToRow(lastIndex, this.props);
|
|
const areUnreadBelowCurrentPosition = Boolean(
|
|
isNumber(unreadCount) &&
|
|
unreadCount > 0 &&
|
|
(!haveNewest || newestFullyVisible.row < lastItemRow)
|
|
);
|
|
|
|
const shouldShowScrollDownButton = Boolean(
|
|
!haveNewest ||
|
|
areUnreadBelowCurrentPosition ||
|
|
newestFullyVisible.row < newestRow - SCROLL_DOWN_BUTTON_THRESHOLD
|
|
);
|
|
|
|
this.setState({
|
|
shouldShowScrollDownButton,
|
|
areUnreadBelowCurrentPosition,
|
|
});
|
|
},
|
|
500,
|
|
{ maxWait: 500 }
|
|
);
|
|
|
|
private loadOlderMessages = (): void => {
|
|
const { haveOldest, isLoadingMessages, items, loadOlderMessages } =
|
|
this.props;
|
|
|
|
if (this.loadCountdownTimeout) {
|
|
clearTimeout(this.loadCountdownTimeout);
|
|
this.loadCountdownTimeout = null;
|
|
}
|
|
|
|
if (isLoadingMessages || haveOldest || !items || items.length < 1) {
|
|
return;
|
|
}
|
|
|
|
const oldestId = items[0];
|
|
loadOlderMessages(oldestId);
|
|
};
|
|
|
|
private rowRenderer = ({
|
|
index: rowIndex,
|
|
key,
|
|
parent,
|
|
style,
|
|
}: Readonly<ListRowProps>): JSX.Element => {
|
|
const {
|
|
id,
|
|
i18n,
|
|
haveOldest,
|
|
items,
|
|
renderItem,
|
|
renderHeroRow,
|
|
renderLastSeenIndicator,
|
|
renderTypingBubble,
|
|
unblurAvatar,
|
|
updateSharedGroups,
|
|
} = this.props;
|
|
const { lastMeasuredWarningHeight, widthBreakpoint } = this.state;
|
|
|
|
const commonProps = {
|
|
'data-row': rowIndex,
|
|
style: {
|
|
...style,
|
|
width: `${this.mostRecentWidth}px`,
|
|
},
|
|
role: 'row',
|
|
};
|
|
|
|
let rowContents: ReactChild;
|
|
switch (rowIndex) {
|
|
case getHeroRow(this.props):
|
|
rowContents = (
|
|
<div {...commonProps}>
|
|
{Timeline.getWarning(this.props, this.state) ? (
|
|
<div style={{ height: lastMeasuredWarningHeight }} />
|
|
) : null}
|
|
{renderHeroRow(
|
|
id,
|
|
this.resizeHeroRow,
|
|
unblurAvatar,
|
|
updateSharedGroups
|
|
)}
|
|
</div>
|
|
);
|
|
break;
|
|
case getLastSeenIndicatorRow(this.props):
|
|
rowContents = <div {...commonProps}>{renderLastSeenIndicator(id)}</div>;
|
|
break;
|
|
case getTypingBubbleRow(this.props):
|
|
rowContents = (
|
|
<div {...commonProps} className="module-timeline__message-container">
|
|
{renderTypingBubble(id)}
|
|
</div>
|
|
);
|
|
break;
|
|
default:
|
|
{
|
|
const itemIndex = fromRowToItemIndex(rowIndex, this.props);
|
|
if (typeof itemIndex !== 'number') {
|
|
throw new Error(
|
|
`Attempted to render item with undefined index - row ${rowIndex}`
|
|
);
|
|
}
|
|
const previousMessageId: undefined | string = items[itemIndex - 1];
|
|
const messageId = items[itemIndex];
|
|
const nextMessageId: undefined | string = items[itemIndex + 1];
|
|
|
|
const actionProps = getActions(this.props);
|
|
|
|
rowContents = (
|
|
<div
|
|
{...commonProps}
|
|
id={messageId}
|
|
className="module-timeline__message-container"
|
|
>
|
|
<ErrorBoundary
|
|
i18n={i18n}
|
|
showDebugLog={() => window.showDebugLog()}
|
|
>
|
|
{renderItem({
|
|
actionProps,
|
|
containerElementRef: this.containerRef,
|
|
containerWidthBreakpoint: widthBreakpoint,
|
|
conversationId: id,
|
|
isOldestTimelineItem: haveOldest && itemIndex === 0,
|
|
messageId,
|
|
nextMessageId,
|
|
onHeightChange: this.resizeMessage,
|
|
previousMessageId,
|
|
})}
|
|
</ErrorBoundary>
|
|
</div>
|
|
);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return (
|
|
<CellMeasurer
|
|
cache={this.cellSizeCache}
|
|
columnIndex={0}
|
|
key={key}
|
|
parent={parent}
|
|
rowIndex={rowIndex}
|
|
width={this.mostRecentWidth}
|
|
>
|
|
{rowContents}
|
|
</CellMeasurer>
|
|
);
|
|
};
|
|
|
|
private getRowHeightFromCache = ({
|
|
index,
|
|
}: Readonly<{ index: number }>): number =>
|
|
this.cellSizeCache.getHeight(index);
|
|
|
|
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);
|
|
}
|
|
|
|
const oneTimeScrollRow =
|
|
items && items.length > 0 ? items.length - 1 : undefined;
|
|
|
|
this.setState({
|
|
propScrollToIndex: undefined,
|
|
oneTimeScrollRow,
|
|
});
|
|
};
|
|
|
|
private onClickScrollDownButton = (): void => {
|
|
this.scrollDown(false);
|
|
};
|
|
|
|
private scrollDown = (setFocus?: boolean): void => {
|
|
const {
|
|
haveNewest,
|
|
id,
|
|
isLoadingMessages,
|
|
items,
|
|
loadNewestMessages,
|
|
oldestUnreadIndex,
|
|
selectMessage,
|
|
} = this.props;
|
|
if (!items || items.length < 1) {
|
|
return;
|
|
}
|
|
|
|
const lastId = items[items.length - 1];
|
|
const lastSeenIndicatorRow = getLastSeenIndicatorRow(this.props);
|
|
|
|
const { visibleRows } = this.state;
|
|
if (!visibleRows) {
|
|
if (haveNewest) {
|
|
this.scrollToBottom(setFocus);
|
|
} else if (!isLoadingMessages) {
|
|
loadNewestMessages(lastId, setFocus);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
const { newestFullyVisible } = visibleRows;
|
|
|
|
if (
|
|
newestFullyVisible &&
|
|
isNumber(lastSeenIndicatorRow) &&
|
|
newestFullyVisible.row < lastSeenIndicatorRow
|
|
) {
|
|
if (setFocus && isNumber(oldestUnreadIndex)) {
|
|
const messageId = items[oldestUnreadIndex];
|
|
selectMessage(messageId, id);
|
|
}
|
|
this.setState({
|
|
oneTimeScrollRow: lastSeenIndicatorRow,
|
|
});
|
|
} else if (haveNewest) {
|
|
this.scrollToBottom(setFocus);
|
|
} else if (!isLoadingMessages) {
|
|
loadNewestMessages(lastId, setFocus);
|
|
}
|
|
};
|
|
|
|
public override componentDidMount(): void {
|
|
this.updateWithVisibleRows();
|
|
window.registerForActive(this.updateWithVisibleRows);
|
|
|
|
this.delayedPeekTimeout = setTimeout(() => {
|
|
const { id, peekGroupCallForTheFirstTime } = this.props;
|
|
peekGroupCallForTheFirstTime(id);
|
|
}, 500);
|
|
}
|
|
|
|
public override componentWillUnmount(): void {
|
|
const { delayedPeekTimeout } = this;
|
|
|
|
window.unregisterForActive(this.updateWithVisibleRows);
|
|
|
|
if (delayedPeekTimeout) {
|
|
clearTimeout(delayedPeekTimeout);
|
|
}
|
|
}
|
|
|
|
public override componentDidUpdate(
|
|
prevProps: Readonly<PropsType>,
|
|
prevState: Readonly<StateType>
|
|
): void {
|
|
const {
|
|
clearChangedMessages,
|
|
haveOldest,
|
|
id,
|
|
isIncomingMessageRequest,
|
|
items,
|
|
messageHeightChangeIndex,
|
|
messageHeightChangeBaton,
|
|
oldestUnreadIndex,
|
|
resetCounter,
|
|
scrollToIndex,
|
|
typingContactId,
|
|
} = this.props;
|
|
|
|
// We recompute the hero row's height if:
|
|
//
|
|
// 1. We just started showing it (the user has scrolled up to see the hero row)
|
|
// 2. Warnings were shown (they add padding to the hero for the floating warning)
|
|
const hadOldest = prevProps.haveOldest;
|
|
const hadWarning = Boolean(Timeline.getWarning(prevProps, prevState));
|
|
const haveWarning = Boolean(Timeline.getWarning(this.props, this.state));
|
|
const shouldRecomputeRowHeights =
|
|
(!hadOldest && haveOldest) || hadWarning !== haveWarning;
|
|
if (shouldRecomputeRowHeights) {
|
|
this.resizeHeroRow();
|
|
}
|
|
|
|
// There are a number of situations which can necessitate that we forget about row
|
|
// heights previously calculated. We reset the minimum number of rows to minimize
|
|
// unexpected changes to the scroll position. Those changes happen because
|
|
// react-virtualized doesn't know what to expect (variable row heights) when it
|
|
// renders, so it does have a fixed row it's attempting to scroll to, and you ask it
|
|
// to render a given point it space, it will do pretty random things.
|
|
|
|
if (
|
|
!prevProps.items ||
|
|
prevProps.items.length === 0 ||
|
|
resetCounter !== prevProps.resetCounter
|
|
) {
|
|
if (prevProps.items && prevProps.items.length > 0) {
|
|
this.resize();
|
|
}
|
|
|
|
// We want to come in at the top of the conversation if it's a message request
|
|
const oneTimeScrollRow = isIncomingMessageRequest
|
|
? undefined
|
|
: getLastSeenIndicatorRow(this.props);
|
|
const atBottom = !isIncomingMessageRequest;
|
|
|
|
// TODO: DESKTOP-688
|
|
// eslint-disable-next-line react/no-did-update-set-state
|
|
this.setState({
|
|
oneTimeScrollRow,
|
|
atBottom,
|
|
propScrollToIndex: scrollToIndex,
|
|
prevPropScrollToIndex: scrollToIndex,
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
let resizeStartRow: number | undefined;
|
|
|
|
if (isNumber(messageHeightChangeIndex)) {
|
|
resizeStartRow = fromItemIndexToRow(messageHeightChangeIndex, this.props);
|
|
clearChangedMessages(id, messageHeightChangeBaton);
|
|
}
|
|
|
|
if (
|
|
items !== prevProps.items ||
|
|
oldestUnreadIndex !== prevProps.oldestUnreadIndex ||
|
|
Boolean(typingContactId) !== Boolean(prevProps.typingContactId)
|
|
) {
|
|
const { atTop } = this.state;
|
|
|
|
// This clause handles prepended messages when user scrolls up. New
|
|
// messages are added to `items`, but we want to keep the scroll position
|
|
// at the first previously visible message even though the row numbers
|
|
// have now changed.
|
|
if (atTop) {
|
|
const oldFirstIndex = 0;
|
|
const oldFirstId = prevProps.items[oldFirstIndex];
|
|
|
|
const newFirstIndex = items.findIndex(item => item === oldFirstId);
|
|
if (newFirstIndex < 0) {
|
|
this.resize();
|
|
|
|
return;
|
|
}
|
|
|
|
const newRow = fromItemIndexToRow(newFirstIndex, this.props);
|
|
if (newRow > 0) {
|
|
// We're loading more new messages at the top; we want to stay at the top
|
|
this.resize();
|
|
// TODO: DESKTOP-688
|
|
// eslint-disable-next-line react/no-did-update-set-state
|
|
this.setState({ oneTimeScrollRow: newRow });
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Compare current rows against previous rows to identify the number of
|
|
// consecutive rows (from start of the list) the are the same in both
|
|
// lists.
|
|
const rowsIterator = getEphemeralRows(this.props);
|
|
const prevRowsIterator = getEphemeralRows(prevProps);
|
|
|
|
let firstChangedRow = 0;
|
|
// eslint-disable-next-line no-constant-condition
|
|
while (true) {
|
|
const row = rowsIterator.next();
|
|
if (row.done) {
|
|
break;
|
|
}
|
|
|
|
const prevRow = prevRowsIterator.next();
|
|
if (prevRow.done) {
|
|
break;
|
|
}
|
|
|
|
if (prevRow.value !== row.value) {
|
|
break;
|
|
}
|
|
|
|
firstChangedRow += 1;
|
|
}
|
|
|
|
// If either:
|
|
//
|
|
// - Row count has changed after props update
|
|
// - There are some different rows (and the loop above was interrupted)
|
|
//
|
|
// Recompute heights of all rows starting from the first changed row or
|
|
// the last row in the previous row list.
|
|
if (!rowsIterator.next().done || !prevRowsIterator.next().done) {
|
|
resizeStartRow = Math.min(
|
|
resizeStartRow ?? firstChangedRow,
|
|
firstChangedRow
|
|
);
|
|
}
|
|
}
|
|
|
|
if (this.resizeFlag) {
|
|
this.resize();
|
|
|
|
return;
|
|
}
|
|
|
|
if (resizeStartRow !== undefined) {
|
|
this.resize(resizeStartRow);
|
|
}
|
|
|
|
this.updateWithVisibleRows();
|
|
}
|
|
|
|
private getScrollTarget = (): number | undefined => {
|
|
const { oneTimeScrollRow, atBottom, propScrollToIndex } = this.state;
|
|
|
|
const rowCount = getRowCount(this.props);
|
|
const targetMessageRow = isNumber(propScrollToIndex)
|
|
? fromItemIndexToRow(propScrollToIndex, this.props)
|
|
: undefined;
|
|
const scrollToBottom = atBottom ? rowCount - 1 : undefined;
|
|
|
|
if (isNumber(targetMessageRow)) {
|
|
return targetMessageRow;
|
|
}
|
|
|
|
if (isNumber(oneTimeScrollRow)) {
|
|
return oneTimeScrollRow;
|
|
}
|
|
|
|
return scrollToBottom;
|
|
};
|
|
|
|
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<HTMLDivElement>
|
|
): 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') {
|
|
this.setState({ oneTimeScrollRow: 0 });
|
|
|
|
const firstMessageId = items[0];
|
|
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,
|
|
areWeAdmin,
|
|
clearInvitedUuidsForNewlyCreatedGroup,
|
|
closeContactSpoofingReview,
|
|
contactSpoofingReview,
|
|
getPreferredBadge,
|
|
getTimestampForMessage,
|
|
haveOldest,
|
|
i18n,
|
|
id,
|
|
invitedContactsForNewlyCreatedGroup,
|
|
isGroupV1AndDisabled,
|
|
isLoadingMessages,
|
|
items,
|
|
onBlock,
|
|
onBlockAndReportSpam,
|
|
onDelete,
|
|
onUnblock,
|
|
showContactModal,
|
|
removeMember,
|
|
reviewGroupMemberNameCollision,
|
|
reviewMessageRequestNameCollision,
|
|
theme,
|
|
} = this.props;
|
|
const {
|
|
shouldShowScrollDownButton,
|
|
areUnreadBelowCurrentPosition,
|
|
hasRecentlyScrolled,
|
|
lastMeasuredWarningHeight,
|
|
visibleRows,
|
|
widthBreakpoint,
|
|
} = this.state;
|
|
|
|
const rowCount = getRowCount(this.props);
|
|
const scrollToIndex = this.getScrollTarget();
|
|
|
|
if (!items || rowCount === 0) {
|
|
log.error('<Timeline> row count is 0');
|
|
return null;
|
|
}
|
|
|
|
let floatingHeader: ReactNode;
|
|
const oldestPartiallyVisibleMessageId =
|
|
visibleRows?.oldestPartiallyVisibleMessageId;
|
|
// 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 (oldestPartiallyVisibleMessageTimestamp) {
|
|
floatingHeader = (
|
|
<TimelineFloatingHeader
|
|
i18n={i18n}
|
|
isLoading={isLoadingMessages}
|
|
style={
|
|
lastMeasuredWarningHeight
|
|
? { marginTop: lastMeasuredWarningHeight }
|
|
: undefined
|
|
}
|
|
timestamp={oldestPartiallyVisibleMessageTimestamp}
|
|
visible={
|
|
(hasRecentlyScrolled || isLoadingMessages) &&
|
|
(!haveOldest || oldestPartiallyVisibleMessageId !== items[0])
|
|
}
|
|
/>
|
|
);
|
|
}
|
|
|
|
const autoSizer = (
|
|
<AutoSizer>
|
|
{({ height, width }) => {
|
|
if (this.mostRecentWidth && this.mostRecentWidth !== width) {
|
|
this.resizeFlag = true;
|
|
|
|
setTimeout(this.resize, 0);
|
|
} else if (
|
|
this.mostRecentHeight &&
|
|
this.mostRecentHeight !== height
|
|
) {
|
|
setTimeout(this.onHeightOnlyChange, 0);
|
|
}
|
|
|
|
this.mostRecentWidth = width;
|
|
this.mostRecentHeight = height;
|
|
|
|
return (
|
|
<List
|
|
// React Virtualized has an incorrect type for this prop. Until [a fix][0]
|
|
// is merged, we have to do this cast.
|
|
// [0]: https://github.com/DefinitelyTyped/DefinitelyTyped/pull/58705
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
deferredMeasurementCache={this.cellSizeCache as any}
|
|
height={height}
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
onScroll={this.onScroll as any}
|
|
overscanRowCount={10}
|
|
onRowsRendered={this.onRowsRendered}
|
|
ref={this.listRef}
|
|
rowCount={rowCount}
|
|
rowHeight={this.getRowHeightFromCache}
|
|
rowRenderer={this.rowRenderer}
|
|
scrollToAlignment="start"
|
|
scrollToIndex={scrollToIndex}
|
|
tabIndex={-1}
|
|
width={width}
|
|
style={{
|
|
// `overlay` is [a nonstandard value][0] so it's not supported. See [this
|
|
// issue][1].
|
|
//
|
|
// [0]: https://developer.mozilla.org/en-US/docs/Web/CSS/overflow#values
|
|
// [1]: https://github.com/frenic/csstype/issues/62#issuecomment-937238313
|
|
//
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
overflowY: 'overlay' as any,
|
|
}}
|
|
/>
|
|
);
|
|
}}
|
|
</AutoSizer>
|
|
);
|
|
|
|
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 = (
|
|
<Intl
|
|
i18n={i18n}
|
|
id="ContactSpoofing__same-name"
|
|
components={{
|
|
link: (
|
|
<TimelineWarning.Link
|
|
onClick={() => {
|
|
reviewMessageRequestNameCollision({
|
|
safeConversationId: warning.safeConversation.id,
|
|
});
|
|
}}
|
|
>
|
|
{i18n('ContactSpoofing__same-name__link')}
|
|
</TimelineWarning.Link>
|
|
),
|
|
}}
|
|
/>
|
|
);
|
|
onClose = () => {
|
|
this.setState({
|
|
hasDismissedDirectContactSpoofingWarning: true,
|
|
});
|
|
};
|
|
break;
|
|
case ContactSpoofingType.MultipleGroupMembersWithSameTitle: {
|
|
const { groupNameCollisions } = warning;
|
|
text = (
|
|
<Intl
|
|
i18n={i18n}
|
|
id="ContactSpoofing__same-name-in-group"
|
|
components={{
|
|
count: Object.values(groupNameCollisions)
|
|
.reduce(
|
|
(result, conversations) => result + conversations.length,
|
|
0
|
|
)
|
|
.toString(),
|
|
link: (
|
|
<TimelineWarning.Link
|
|
onClick={() => {
|
|
reviewGroupMemberNameCollision(id);
|
|
}}
|
|
>
|
|
{i18n('ContactSpoofing__same-name-in-group__link')}
|
|
</TimelineWarning.Link>
|
|
),
|
|
}}
|
|
/>
|
|
);
|
|
onClose = () => {
|
|
acknowledgeGroupMemberNameCollisions(groupNameCollisions);
|
|
};
|
|
break;
|
|
}
|
|
default:
|
|
throw missingCaseError(warning);
|
|
}
|
|
|
|
timelineWarning = (
|
|
<Measure
|
|
bounds
|
|
onResize={({ bounds }) => {
|
|
if (!bounds) {
|
|
assert(false, 'We should be measuring the bounds');
|
|
return;
|
|
}
|
|
this.setState({ lastMeasuredWarningHeight: bounds.height });
|
|
}}
|
|
>
|
|
{({ measureRef }) => (
|
|
<TimelineWarnings ref={measureRef}>
|
|
<TimelineWarning i18n={i18n} onClose={onClose}>
|
|
<TimelineWarning.IconContainer>
|
|
<TimelineWarning.GenericIcon />
|
|
</TimelineWarning.IconContainer>
|
|
<TimelineWarning.Text>{text}</TimelineWarning.Text>
|
|
</TimelineWarning>
|
|
</TimelineWarnings>
|
|
)}
|
|
</Measure>
|
|
);
|
|
}
|
|
|
|
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 = (
|
|
<ContactSpoofingReviewDialog
|
|
{...commonProps}
|
|
type={ContactSpoofingType.DirectConversationWithSameTitle}
|
|
possiblyUnsafeConversation={
|
|
contactSpoofingReview.possiblyUnsafeConversation
|
|
}
|
|
safeConversation={contactSpoofingReview.safeConversation}
|
|
/>
|
|
);
|
|
break;
|
|
case ContactSpoofingType.MultipleGroupMembersWithSameTitle:
|
|
contactSpoofingReviewDialog = (
|
|
<ContactSpoofingReviewDialog
|
|
{...commonProps}
|
|
type={ContactSpoofingType.MultipleGroupMembersWithSameTitle}
|
|
areWeAdmin={Boolean(areWeAdmin)}
|
|
collisionInfoByTitle={contactSpoofingReview.collisionInfoByTitle}
|
|
/>
|
|
);
|
|
break;
|
|
default:
|
|
throw missingCaseError(contactSpoofingReview);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<Measure
|
|
bounds
|
|
onResize={({ bounds }) => {
|
|
this.setState({
|
|
widthBreakpoint: getWidthBreakpoint(bounds?.width || 0),
|
|
});
|
|
}}
|
|
>
|
|
{({ measureRef }) => (
|
|
<div
|
|
className={classNames(
|
|
'module-timeline',
|
|
isGroupV1AndDisabled ? 'module-timeline--disabled' : null,
|
|
`module-timeline--width-${widthBreakpoint}`
|
|
)}
|
|
role="presentation"
|
|
tabIndex={-1}
|
|
onBlur={this.handleBlur}
|
|
onKeyDown={this.handleKeyDown}
|
|
ref={this.containerRefMerger(measureRef)}
|
|
>
|
|
{timelineWarning}
|
|
|
|
{floatingHeader}
|
|
|
|
{autoSizer}
|
|
|
|
{shouldShowScrollDownButton ? (
|
|
<ScrollDownButton
|
|
conversationId={id}
|
|
withNewMessages={areUnreadBelowCurrentPosition}
|
|
scrollDown={this.onClickScrollDownButton}
|
|
i18n={i18n}
|
|
/>
|
|
) : null}
|
|
</div>
|
|
)}
|
|
</Measure>
|
|
|
|
{Boolean(invitedContactsForNewlyCreatedGroup.length) && (
|
|
<NewlyCreatedGroupInvitedContactsDialog
|
|
contacts={invitedContactsForNewlyCreatedGroup}
|
|
getPreferredBadge={getPreferredBadge}
|
|
i18n={i18n}
|
|
onClose={clearInvitedUuidsForNewlyCreatedGroup}
|
|
theme={theme}
|
|
/>
|
|
)}
|
|
|
|
{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);
|
|
}
|
|
}
|
|
}
|