signal-desktop/ts/components/ConversationList.tsx

600 lines
19 KiB
TypeScript
Raw Normal View History

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactNode } from 'react';
import React, { useCallback } from 'react';
import type { ListRowRenderer } from 'react-virtualized';
import classNames from 'classnames';
import { get, pick } from 'lodash';
import { missingCaseError } from '../util/missingCaseError';
import { assertDev } from '../util/assert';
import type { ParsedE164Type } from '../util/libphonenumberInstance';
2021-11-02 23:01:13 +00:00
import type { LocalizerType, ThemeType } from '../types/Util';
import { ScrollBehavior } from '../types/Util';
2023-08-09 00:53:06 +00:00
import { getNavSidebarWidthBreakpoint } from './_util';
2021-11-17 21:11:21 +00:00
import type { PreferredBadgeSelectorType } from '../state/selectors/badges';
import type { LookupConversationWithoutServiceIdActionsType } from '../util/lookupConversationWithoutServiceId';
2022-06-16 19:12:50 +00:00
import type { ShowConversationType } from '../state/ducks/conversations';
import type { PropsData as ConversationListItemPropsType } from './conversationList/ConversationListItem';
import type { ContactCheckboxDisabledReason } from './conversationList/ContactCheckbox';
2021-11-17 21:11:21 +00:00
import type { ContactListItemConversationType as ContactListItemPropsType } from './conversationList/ContactListItem';
import type { GroupListItemConversationType } from './conversationList/GroupListItem';
import { ConversationListItem } from './conversationList/ConversationListItem';
import { ContactListItem } from './conversationList/ContactListItem';
import { ContactCheckbox as ContactCheckboxComponent } from './conversationList/ContactCheckbox';
import { PhoneNumberCheckbox as PhoneNumberCheckboxComponent } from './conversationList/PhoneNumberCheckbox';
2022-06-17 00:38:28 +00:00
import { UsernameCheckbox as UsernameCheckboxComponent } from './conversationList/UsernameCheckbox';
import {
ComposeStepButton,
Icon as ComposeStepButtonIcon,
} from './conversationList/ComposeStepButton';
import { StartNewConversation as StartNewConversationComponent } from './conversationList/StartNewConversation';
import { SearchResultsLoadingFakeHeader as SearchResultsLoadingFakeHeaderComponent } from './conversationList/SearchResultsLoadingFakeHeader';
import { SearchResultsLoadingFakeRow as SearchResultsLoadingFakeRowComponent } from './conversationList/SearchResultsLoadingFakeRow';
2021-11-12 01:17:29 +00:00
import { UsernameSearchResultListItem } from './conversationList/UsernameSearchResultListItem';
import { GroupListItem } from './conversationList/GroupListItem';
import { ListView } from './ListView';
2024-11-13 20:24:00 +00:00
import { Button, ButtonVariant } from './Button';
export enum RowType {
2022-06-17 00:38:28 +00:00
ArchiveButton = 'ArchiveButton',
Blank = 'Blank',
Contact = 'Contact',
2024-11-13 20:24:00 +00:00
ClearFilterButton = 'ClearFilterButton',
2022-06-17 00:38:28 +00:00
ContactCheckbox = 'ContactCheckbox',
PhoneNumberCheckbox = 'PhoneNumberCheckbox',
UsernameCheckbox = 'UsernameCheckbox',
Conversation = 'Conversation',
CreateNewGroup = 'CreateNewGroup',
FindByUsername = 'FindByUsername',
FindByPhoneNumber = 'FindByPhoneNumber',
2022-06-17 00:38:28 +00:00
Header = 'Header',
MessageSearchResult = 'MessageSearchResult',
SearchResultsLoadingFakeHeader = 'SearchResultsLoadingFakeHeader',
SearchResultsLoadingFakeRow = 'SearchResultsLoadingFakeRow',
// this could later be expanded to SelectSingleConversation
SelectSingleGroup = 'SelectSingleGroup',
2022-06-17 00:38:28 +00:00
StartNewConversation = 'StartNewConversation',
UsernameSearchResult = 'UsernameSearchResult',
}
type ArchiveButtonRowType = {
type: RowType.ArchiveButton;
archivedConversationsCount: number;
};
2021-03-03 20:09:58 +00:00
type BlankRowType = { type: RowType.Blank };
type ContactRowType = {
type: RowType.Contact;
contact: ContactListItemPropsType;
2021-03-03 20:09:58 +00:00
isClickable?: boolean;
2023-04-05 20:48:00 +00:00
hasContextMenu?: boolean;
2021-03-03 20:09:58 +00:00
};
2024-11-13 20:24:00 +00:00
type ClearFilterButtonRowType = {
type: RowType.ClearFilterButton;
isOnNoResultsPage: boolean;
};
2021-03-03 20:09:58 +00:00
type ContactCheckboxRowType = {
type: RowType.ContactCheckbox;
contact: ContactListItemPropsType;
isChecked: boolean;
disabledReason?: ContactCheckboxDisabledReason;
};
type PhoneNumberCheckboxRowType = {
type: RowType.PhoneNumberCheckbox;
phoneNumber: ParsedE164Type;
isChecked: boolean;
isFetching: boolean;
};
2022-06-17 00:38:28 +00:00
type UsernameCheckboxRowType = {
type: RowType.UsernameCheckbox;
username: string;
isChecked: boolean;
isFetching: boolean;
};
type ConversationRowType = {
type: RowType.Conversation;
conversation: ConversationListItemPropsType;
};
2021-03-03 20:09:58 +00:00
type CreateNewGroupRowType = {
type: RowType.CreateNewGroup;
};
type FindByUsername = {
type: RowType.FindByUsername;
};
type FindByPhoneNumber = {
type: RowType.FindByPhoneNumber;
};
type MessageRowType = {
type: RowType.MessageSearchResult;
messageId: string;
};
type HeaderRowType = {
type: RowType.Header;
getHeaderText: (i18n: LocalizerType) => string;
};
// Exported for tests across multiple files
export function _testHeaderText(row: Row | void): string | null {
if (row?.type === RowType.Header) {
return row.getHeaderText(((key: string) => key) as LocalizerType);
}
return null;
}
type SearchResultsLoadingFakeHeaderType = {
type: RowType.SearchResultsLoadingFakeHeader;
};
type SearchResultsLoadingFakeRowType = {
type: RowType.SearchResultsLoadingFakeRow;
};
type SelectSingleGroupRowType = {
type: RowType.SelectSingleGroup;
group: GroupListItemConversationType;
};
type StartNewConversationRowType = {
type: RowType.StartNewConversation;
phoneNumber: ParsedE164Type;
isFetching: boolean;
};
2021-11-12 01:17:29 +00:00
type UsernameRowType = {
type: RowType.UsernameSearchResult;
username: string;
isFetchingUsername: boolean;
};
export type Row =
| ArchiveButtonRowType
2021-03-03 20:09:58 +00:00
| BlankRowType
| ContactRowType
2021-03-03 20:09:58 +00:00
| ContactCheckboxRowType
2024-11-13 20:24:00 +00:00
| ClearFilterButtonRowType
| PhoneNumberCheckboxRowType
2022-06-17 00:38:28 +00:00
| UsernameCheckboxRowType
| ConversationRowType
2021-03-03 20:09:58 +00:00
| CreateNewGroupRowType
| FindByUsername
| FindByPhoneNumber
| MessageRowType
| HeaderRowType
| SearchResultsLoadingFakeHeaderType
| SearchResultsLoadingFakeRowType
2021-11-12 01:17:29 +00:00
| StartNewConversationRowType
| SelectSingleGroupRowType
2021-11-12 01:17:29 +00:00
| UsernameRowType;
export type PropsType = {
dimensions?: {
width: number;
height: number;
};
rowCount: number;
// If `getRow` is called with an invalid index, it should return `undefined`. However,
// this should only happen if there is a bug somewhere. For example, an inaccurate
// `rowCount`.
getRow: (index: number) => undefined | Row;
scrollBehavior?: ScrollBehavior;
scrollToRowIndex?: number;
shouldRecomputeRowHeights: boolean;
scrollable?: boolean;
2021-11-17 21:11:21 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
i18n: LocalizerType;
2021-11-02 23:01:13 +00:00
theme: ThemeType;
2023-04-05 20:48:00 +00:00
blockConversation: (conversationId: string) => void;
onClickArchiveButton: () => void;
2021-03-03 20:09:58 +00:00
onClickContactCheckbox: (
conversationId: string,
disabledReason: undefined | ContactCheckboxDisabledReason
) => void;
2024-11-13 20:24:00 +00:00
onClickClearFilterButton: () => void;
onPreloadConversation: (conversationId: string, messageId?: string) => void;
2021-03-03 20:09:58 +00:00
onSelectConversation: (conversationId: string, messageId?: string) => void;
2023-04-05 20:48:00 +00:00
onOutgoingAudioCallInConversation: (conversationId: string) => void;
onOutgoingVideoCallInConversation: (conversationId: string) => void;
2023-10-25 23:01:16 +00:00
removeConversation: (conversationId: string) => void;
2022-11-18 00:45:19 +00:00
renderMessageSearchResult?: (id: string) => JSX.Element;
2021-03-03 20:09:58 +00:00
showChooseGroupMembers: () => void;
showFindByUsername: () => void;
showFindByPhoneNumber: () => void;
2022-06-16 19:12:50 +00:00
showConversation: ShowConversationType;
} & LookupConversationWithoutServiceIdActionsType;
const NORMAL_ROW_HEIGHT = 76;
const SELECT_ROW_HEIGHT = 52;
const HEADER_ROW_HEIGHT = 40;
2022-11-18 00:45:19 +00:00
export function ConversationList({
dimensions,
2021-11-17 21:11:21 +00:00
getPreferredBadge,
getRow,
i18n,
2023-04-05 20:48:00 +00:00
blockConversation,
onClickArchiveButton,
2021-03-03 20:09:58 +00:00
onClickContactCheckbox,
2024-11-13 20:24:00 +00:00
onClickClearFilterButton,
onPreloadConversation,
onSelectConversation,
2023-04-05 20:48:00 +00:00
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
removeConversation,
renderMessageSearchResult,
rowCount,
scrollBehavior = ScrollBehavior.Default,
scrollToRowIndex,
scrollable = true,
shouldRecomputeRowHeights,
2021-03-03 20:09:58 +00:00
showChooseGroupMembers,
showFindByUsername,
showFindByPhoneNumber,
lookupConversationWithoutServiceId,
showUserNotFoundModal,
setIsFetchingUUID,
showConversation,
2021-11-02 23:01:13 +00:00
theme,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element | null {
const calculateRowHeight = useCallback(
(index: number): number => {
const row = getRow(index);
if (!row) {
assertDev(false, `Expected a row at index ${index}`);
return NORMAL_ROW_HEIGHT;
}
switch (row.type) {
case RowType.Header:
case RowType.SearchResultsLoadingFakeHeader:
return HEADER_ROW_HEIGHT;
case RowType.SelectSingleGroup:
case RowType.ContactCheckbox:
case RowType.Contact:
case RowType.CreateNewGroup:
case RowType.FindByUsername:
case RowType.FindByPhoneNumber:
return SELECT_ROW_HEIGHT;
default:
return NORMAL_ROW_HEIGHT;
}
},
[getRow]
);
const renderRow: ListRowRenderer = useCallback(
({ key, index, style }) => {
const row = getRow(index);
if (!row) {
assertDev(false, `Expected a row at index ${index}`);
return <div key={key} style={style} />;
}
2021-08-11 16:23:21 +00:00
let result: ReactNode;
switch (row.type) {
case RowType.ArchiveButton:
2021-08-11 16:23:21 +00:00
result = (
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:archivedConversations')}
className="module-conversation-list__item--archive-button"
onClick={onClickArchiveButton}
type="button"
>
<div className="module-conversation-list__item--archive-button__icon" />
<span className="module-conversation-list__item--archive-button__text">
2023-03-30 00:03:25 +00:00
{i18n('icu:archivedConversations')}
</span>
<span className="module-conversation-list__item--archive-button__archived-count">
{row.archivedConversationsCount}
</span>
</button>
);
2021-08-11 16:23:21 +00:00
break;
2021-03-03 20:09:58 +00:00
case RowType.Blank:
2022-11-18 00:45:19 +00:00
result = undefined;
2021-08-11 16:23:21 +00:00
break;
2021-03-03 20:09:58 +00:00
case RowType.Contact: {
2023-04-05 20:48:00 +00:00
const { isClickable = true, hasContextMenu = false } = row;
2021-08-11 16:23:21 +00:00
result = (
<ContactListItem
{...row.contact}
2021-11-17 21:11:21 +00:00
badge={getPreferredBadge(row.contact.badges)}
2021-03-03 20:09:58 +00:00
onClick={isClickable ? onSelectConversation : undefined}
i18n={i18n}
2021-11-17 21:11:21 +00:00
theme={theme}
2023-04-05 20:48:00 +00:00
hasContextMenu={hasContextMenu}
onAudioCall={
isClickable ? onOutgoingAudioCallInConversation : undefined
}
onVideoCall={
isClickable ? onOutgoingVideoCallInConversation : undefined
}
onBlock={isClickable ? blockConversation : undefined}
onRemove={isClickable ? removeConversation : undefined}
2021-03-03 20:09:58 +00:00
/>
);
2021-08-11 16:23:21 +00:00
break;
2021-03-03 20:09:58 +00:00
}
case RowType.ContactCheckbox:
2021-08-11 16:23:21 +00:00
result = (
2021-03-03 20:09:58 +00:00
<ContactCheckboxComponent
{...row.contact}
2021-11-17 21:11:21 +00:00
badge={getPreferredBadge(row.contact.badges)}
2021-03-03 20:09:58 +00:00
isChecked={row.isChecked}
disabledReason={row.disabledReason}
onClick={onClickContactCheckbox}
i18n={i18n}
2021-11-17 21:11:21 +00:00
theme={theme}
/>
);
2021-08-11 16:23:21 +00:00
break;
2024-11-13 20:24:00 +00:00
case RowType.ClearFilterButton:
result = (
<div className="ClearFilterButton module-conversation-list__item--clear-filter-button">
<Button
variant={ButtonVariant.SecondaryAffirmative}
className={classNames('ClearFilterButton__inner', {
// The clear filter button should be closer to the empty state
// text than to the search results.
'ClearFilterButton__inner-vertical-center':
!row.isOnNoResultsPage,
})}
onClick={onClickClearFilterButton}
>
{i18n('icu:clearFilterButton')}
</Button>
</div>
);
break;
case RowType.PhoneNumberCheckbox:
result = (
<PhoneNumberCheckboxComponent
phoneNumber={row.phoneNumber}
lookupConversationWithoutServiceId={
lookupConversationWithoutServiceId
}
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
toggleConversationInChooseMembers={conversationId =>
onClickContactCheckbox(conversationId, undefined)
}
isChecked={row.isChecked}
isFetching={row.isFetching}
2022-06-17 00:38:28 +00:00
i18n={i18n}
theme={theme}
/>
);
break;
case RowType.UsernameCheckbox:
result = (
<UsernameCheckboxComponent
username={row.username}
lookupConversationWithoutServiceId={
lookupConversationWithoutServiceId
}
2022-06-17 00:38:28 +00:00
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
toggleConversationInChooseMembers={conversationId =>
onClickContactCheckbox(conversationId, undefined)
}
isChecked={row.isChecked}
isFetching={row.isFetching}
i18n={i18n}
theme={theme}
/>
);
break;
2021-08-11 16:23:21 +00:00
case RowType.Conversation: {
const itemProps = pick(row.conversation, [
'acceptedMessageRequest',
2024-07-11 19:44:09 +00:00
'avatarUrl',
2021-11-02 23:01:13 +00:00
'badges',
2021-08-11 16:23:21 +00:00
'color',
'draftPreview',
2023-01-13 00:24:59 +00:00
'groupId',
2021-08-11 16:23:21 +00:00
'id',
2024-04-02 16:41:28 +00:00
'isBlocked',
2021-08-11 16:23:21 +00:00
'isMe',
'isSelected',
2022-01-31 22:01:34 +00:00
'isPinned',
2021-08-11 16:23:21 +00:00
'lastMessage',
'lastUpdated',
'markedUnread',
'muteExpiresAt',
'phoneNumber',
'profileName',
2023-04-05 20:48:00 +00:00
'removalStage',
2021-08-11 16:23:21 +00:00
'sharedGroupNames',
'shouldShowDraft',
'title',
'type',
2023-09-27 21:23:52 +00:00
'typingContactIdTimestamps',
2024-07-11 19:44:09 +00:00
'unblurredAvatarUrl',
2021-08-11 16:23:21 +00:00
'unreadCount',
'unreadMentionsCount',
2023-08-16 20:54:39 +00:00
'serviceId',
2021-08-11 16:23:21 +00:00
]);
2021-11-02 23:01:13 +00:00
const { badges, title, unreadCount, lastMessage } = itemProps;
2021-08-11 16:23:21 +00:00
result = (
2023-04-21 21:23:30 +00:00
<ConversationListItem
{...itemProps}
buttonAriaLabel={i18n('icu:ConversationList__aria-label', {
lastMessage:
get(lastMessage, 'text') ||
2023-03-30 00:03:25 +00:00
i18n('icu:ConversationList__last-message-undefined'),
title,
2024-03-04 18:03:11 +00:00
unreadCount: unreadCount ?? 0,
})}
2023-04-21 21:23:30 +00:00
key={key}
badge={getPreferredBadge(badges)}
onMouseDown={onPreloadConversation}
2023-04-21 21:23:30 +00:00
onClick={onSelectConversation}
i18n={i18n}
theme={theme}
/>
);
2021-08-11 16:23:21 +00:00
break;
}
2021-03-03 20:09:58 +00:00
case RowType.CreateNewGroup:
2021-08-11 16:23:21 +00:00
result = (
<ComposeStepButton
icon={ComposeStepButtonIcon.Group}
title={i18n('icu:createNewGroupButton')}
2021-03-03 20:09:58 +00:00
onClick={showChooseGroupMembers}
/>
);
2021-08-11 16:23:21 +00:00
break;
case RowType.FindByUsername:
result = (
<ComposeStepButton
icon={ComposeStepButtonIcon.Username}
title={i18n('icu:LeftPane__compose__findByUsername')}
onClick={showFindByUsername}
/>
);
break;
case RowType.FindByPhoneNumber:
result = (
<ComposeStepButton
icon={ComposeStepButtonIcon.PhoneNumber}
title={i18n('icu:LeftPane__compose__findByPhoneNumber')}
onClick={showFindByPhoneNumber}
/>
);
break;
case RowType.Header: {
const headerText = row.getHeaderText(i18n);
2021-08-11 16:23:21 +00:00
result = (
<div
className="module-conversation-list__item--header"
aria-label={headerText}
>
{headerText}
</div>
);
2021-08-11 16:23:21 +00:00
break;
}
case RowType.MessageSearchResult:
2022-11-18 00:45:19 +00:00
result = <>{renderMessageSearchResult?.(row.messageId)}</>;
2021-08-11 16:23:21 +00:00
break;
case RowType.SearchResultsLoadingFakeHeader:
2021-08-11 16:23:21 +00:00
result = <SearchResultsLoadingFakeHeaderComponent />;
break;
case RowType.SearchResultsLoadingFakeRow:
2021-08-11 16:23:21 +00:00
result = <SearchResultsLoadingFakeRowComponent />;
break;
case RowType.SelectSingleGroup:
result = (
<GroupListItem
i18n={i18n}
group={row.group}
onSelectGroup={onSelectConversation}
/>
);
break;
case RowType.StartNewConversation:
2021-08-11 16:23:21 +00:00
result = (
<StartNewConversationComponent
i18n={i18n}
phoneNumber={row.phoneNumber}
isFetching={row.isFetching}
lookupConversationWithoutServiceId={
lookupConversationWithoutServiceId
}
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
showConversation={showConversation}
/>
);
2021-08-11 16:23:21 +00:00
break;
2021-11-12 01:17:29 +00:00
case RowType.UsernameSearchResult:
result = (
<UsernameSearchResultListItem
i18n={i18n}
username={row.username}
isFetchingUsername={row.isFetchingUsername}
lookupConversationWithoutServiceId={
lookupConversationWithoutServiceId
}
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
showConversation={showConversation}
2021-11-12 01:17:29 +00:00
/>
);
break;
default:
throw missingCaseError(row);
}
2021-08-11 16:23:21 +00:00
return (
<span aria-rowindex={index + 1} role="row" style={style} key={key}>
2021-10-12 15:23:13 +00:00
<span role="gridcell" aria-colindex={1}>
{result}
2021-10-12 15:23:13 +00:00
</span>
2021-08-11 16:23:21 +00:00
</span>
);
},
[
2023-04-05 20:48:00 +00:00
blockConversation,
2021-11-17 21:11:21 +00:00
getPreferredBadge,
getRow,
i18n,
lookupConversationWithoutServiceId,
onClickArchiveButton,
2024-11-13 20:24:00 +00:00
onClickClearFilterButton,
2021-03-03 20:09:58 +00:00
onClickContactCheckbox,
2023-04-05 20:48:00 +00:00
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
onPreloadConversation,
onSelectConversation,
2023-04-05 20:48:00 +00:00
removeConversation,
renderMessageSearchResult,
2023-04-05 20:48:00 +00:00
setIsFetchingUUID,
2021-03-03 20:09:58 +00:00
showChooseGroupMembers,
showFindByUsername,
showFindByPhoneNumber,
showConversation,
2023-04-05 20:48:00 +00:00
showUserNotFoundModal,
2021-11-02 23:01:13 +00:00
theme,
]
);
if (dimensions == null) {
return null;
}
2023-08-09 00:53:06 +00:00
const widthBreakpoint = getNavSidebarWidthBreakpoint(dimensions.width);
return (
<ListView
className={classNames(
'module-conversation-list',
`module-conversation-list--width-${widthBreakpoint}`
)}
width={dimensions.width}
height={dimensions.height}
rowCount={rowCount}
calculateRowHeight={calculateRowHeight}
rowRenderer={renderRow}
scrollToIndex={scrollToRowIndex}
shouldRecomputeRowHeights={shouldRecomputeRowHeights}
scrollable={scrollable}
scrollBehavior={scrollBehavior}
/>
);
2022-11-18 00:45:19 +00:00
}