// Copyright 2021-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactChild, ChangeEvent } from 'react'; import React from 'react'; import { LeftPaneHelper } from './LeftPaneHelper'; import type { Row } from '../ConversationList'; import { RowType } from '../ConversationList'; import type { ConversationType } from '../../state/ducks/conversations'; import { ContactCheckboxDisabledReason } from '../conversationList/ContactCheckbox'; import { ContactPills } from '../ContactPills'; import { ContactPill } from '../ContactPill'; import { SearchInput } from '../SearchInput'; import { AddGroupMemberErrorDialog, AddGroupMemberErrorDialogMode, } from '../AddGroupMemberErrorDialog'; import { Button } from '../Button'; import type { LocalizerType } from '../../types/Util'; import { getGroupSizeRecommendedLimit, getGroupSizeHardLimit, } from '../../groups/limits'; export type LeftPaneChooseGroupMembersPropsType = { candidateContacts: ReadonlyArray; isShowingRecommendedGroupSizeModal: boolean; isShowingMaximumGroupSizeModal: boolean; searchTerm: string; selectedContacts: Array; }; export class LeftPaneChooseGroupMembersHelper extends LeftPaneHelper { private readonly candidateContacts: ReadonlyArray; private readonly isShowingMaximumGroupSizeModal: boolean; private readonly isShowingRecommendedGroupSizeModal: boolean; private readonly searchTerm: string; private readonly selectedContacts: Array; private readonly selectedConversationIdsSet: Set; constructor({ candidateContacts, isShowingMaximumGroupSizeModal, isShowingRecommendedGroupSizeModal, searchTerm, selectedContacts, }: Readonly) { super(); this.candidateContacts = candidateContacts; this.isShowingMaximumGroupSizeModal = isShowingMaximumGroupSizeModal; this.isShowingRecommendedGroupSizeModal = isShowingRecommendedGroupSizeModal; this.searchTerm = searchTerm; this.selectedContacts = selectedContacts; this.selectedConversationIdsSet = new Set( selectedContacts.map(contact => contact.id) ); } override getHeaderContents({ i18n, startComposing, }: Readonly<{ i18n: LocalizerType; startComposing: () => void; }>): ReactChild { const backButtonLabel = i18n('chooseGroupMembers__back-button'); return (
); } override getBackAction({ startComposing, }: { startComposing: () => void; }): () => void { return startComposing; } override getSearchInput({ i18n, onChangeComposeSearchTerm, }: Readonly<{ i18n: LocalizerType; onChangeComposeSearchTerm: ( event: ChangeEvent ) => unknown; }>): ReactChild { return ( ); } override getPreRowsNode({ closeMaximumGroupSizeModal, closeRecommendedGroupSizeModal, i18n, removeSelectedContact, }: Readonly<{ closeMaximumGroupSizeModal: () => unknown; closeRecommendedGroupSizeModal: () => unknown; i18n: LocalizerType; removeSelectedContact: (conversationId: string) => unknown; }>): ReactChild { let modalNode: undefined | ReactChild; if (this.isShowingMaximumGroupSizeModal) { modalNode = ( ); } else if (this.isShowingRecommendedGroupSizeModal) { modalNode = ( ); } return ( <> {Boolean(this.selectedContacts.length) && ( {this.selectedContacts.map(contact => ( ))} )} {this.getRowCount() ? null : (
{i18n('noContactsFound')}
)} {modalNode} ); } override getFooterContents({ i18n, startSettingGroupMetadata, }: Readonly<{ i18n: LocalizerType; startSettingGroupMetadata: () => void; }>): ReactChild { return ( ); } 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); const disabledReason = !isChecked && this.hasSelectedMaximumNumberOfContacts() ? ContactCheckboxDisabledReason.MaximumContactsSelected : undefined; 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 ): undefined { return undefined; } getConversationAndMessageInDirection( ..._args: ReadonlyArray ): 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(); } }