2021-03-03 20:09:58 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ReactChild, ChangeEvent } from 'react';
|
|
|
|
import React from 'react';
|
2021-03-03 20:09:58 +00:00
|
|
|
|
|
|
|
import { LeftPaneHelper } from './LeftPaneHelper';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { Row } from '../ConversationList';
|
|
|
|
import { RowType } from '../ConversationList';
|
|
|
|
import type { ConversationType } from '../../state/ducks/conversations';
|
2021-03-03 20:09:58 +00:00
|
|
|
import { ContactCheckboxDisabledReason } from '../conversationList/ContactCheckbox';
|
|
|
|
import { ContactPills } from '../ContactPills';
|
|
|
|
import { ContactPill } from '../ContactPill';
|
2021-05-11 00:50:43 +00:00
|
|
|
import { SearchInput } from '../SearchInput';
|
2021-03-11 21:29:31 +00:00
|
|
|
import {
|
|
|
|
AddGroupMemberErrorDialog,
|
|
|
|
AddGroupMemberErrorDialogMode,
|
|
|
|
} from '../AddGroupMemberErrorDialog';
|
2021-03-03 20:09:58 +00:00
|
|
|
import { Button } from '../Button';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { LocalizerType } from '../../types/Util';
|
2021-03-03 20:09:58 +00:00
|
|
|
import {
|
|
|
|
getGroupSizeRecommendedLimit,
|
|
|
|
getGroupSizeHardLimit,
|
|
|
|
} from '../../groups/limits';
|
|
|
|
|
|
|
|
export type LeftPaneChooseGroupMembersPropsType = {
|
|
|
|
candidateContacts: ReadonlyArray<ConversationType>;
|
|
|
|
cantAddContactForModal: undefined | ConversationType;
|
|
|
|
isShowingRecommendedGroupSizeModal: boolean;
|
|
|
|
isShowingMaximumGroupSizeModal: boolean;
|
|
|
|
searchTerm: string;
|
|
|
|
selectedContacts: Array<ConversationType>;
|
|
|
|
};
|
|
|
|
|
2021-04-26 16:38:50 +00:00
|
|
|
export class LeftPaneChooseGroupMembersHelper extends LeftPaneHelper<LeftPaneChooseGroupMembersPropsType> {
|
2021-03-03 20:09:58 +00:00
|
|
|
private readonly candidateContacts: ReadonlyArray<ConversationType>;
|
|
|
|
|
|
|
|
private readonly cantAddContactForModal:
|
|
|
|
| undefined
|
|
|
|
| Readonly<{ title: string }>;
|
|
|
|
|
|
|
|
private readonly isShowingMaximumGroupSizeModal: boolean;
|
|
|
|
|
|
|
|
private readonly isShowingRecommendedGroupSizeModal: boolean;
|
|
|
|
|
|
|
|
private readonly searchTerm: string;
|
|
|
|
|
|
|
|
private readonly selectedContacts: Array<ConversationType>;
|
|
|
|
|
|
|
|
private readonly selectedConversationIdsSet: Set<string>;
|
|
|
|
|
|
|
|
constructor({
|
|
|
|
candidateContacts,
|
|
|
|
cantAddContactForModal,
|
|
|
|
isShowingMaximumGroupSizeModal,
|
|
|
|
isShowingRecommendedGroupSizeModal,
|
|
|
|
searchTerm,
|
|
|
|
selectedContacts,
|
|
|
|
}: Readonly<LeftPaneChooseGroupMembersPropsType>) {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.candidateContacts = candidateContacts;
|
|
|
|
this.cantAddContactForModal = cantAddContactForModal;
|
|
|
|
this.isShowingMaximumGroupSizeModal = isShowingMaximumGroupSizeModal;
|
|
|
|
this.isShowingRecommendedGroupSizeModal = isShowingRecommendedGroupSizeModal;
|
|
|
|
this.searchTerm = searchTerm;
|
|
|
|
this.selectedContacts = selectedContacts;
|
|
|
|
|
|
|
|
this.selectedConversationIdsSet = new Set(
|
|
|
|
selectedContacts.map(contact => contact.id)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
getHeaderContents({
|
|
|
|
i18n,
|
|
|
|
startComposing,
|
|
|
|
}: Readonly<{
|
|
|
|
i18n: LocalizerType;
|
|
|
|
startComposing: () => void;
|
|
|
|
}>): ReactChild {
|
|
|
|
const backButtonLabel = i18n('chooseGroupMembers__back-button');
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="module-left-pane__header__contents">
|
|
|
|
<button
|
|
|
|
aria-label={backButtonLabel}
|
|
|
|
className="module-left-pane__header__contents__back-button"
|
2021-04-02 21:43:39 +00:00
|
|
|
onClick={this.getBackAction({ startComposing })}
|
2021-03-03 20:09:58 +00:00
|
|
|
title={backButtonLabel}
|
|
|
|
type="button"
|
|
|
|
/>
|
|
|
|
<div className="module-left-pane__header__contents__text">
|
|
|
|
{i18n('chooseGroupMembers__title')}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-04-02 21:43:39 +00:00
|
|
|
getBackAction({
|
|
|
|
startComposing,
|
|
|
|
}: {
|
|
|
|
startComposing: () => void;
|
|
|
|
}): () => void {
|
|
|
|
return startComposing;
|
|
|
|
}
|
|
|
|
|
2021-03-03 20:09:58 +00:00
|
|
|
getPreRowsNode({
|
|
|
|
closeCantAddContactToGroupModal,
|
|
|
|
closeMaximumGroupSizeModal,
|
|
|
|
closeRecommendedGroupSizeModal,
|
|
|
|
i18n,
|
|
|
|
onChangeComposeSearchTerm,
|
|
|
|
removeSelectedContact,
|
|
|
|
}: Readonly<{
|
|
|
|
closeCantAddContactToGroupModal: () => unknown;
|
|
|
|
closeMaximumGroupSizeModal: () => unknown;
|
|
|
|
closeRecommendedGroupSizeModal: () => unknown;
|
|
|
|
i18n: LocalizerType;
|
|
|
|
onChangeComposeSearchTerm: (
|
|
|
|
event: ChangeEvent<HTMLInputElement>
|
|
|
|
) => unknown;
|
|
|
|
removeSelectedContact: (conversationId: string) => unknown;
|
|
|
|
}>): ReactChild {
|
2021-03-11 21:29:31 +00:00
|
|
|
let modalNode: undefined | ReactChild;
|
2021-03-03 20:09:58 +00:00
|
|
|
if (this.isShowingMaximumGroupSizeModal) {
|
2021-03-11 21:29:31 +00:00
|
|
|
modalNode = (
|
|
|
|
<AddGroupMemberErrorDialog
|
|
|
|
i18n={i18n}
|
|
|
|
maximumNumberOfContacts={this.getMaximumNumberOfContacts()}
|
|
|
|
mode={AddGroupMemberErrorDialogMode.MaximumGroupSize}
|
|
|
|
onClose={closeMaximumGroupSizeModal}
|
|
|
|
/>
|
|
|
|
);
|
2021-03-03 20:09:58 +00:00
|
|
|
} else if (this.isShowingRecommendedGroupSizeModal) {
|
2021-03-11 21:29:31 +00:00
|
|
|
modalNode = (
|
|
|
|
<AddGroupMemberErrorDialog
|
|
|
|
i18n={i18n}
|
|
|
|
recommendedMaximumNumberOfContacts={this.getRecommendedMaximumNumberOfContacts()}
|
|
|
|
mode={AddGroupMemberErrorDialogMode.RecommendedMaximumGroupSize}
|
|
|
|
onClose={closeRecommendedGroupSizeModal}
|
|
|
|
/>
|
|
|
|
);
|
2021-03-03 20:09:58 +00:00
|
|
|
} else if (this.cantAddContactForModal) {
|
2021-03-11 21:29:31 +00:00
|
|
|
modalNode = (
|
|
|
|
<AddGroupMemberErrorDialog
|
|
|
|
i18n={i18n}
|
|
|
|
contact={this.cantAddContactForModal}
|
|
|
|
mode={AddGroupMemberErrorDialogMode.CantAddContact}
|
|
|
|
onClose={closeCantAddContactToGroupModal}
|
|
|
|
/>
|
|
|
|
);
|
2021-03-03 20:09:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
2021-05-11 00:50:43 +00:00
|
|
|
<SearchInput
|
|
|
|
moduleClassName="module-left-pane__compose-search-form"
|
|
|
|
onChange={onChangeComposeSearchTerm}
|
|
|
|
placeholder={i18n('contactSearchPlaceholder')}
|
|
|
|
ref={focusRef}
|
|
|
|
value={this.searchTerm}
|
|
|
|
/>
|
2021-03-03 20:09:58 +00:00
|
|
|
|
|
|
|
{Boolean(this.selectedContacts.length) && (
|
|
|
|
<ContactPills>
|
|
|
|
{this.selectedContacts.map(contact => (
|
|
|
|
<ContactPill
|
|
|
|
key={contact.id}
|
2021-05-07 22:21:10 +00:00
|
|
|
acceptedMessageRequest={contact.acceptedMessageRequest}
|
2021-03-03 20:09:58 +00:00
|
|
|
avatarPath={contact.avatarPath}
|
|
|
|
color={contact.color}
|
|
|
|
firstName={contact.firstName}
|
|
|
|
i18n={i18n}
|
|
|
|
id={contact.id}
|
2021-05-07 22:21:10 +00:00
|
|
|
isMe={contact.isMe}
|
2021-03-03 20:09:58 +00:00
|
|
|
name={contact.name}
|
|
|
|
phoneNumber={contact.phoneNumber}
|
|
|
|
profileName={contact.profileName}
|
2021-05-07 22:21:10 +00:00
|
|
|
sharedGroupNames={contact.sharedGroupNames}
|
2021-03-03 20:09:58 +00:00
|
|
|
title={contact.title}
|
|
|
|
onClickRemove={removeSelectedContact}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</ContactPills>
|
|
|
|
)}
|
|
|
|
|
|
|
|
{this.getRowCount() ? null : (
|
|
|
|
<div className="module-left-pane__compose-no-contacts">
|
2021-03-11 21:29:31 +00:00
|
|
|
{i18n('noContactsFound')}
|
2021-03-03 20:09:58 +00:00
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
|
2021-03-11 21:29:31 +00:00
|
|
|
{modalNode}
|
2021-03-03 20:09:58 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
getFooterContents({
|
|
|
|
i18n,
|
|
|
|
startSettingGroupMetadata,
|
|
|
|
}: Readonly<{
|
|
|
|
i18n: LocalizerType;
|
|
|
|
startSettingGroupMetadata: () => void;
|
|
|
|
}>): ReactChild {
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
disabled={this.hasExceededMaximumNumberOfContacts()}
|
|
|
|
onClick={startSettingGroupMetadata}
|
|
|
|
>
|
|
|
|
{this.selectedContacts.length
|
|
|
|
? i18n('chooseGroupMembers__next')
|
|
|
|
: i18n('chooseGroupMembers__skip')}
|
|
|
|
</Button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
getRowCount(): number {
|
|
|
|
if (!this.candidateContacts.length) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return this.candidateContacts.length + 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
getRow(rowIndex: number): undefined | Row {
|
|
|
|
if (!this.candidateContacts.length) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rowIndex === 0) {
|
|
|
|
return {
|
|
|
|
type: RowType.Header,
|
|
|
|
i18nKey: 'contactsHeader',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// This puts a blank row for the footer.
|
|
|
|
if (rowIndex === this.candidateContacts.length + 1) {
|
|
|
|
return { type: RowType.Blank };
|
|
|
|
}
|
|
|
|
|
|
|
|
const contact = this.candidateContacts[rowIndex - 1];
|
|
|
|
if (!contact) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const isChecked = this.selectedConversationIdsSet.has(contact.id);
|
|
|
|
|
|
|
|
let disabledReason: undefined | ContactCheckboxDisabledReason;
|
|
|
|
if (!isChecked) {
|
|
|
|
if (this.hasSelectedMaximumNumberOfContacts()) {
|
|
|
|
disabledReason = ContactCheckboxDisabledReason.MaximumContactsSelected;
|
|
|
|
} else if (!contact.isGroupV2Capable) {
|
|
|
|
disabledReason = ContactCheckboxDisabledReason.NotCapable;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: RowType.ContactCheckbox,
|
|
|
|
contact,
|
|
|
|
isChecked,
|
|
|
|
disabledReason,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is deliberately unimplemented because these keyboard shortcuts shouldn't work in
|
|
|
|
// the composer. The same is true for the "in direction" function below.
|
|
|
|
getConversationAndMessageAtIndex(
|
|
|
|
..._args: ReadonlyArray<unknown>
|
|
|
|
): undefined {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
getConversationAndMessageInDirection(
|
|
|
|
..._args: ReadonlyArray<unknown>
|
|
|
|
): undefined {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
shouldRecomputeRowHeights(_old: unknown): boolean {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
private hasSelectedMaximumNumberOfContacts(): boolean {
|
|
|
|
return this.selectedContacts.length >= this.getMaximumNumberOfContacts();
|
|
|
|
}
|
|
|
|
|
|
|
|
private hasExceededMaximumNumberOfContacts(): boolean {
|
|
|
|
// It should be impossible to reach this state. This is here as a failsafe.
|
|
|
|
return this.selectedContacts.length > this.getMaximumNumberOfContacts();
|
|
|
|
}
|
|
|
|
|
|
|
|
private getRecommendedMaximumNumberOfContacts(): number {
|
|
|
|
return getGroupSizeRecommendedLimit(151) - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
private getMaximumNumberOfContacts(): number {
|
|
|
|
return getGroupSizeHardLimit(1001) - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function focusRef(el: HTMLElement | null) {
|
|
|
|
if (el) {
|
|
|
|
el.focus();
|
|
|
|
}
|
|
|
|
}
|