signal-desktop/ts/test-both/state/selectors/conversations_test.ts

1874 lines
54 KiB
TypeScript
Raw Normal View History

// Copyright 2019-2021 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
2019-01-14 21:49:58 +00:00
import { assert } from 'chai';
import {
2021-03-03 20:09:58 +00:00
OneTimeModalState,
ComposerStep,
ConversationLookupType,
ConversationType,
getEmptyState,
} from '../../../state/ducks/conversations';
2019-01-14 21:49:58 +00:00
import {
_getConversationComparator,
2019-03-12 00:20:16 +00:00
_getLeftPaneLists,
2021-04-28 20:44:48 +00:00
getAllComposableConversations,
2021-03-11 21:29:31 +00:00
getCandidateContactsForNewGroup,
2021-03-03 20:09:58 +00:00
getCantAddContactForModal,
getComposableContacts,
getComposableGroups,
2021-03-03 20:09:58 +00:00
getComposeGroupAvatar,
getComposeGroupName,
getComposerConversationSearchTerm,
2021-03-03 20:09:58 +00:00
getComposerStep,
getComposeSelectedContacts,
2021-05-28 16:15:17 +00:00
getContactNameColorSelector,
getConversationByIdSelector,
getConversationsByTitleSelector,
getConversationSelector,
getFilteredCandidateContactsForNewGroup,
getFilteredComposeContacts,
getFilteredComposeGroups,
2021-03-03 20:09:58 +00:00
getInvitedContactsForNewlyCreatedGroup,
getMaximumGroupSizeModalState,
getPlaceholderContact,
2021-03-03 20:09:58 +00:00
getRecommendedGroupSizeModalState,
getSelectedConversation,
getSelectedConversationId,
2021-03-03 20:09:58 +00:00
hasGroupCreationError,
isCreatingGroup,
2019-01-14 21:49:58 +00:00
} from '../../../state/selectors/conversations';
import { noopAction } from '../../../state/ducks/noop';
import { StateType, reducer as rootReducer } from '../../../state/reducer';
import { setup as setupI18n } from '../../../../js/modules/i18n';
import enMessages from '../../../../_locales/en/messages.json';
2021-05-07 22:21:10 +00:00
import { getDefaultConversation } from '../../helpers/getDefaultConversation';
2019-01-14 21:49:58 +00:00
describe('both/state/selectors/conversations', () => {
const getEmptyRootState = (): StateType => {
return rootReducer(undefined, noopAction());
};
2021-05-07 22:21:10 +00:00
function makeConversation(id: string): ConversationType {
return getDefaultConversation({
id,
2021-04-27 22:35:35 +00:00
searchableTitle: `${id} title`,
title: `${id} title`,
2021-05-07 22:21:10 +00:00
});
}
const i18n = setupI18n('en', enMessages);
describe('#getConversationByIdSelector', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
2021-05-07 22:21:10 +00:00
conversationLookup: { abc123: makeConversation('abc123') },
},
};
it('returns undefined if the conversation is not in the lookup', () => {
const selector = getConversationByIdSelector(state);
const actual = selector('xyz');
assert.isUndefined(actual);
});
it('returns the conversation in the lookup if it exists', () => {
const selector = getConversationByIdSelector(state);
const actual = selector('abc123');
assert.strictEqual(actual?.title, 'abc123 title');
});
});
describe('#getConversationSelector', () => {
it('returns empty placeholder if falsey id provided', () => {
const state = getEmptyRootState();
const selector = getConversationSelector(state);
const actual = selector(undefined);
assert.deepEqual(actual, getPlaceholderContact());
});
it('returns empty placeholder if no match', () => {
const state = {
...getEmptyRootState(),
};
const selector = getConversationSelector(state);
const actual = selector('random-id');
assert.deepEqual(actual, getPlaceholderContact());
});
it('returns conversation by e164 first', () => {
const id = 'id';
2021-05-07 22:21:10 +00:00
const conversation = makeConversation(id);
const wrongConversation = makeConversation('wrong');
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: wrongConversation,
},
conversationsByE164: {
[id]: conversation,
},
conversationsByUuid: {
[id]: wrongConversation,
},
conversationsByGroupId: {
[id]: wrongConversation,
},
},
};
const selector = getConversationSelector(state);
const actual = selector(id);
assert.strictEqual(actual, conversation);
});
it('returns conversation by uuid', () => {
const id = 'id';
2021-05-07 22:21:10 +00:00
const conversation = makeConversation(id);
const wrongConversation = makeConversation('wrong');
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: wrongConversation,
},
conversationsByUuid: {
[id]: conversation,
},
conversationsByGroupId: {
[id]: wrongConversation,
},
},
};
const selector = getConversationSelector(state);
const actual = selector(id);
assert.strictEqual(actual, conversation);
});
it('returns conversation by groupId', () => {
const id = 'id';
2021-05-07 22:21:10 +00:00
const conversation = makeConversation(id);
const wrongConversation = makeConversation('wrong');
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: wrongConversation,
},
conversationsByGroupId: {
[id]: conversation,
},
},
};
const selector = getConversationSelector(state);
const actual = selector(id);
assert.strictEqual(actual, conversation);
});
it('returns conversation by conversationId', () => {
const id = 'id';
2021-05-07 22:21:10 +00:00
const conversation = makeConversation(id);
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: conversation,
},
},
};
const selector = getConversationSelector(state);
const actual = selector(id);
assert.strictEqual(actual, conversation);
});
// Less important now, given that all prop-generation for conversations is in
// models/conversation.getProps() and not here.
it('does proper caching of result', () => {
const id = 'id';
2021-05-07 22:21:10 +00:00
const conversation = makeConversation(id);
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: conversation,
},
},
};
const selector = getConversationSelector(state);
const actual = selector(id);
const secondState = {
...state,
conversations: {
...getEmptyState(),
conversationLookup: {
[id]: conversation,
},
},
};
const secondSelector = getConversationSelector(secondState);
const secondActual = secondSelector(id);
assert.strictEqual(actual, secondActual);
const thirdState = {
...state,
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
[id]: makeConversation('third'),
},
},
};
const thirdSelector = getConversationSelector(thirdState);
const thirdActual = thirdSelector(id);
assert.notStrictEqual(actual, thirdActual);
});
});
2021-03-03 20:09:58 +00:00
describe('#getInvitedContactsForNewlyCreatedGroup', () => {
it('returns an empty array if there are no invited contacts', () => {
const state = getEmptyRootState();
assert.deepEqual(getInvitedContactsForNewlyCreatedGroup(state), []);
});
it('returns "hydrated" invited contacts', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc: makeConversation('abc'),
def: makeConversation('def'),
2021-03-03 20:09:58 +00:00
},
invitedConversationIdsForNewlyCreatedGroup: ['def', 'abc'],
},
};
const result = getInvitedContactsForNewlyCreatedGroup(state);
const titles = result.map(conversation => conversation.title);
assert.deepEqual(titles, ['def title', 'abc title']);
});
});
describe('#getComposerStep', () => {
it("returns undefined if the composer isn't open", () => {
const state = getEmptyRootState();
const result = getComposerStep(state);
assert.isUndefined(result);
});
it('returns the first step of the composer', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.StartDirectConversation as const,
searchTerm: 'foo',
2021-03-03 20:09:58 +00:00
},
},
};
const result = getComposerStep(state);
assert.strictEqual(result, ComposerStep.StartDirectConversation);
});
it('returns the second step of the composer', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.ChooseGroupMembers as const,
searchTerm: 'foo',
2021-03-03 20:09:58 +00:00
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
},
},
};
const result = getComposerStep(state);
assert.strictEqual(result, ComposerStep.ChooseGroupMembers);
});
it('returns the third step of the composer', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: false,
hasError: false as const,
},
},
};
const result = getComposerStep(state);
assert.strictEqual(result, ComposerStep.SetGroupMetadata);
});
});
describe('#hasGroupCreationError', () => {
it('returns false if not in the "set group metadata" composer step', () => {
assert.isFalse(hasGroupCreationError(getEmptyRootState()));
assert.isFalse(
hasGroupCreationError({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.StartDirectConversation,
searchTerm: '',
2021-03-03 20:09:58 +00:00
},
},
})
);
});
2021-03-03 20:09:58 +00:00
it('returns false if there is no group creation error', () => {
assert.isFalse(
hasGroupCreationError({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: [],
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: false as const,
hasError: false as const,
},
},
})
);
});
it('returns true if there is a group creation error', () => {
assert.isTrue(
2021-03-03 20:09:58 +00:00
hasGroupCreationError({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
2021-03-03 20:09:58 +00:00
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: [],
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: false as const,
hasError: true as const,
},
},
})
);
});
});
describe('#isCreatingGroup', () => {
it('returns false if not in the "set group metadata" composer step', () => {
assert.isFalse(hasGroupCreationError(getEmptyRootState()));
assert.isFalse(
isCreatingGroup({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.StartDirectConversation,
searchTerm: '',
},
},
})
);
});
2021-03-03 20:09:58 +00:00
it('returns false if the group is not being created', () => {
assert.isFalse(
isCreatingGroup({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: [],
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: false as const,
hasError: true as const,
},
},
})
);
});
it('returns true if the group is being created', () => {
assert.isTrue(
isCreatingGroup({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: [],
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: true as const,
hasError: false as const,
},
},
})
);
});
});
2021-04-28 20:44:48 +00:00
describe('#getAllComposableConversations', () => {
const getRootState = (): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
2021-04-28 20:44:48 +00:00
isMe: true,
profileName: 'My own name',
2021-04-28 20:44:48 +00:00
},
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
const getRootStateWithConversations = (): StateType => {
const result = getRootState();
Object.assign(result.conversations.conversationLookup, {
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
type: 'direct',
profileName: 'A',
2021-04-28 20:44:48 +00:00
title: 'A',
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
2021-04-28 20:44:48 +00:00
type: 'group',
isGroupV1AndDisabled: true,
name: '2',
2021-04-28 20:44:48 +00:00
title: 'Should Be Dropped (GV1)',
},
'convo-3': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-3'),
2021-04-28 20:44:48 +00:00
type: 'group',
name: 'B',
2021-04-28 20:44:48 +00:00
title: 'B',
},
'convo-4': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-4'),
2021-04-28 20:44:48 +00:00
isBlocked: true,
name: '4',
2021-04-28 20:44:48 +00:00
title: 'Should Be Dropped (blocked)',
},
'convo-5': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-5'),
2021-04-28 20:44:48 +00:00
discoveredUnregisteredAt: new Date(1999, 3, 20).getTime(),
name: 'C',
2021-04-28 20:44:48 +00:00
title: 'C',
},
'convo-6': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-6'),
2021-04-28 20:44:48 +00:00
profileSharing: true,
name: 'Should Be Droped (no title)',
title: null,
},
'convo-7': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-7'),
2021-04-28 20:44:48 +00:00
discoveredUnregisteredAt: Date.now(),
name: '7',
2021-04-28 20:44:48 +00:00
title: 'Should Be Dropped (unregistered)',
},
});
return result;
};
it('returns no gv1, no blocked, no missing titles', () => {
const state = getRootStateWithConversations();
const result = getAllComposableConversations(state);
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, [
'our-conversation-id',
'convo-1',
'convo-3',
'convo-5',
]);
});
});
describe('#getComposableContacts', () => {
const getRootState = (): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
isMe: true,
},
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
it('returns only direct contacts, including me', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
isMe: true,
profileSharing: false,
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
type: 'group' as const,
name: 'Friends!',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
name: 'Alice',
},
},
},
};
const result = getComposableContacts(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-0', 'convo-2']);
});
it('excludes blocked, unregistered, and missing name/profileSharing', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
name: 'Ex',
isBlocked: true,
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
name: 'Bob',
discoveredUnregisteredAt: Date.now(),
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
name: 'Charlie',
},
},
},
};
const result = getComposableContacts(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-2']);
});
});
describe('#getCandidateContactsForNewGroup', () => {
const getRootState = (): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
isMe: true,
},
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
it('returns only direct contacts, without me', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
isMe: true,
name: 'Me!',
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
type: 'group' as const,
name: 'Friends!',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
name: 'Alice',
},
},
},
};
const result = getCandidateContactsForNewGroup(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-2']);
});
it('excludes blocked, unregistered, and missing name/profileSharing', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
name: 'Ex',
isBlocked: true,
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
name: 'Bob',
discoveredUnregisteredAt: Date.now(),
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
name: 'Charlie',
},
},
},
};
const result = getCandidateContactsForNewGroup(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-2']);
});
});
describe('#getComposableGroups', () => {
const getRootState = (): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
isMe: true,
},
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
it('returns only groups with name', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
isMe: true,
name: 'Me!',
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
type: 'group' as const,
name: 'Friends!',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
type: 'group' as const,
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
},
},
};
const result = getComposableGroups(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-1']);
});
it('excludes blocked, and missing name/profileSharing', () => {
const state = {
...getRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-0': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-0'),
type: 'group' as const,
name: 'Family!',
isBlocked: true,
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
type: 'group' as const,
name: 'Friends!',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
type: 'group' as const,
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
},
},
};
const result = getComposableGroups(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-1']);
});
});
describe('#getFilteredComposeContacts', () => {
const getRootState = (searchTerm = ''): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
name: 'Me, Myself, and I',
title: 'Me, Myself, and I',
searchableTitle: 'Note to Self',
isMe: true,
},
},
composer: {
2021-03-03 20:09:58 +00:00
step: ComposerStep.StartDirectConversation,
searchTerm,
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
const getRootStateWithConversations = (searchTerm = ''): StateType => {
const result = getRootState(searchTerm);
Object.assign(result.conversations.conversationLookup, {
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
name: 'In System Contacts',
title: 'A. Sorted First',
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
title: 'Should Be Dropped (no name, no profile sharing)',
},
'convo-3': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-3'),
type: 'group',
title: 'Should Be Dropped (group)',
},
'convo-4': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-4'),
isBlocked: true,
title: 'Should Be Dropped (blocked)',
},
'convo-5': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-5'),
discoveredUnregisteredAt: new Date(1999, 3, 20).getTime(),
name: 'In System Contacts (and unregistered too long ago)',
title: 'B. Sorted Second',
},
'convo-6': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-6'),
profileSharing: true,
profileName: 'C. Has Profile Sharing',
title: 'C. Has Profile Sharing',
},
'convo-7': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-7'),
discoveredUnregisteredAt: Date.now(),
title: 'Should Be Dropped (unregistered)',
},
});
return result;
};
2021-04-27 22:35:35 +00:00
it('returns no results when there are no contacts', () => {
const state = getRootState('foo bar baz');
const result = getFilteredComposeContacts(state);
assert.isEmpty(result);
});
it('includes Note to Self with no search term', () => {
const state = getRootStateWithConversations();
const result = getFilteredComposeContacts(state);
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, [
'convo-1',
'convo-5',
'convo-6',
'our-conversation-id',
]);
});
it('can search for contacts', () => {
const state = getRootStateWithConversations('in system');
const result = getFilteredComposeContacts(state);
const ids = result.map(contact => contact.id);
2021-03-29 22:19:59 +00:00
// NOTE: convo-6 matches because you can't write "Sharing" without "in"
assert.deepEqual(ids, ['convo-1', 'convo-5', 'convo-6']);
});
2021-04-27 22:35:35 +00:00
it('can search for note to self', () => {
const state = getRootStateWithConversations('note');
const result = getFilteredComposeContacts(state);
2021-04-27 22:35:35 +00:00
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, ['our-conversation-id']);
2021-04-27 22:35:35 +00:00
});
2021-04-28 20:44:48 +00:00
it('returns note to self when searching for your own name', () => {
2021-04-27 22:35:35 +00:00
const state = getRootStateWithConversations('Myself');
const result = getFilteredComposeContacts(state);
2021-04-27 22:35:35 +00:00
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, ['our-conversation-id']);
2021-04-27 22:35:35 +00:00
});
});
describe('#getFilteredComposeGroups', () => {
const getState = (searchTerm = ''): StateType => {
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
isMe: true,
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
name: 'In System Contacts',
title: 'Should be dropped (contact)',
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
title: 'Should be dropped (contact)',
},
'convo-3': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-3'),
type: 'group',
name: 'Hello World',
title: 'Hello World',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-4': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-4'),
type: 'group',
isBlocked: true,
title: 'Should be dropped (blocked)',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-5': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-5'),
type: 'group',
title: 'Unknown Group',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-6': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-6'),
type: 'group',
name: 'Signal',
title: 'Signal',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
'convo-7': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-7'),
profileSharing: false,
type: 'group',
name: 'Signal Fake',
title: 'Signal Fake',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
},
},
composer: {
step: ComposerStep.StartDirectConversation,
searchTerm,
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
it('can search for groups', () => {
const state = getState('hello');
const result = getFilteredComposeGroups(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-3']);
});
it('does not return unknown groups when getting all groups (no search term)', () => {
const state = getState();
const result = getFilteredComposeGroups(state);
const ids = result.map(group => group.id);
assert.deepEqual(ids, ['convo-3', 'convo-6', 'convo-7']);
});
});
describe('#getFilteredCandidateContactsForNewGroup', () => {
const getRootState = (searchTerm = ''): StateType => {
2021-03-03 20:09:58 +00:00
const rootState = getEmptyRootState();
return {
...rootState,
conversations: {
...getEmptyState(),
conversationLookup: {
'our-conversation-id': {
2021-05-07 22:21:10 +00:00
...makeConversation('our-conversation-id'),
2021-03-03 20:09:58 +00:00
isMe: true,
},
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
2021-03-03 20:09:58 +00:00
name: 'In System Contacts',
title: 'A. Sorted First',
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
2021-03-11 21:29:31 +00:00
title: 'Should be dropped (has no name)',
2021-03-03 20:09:58 +00:00
},
'convo-3': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-3'),
2021-03-03 20:09:58 +00:00
type: 'group',
title: 'Should Be Dropped (group)',
2021-05-07 22:21:10 +00:00
sharedGroupNames: [],
2021-03-03 20:09:58 +00:00
},
'convo-4': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-4'),
2021-03-03 20:09:58 +00:00
isBlocked: true,
2021-03-11 21:29:31 +00:00
name: 'My Name',
2021-03-03 20:09:58 +00:00
title: 'Should Be Dropped (blocked)',
},
'convo-5': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-5'),
2021-03-03 20:09:58 +00:00
discoveredUnregisteredAt: new Date(1999, 3, 20).getTime(),
name: 'In System Contacts (and unregistered too long ago)',
title: 'C. Sorted Third',
2021-03-03 20:09:58 +00:00
},
'convo-6': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-6'),
2021-03-03 20:09:58 +00:00
discoveredUnregisteredAt: Date.now(),
name: 'My Name',
title: 'Should Be Dropped (unregistered)',
2021-03-03 20:09:58 +00:00
},
},
composer: {
step: ComposerStep.ChooseGroupMembers,
searchTerm,
2021-03-03 20:09:58 +00:00
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
},
},
user: {
...rootState.user,
ourConversationId: 'our-conversation-id',
i18n,
},
};
};
it('returns sorted contacts when there is no search term', () => {
const state = getRootState();
const result = getFilteredCandidateContactsForNewGroup(state);
2021-03-03 20:09:58 +00:00
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, ['convo-1', 'convo-5']);
2021-03-03 20:09:58 +00:00
});
it('can search for contacts', () => {
const state = getRootState('system contacts');
const result = getFilteredCandidateContactsForNewGroup(state);
2021-03-03 20:09:58 +00:00
const ids = result.map(contact => contact.id);
assert.deepEqual(ids, ['convo-1', 'convo-5']);
2021-03-03 20:09:58 +00:00
});
});
describe('#getCantAddContactForModal', () => {
it('returns undefined if not in the "choose group members" composer step', () => {
assert.isUndefined(getCantAddContactForModal(getEmptyRootState()));
assert.isUndefined(
getCantAddContactForModal({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.StartDirectConversation,
searchTerm: '',
2021-03-03 20:09:58 +00:00
},
},
})
);
});
it("returns undefined if there's no contact marked", () => {
assert.isUndefined(
getCantAddContactForModal({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
cantAddContactIdForModal: undefined,
searchTerm: '',
2021-03-03 20:09:58 +00:00
groupAvatar: undefined,
groupName: '',
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
selectedConversationIds: [],
step: ComposerStep.ChooseGroupMembers as const,
},
},
})
);
});
it('returns the marked contact', () => {
2021-05-07 22:21:10 +00:00
const conversation = makeConversation('abc123');
2021-03-03 20:09:58 +00:00
assert.deepEqual(
getCantAddContactForModal({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: { abc123: conversation },
composer: {
cantAddContactIdForModal: 'abc123',
searchTerm: '',
2021-03-03 20:09:58 +00:00
groupAvatar: undefined,
groupName: '',
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
selectedConversationIds: [],
step: ComposerStep.ChooseGroupMembers as const,
},
},
}),
conversation
);
});
});
describe('#getComposerConversationSearchTerm', () => {
it("returns the composer's contact search term", () => {
assert.strictEqual(
getComposerConversationSearchTerm({
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
2021-03-03 20:09:58 +00:00
step: ComposerStep.StartDirectConversation,
searchTerm: 'foo bar',
},
},
}),
'foo bar'
);
});
});
describe('#_getLeftPaneLists', () => {
2019-01-14 21:49:58 +00:00
it('sorts conversations based on timestamp then by intl-friendly title', () => {
2019-03-12 00:20:16 +00:00
const data: ConversationLookupType = {
2021-05-07 22:21:10 +00:00
id1: getDefaultConversation({
2019-01-14 21:49:58 +00:00
id: 'id1',
e164: '+18005551111',
2019-01-14 21:49:58 +00:00
activeAt: Date.now(),
name: 'No timestamp',
timestamp: 0,
inboxPosition: 0,
2019-01-14 21:49:58 +00:00
phoneNumber: 'notused',
2019-03-12 00:20:16 +00:00
isArchived: false,
markedUnread: false,
2019-01-14 21:49:58 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'No timestamp',
2019-01-14 21:49:58 +00:00
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
2020-05-27 21:37:06 +00:00
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
id2: getDefaultConversation({
2019-01-14 21:49:58 +00:00
id: 'id2',
e164: '+18005551111',
2019-01-14 21:49:58 +00:00
activeAt: Date.now(),
name: 'B',
timestamp: 20,
inboxPosition: 21,
2019-01-14 21:49:58 +00:00
phoneNumber: 'notused',
2019-03-12 00:20:16 +00:00
isArchived: false,
markedUnread: false,
2019-01-14 21:49:58 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'B',
2019-01-14 21:49:58 +00:00
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
2020-05-27 21:37:06 +00:00
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
id3: getDefaultConversation({
2019-01-14 21:49:58 +00:00
id: 'id3',
e164: '+18005551111',
2019-01-14 21:49:58 +00:00
activeAt: Date.now(),
name: 'C',
timestamp: 20,
inboxPosition: 22,
2019-01-14 21:49:58 +00:00
phoneNumber: 'notused',
2019-03-12 00:20:16 +00:00
isArchived: false,
markedUnread: false,
2019-01-14 21:49:58 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'C',
2019-01-14 21:49:58 +00:00
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
2020-05-27 21:37:06 +00:00
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
id4: getDefaultConversation({
2019-01-14 21:49:58 +00:00
id: 'id4',
e164: '+18005551111',
2019-01-14 21:49:58 +00:00
activeAt: Date.now(),
name: 'Á',
timestamp: 20,
inboxPosition: 20,
2019-01-14 21:49:58 +00:00
phoneNumber: 'notused',
2019-03-12 00:20:16 +00:00
isArchived: false,
markedUnread: false,
2019-01-14 21:49:58 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'A',
2019-01-14 21:49:58 +00:00
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
2020-05-27 21:37:06 +00:00
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
id5: getDefaultConversation({
2019-01-14 21:49:58 +00:00
id: 'id5',
e164: '+18005551111',
2019-01-14 21:49:58 +00:00
activeAt: Date.now(),
name: 'First!',
timestamp: 30,
inboxPosition: 30,
2019-01-14 21:49:58 +00:00
phoneNumber: 'notused',
2019-03-12 00:20:16 +00:00
isArchived: false,
markedUnread: false,
2019-01-14 21:49:58 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'First!',
2019-01-14 21:49:58 +00:00
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
2020-05-27 21:37:06 +00:00
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
2019-01-14 21:49:58 +00:00
};
2020-07-24 01:35:32 +00:00
const comparator = _getConversationComparator();
const {
archivedConversations,
conversations,
pinnedConversations,
} = _getLeftPaneLists(data, comparator);
2019-01-14 21:49:58 +00:00
2019-03-12 00:20:16 +00:00
assert.strictEqual(conversations[0].name, 'First!');
assert.strictEqual(conversations[1].name, 'Á');
assert.strictEqual(conversations[2].name, 'B');
assert.strictEqual(conversations[3].name, 'C');
assert.strictEqual(conversations[4].name, 'No timestamp');
assert.strictEqual(conversations.length, 5);
assert.strictEqual(archivedConversations.length, 0);
assert.strictEqual(pinnedConversations.length, 0);
2019-01-14 21:49:58 +00:00
});
2020-10-10 14:25:17 +00:00
describe('given pinned conversations', () => {
it('sorts pinned conversations based on order in storage', () => {
const data: ConversationLookupType = {
2021-05-07 22:21:10 +00:00
pin2: getDefaultConversation({
2020-10-10 14:25:17 +00:00
id: 'pin2',
e164: '+18005551111',
activeAt: Date.now(),
name: 'Pin Two',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: true,
markedUnread: false,
2020-10-10 14:25:17 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin Two',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin3: getDefaultConversation({
2020-10-10 14:25:17 +00:00
id: 'pin3',
e164: '+18005551111',
activeAt: Date.now(),
name: 'Pin Three',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: true,
markedUnread: false,
2020-10-10 14:25:17 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin Three',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin1: getDefaultConversation({
2020-10-10 14:25:17 +00:00
id: 'pin1',
e164: '+18005551111',
activeAt: Date.now(),
name: 'Pin One',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: true,
markedUnread: false,
2020-10-10 14:25:17 +00:00
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin One',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
2020-10-10 14:25:17 +00:00
};
const pinnedConversationIds = ['pin1', 'pin2', 'pin3'];
2020-10-10 14:25:17 +00:00
const comparator = _getConversationComparator();
const {
archivedConversations,
conversations,
pinnedConversations,
} = _getLeftPaneLists(
data,
comparator,
undefined,
pinnedConversationIds
);
2020-10-10 14:25:17 +00:00
assert.strictEqual(pinnedConversations[0].name, 'Pin One');
assert.strictEqual(pinnedConversations[1].name, 'Pin Two');
assert.strictEqual(pinnedConversations[2].name, 'Pin Three');
assert.strictEqual(archivedConversations.length, 0);
assert.strictEqual(conversations.length, 0);
2020-10-10 14:25:17 +00:00
});
it('includes archived and pinned conversations with no active_at', () => {
const data: ConversationLookupType = {
2021-05-07 22:21:10 +00:00
pin2: getDefaultConversation({
id: 'pin2',
e164: '+18005551111',
name: 'Pin Two',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: true,
markedUnread: false,
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin Two',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin3: getDefaultConversation({
id: 'pin3',
e164: '+18005551111',
name: 'Pin Three',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: true,
markedUnread: false,
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin Three',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin1: getDefaultConversation({
id: 'pin1',
e164: '+18005551111',
name: 'Pin One',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: true,
isPinned: true,
markedUnread: false,
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin One',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin4: getDefaultConversation({
id: 'pin1',
e164: '+18005551111',
name: 'Pin Four',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
activeAt: Date.now(),
isArchived: true,
isPinned: false,
markedUnread: false,
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin One',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
pin5: getDefaultConversation({
id: 'pin1',
e164: '+18005551111',
name: 'Pin Five',
timestamp: 30,
inboxPosition: 30,
phoneNumber: 'notused',
isArchived: false,
isPinned: false,
markedUnread: false,
type: 'direct',
isMe: false,
lastUpdated: Date.now(),
title: 'Pin One',
unreadCount: 1,
isSelected: false,
typingContact: {
name: 'Someone There',
phoneNumber: '+18005551111',
},
acceptedMessageRequest: true,
2021-05-07 22:21:10 +00:00
}),
};
const pinnedConversationIds = ['pin1', 'pin2', 'pin3'];
const comparator = _getConversationComparator();
const {
archivedConversations,
conversations,
pinnedConversations,
} = _getLeftPaneLists(
data,
comparator,
undefined,
pinnedConversationIds
);
assert.strictEqual(pinnedConversations[0].name, 'Pin One');
assert.strictEqual(pinnedConversations[1].name, 'Pin Two');
assert.strictEqual(pinnedConversations[2].name, 'Pin Three');
assert.strictEqual(pinnedConversations.length, 3);
assert.strictEqual(archivedConversations[0].name, 'Pin Four');
assert.strictEqual(archivedConversations.length, 1);
assert.strictEqual(conversations.length, 0);
});
2020-10-10 14:25:17 +00:00
});
2019-01-14 21:49:58 +00:00
});
2021-03-03 20:09:58 +00:00
describe('#getMaximumGroupSizeModalState', () => {
it('returns the modal state', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
cantAddContactIdForModal: undefined,
searchTerm: 'to be cleared',
2021-03-03 20:09:58 +00:00
groupAvatar: undefined,
groupName: '',
maximumGroupSizeModalState: OneTimeModalState.Showing,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
selectedConversationIds: [],
step: ComposerStep.ChooseGroupMembers as const,
},
},
};
assert.strictEqual(
getMaximumGroupSizeModalState(state),
OneTimeModalState.Showing
);
});
});
describe('#getRecommendedGroupSizeModalState', () => {
it('returns the modal state', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
cantAddContactIdForModal: undefined,
searchTerm: 'to be cleared',
2021-03-03 20:09:58 +00:00
groupAvatar: undefined,
groupName: '',
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
recommendedGroupSizeModalState: OneTimeModalState.Showing,
selectedConversationIds: [],
step: ComposerStep.ChooseGroupMembers as const,
},
},
};
assert.strictEqual(
getRecommendedGroupSizeModalState(state),
OneTimeModalState.Showing
);
});
});
describe('#getComposeGroupAvatar', () => {
it('returns undefined if there is no group avatar', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: undefined,
isCreating: false,
hasError: false as const,
},
},
};
assert.isUndefined(getComposeGroupAvatar(state));
});
it('returns the group avatar', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: '',
groupAvatar: new Uint8Array([1, 2, 3]).buffer,
isCreating: false,
hasError: false as const,
},
},
};
assert.deepEqual(
getComposeGroupAvatar(state),
new Uint8Array([1, 2, 3]).buffer
);
});
});
describe('#getComposeGroupName', () => {
it('returns the group name', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: ['abc'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: 'foo bar',
groupAvatar: undefined,
isCreating: false,
hasError: false as const,
},
},
};
assert.deepEqual(getComposeGroupName(state), 'foo bar');
});
});
describe('#getComposeSelectedContacts', () => {
it("returns the composer's selected contacts", () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
'convo-1': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-1'),
2021-03-03 20:09:58 +00:00
title: 'Person One',
},
'convo-2': {
2021-05-07 22:21:10 +00:00
...makeConversation('convo-2'),
2021-03-03 20:09:58 +00:00
title: 'Person Two',
},
},
composer: {
step: ComposerStep.SetGroupMetadata as const,
selectedConversationIds: ['convo-2', 'convo-1'],
cantAddContactIdForModal: undefined,
recommendedGroupSizeModalState: OneTimeModalState.NeverShown,
maximumGroupSizeModalState: OneTimeModalState.NeverShown,
groupName: 'foo bar',
groupAvatar: undefined,
isCreating: false,
hasError: false as const,
},
},
};
const titles = getComposeSelectedContacts(state).map(
contact => contact.title
);
assert.deepEqual(titles, ['Person Two', 'Person One']);
});
});
2021-04-21 16:31:12 +00:00
describe('#getConversationsByTitleSelector', () => {
it('returns a selector that finds conversations by title', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc: { ...makeConversation('abc'), title: 'Janet' },
def: { ...makeConversation('def'), title: 'Janet' },
geh: { ...makeConversation('geh'), title: 'Rick' },
2021-04-21 16:31:12 +00:00
},
},
};
const selector = getConversationsByTitleSelector(state);
assert.sameMembers(
selector('Janet').map(c => c.id),
['abc', 'def']
);
assert.sameMembers(
selector('Rick').map(c => c.id),
['geh']
);
assert.isEmpty(selector('abc'));
assert.isEmpty(selector('xyz'));
});
});
describe('#getSelectedConversationId', () => {
it('returns undefined if no conversation is selected', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc123: makeConversation('abc123'),
},
},
};
assert.isUndefined(getSelectedConversationId(state));
});
it('returns the selected conversation ID', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc123: makeConversation('abc123'),
},
selectedConversationId: 'abc123',
},
};
assert.strictEqual(getSelectedConversationId(state), 'abc123');
});
});
describe('#getSelectedConversation', () => {
it('returns undefined if no conversation is selected', () => {
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc123: makeConversation('abc123'),
},
},
};
assert.isUndefined(getSelectedConversation(state));
});
2021-05-07 22:21:10 +00:00
it('returns the selected conversation', () => {
const conversation = makeConversation('abc123');
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
2021-05-07 22:21:10 +00:00
abc123: conversation,
},
selectedConversationId: 'abc123',
},
};
2021-05-07 22:21:10 +00:00
assert.strictEqual(getSelectedConversation(state), conversation);
});
});
2021-05-28 16:15:17 +00:00
describe('#getContactNameColorSelector', () => {
function makeConversationWithUuid(id: string): ConversationType {
const convo = makeConversation(id);
convo.uuid = id;
return convo;
}
it('returns the right color order sorted by UUID ASC', () => {
const group = makeConversation('group');
group.sortedGroupMembers = [
makeConversationWithUuid('zyx'),
makeConversationWithUuid('vut'),
makeConversationWithUuid('srq'),
makeConversationWithUuid('pon'),
makeConversationWithUuid('mlk'),
makeConversationWithUuid('jih'),
makeConversationWithUuid('gfe'),
];
const state = {
...getEmptyRootState(),
conversations: {
...getEmptyState(),
conversationLookup: {
group,
},
},
};
const contactNameColorSelector = getContactNameColorSelector(state);
2021-06-01 23:37:12 +00:00
assert.equal(contactNameColorSelector('group', 'gfe'), '200');
assert.equal(contactNameColorSelector('group', 'jih'), '150');
assert.equal(contactNameColorSelector('group', 'mlk'), '010');
assert.equal(contactNameColorSelector('group', 'pon'), '300');
assert.equal(contactNameColorSelector('group', 'srq'), '210');
assert.equal(contactNameColorSelector('group', 'vut'), '090');
assert.equal(contactNameColorSelector('group', 'zyx'), '330');
2021-05-28 16:15:17 +00:00
});
});
2019-01-14 21:49:58 +00:00
});