// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactChild } from 'react'; import React from 'react'; import type { ToFindType } from './LeftPaneHelper'; import { LeftPaneHelper } from './LeftPaneHelper'; import type { LocalizerType } from '../../types/Util'; import type { Row } from '../ConversationList'; import { RowType } from '../ConversationList'; import type { PropsData as ConversationListItemPropsType } from '../conversationList/ConversationListItem'; import { handleKeydownForSearch } from './handleKeydownForSearch'; import { Intl } from '../Intl'; import { Emojify } from '../conversation/Emojify'; import { assert } from '../../util/assert'; // The "correct" thing to do is to measure the size of the left pane and render enough // search results for the container height. But (1) that's slow (2) the list is // virtualized (3) 99 rows is over 7500px tall, taller than most monitors (4) it's fine // if, in some extremely tall window, we have some empty space. So we just hard-code a // fairly big number. const SEARCH_RESULTS_FAKE_ROW_COUNT = 99; type MaybeLoadedSearchResultsType = | { isLoading: true } | { isLoading: false; results: Array }; export type LeftPaneSearchPropsType = { conversationResults: MaybeLoadedSearchResultsType; contactResults: MaybeLoadedSearchResultsType; messageResults: MaybeLoadedSearchResultsType<{ id: string; conversationId: string; }>; searchConversationName?: string; primarySendsSms: boolean; searchTerm: string; }; const searchResultKeys: Array< 'conversationResults' | 'contactResults' | 'messageResults' > = ['conversationResults', 'contactResults', 'messageResults']; export class LeftPaneSearchHelper extends LeftPaneHelper { private readonly conversationResults: MaybeLoadedSearchResultsType; private readonly contactResults: MaybeLoadedSearchResultsType; private readonly messageResults: MaybeLoadedSearchResultsType<{ id: string; conversationId: string; }>; private readonly searchConversationName?: string; private readonly primarySendsSms: boolean; private readonly searchTerm: string; constructor({ conversationResults, contactResults, messageResults, searchConversationName, primarySendsSms, searchTerm, }: Readonly) { super(); this.conversationResults = conversationResults; this.contactResults = contactResults; this.messageResults = messageResults; this.searchConversationName = searchConversationName; this.primarySendsSms = primarySendsSms; this.searchTerm = searchTerm; } override getPreRowsNode({ i18n, }: Readonly<{ i18n: LocalizerType }>): null | ReactChild { const mightHaveSearchResults = this.allResults().some( searchResult => searchResult.isLoading || searchResult.results.length ); if (mightHaveSearchResults) { return null; } const { searchConversationName, primarySendsSms, searchTerm } = this; let noResults: ReactChild; if (searchConversationName) { noResults = ( ), }} /> ); } else { noResults = ( <>
{i18n('noSearchResults', [searchTerm])}
{primarySendsSms && (
{i18n('noSearchResults--sms-only')}
)} ); } return !searchConversationName || searchTerm ? (
{noResults}
) : null; } getRowCount(): number { if (this.isLoading()) { // 1 for the header. return 1 + SEARCH_RESULTS_FAKE_ROW_COUNT; } return this.allResults().reduce( (result: number, searchResults) => result + getRowCountForLoadedSearchResults(searchResults), 0 ); } // This is currently unimplemented. See DESKTOP-1170. override getRowIndexToScrollTo( _selectedConversationId: undefined | string ): undefined | number { return undefined; } getRow(rowIndex: number): undefined | Row { const { conversationResults, contactResults, messageResults } = this; if (this.isLoading()) { if (rowIndex === 0) { return { type: RowType.SearchResultsLoadingFakeHeader }; } if (rowIndex + 1 <= SEARCH_RESULTS_FAKE_ROW_COUNT) { return { type: RowType.SearchResultsLoadingFakeRow }; } return undefined; } const conversationRowCount = getRowCountForLoadedSearchResults(conversationResults); const contactRowCount = getRowCountForLoadedSearchResults(contactResults); const messageRowCount = getRowCountForLoadedSearchResults(messageResults); if (rowIndex < conversationRowCount) { if (rowIndex === 0) { return { type: RowType.Header, i18nKey: 'conversationsHeader', }; } assert( !conversationResults.isLoading, "We shouldn't get here with conversation results still loading" ); const conversation = conversationResults.results[rowIndex - 1]; return conversation ? { type: RowType.Conversation, conversation, } : undefined; } if (rowIndex < conversationRowCount + contactRowCount) { const localIndex = rowIndex - conversationRowCount; if (localIndex === 0) { return { type: RowType.Header, i18nKey: 'contactsHeader', }; } assert( !contactResults.isLoading, "We shouldn't get here with contact results still loading" ); const conversation = contactResults.results[localIndex - 1]; return conversation ? { type: RowType.Conversation, conversation, } : undefined; } if (rowIndex >= conversationRowCount + contactRowCount + messageRowCount) { return undefined; } const localIndex = rowIndex - conversationRowCount - contactRowCount; if (localIndex === 0) { return { type: RowType.Header, i18nKey: 'messagesHeader', }; } assert( !messageResults.isLoading, "We shouldn't get here with message results still loading" ); const message = messageResults.results[localIndex - 1]; return message ? { type: RowType.MessageSearchResult, messageId: message.id, } : undefined; } override isScrollable(): boolean { return !this.isLoading(); } shouldRecomputeRowHeights(old: Readonly): boolean { const oldIsLoading = new LeftPaneSearchHelper(old).isLoading(); const newIsLoading = this.isLoading(); if (oldIsLoading && newIsLoading) { return false; } if (oldIsLoading !== newIsLoading) { return true; } return searchResultKeys.some( key => getRowCountForLoadedSearchResults(old[key]) !== getRowCountForLoadedSearchResults(this[key]) ); } // This is currently unimplemented. See DESKTOP-1170. getConversationAndMessageAtIndex( _conversationIndex: number ): undefined | { conversationId: string; messageId?: string } { return undefined; } // This is currently unimplemented. See DESKTOP-1170. getConversationAndMessageInDirection( _toFind: Readonly, _selectedConversationId: undefined | string, _selectedMessageId: unknown ): undefined | { conversationId: string } { return undefined; } override onKeyDown( event: KeyboardEvent, options: Readonly<{ searchInConversation: (conversationId: string) => unknown; selectedConversationId: undefined | string; startSearch: () => unknown; }> ): void { handleKeydownForSearch(event, options); } private allResults() { return [this.conversationResults, this.contactResults, this.messageResults]; } private isLoading(): boolean { return this.allResults().some(results => results.isLoading); } } function getRowCountForLoadedSearchResults( searchResults: Readonly> ): number { // It's possible to call this helper with invalid results (e.g., ones that are loading). // We could change the parameter of this function, but that adds a bunch of redundant // checks that are, in the author's opinion, less clear. if (searchResults.isLoading) { assert( false, 'getRowCountForLoadedSearchResults: Expected this to be called with loaded search results. Returning 0' ); return 0; } const resultRows = searchResults.results.length; const hasHeader = Boolean(resultRows); return (hasHeader ? 1 : 0) + resultRows; }