// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactChild, ChangeEvent } from 'react'; import React from 'react'; import type { PhoneNumber } from 'google-libphonenumber'; import { LeftPaneHelper } from './LeftPaneHelper'; import type { Row } from '../ConversationList'; import { RowType } from '../ConversationList'; import type { PropsDataType as ContactListItemPropsType } from '../conversationList/ContactListItem'; import type { PropsData as ConversationListItemPropsType } from '../conversationList/ConversationListItem'; import { SearchInput } from '../SearchInput'; import type { LocalizerType } from '../../types/Util'; import { instance as phoneNumberInstance, PhoneNumberFormat, } from '../../util/libphonenumberInstance'; import { assert } from '../../util/assert'; import { missingCaseError } from '../../util/missingCaseError'; export type LeftPaneComposePropsType = { composeContacts: ReadonlyArray; composeGroups: ReadonlyArray; regionCode: string; searchTerm: string; }; enum TopButton { None, CreateNewGroup, StartNewConversation, } export class LeftPaneComposeHelper extends LeftPaneHelper { private readonly composeContacts: ReadonlyArray; private readonly composeGroups: ReadonlyArray; private readonly searchTerm: string; private readonly phoneNumber: undefined | PhoneNumber; constructor({ composeContacts, composeGroups, regionCode, searchTerm, }: Readonly) { super(); this.searchTerm = searchTerm; this.phoneNumber = parsePhoneNumber(searchTerm, regionCode); this.composeGroups = composeGroups; this.composeContacts = composeContacts; } getHeaderContents({ i18n, showInbox, }: Readonly<{ i18n: LocalizerType; showInbox: () => void; }>): ReactChild { return (
); } getBackAction({ showInbox }: { showInbox: () => void }): () => void { return showInbox; } getPreRowsNode({ i18n, onChangeComposeSearchTerm, }: Readonly<{ i18n: LocalizerType; onChangeComposeSearchTerm: ( event: ChangeEvent ) => unknown; }>): ReactChild { return ( <> {this.getRowCount() ? null : (
{i18n('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; } 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.StartNewConversation: assert( this.phoneNumber, 'LeftPaneComposeHelper: we should have a phone number if the top button is "Start new conversation"' ); return { type: RowType.StartNewConversation, phoneNumber: phoneNumberInstance.format( this.phoneNumber, PhoneNumberFormat.E164 ), }; case TopButton.CreateNewGroup: return { type: RowType.CreateNewGroup }; default: throw missingCaseError(topButton); } } virtualRowIndex -= 1; } if (this.hasContactsHeader()) { if (virtualRowIndex === 0) { return { type: RowType.Header, i18nKey: 'contactsHeader', }; } virtualRowIndex -= 1; const contact = this.composeContacts[virtualRowIndex]; if (contact) { return { type: RowType.Contact, contact, }; } virtualRowIndex -= this.composeContacts.length; } if (this.hasGroupsHeader()) { if (virtualRowIndex === 0) { return { type: RowType.Header, i18nKey: 'groupsHeader', }; } virtualRowIndex -= 1; const group = this.composeGroups[virtualRowIndex]; return { type: RowType.Conversation, conversation: group, }; } 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 ); } private getTopButton(): TopButton { if (this.phoneNumber) { return TopButton.StartNewConversation; } 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; } { let top: number | undefined; let contact: number | undefined; let group: number | undefined; let rowCount = 0; if (this.hasTopButton()) { top = 0; rowCount += 1; } if (this.composeContacts.length) { contact = rowCount; rowCount += this.composeContacts.length; } if (this.composeGroups.length) { group = rowCount; } return { top, contact, group, }; } } function focusRef(el: HTMLElement | null) { if (el) { el.focus(); } } function parsePhoneNumber( str: string, regionCode: string ): undefined | PhoneNumber { let result: PhoneNumber; try { result = phoneNumberInstance.parse(str, regionCode); } catch (err) { return undefined; } if (!phoneNumberInstance.isValidNumber(result)) { return undefined; } return result; }