// Copyright 2021 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 { ContactListItemConversationType } from '../conversationList/ContactListItem'; import { SearchInput } from '../SearchInput'; import type { LocalizerType } from '../../types/Util'; import type { ParsedE164Type } from '../../util/libphonenumberInstance'; import { parseAndFormatPhoneNumber } from '../../util/libphonenumberInstance'; import { missingCaseError } from '../../util/missingCaseError'; import { getUsernameFromSearch } from '../../types/Username'; import type { UUIDFetchStateType } from '../../util/uuidFetchState'; import { isFetchingByUsername, isFetchingByE164, } from '../../util/uuidFetchState'; import type { GroupListItemConversationType } from '../conversationList/GroupListItem'; export type LeftPaneComposePropsType = { composeContacts: ReadonlyArray; composeGroups: ReadonlyArray; regionCode: string | undefined; searchTerm: string; uuidFetchState: UUIDFetchStateType; isUsernamesEnabled: boolean; }; enum TopButton { None, CreateNewGroup, } export class LeftPaneComposeHelper extends LeftPaneHelper { private readonly composeContacts: ReadonlyArray; private readonly composeGroups: ReadonlyArray; private readonly uuidFetchState: UUIDFetchStateType; private readonly searchTerm: string; private readonly phoneNumber: ParsedE164Type | undefined; private readonly isPhoneNumberVisible: boolean; private readonly username: string | undefined; private readonly isUsernameVisible: boolean; constructor({ composeContacts, composeGroups, regionCode, searchTerm, isUsernamesEnabled, uuidFetchState, }: Readonly) { super(); this.composeContacts = composeContacts; this.composeGroups = composeGroups; this.searchTerm = searchTerm; this.uuidFetchState = uuidFetchState; const username = getUsernameFromSearch(this.searchTerm); if (isUsernamesEnabled) { this.username = username; this.isUsernameVisible = isUsernamesEnabled && Boolean(username) && this.composeContacts.every(contact => contact.username !== username); } else { this.isUsernameVisible = false; } const phoneNumber = parseAndFormatPhoneNumber(searchTerm, regionCode); if (!username && phoneNumber) { this.phoneNumber = phoneNumber; this.isPhoneNumberVisible = this.composeContacts.every( contact => contact.e164 !== phoneNumber.e164 ); } else { this.isPhoneNumberVisible = false; } } override getHeaderContents({ i18n, showInbox, }: Readonly<{ i18n: LocalizerType; showInbox: () => void; }>): ReactChild { return (
); } override getBackAction({ showInbox }: { showInbox: () => void }): () => void { return showInbox; } override getSearchInput({ i18n, onChangeComposeSearchTerm, }: Readonly<{ i18n: LocalizerType; onChangeComposeSearchTerm: ( event: ChangeEvent ) => unknown; }>): ReactChild { return ( ); } override getPreRowsNode({ i18n, }: Readonly<{ i18n: LocalizerType; }>): ReactChild | null { return this.getRowCount() ? null : (
{i18n('icu:noConversationsFound')}
); } getRowCount(): number { let result = this.composeContacts.length + this.composeGroups.length; if (this.hasTopButton()) { result += 1; } if (this.hasContactsHeader()) { result += 1; } if (this.hasGroupsHeader()) { result += 1; } if (this.isUsernameVisible) { result += 2; } if (this.isPhoneNumberVisible) { result += 2; } return result; } getRow(actualRowIndex: number): undefined | Row { let virtualRowIndex = actualRowIndex; if (this.hasTopButton()) { if (virtualRowIndex === 0) { const topButton = this.getTopButton(); switch (topButton) { case TopButton.None: break; case TopButton.CreateNewGroup: return { type: RowType.CreateNewGroup }; default: throw missingCaseError(topButton); } } virtualRowIndex -= 1; } if (this.hasContactsHeader()) { if (virtualRowIndex === 0) { return { type: RowType.Header, getHeaderText: i18n => i18n('icu:contactsHeader'), }; } virtualRowIndex -= 1; const contact = this.composeContacts[virtualRowIndex]; if (contact) { return { type: RowType.Contact, contact, hasContextMenu: true, }; } virtualRowIndex -= this.composeContacts.length; } if (this.hasGroupsHeader()) { if (virtualRowIndex === 0) { return { type: RowType.Header, getHeaderText: i18n => i18n('icu:groupsHeader'), }; } virtualRowIndex -= 1; const group = this.composeGroups[virtualRowIndex]; if (group) { return { type: RowType.SelectSingleGroup, group, }; } virtualRowIndex -= this.composeGroups.length; } if (this.username && this.isUsernameVisible) { if (virtualRowIndex === 0) { return { type: RowType.Header, getHeaderText: i18n => i18n('icu:findByUsernameHeader'), }; } virtualRowIndex -= 1; if (virtualRowIndex === 0) { return { type: RowType.UsernameSearchResult, username: this.username, isFetchingUsername: isFetchingByUsername( this.uuidFetchState, this.username ), }; } } if (this.phoneNumber && this.isPhoneNumberVisible) { if (virtualRowIndex === 0) { return { type: RowType.Header, getHeaderText: i18n => i18n('icu:findByPhoneNumberHeader'), }; } virtualRowIndex -= 1; if (virtualRowIndex === 0) { return { type: RowType.StartNewConversation, phoneNumber: this.phoneNumber, isFetching: isFetchingByE164( this.uuidFetchState, this.phoneNumber.e164 ), }; } } return undefined; } // 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( exProps: Readonly ): boolean { const prev = new LeftPaneComposeHelper(exProps); const currHeaderIndices = this.getHeaderIndices(); const prevHeaderIndices = prev.getHeaderIndices(); return ( currHeaderIndices.top !== prevHeaderIndices.top || currHeaderIndices.contact !== prevHeaderIndices.contact || currHeaderIndices.group !== prevHeaderIndices.group || currHeaderIndices.username !== prevHeaderIndices.username || currHeaderIndices.phoneNumber !== prevHeaderIndices.phoneNumber ); } private getTopButton(): TopButton { if (this.searchTerm) { return TopButton.None; } return TopButton.CreateNewGroup; } private hasTopButton(): boolean { return this.getTopButton() !== TopButton.None; } private hasContactsHeader(): boolean { return Boolean(this.composeContacts.length); } private hasGroupsHeader(): boolean { return Boolean(this.composeGroups.length); } private getHeaderIndices(): { top?: number; contact?: number; group?: number; phoneNumber?: number; username?: number; } { let top: number | undefined; let contact: number | undefined; let group: number | undefined; let phoneNumber: number | undefined; let username: number | undefined; let rowCount = 0; if (this.hasTopButton()) { top = 0; rowCount += 1; } if (this.hasContactsHeader()) { contact = rowCount; rowCount += this.composeContacts.length; } if (this.hasGroupsHeader()) { group = rowCount; rowCount += this.composeContacts.length; } if (this.phoneNumber) { phoneNumber = rowCount; } if (this.username) { username = rowCount; } return { top, contact, group, phoneNumber, username, }; } } function focusRef(el: HTMLElement | null) { if (el) { el.focus(); } }