// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { assert } from 'chai'; import * as sinon from 'sinon'; import { v4 as uuid } from 'uuid'; import { RowType, _testHeaderText } from '../../../components/ConversationList'; import { getDefaultConversation } from '../../../test-both/helpers/getDefaultConversation'; import { LeftPaneSearchHelper } from '../../../components/leftPane/LeftPaneSearchHelper'; const baseSearchHelperArgs = { conversationResults: { isLoading: false, results: [] }, contactResults: { isLoading: false, results: [] }, filterByUnread: false, messageResults: { isLoading: false, results: [] }, isSearchingGlobally: true, searchTerm: 'foo', primarySendsSms: false, searchConversation: undefined, searchDisabled: false, startSearchCounter: 0, }; describe('LeftPaneSearchHelper', () => { const fakeMessage = () => ({ id: uuid(), type: 'outgoing', conversationId: uuid(), }); describe('getBackAction', () => { it('returns undefined; going back is handled elsewhere in the app', () => { const helper = new LeftPaneSearchHelper(baseSearchHelperArgs); assert.isUndefined( helper.getBackAction({ showChooseGroupMembers: sinon.fake(), showInbox: sinon.fake(), startComposing: sinon.fake(), }) ); }); }); describe('getRowCount', () => { it('returns 100 if any results are loading', () => { assert.strictEqual( new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }).getRowCount(), 100 ); assert.strictEqual( new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }).getRowCount(), 100 ); assert.strictEqual( new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: false, results: [fakeMessage()] }, }).getRowCount(), 100 ); }); it('returns 0 when there are no search results', () => { const helper = new LeftPaneSearchHelper({ conversationResults: { isLoading: false, results: [] }, contactResults: { isLoading: false, results: [] }, messageResults: { isLoading: false, results: [] }, isSearchingGlobally: true, searchTerm: 'foo', primarySendsSms: false, searchConversation: undefined, searchDisabled: false, startSearchCounter: 0, filterByUnread: false, }); assert.strictEqual(helper.getRowCount(), 0); }); it('returns 1 + the number of results, dropping empty sections', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: false, results: [] }, messageResults: { isLoading: false, results: [fakeMessage()] }, }); assert.strictEqual(helper.getRowCount(), 5); }); }); describe('getRow', () => { it('returns a "loading search results" row if any results are loading', () => { const helpers = [ new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }), new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }), new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: false, results: [fakeMessage()] }, }), ]; helpers.forEach(helper => { assert.deepEqual(helper.getRow(0), { type: RowType.SearchResultsLoadingFakeHeader, }); for (let i = 1; i < 99; i += 1) { assert.deepEqual(helper.getRow(i), { type: RowType.SearchResultsLoadingFakeRow, }); } assert.isUndefined(helper.getRow(100)); }); }); it('returns header + results when all sections have loaded with results', () => { const conversations = [ getDefaultConversation(), getDefaultConversation(), ]; const contacts = [getDefaultConversation()]; const messages = [fakeMessage(), fakeMessage()]; const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: conversations, }, contactResults: { isLoading: false, results: contacts }, messageResults: { isLoading: false, results: messages }, }); assert.deepEqual( _testHeaderText(helper.getRow(0)), 'icu:conversationsHeader' ); assert.deepEqual(helper.getRow(1), { type: RowType.Conversation, conversation: conversations[0], }); assert.deepEqual(helper.getRow(2), { type: RowType.Conversation, conversation: conversations[1], }); assert.deepEqual(_testHeaderText(helper.getRow(3)), 'icu:contactsHeader'); assert.deepEqual(helper.getRow(4), { type: RowType.Conversation, conversation: contacts[0], }); assert.deepEqual(_testHeaderText(helper.getRow(5)), 'icu:messagesHeader'); assert.deepEqual(helper.getRow(6), { type: RowType.MessageSearchResult, messageId: messages[0].id, }); assert.deepEqual(helper.getRow(7), { type: RowType.MessageSearchResult, messageId: messages[1].id, }); }); it('omits conversations when there are no conversation results', () => { const contacts = [getDefaultConversation()]; const messages = [fakeMessage(), fakeMessage()]; const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, contactResults: { isLoading: false, results: contacts }, messageResults: { isLoading: false, results: messages }, }); assert.deepEqual(_testHeaderText(helper.getRow(0)), 'icu:contactsHeader'); assert.deepEqual(helper.getRow(1), { type: RowType.Conversation, conversation: contacts[0], }); assert.deepEqual(_testHeaderText(helper.getRow(2)), 'icu:messagesHeader'); assert.deepEqual(helper.getRow(3), { type: RowType.MessageSearchResult, messageId: messages[0].id, }); assert.deepEqual(helper.getRow(4), { type: RowType.MessageSearchResult, messageId: messages[1].id, }); }); it('omits contacts when there are no contact results', () => { const conversations = [ getDefaultConversation(), getDefaultConversation(), ]; const messages = [fakeMessage(), fakeMessage()]; const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: conversations, }, messageResults: { isLoading: false, results: messages }, }); assert.deepEqual( _testHeaderText(helper.getRow(0)), 'icu:conversationsHeader' ); assert.deepEqual(helper.getRow(1), { type: RowType.Conversation, conversation: conversations[0], }); assert.deepEqual(helper.getRow(2), { type: RowType.Conversation, conversation: conversations[1], }); assert.deepEqual(_testHeaderText(helper.getRow(3)), 'icu:messagesHeader'); assert.deepEqual(helper.getRow(4), { type: RowType.MessageSearchResult, messageId: messages[0].id, }); assert.deepEqual(helper.getRow(5), { type: RowType.MessageSearchResult, messageId: messages[1].id, }); }); }); it('omits messages when there are no message results', () => { const conversations = [getDefaultConversation(), getDefaultConversation()]; const contacts = [getDefaultConversation()]; const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: conversations, }, contactResults: { isLoading: false, results: contacts }, }); assert.deepEqual( _testHeaderText(helper.getRow(0)), 'icu:conversationsHeader' ); assert.deepEqual(helper.getRow(1), { type: RowType.Conversation, conversation: conversations[0], }); assert.deepEqual(helper.getRow(2), { type: RowType.Conversation, conversation: conversations[1], }); assert.deepEqual(_testHeaderText(helper.getRow(3)), 'icu:contactsHeader'); assert.deepEqual(helper.getRow(4), { type: RowType.Conversation, conversation: contacts[0], }); assert.isUndefined(helper.getRow(5)); }); describe('isScrollable', () => { it('returns false if any results are loading', () => { const helpers = [ new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }), new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }), new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: false, results: [fakeMessage()] }, }), ]; helpers.forEach(helper => { assert.isFalse(helper.isScrollable()); }); }); it('returns true if all results have loaded', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }); assert.isTrue(helper.isScrollable()); }); }); describe('shouldRecomputeRowHeights', () => { it("returns false if the number of results doesn't change", () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }); assert.isFalse( helper.shouldRecomputeRowHeights({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: false, results: [] }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }) ); }); it('returns false when a section completes loading, but not all sections are done (because the pane is still loading overall)', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }); assert.isFalse( helper.shouldRecomputeRowHeights({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation()], }, contactResults: { isLoading: true }, messageResults: { isLoading: true }, }) ); }); it('returns true when all sections finish loading', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: false, results: [fakeMessage()] }, }); assert.isTrue( helper.shouldRecomputeRowHeights({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: false, results: [] }, messageResults: { isLoading: false, results: [fakeMessage()] }, }) ); }); it('returns true if the number of results in a section changes', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, }); assert.isTrue( helper.shouldRecomputeRowHeights({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation()], }, }) ); }); }); describe('getConversationAndMessageAtIndex', () => { it('returns correct conversation at given index', () => { const expected = getDefaultConversation(); const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [expected, getDefaultConversation()], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }); assert.strictEqual( helper.getConversationAndMessageAtIndex(0)?.conversationId, expected.id ); }); it('returns correct contact at given index', () => { const expected = getDefaultConversation(); const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, contactResults: { isLoading: false, results: [expected], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }); assert.strictEqual( helper.getConversationAndMessageAtIndex(2)?.conversationId, expected.id ); }); it('returns correct message at given index', () => { const expected = fakeMessage(); const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), expected], }, }); assert.strictEqual( helper.getConversationAndMessageAtIndex(4)?.messageId, expected.id ); }); it('returns correct message at given index skipping not loaded results', () => { const expected = fakeMessage(); const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: true }, contactResults: { isLoading: true }, messageResults: { isLoading: false, results: [fakeMessage(), expected, fakeMessage()], }, }); assert.strictEqual( helper.getConversationAndMessageAtIndex(1)?.messageId, expected.id ); }); it('returns undefined if search candidate with given index does not exist', () => { const helper = new LeftPaneSearchHelper({ ...baseSearchHelperArgs, conversationResults: { isLoading: false, results: [getDefaultConversation(), getDefaultConversation()], }, messageResults: { isLoading: false, results: [fakeMessage(), fakeMessage(), fakeMessage()], }, }); assert.isUndefined( helper.getConversationAndMessageAtIndex(100)?.messageId ); assert.isUndefined( helper.getConversationAndMessageAtIndex(-100)?.messageId ); }); }); });