signal-desktop/ts/components/conversation/conversation-details/AddGroupMembersModal/ChooseGroupMembersModal.tsx

502 lines
16 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2021 Signal Messenger, LLC
2021-03-11 21:29:31 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import React, {
useEffect,
useMemo,
useState,
useRef,
useCallback,
} from 'react';
2023-03-09 21:46:01 +00:00
import { omit } from 'lodash';
import type { MeasuredComponentProps } from 'react-measure';
import Measure from 'react-measure';
import type { ListRowProps } from 'react-virtualized';
2021-03-11 21:29:31 +00:00
2021-11-02 23:01:13 +00:00
import type { LocalizerType, ThemeType } from '../../../../types/Util';
import { getUsernameFromSearch } from '../../../../types/Username';
import { strictAssert, assertDev } from '../../../../util/assert';
import { refMerger } from '../../../../util/refMerger';
import { useRestoreFocus } from '../../../../hooks/useRestoreFocus';
2021-03-11 21:29:31 +00:00
import { missingCaseError } from '../../../../util/missingCaseError';
import type { LookupConversationWithoutUuidActionsType } from '../../../../util/lookupConversationWithoutUuid';
import { parseAndFormatPhoneNumber } from '../../../../util/libphonenumberInstance';
2023-01-02 17:25:44 +00:00
import type { ParsedE164Type } from '../../../../util/libphonenumberInstance';
import { filterAndSortConversationsByRecent } from '../../../../util/filterAndSortConversations';
import type { ConversationType } from '../../../../state/ducks/conversations';
2021-11-17 21:11:21 +00:00
import type { PreferredBadgeSelectorType } from '../../../../state/selectors/badges';
import type {
UUIDFetchStateKeyType,
UUIDFetchStateType,
} from '../../../../util/uuidFetchState';
2022-06-17 00:38:28 +00:00
import {
isFetchingByE164,
isFetchingByUsername,
} from '../../../../util/uuidFetchState';
2021-03-11 21:29:31 +00:00
import { ModalHost } from '../../../ModalHost';
import { ContactPills } from '../../../ContactPills';
import { ContactPill } from '../../../ContactPill';
import type { Row } from '../../../ConversationList';
import { RowType } from '../../../ConversationList';
import {
ContactCheckbox,
ContactCheckboxDisabledReason,
} from '../../../conversationList/ContactCheckbox';
2021-03-11 21:29:31 +00:00
import { Button, ButtonVariant } from '../../../Button';
2021-05-11 00:50:43 +00:00
import { SearchInput } from '../../../SearchInput';
import { ListView } from '../../../ListView';
import { UsernameCheckbox } from '../../../conversationList/UsernameCheckbox';
import { PhoneNumberCheckbox } from '../../../conversationList/PhoneNumberCheckbox';
2021-03-11 21:29:31 +00:00
export type StatePropsType = {
regionCode: string | undefined;
2021-03-11 21:29:31 +00:00
candidateContacts: ReadonlyArray<ConversationType>;
conversationIdsAlreadyInGroup: Set<string>;
2021-11-17 21:11:21 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
2021-03-11 21:29:31 +00:00
i18n: LocalizerType;
theme: ThemeType;
2021-03-11 21:29:31 +00:00
maxGroupSize: number;
searchTerm: string;
selectedContacts: ReadonlyArray<ConversationType>;
confirmAdds: () => void;
onClose: () => void;
removeSelectedContact: (_: string) => void;
2021-03-11 21:29:31 +00:00
setSearchTerm: (_: string) => void;
toggleSelectedContact: (conversationId: string) => void;
2022-06-17 00:38:28 +00:00
isUsernamesEnabled: boolean;
} & Pick<
LookupConversationWithoutUuidActionsType,
'lookupConversationWithoutUuid'
>;
type ActionPropsType = Omit<
LookupConversationWithoutUuidActionsType,
'setIsFetchingUUID' | 'lookupConversationWithoutUuid'
>;
type PropsType = StatePropsType & ActionPropsType;
2021-03-11 21:29:31 +00:00
// TODO: This should use <Modal>. See DESKTOP-1038.
2022-11-18 00:45:19 +00:00
export function ChooseGroupMembersModal({
regionCode,
2021-03-11 21:29:31 +00:00
candidateContacts,
confirmAdds,
conversationIdsAlreadyInGroup,
i18n,
maxGroupSize,
onClose,
removeSelectedContact,
searchTerm,
selectedContacts,
setSearchTerm,
2021-11-02 23:01:13 +00:00
theme,
2021-03-11 21:29:31 +00:00
toggleSelectedContact,
lookupConversationWithoutUuid,
showUserNotFoundModal,
2022-06-17 00:38:28 +00:00
isUsernamesEnabled,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
const [focusRef] = useRestoreFocus();
2022-06-17 00:38:28 +00:00
let username: string | undefined;
let isUsernameChecked = false;
let isUsernameVisible = false;
2022-10-18 17:12:02 +00:00
if (isUsernamesEnabled) {
2022-06-17 00:38:28 +00:00
username = getUsernameFromSearch(searchTerm);
isUsernameChecked = selectedContacts.some(
contact => contact.username === username
);
2022-07-08 20:46:25 +00:00
isUsernameVisible =
Boolean(username) &&
candidateContacts.every(contact => contact.username !== username);
2022-06-17 00:38:28 +00:00
}
2023-01-02 17:25:44 +00:00
let phoneNumber: ParsedE164Type | undefined;
if (!username) {
phoneNumber = parseAndFormatPhoneNumber(searchTerm, regionCode);
}
2022-10-18 17:12:02 +00:00
let isPhoneNumberChecked = false;
2023-01-02 17:25:44 +00:00
let isPhoneNumberVisible = false;
if (phoneNumber) {
const { e164 } = phoneNumber;
2022-10-18 17:12:02 +00:00
isPhoneNumberChecked =
phoneNumber.isValid &&
2023-01-02 17:25:44 +00:00
selectedContacts.some(contact => contact.e164 === e164);
2022-10-18 17:12:02 +00:00
2023-01-02 17:25:44 +00:00
isPhoneNumberVisible = candidateContacts.every(
contact => contact.e164 !== e164
);
}
2022-10-18 17:12:02 +00:00
2021-03-11 21:29:31 +00:00
const inputRef = useRef<null | HTMLInputElement>(null);
const numberOfContactsAlreadyInGroup = conversationIdsAlreadyInGroup.size;
const hasSelectedMaximumNumberOfContacts =
selectedContacts.length + numberOfContactsAlreadyInGroup >= maxGroupSize;
const selectedConversationIdsSet: Set<string> = useMemo(
() => new Set(selectedContacts.map(contact => contact.id)),
[selectedContacts]
);
const canContinue = Boolean(selectedContacts.length);
const [filteredContacts, setFilteredContacts] = useState(
filterAndSortConversationsByRecent(candidateContacts, '', regionCode)
2021-03-11 21:29:31 +00:00
);
const normalizedSearchTerm = searchTerm.trim();
useEffect(() => {
const timeout = setTimeout(() => {
setFilteredContacts(
filterAndSortConversationsByRecent(
2021-04-28 20:44:48 +00:00
candidateContacts,
normalizedSearchTerm,
regionCode
2021-04-28 20:44:48 +00:00
)
2021-03-11 21:29:31 +00:00
);
}, 200);
return () => {
clearTimeout(timeout);
};
}, [
candidateContacts,
normalizedSearchTerm,
setFilteredContacts,
regionCode,
]);
2021-03-11 21:29:31 +00:00
const [uuidFetchState, setUuidFetchState] = useState<UUIDFetchStateType>({});
const setIsFetchingUUID = useCallback(
(identifier: UUIDFetchStateKeyType, isFetching: boolean) => {
setUuidFetchState(prevState => {
return isFetching
? {
...prevState,
[identifier]: isFetching,
}
: omit(prevState, identifier);
});
},
[setUuidFetchState]
);
let rowCount = 0;
if (filteredContacts.length) {
rowCount += filteredContacts.length;
}
2022-06-17 00:38:28 +00:00
if (isPhoneNumberVisible || isUsernameVisible) {
// "Contacts" header
if (filteredContacts.length) {
rowCount += 1;
}
// "Find by phone number" + phone number
2022-06-17 00:38:28 +00:00
// or
// "Find by username" + username
rowCount += 2;
}
2021-03-11 21:29:31 +00:00
const getRow = (index: number): undefined | Row => {
let virtualIndex = index;
2022-06-17 00:38:28 +00:00
if (
(isPhoneNumberVisible || isUsernameVisible) &&
filteredContacts.length
) {
if (virtualIndex === 0) {
return {
type: RowType.Header,
// eslint-disable-next-line @typescript-eslint/no-shadow
getHeaderText: i18n => i18n('contactsHeader'),
};
}
virtualIndex -= 1;
2021-03-11 21:29:31 +00:00
}
if (virtualIndex < filteredContacts.length) {
const contact = filteredContacts[virtualIndex];
2021-03-11 21:29:31 +00:00
const isSelected = selectedConversationIdsSet.has(contact.id);
const isAlreadyInGroup = conversationIdsAlreadyInGroup.has(contact.id);
let disabledReason: undefined | ContactCheckboxDisabledReason;
if (isAlreadyInGroup) {
disabledReason = ContactCheckboxDisabledReason.AlreadyAdded;
} else if (hasSelectedMaximumNumberOfContacts && !isSelected) {
disabledReason = ContactCheckboxDisabledReason.MaximumContactsSelected;
}
return {
type: RowType.ContactCheckbox,
contact,
isChecked: isSelected || isAlreadyInGroup,
disabledReason,
};
2021-03-11 21:29:31 +00:00
}
virtualIndex -= filteredContacts.length;
if (isPhoneNumberVisible) {
2023-01-02 17:25:44 +00:00
strictAssert(
phoneNumber !== undefined,
"phone number can't be visible if not present"
);
if (virtualIndex === 0) {
return {
type: RowType.Header,
// eslint-disable-next-line @typescript-eslint/no-shadow
getHeaderText: i18n => i18n('findByPhoneNumberHeader'),
};
}
if (virtualIndex === 1) {
return {
type: RowType.PhoneNumberCheckbox,
isChecked: isPhoneNumberChecked,
isFetching: isFetchingByE164(uuidFetchState, phoneNumber.e164),
phoneNumber,
};
}
virtualIndex -= 2;
}
2022-06-17 00:38:28 +00:00
if (username) {
if (virtualIndex === 0) {
return {
type: RowType.Header,
// eslint-disable-next-line @typescript-eslint/no-shadow
getHeaderText: i18n => i18n('findByUsernameHeader'),
2022-06-17 00:38:28 +00:00
};
}
if (virtualIndex === 1) {
return {
type: RowType.UsernameCheckbox,
isChecked: isUsernameChecked,
isFetching: isFetchingByUsername(uuidFetchState, username),
username,
};
}
virtualIndex -= 2;
}
return undefined;
2021-03-11 21:29:31 +00:00
};
const handleContactClick = (
conversationId: string,
disabledReason: undefined | ContactCheckboxDisabledReason
) => {
switch (disabledReason) {
case undefined:
toggleSelectedContact(conversationId);
break;
case ContactCheckboxDisabledReason.AlreadyAdded:
case ContactCheckboxDisabledReason.MaximumContactsSelected:
// These are no-ops.
break;
default:
throw missingCaseError(disabledReason);
}
};
const renderItem = ({ key, index, style }: ListRowProps) => {
const row = getRow(index);
let item;
switch (row?.type) {
case RowType.Header: {
const headerText = row.getHeaderText(i18n);
item = (
<div
className="module-conversation-list__item--header"
aria-label={headerText}
>
{headerText}
</div>
);
break;
}
case RowType.ContactCheckbox:
item = (
<ContactCheckbox
i18n={i18n}
theme={theme}
{...row.contact}
onClick={handleContactClick}
isChecked={row.isChecked}
badge={undefined}
/>
);
break;
case RowType.UsernameCheckbox:
item = (
<UsernameCheckbox
i18n={i18n}
theme={theme}
username={row.username}
isChecked={row.isChecked}
isFetching={row.isFetching}
toggleConversationInChooseMembers={conversationId =>
handleContactClick(conversationId, undefined)
}
2023-03-09 21:46:01 +00:00
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
2023-03-09 21:46:01 +00:00
lookupConversationWithoutUuid={lookupConversationWithoutUuid}
/>
);
break;
case RowType.PhoneNumberCheckbox:
item = (
<PhoneNumberCheckbox
phoneNumber={row.phoneNumber}
lookupConversationWithoutUuid={lookupConversationWithoutUuid}
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
toggleConversationInChooseMembers={conversationId =>
handleContactClick(conversationId, undefined)
}
isChecked={row.isChecked}
isFetching={row.isFetching}
i18n={i18n}
theme={theme}
/>
);
break;
default:
}
return (
<div key={key} style={style}>
{item}
</div>
);
};
2021-03-11 21:29:31 +00:00
return (
2022-09-27 20:24:21 +00:00
<ModalHost
modalName="AddGroupMembersModal.ChooseGroupMembersModal"
onClose={onClose}
>
2021-03-11 21:29:31 +00:00
<div className="module-AddGroupMembersModal module-AddGroupMembersModal--choose-members">
<button
aria-label={i18n('close')}
className="module-AddGroupMembersModal__close-button"
type="button"
onClick={() => {
onClose();
}}
/>
<h1 className="module-AddGroupMembersModal__header">
{i18n('AddGroupMembersModal--title')}
</h1>
2021-05-11 00:50:43 +00:00
<SearchInput
2022-02-14 17:57:11 +00:00
i18n={i18n}
2021-03-11 21:29:31 +00:00
placeholder={i18n('contactSearchPlaceholder')}
onChange={event => {
setSearchTerm(event.target.value);
}}
onKeyDown={event => {
if (canContinue && event.key === 'Enter') {
confirmAdds();
}
}}
ref={refMerger<HTMLInputElement>(inputRef, focusRef)}
2021-03-11 21:29:31 +00:00
value={searchTerm}
/>
{Boolean(selectedContacts.length) && (
<ContactPills>
{selectedContacts.map(contact => (
<ContactPill
key={contact.id}
acceptedMessageRequest={contact.acceptedMessageRequest}
2021-03-11 21:29:31 +00:00
avatarPath={contact.avatarPath}
color={contact.color}
firstName={contact.systemGivenName ?? contact.firstName}
2021-03-11 21:29:31 +00:00
i18n={i18n}
2021-05-07 22:21:10 +00:00
isMe={contact.isMe}
2021-03-11 21:29:31 +00:00
id={contact.id}
phoneNumber={contact.phoneNumber}
profileName={contact.profileName}
2021-05-07 22:21:10 +00:00
sharedGroupNames={contact.sharedGroupNames}
2021-03-11 21:29:31 +00:00
title={contact.title}
onClickRemove={() => {
removeSelectedContact(contact.id);
}}
/>
))}
</ContactPills>
)}
{rowCount ? (
2021-03-11 21:29:31 +00:00
<Measure bounds>
{({ contentRect, measureRef }: MeasuredComponentProps) => {
// Though `width` and `height` are required properties, we want to be
// careful in case the caller sends bogus data. Notably, react-measure's
// types seem to be inaccurate.
const { width = 100, height = 100 } = contentRect.bounds || {};
if (!width || !height) {
return null;
}
2021-03-11 21:29:31 +00:00
// We disable this ESLint rule because we're capturing a bubbled keydown
// event. See [this note in the jsx-a11y docs][0].
//
// [0]: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/c275964f52c35775208bd00cb612c6f82e42e34f/docs/rules/no-static-element-interactions.md#case-the-event-handler-is-only-being-used-to-capture-bubbled-events
/* eslint-disable jsx-a11y/no-static-element-interactions */
return (
<div
className="module-AddGroupMembersModal__list-wrapper"
ref={measureRef}
onKeyDown={event => {
if (event.key === 'Enter') {
inputRef.current?.focus();
}
}}
>
<ListView
width={width}
height={height}
rowCount={rowCount}
calculateRowHeight={index => {
const row = getRow(index);
if (!row) {
assertDev(false, `Expected a row at index ${index}`);
return 52;
}
switch (row.type) {
case RowType.Header:
return 40;
2021-03-11 21:29:31 +00:00
default:
return 52;
2021-03-11 21:29:31 +00:00
}
}}
rowRenderer={renderItem}
2021-03-11 21:29:31 +00:00
/>
</div>
);
/* eslint-enable jsx-a11y/no-static-element-interactions */
}}
</Measure>
) : (
<div className="module-AddGroupMembersModal__no-candidate-contacts">
{i18n('noContactsFound')}
</div>
)}
<div className="module-AddGroupMembersModal__button-container">
<Button onClick={onClose} variant={ButtonVariant.Secondary}>
{i18n('cancel')}
</Button>
<Button disabled={!canContinue} onClick={confirmAdds}>
{i18n('AddGroupMembersModal--continue-to-confirm')}
</Button>
</div>
</div>
</ModalHost>
);
2022-11-18 00:45:19 +00:00
}