2021-02-23 20:34:28 +00:00
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { assert } from 'chai' ;
2021-04-02 21:43:39 +00:00
import * as sinon from 'sinon' ;
2021-02-23 20:34:28 +00:00
import { v4 as uuid } from 'uuid' ;
import { RowType } from '../../../components/ConversationList' ;
import { FindDirection } from '../../../components/leftPane/LeftPaneHelper' ;
import { LeftPaneInboxHelper } from '../../../components/leftPane/LeftPaneInboxHelper' ;
describe ( 'LeftPaneInboxHelper' , ( ) = > {
const fakeConversation = ( ) = > ( {
id : uuid ( ) ,
title : uuid ( ) ,
type : 'direct' as const ,
} ) ;
2021-04-02 21:43:39 +00:00
describe ( 'getBackAction' , ( ) = > {
it ( "returns undefined; you can't go back from the main inbox" , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . isUndefined (
helper . getBackAction ( {
showChooseGroupMembers : sinon.fake ( ) ,
showInbox : sinon.fake ( ) ,
startComposing : sinon.fake ( ) ,
} )
) ;
} ) ;
} ) ;
2021-02-23 20:34:28 +00:00
describe ( 'getRowCount' , ( ) = > {
it ( 'returns 0 if there are no conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 0 ) ;
} ) ;
it ( 'returns 1 if there are only archived conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 1 ) ;
} ) ;
it ( "returns the number of non-pinned conversations if that's all there is" , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 3 ) ;
} ) ;
it ( "returns the number of pinned conversations if that's all there is" , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 3 ) ;
} ) ;
it ( 'adds 2 rows for each header if there are pinned and non-pinned conversations,' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) ] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 6 ) ;
} ) ;
it ( 'adds 1 row for the archive button if there are any archived conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . strictEqual ( helper . getRowCount ( ) , 4 ) ;
} ) ;
} ) ;
describe ( 'getRowIndexToScrollTo' , ( ) = > {
it ( 'returns undefined if no conversation is selected' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
pinnedConversations : [ fakeConversation ( ) ] ,
archivedConversations : [ ] ,
} ) ;
assert . isUndefined ( helper . getRowIndexToScrollTo ( undefined ) ) ;
} ) ;
it ( 'returns undefined if the selected conversation is not pinned or non-pinned' , ( ) = > {
const archivedConversations = [ fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
pinnedConversations : [ fakeConversation ( ) ] ,
archivedConversations ,
} ) ;
assert . isUndefined (
helper . getRowIndexToScrollTo ( archivedConversations [ 0 ] . id )
) ;
} ) ;
it ( "returns the pinned conversation's index if there are only pinned conversations" , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getRowIndexToScrollTo ( pinnedConversations [ 0 ] . id ) ,
0
) ;
assert . strictEqual (
helper . getRowIndexToScrollTo ( pinnedConversations [ 1 ] . id ) ,
1
) ;
} ) ;
it ( "returns the conversation's index if there are only non-pinned conversations" , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowIndexToScrollTo ( conversations [ 0 ] . id ) , 0 ) ;
assert . strictEqual ( helper . getRowIndexToScrollTo ( conversations [ 1 ] . id ) , 1 ) ;
} ) ;
it ( "returns the pinned conversation's index + 1 (for the header) if there are both pinned and non-pinned conversations" , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getRowIndexToScrollTo ( pinnedConversations [ 0 ] . id ) ,
1
) ;
assert . strictEqual (
helper . getRowIndexToScrollTo ( pinnedConversations [ 1 ] . id ) ,
2
) ;
} ) ;
it ( "returns the non-pinned conversation's index + pinnedConversations.length + 2 (for the headers) if there are both pinned and non-pinned conversations" , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual ( helper . getRowIndexToScrollTo ( conversations [ 0 ] . id ) , 5 ) ;
assert . strictEqual ( helper . getRowIndexToScrollTo ( conversations [ 1 ] . id ) , 6 ) ;
} ) ;
} ) ;
describe ( 'getRow' , ( ) = > {
it ( 'returns the archive button if there are only archived conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . ArchiveButton ,
archivedConversationsCount : 2 ,
} ) ;
assert . isUndefined ( helper . getRow ( 1 ) ) ;
} ) ;
it ( "returns pinned conversations if that's all there are" , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 1 ] ,
} ) ;
assert . isUndefined ( helper . getRow ( 2 ) ) ;
} ) ;
it ( 'returns pinned conversations and an archive button if there are no non-pinned conversations' , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 2 ) , {
type : RowType . ArchiveButton ,
archivedConversationsCount : 1 ,
} ) ;
assert . isUndefined ( helper . getRow ( 3 ) ) ;
} ) ;
it ( "returns non-pinned conversations if that's all there are" , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Conversation ,
conversation : conversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : conversations [ 1 ] ,
} ) ;
assert . isUndefined ( helper . getRow ( 2 ) ) ;
} ) ;
it ( 'returns non-pinned conversations and an archive button if there are no pinned conversations' , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Conversation ,
conversation : conversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : conversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 2 ) , {
type : RowType . ArchiveButton ,
archivedConversationsCount : 1 ,
} ) ;
assert . isUndefined ( helper . getRow ( 3 ) ) ;
} ) ;
it ( 'returns headers if there are both pinned and non-pinned conversations' , ( ) = > {
const conversations = [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ;
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Header ,
i18nKey : 'LeftPane--pinned' ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 2 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 3 ) , {
type : RowType . Header ,
i18nKey : 'LeftPane--chats' ,
} ) ;
assert . deepEqual ( helper . getRow ( 4 ) , {
type : RowType . Conversation ,
conversation : conversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 5 ) , {
type : RowType . Conversation ,
conversation : conversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 6 ) , {
type : RowType . Conversation ,
conversation : conversations [ 2 ] ,
} ) ;
assert . isUndefined ( helper . getRow ( 7 ) ) ;
} ) ;
it ( 'returns headers if there are both pinned and non-pinned conversations, and an archive button' , ( ) = > {
const conversations = [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ;
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 0 ) , {
type : RowType . Header ,
i18nKey : 'LeftPane--pinned' ,
} ) ;
assert . deepEqual ( helper . getRow ( 1 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 2 ) , {
type : RowType . Conversation ,
conversation : pinnedConversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 3 ) , {
type : RowType . Header ,
i18nKey : 'LeftPane--chats' ,
} ) ;
assert . deepEqual ( helper . getRow ( 4 ) , {
type : RowType . Conversation ,
conversation : conversations [ 0 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 5 ) , {
type : RowType . Conversation ,
conversation : conversations [ 1 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 6 ) , {
type : RowType . Conversation ,
conversation : conversations [ 2 ] ,
} ) ;
assert . deepEqual ( helper . getRow ( 7 ) , {
type : RowType . ArchiveButton ,
archivedConversationsCount : 1 ,
} ) ;
assert . isUndefined ( helper . getRow ( 8 ) ) ;
} ) ;
} ) ;
describe ( 'getConversationAndMessageAtIndex' , ( ) = > {
it ( 'returns pinned converastions, then non-pinned conversations' , ( ) = > {
const conversations = [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ;
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 0 ) ? . conversationId ,
pinnedConversations [ 0 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 1 ) ? . conversationId ,
pinnedConversations [ 1 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 2 ) ? . conversationId ,
conversations [ 0 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 3 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 4 ) ? . conversationId ,
conversations [ 2 ] . id
) ;
} ) ;
it ( "when requesting an index out of bounds, returns the last pinned conversation when that's all there is" , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 2 ) ? . conversationId ,
pinnedConversations [ 1 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 99 ) ? . conversationId ,
pinnedConversations [ 1 ] . id
) ;
// This is mostly a resilience measure in case we're ever called with an invalid
// index.
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( - 1 ) ? . conversationId ,
pinnedConversations [ 1 ] . id
) ;
} ) ;
it ( "when requesting an index out of bounds, returns the last non-pinned conversation when that's all there is" , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations : [ ] ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 2 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 99 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
// This is mostly a resilience measure in case we're ever called with an invalid
// index.
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( - 1 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
} ) ;
it ( 'when requesting an index out of bounds, returns the last non-pinned conversation when there are both pinned and non-pinned conversations' , ( ) = > {
const conversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 4 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( 99 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
// This is mostly a resilience measure in case we're ever called with an invalid
// index.
assert . strictEqual (
helper . getConversationAndMessageAtIndex ( - 1 ) ? . conversationId ,
conversations [ 1 ] . id
) ;
} ) ;
it ( 'returns undefined if there are no conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ ] ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . isUndefined ( helper . getConversationAndMessageAtIndex ( 0 ) ) ;
assert . isUndefined ( helper . getConversationAndMessageAtIndex ( 1 ) ) ;
assert . isUndefined ( helper . getConversationAndMessageAtIndex ( - 1 ) ) ;
} ) ;
} ) ;
describe ( 'getConversationAndMessageInDirection' , ( ) = > {
it ( 'returns the next conversation when searching downward' , ( ) = > {
const pinnedConversations = [ fakeConversation ( ) , fakeConversation ( ) ] ;
const conversations = [ fakeConversation ( ) ] ;
const helper = new LeftPaneInboxHelper ( {
conversations ,
pinnedConversations ,
archivedConversations : [ ] ,
} ) ;
assert . deepEqual (
helper . getConversationAndMessageInDirection (
{ direction : FindDirection.Down , unreadOnly : false } ,
pinnedConversations [ 1 ] . id ,
undefined
) ,
{ conversationId : conversations [ 0 ] . id }
) ;
} ) ;
// Additional tests are found with `getConversationInDirection`.
} ) ;
describe ( 'shouldRecomputeRowHeights' , ( ) = > {
it ( "returns false if the number of conversations in each section doesn't change" , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . isFalse (
helper . shouldRecomputeRowHeights ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
} )
) ;
} ) ;
it ( 'returns false if the only thing changed is whether conversations are archived' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . isFalse (
helper . shouldRecomputeRowHeights ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ ] ,
} )
) ;
} ) ;
it ( 'returns false if the only thing changed is the number of non-pinned conversations' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . isFalse (
helper . shouldRecomputeRowHeights ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
} )
) ;
} ) ;
it ( 'returns true if the number of pinned conversations changes' , ( ) = > {
const helper = new LeftPaneInboxHelper ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations : [ fakeConversation ( ) , fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} ) ;
assert . isTrue (
helper . shouldRecomputeRowHeights ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations : [
fakeConversation ( ) ,
fakeConversation ( ) ,
fakeConversation ( ) ,
] ,
archivedConversations : [ fakeConversation ( ) ] ,
} )
) ;
assert . isTrue (
helper . shouldRecomputeRowHeights ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations : [ fakeConversation ( ) ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} )
) ;
assert . isTrue (
helper . shouldRecomputeRowHeights ( {
conversations : [ fakeConversation ( ) ] ,
pinnedConversations : [ ] ,
archivedConversations : [ fakeConversation ( ) ] ,
} )
) ;
} ) ;
} ) ;
} ) ;