// Copyright 2019-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import memoizee from 'memoizee'; import { isNumber } from 'lodash'; import { createSelector } from 'reselect'; import type { StateType } from '../reducer'; import type { ConversationLookupType, ConversationMessageType, ConversationsStateType, ConversationType, ConversationVerificationData, MessageLookupType, MessagesByConversationType, PreJoinConversationType, } from '../ducks/conversations'; import type { UsernameSaveState } from '../ducks/conversationsEnums'; import { ComposerStep, OneTimeModalState, ConversationVerificationState, } from '../ducks/conversationsEnums'; import { getOwn } from '../../util/getOwn'; import { isNotNil } from '../../util/isNotNil'; import type { UUIDFetchStateType } from '../../util/uuidFetchState'; import { deconstructLookup } from '../../util/deconstructLookup'; import type { PropsDataType as TimelinePropsType } from '../../components/conversation/Timeline'; import type { TimelineItemType } from '../../components/conversation/TimelineItem'; import { assert } from '../../util/assert'; import { isConversationUnregistered } from '../../util/isConversationUnregistered'; import { filterAndSortConversationsByTitle } from '../../util/filterAndSortConversations'; import type { ContactNameColorType } from '../../types/Colors'; import { ContactNameColors } from '../../types/Colors'; import type { AvatarDataType } from '../../types/Avatar'; import type { UUIDStringType } from '../../types/UUID'; import { isInSystemContacts } from '../../util/isInSystemContacts'; import { sortByTitle } from '../../util/sortByTitle'; import { isDirectConversation, isGroupV1, isGroupV2, } from '../../util/whatTypeOfConversation'; import { getIntl, getRegionCode, getUserConversationId, getUserNumber, getUserUuid, } from './user'; import { getPinnedConversationIds } from './items'; import { getPropsForBubble } from './message'; import type { CallSelectorType, CallStateType } from './calling'; import { getActiveCall, getCallSelector } from './calling'; import type { AccountSelectorType } from './accounts'; import { getAccountSelector } from './accounts'; import * as log from '../../logging/log'; import { TimelineMessageLoadingState } from '../../util/timelineUtil'; let placeholderContact: ConversationType; export const getPlaceholderContact = (): ConversationType => { if (placeholderContact) { return placeholderContact; } placeholderContact = { acceptedMessageRequest: false, badges: [], id: 'placeholder-contact', type: 'direct', title: window.i18n('unknownContact'), isMe: false, sharedGroupNames: [], }; return placeholderContact; }; export const getConversations = (state: StateType): ConversationsStateType => state.conversations; export const getPreJoinConversation = createSelector( getConversations, (state: ConversationsStateType): PreJoinConversationType | undefined => { return state.preJoinConversation; } ); export const getConversationLookup = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationLookup; } ); export const getConversationsByUuid = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationsByUuid; } ); export const getConversationsByE164 = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationsByE164; } ); export const getConversationsByGroupId = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationsByGroupId; } ); export const getConversationsByUsername = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationsByUsername; } ); export const getAllConversations = createSelector( getConversationLookup, (lookup): Array => Object.values(lookup) ); export const getConversationsByTitleSelector = createSelector( getAllConversations, (conversations): ((title: string) => Array) => (title: string) => conversations.filter(conversation => conversation.title === title) ); export const getSelectedConversationId = createSelector( getConversations, (state: ConversationsStateType): string | undefined => { return state.selectedConversationId; } ); type SelectedMessageType = { id: string; counter: number; }; export const getSelectedMessage = createSelector( getConversations, (state: ConversationsStateType): SelectedMessageType | undefined => { if (!state.selectedMessage) { return undefined; } return { id: state.selectedMessage, counter: state.selectedMessageCounter, }; } ); export const getUsernameSaveState = createSelector( getConversations, (state: ConversationsStateType): UsernameSaveState => { return state.usernameSaveState; } ); export const getShowArchived = createSelector( getConversations, (state: ConversationsStateType): boolean => { return Boolean(state.showArchived); } ); const getComposerState = createSelector( getConversations, (state: ConversationsStateType) => state.composer ); export const getComposerStep = createSelector( getComposerState, (composerState): undefined | ComposerStep => composerState?.step ); export const hasGroupCreationError = createSelector( getComposerState, (composerState): boolean => { if (composerState?.step === ComposerStep.SetGroupMetadata) { return composerState.hasError; } return false; } ); export const isCreatingGroup = createSelector( getComposerState, (composerState): boolean => composerState?.step === ComposerStep.SetGroupMetadata && composerState.isCreating ); export const isEditingAvatar = createSelector( getComposerState, (composerState): boolean => composerState?.step === ComposerStep.SetGroupMetadata && composerState.isEditingAvatar ); export const getComposeAvatarData = createSelector( getComposerState, (composerState): ReadonlyArray => composerState?.step === ComposerStep.SetGroupMetadata ? composerState.userAvatarData : [] ); export const getMessages = createSelector( getConversations, (state: ConversationsStateType): MessageLookupType => { return state.messagesLookup; } ); export const getMessagesByConversation = createSelector( getConversations, (state: ConversationsStateType): MessagesByConversationType => { return state.messagesByConversation; } ); const collator = new Intl.Collator(); // Note: we will probably want to put i18n and regionCode back when we are formatting // phone numbers and contacts from scratch here again. export const _getConversationComparator = () => { return (left: ConversationType, right: ConversationType): number => { const leftTimestamp = left.timestamp; const rightTimestamp = right.timestamp; if (leftTimestamp && !rightTimestamp) { return -1; } if (rightTimestamp && !leftTimestamp) { return 1; } if (leftTimestamp && rightTimestamp && leftTimestamp !== rightTimestamp) { return rightTimestamp - leftTimestamp; } if ( typeof left.inboxPosition === 'number' && typeof right.inboxPosition === 'number' ) { return right.inboxPosition > left.inboxPosition ? -1 : 1; } if (typeof left.inboxPosition === 'number' && right.inboxPosition == null) { return -1; } if (typeof right.inboxPosition === 'number' && left.inboxPosition == null) { return 1; } return collator.compare(left.title, right.title); }; }; export const getConversationComparator = createSelector( getIntl, getRegionCode, _getConversationComparator ); export const _getLeftPaneLists = ( lookup: ConversationLookupType, comparator: (left: ConversationType, right: ConversationType) => number, selectedConversation?: string, pinnedConversationIds?: Array ): { conversations: Array; archivedConversations: Array; pinnedConversations: Array; } => { const conversations: Array = []; const archivedConversations: Array = []; const pinnedConversations: Array = []; const values = Object.values(lookup); const max = values.length; for (let i = 0; i < max; i += 1) { let conversation = values[i]; if (selectedConversation === conversation.id) { conversation = { ...conversation, isSelected: true, }; } // We always show pinned conversations if (conversation.isPinned) { pinnedConversations.push(conversation); continue; } if (conversation.activeAt) { if (conversation.isArchived) { archivedConversations.push(conversation); } else { conversations.push(conversation); } } } conversations.sort(comparator); archivedConversations.sort(comparator); pinnedConversations.sort( (a, b) => (pinnedConversationIds || []).indexOf(a.id) - (pinnedConversationIds || []).indexOf(b.id) ); return { conversations, archivedConversations, pinnedConversations }; }; export const getLeftPaneLists = createSelector( getConversationLookup, getConversationComparator, getSelectedConversationId, getPinnedConversationIds, _getLeftPaneLists ); export const getMaximumGroupSizeModalState = createSelector( getComposerState, (composerState): OneTimeModalState => { switch (composerState?.step) { case ComposerStep.ChooseGroupMembers: case ComposerStep.SetGroupMetadata: return composerState.maximumGroupSizeModalState; default: assert( false, 'Can\'t get the maximum group size modal state in this composer state; returning "never shown"' ); return OneTimeModalState.NeverShown; } } ); export const getRecommendedGroupSizeModalState = createSelector( getComposerState, (composerState): OneTimeModalState => { switch (composerState?.step) { case ComposerStep.ChooseGroupMembers: case ComposerStep.SetGroupMetadata: return composerState.recommendedGroupSizeModalState; default: assert( false, 'Can\'t get the recommended group size modal state in this composer state; returning "never shown"' ); return OneTimeModalState.NeverShown; } } ); export const getMe = createSelector( [getConversationLookup, getUserConversationId], ( lookup: ConversationLookupType, ourConversationId: string | undefined ): ConversationType => { if (!ourConversationId) { return getPlaceholderContact(); } return lookup[ourConversationId] || getPlaceholderContact(); } ); export const getComposerConversationSearchTerm = createSelector( getComposerState, (composer): string => { if (!composer) { assert(false, 'getComposerConversationSearchTerm: composer is not open'); return ''; } if (composer.step === ComposerStep.SetGroupMetadata) { assert( false, 'getComposerConversationSearchTerm: composer does not have a search term' ); return ''; } return composer.searchTerm; } ); export const getComposerUUIDFetchState = createSelector( getComposerState, (composer): UUIDFetchStateType => { if (!composer) { assert(false, 'getIsFetchingUsername: composer is not open'); return {}; } if ( composer.step !== ComposerStep.StartDirectConversation && composer.step !== ComposerStep.ChooseGroupMembers ) { assert( false, `getComposerUUIDFetchState: step ${composer.step} ` + 'has no uuidFetchState key' ); return {}; } return composer.uuidFetchState; } ); function isTrusted(conversation: ConversationType): boolean { if (conversation.type === 'group') { return true; } return Boolean( isInSystemContacts(conversation) || conversation.sharedGroupNames.length > 0 || conversation.profileSharing || conversation.isMe ); } function hasDisplayInfo(conversation: ConversationType): boolean { if (conversation.type === 'group') { return Boolean(conversation.name); } return Boolean( conversation.name || conversation.profileName || conversation.phoneNumber || conversation.isMe ); } function canComposeConversation(conversation: ConversationType): boolean { return Boolean( !conversation.isBlocked && !isConversationUnregistered(conversation) && hasDisplayInfo(conversation) && isTrusted(conversation) ); } export const getAllComposableConversations = createSelector( getConversationLookup, (conversationLookup: ConversationLookupType): Array => Object.values(conversationLookup).filter( conversation => !conversation.isBlocked && !conversation.isGroupV1AndDisabled && !isConversationUnregistered(conversation) && // All conversation should have a title except in weird cases where // they don't, in that case we don't want to show these for Forwarding. conversation.title && hasDisplayInfo(conversation) ) ); /** * getComposableContacts/getCandidateContactsForNewGroup both return contacts for the * composer and group members, a different list from your primary system contacts. * This list may include false positives, which is better than missing contacts. * * Note: the key difference between them: * getComposableContacts includes Note to Self * getCandidateContactsForNewGroup does not include Note to Self * * Because they filter unregistered contacts and that's (partially) determined by the * current time, it's possible for them to return stale contacts that have unregistered * if no other conversations change. This should be a rare false positive. */ export const getComposableContacts = createSelector( getConversationLookup, (conversationLookup: ConversationLookupType): Array => Object.values(conversationLookup).filter( conversation => conversation.type === 'direct' && canComposeConversation(conversation) ) ); export const getCandidateContactsForNewGroup = createSelector( getConversationLookup, (conversationLookup: ConversationLookupType): Array => Object.values(conversationLookup).filter( conversation => conversation.type === 'direct' && !conversation.isMe && canComposeConversation(conversation) ) ); export const getComposableGroups = createSelector( getConversationLookup, (conversationLookup: ConversationLookupType): Array => Object.values(conversationLookup).filter( conversation => conversation.type === 'group' && canComposeConversation(conversation) ) ); const getNormalizedComposerConversationSearchTerm = createSelector( getComposerConversationSearchTerm, (searchTerm: string): string => searchTerm.trim() ); export const getFilteredComposeContacts = createSelector( getNormalizedComposerConversationSearchTerm, getComposableContacts, getRegionCode, ( searchTerm: string, contacts: Array, regionCode: string | undefined ): Array => { return filterAndSortConversationsByTitle(contacts, searchTerm, regionCode); } ); export const getFilteredComposeGroups = createSelector( getNormalizedComposerConversationSearchTerm, getComposableGroups, getRegionCode, ( searchTerm: string, groups: Array, regionCode: string | undefined ): Array => { return filterAndSortConversationsByTitle(groups, searchTerm, regionCode); } ); export const getFilteredCandidateContactsForNewGroup = createSelector( getCandidateContactsForNewGroup, getNormalizedComposerConversationSearchTerm, getRegionCode, filterAndSortConversationsByTitle ); const getGroupCreationComposerState = createSelector( getComposerState, ( composerState ): { groupName: string; groupAvatar: undefined | Uint8Array; groupExpireTimer: number; selectedConversationIds: Array; } => { switch (composerState?.step) { case ComposerStep.ChooseGroupMembers: case ComposerStep.SetGroupMetadata: return composerState; default: assert( false, 'getSetGroupMetadataComposerState: expected step to be SetGroupMetadata' ); return { groupName: '', groupAvatar: undefined, groupExpireTimer: 0, selectedConversationIds: [], }; } } ); export const getComposeGroupAvatar = createSelector( getGroupCreationComposerState, (composerState): undefined | Uint8Array => composerState.groupAvatar ); export const getComposeGroupName = createSelector( getGroupCreationComposerState, (composerState): string => composerState.groupName ); export const getComposeGroupExpireTimer = createSelector( getGroupCreationComposerState, (composerState): number => composerState.groupExpireTimer ); export const getComposeSelectedContacts = createSelector( getConversationLookup, getGroupCreationComposerState, (conversationLookup, composerState): Array => deconstructLookup(conversationLookup, composerState.selectedConversationIds) ); // This is where we will put Conversation selector logic, replicating what // is currently in models/conversation.getProps() // What needs to happen to pull that selector logic here? // 1) contactTypingTimers - that UI-only state needs to be moved to redux // 2) all of the message selectors need to be reselect-based; today those // Backbone-based prop-generation functions expect to get Conversation information // directly via ConversationController export function _conversationSelector( conversation?: ConversationType // regionCode: string, // userNumber: string ): ConversationType { if (conversation) { return conversation; } return getPlaceholderContact(); } // A little optimization to reset our selector cache when high-level application data // changes: regionCode and userNumber. type CachedConversationSelectorType = ( conversation?: ConversationType ) => ConversationType; export const getCachedSelectorForConversation = createSelector( getRegionCode, getUserNumber, (): CachedConversationSelectorType => { // Note: memoizee will check all parameters provided, and only run our selector // if any of them have changed. return memoizee(_conversationSelector, { max: 2000 }); } ); export type GetConversationByIdType = (id?: string) => ConversationType; export const getConversationSelector = createSelector( getCachedSelectorForConversation, getConversationLookup, getConversationsByUuid, getConversationsByE164, getConversationsByGroupId, ( selector: CachedConversationSelectorType, byId: ConversationLookupType, byUuid: ConversationLookupType, byE164: ConversationLookupType, byGroupId: ConversationLookupType ): GetConversationByIdType => { return (id?: string) => { if (!id) { return selector(undefined); } const onUuid = getOwn(byUuid, id.toLowerCase ? id.toLowerCase() : id); if (onUuid) { return selector(onUuid); } const onE164 = getOwn(byE164, id); if (onE164) { return selector(onE164); } const onGroupId = getOwn(byGroupId, id); if (onGroupId) { return selector(onGroupId); } const onId = getOwn(byId, id); if (onId) { return selector(onId); } log.warn(`getConversationSelector: No conversation found for id ${id}`); // This will return a placeholder contact return selector(undefined); }; } ); export const getConversationByIdSelector = createSelector( getConversationLookup, conversationLookup => (id: string): undefined | ConversationType => getOwn(conversationLookup, id) ); export const getConversationByUuidSelector = createSelector( getConversationsByUuid, conversationsByUuid => (uuid: UUIDStringType): undefined | ConversationType => getOwn(conversationsByUuid, uuid) ); // A little optimization to reset our selector cache whenever high-level application data // changes: regionCode and userNumber. export const getCachedSelectorForMessage = createSelector( getRegionCode, getUserNumber, (): typeof getPropsForBubble => { // Note: memoizee will check all parameters provided, and only run our selector // if any of them have changed. return memoizee(getPropsForBubble, { max: 2000 }); } ); const getCachedConversationMemberColorsSelector = createSelector( getConversationSelector, getUserConversationId, ( conversationSelector: GetConversationByIdType, ourConversationId: string | undefined ) => { return memoizee( (conversationId: string | undefined) => { const contactNameColors: Map = new Map(); const { sortedGroupMembers = [], type, id: theirId, } = conversationSelector(conversationId); if (type === 'direct') { if (ourConversationId) { contactNameColors.set(ourConversationId, ContactNameColors[0]); } contactNameColors.set(theirId, ContactNameColors[0]); return contactNameColors; } [...sortedGroupMembers] .sort((left, right) => String(left.uuid) > String(right.uuid) ? 1 : -1 ) .forEach((member, i) => { contactNameColors.set( member.id, ContactNameColors[i % ContactNameColors.length] ); }); return contactNameColors; }, { max: 100 } ); } ); export type ContactNameColorSelectorType = ( conversationId: string, contactId: string | undefined ) => ContactNameColorType; export const getContactNameColorSelector = createSelector( getCachedConversationMemberColorsSelector, conversationMemberColorsSelector => { return ( conversationId: string, contactId: string | undefined ): ContactNameColorType => { if (!contactId) { log.warn('No color generated for missing contactId'); return ContactNameColors[0]; } const contactNameColors = conversationMemberColorsSelector(conversationId); const color = contactNameColors.get(contactId); if (!color) { log.warn(`No color generated for contact ${contactId}`); return ContactNameColors[0]; } return color; }; } ); type GetMessageByIdType = (id: string) => TimelineItemType | undefined; export const getMessageSelector = createSelector( getCachedSelectorForMessage, getMessages, getSelectedMessage, getConversationSelector, getRegionCode, getUserNumber, getUserUuid, getUserConversationId, getCallSelector, getActiveCall, getAccountSelector, getContactNameColorSelector, ( messageSelector: typeof getPropsForBubble, messageLookup: MessageLookupType, selectedMessage: SelectedMessageType | undefined, conversationSelector: GetConversationByIdType, regionCode: string | undefined, ourNumber: string | undefined, ourUuid: UUIDStringType | undefined, ourConversationId: string | undefined, callSelector: CallSelectorType, activeCall: undefined | CallStateType, accountSelector: AccountSelectorType, contactNameColorSelector: ContactNameColorSelectorType ): GetMessageByIdType => { return (id: string) => { const message = messageLookup[id]; if (!message) { return undefined; } return messageSelector(message, { conversationSelector, ourConversationId, ourNumber, ourUuid, regionCode, selectedMessageId: selectedMessage?.id, selectedMessageCounter: selectedMessage?.counter, contactNameColorSelector, callSelector, activeCall, accountSelector, }); }; } ); export function _conversationMessagesSelector( conversation: ConversationMessageType ): TimelinePropsType { const { isNearBottom, messageIds, messageLoadingState, metrics, scrollToMessageCounter, scrollToMessageId, } = conversation; const firstId = messageIds[0]; const lastId = messageIds.length === 0 ? undefined : messageIds[messageIds.length - 1]; const { oldestUnseen } = metrics; const haveNewest = !metrics.newest || !lastId || lastId === metrics.newest.id; const haveOldest = !metrics.oldest || !firstId || firstId === metrics.oldest.id; const items = messageIds; const oldestUnseenIndex = oldestUnseen ? messageIds.findIndex(id => id === oldestUnseen.id) : undefined; const scrollToIndex = scrollToMessageId ? messageIds.findIndex(id => id === scrollToMessageId) : undefined; const { totalUnseen } = metrics; return { haveNewest, haveOldest, isNearBottom, items, messageLoadingState, oldestUnseenIndex: isNumber(oldestUnseenIndex) && oldestUnseenIndex >= 0 ? oldestUnseenIndex : undefined, scrollToIndex: isNumber(scrollToIndex) && scrollToIndex >= 0 ? scrollToIndex : undefined, scrollToIndexCounter: scrollToMessageCounter, totalUnseen, }; } type CachedConversationMessagesSelectorType = ( conversation: ConversationMessageType ) => TimelinePropsType; export const getCachedSelectorForConversationMessages = createSelector( getRegionCode, getUserNumber, (): CachedConversationMessagesSelectorType => { // Note: memoizee will check all parameters provided, and only run our selector // if any of them have changed. return memoizee(_conversationMessagesSelector, { max: 50 }); } ); export const getConversationMessagesSelector = createSelector( getCachedSelectorForConversationMessages, getMessagesByConversation, ( conversationMessagesSelector: CachedConversationMessagesSelectorType, messagesByConversation: MessagesByConversationType ) => { return (id: string): TimelinePropsType => { const conversation = messagesByConversation[id]; if (!conversation) { // TODO: DESKTOP-2340 return { haveNewest: false, haveOldest: false, messageLoadingState: TimelineMessageLoadingState.DoingInitialLoad, scrollToIndexCounter: 0, totalUnseen: 0, items: [], }; } return conversationMessagesSelector(conversation); }; } ); export const getInvitedContactsForNewlyCreatedGroup = createSelector( getConversationsByUuid, getConversations, ( conversationLookup, { invitedUuidsForNewlyCreatedGroup = [] } ): Array => deconstructLookup(conversationLookup, invitedUuidsForNewlyCreatedGroup) ); export const getConversationsWithCustomColorSelector = createSelector( getAllConversations, conversations => { return (colorId: string): Array => { return conversations.filter( conversation => conversation.customColorId === colorId ); }; } ); export function isMissingRequiredProfileSharing( conversation: ConversationType ): boolean { const doesConversationRequireIt = !conversation.isMe && !conversation.left && (isGroupV1(conversation) || isDirectConversation(conversation)); return Boolean( doesConversationRequireIt && !conversation.profileSharing && window.Signal.RemoteConfig.isEnabled('desktop.mandatoryProfileSharing') && conversation.messageCount && conversation.messageCount > 0 ); } export const getGroupAdminsSelector = createSelector( getConversationSelector, (conversationSelector: GetConversationByIdType) => { return (conversationId: string): Array => { const { groupId, groupVersion, memberships = [], } = conversationSelector(conversationId); if ( !isGroupV2({ groupId, groupVersion, }) ) { return []; } const admins: Array = []; memberships.forEach(membership => { if (membership.isAdmin) { const admin = conversationSelector(membership.uuid); admins.push(admin); } }); return admins; }; } ); const getConversationVerificationData = createSelector( getConversations, ( conversations: Readonly ): Record => conversations.verificationDataByConversation ); export const getConversationIdsStoppedForVerification = createSelector( getConversationVerificationData, (verificationDataByConversation): Array => Object.keys(verificationDataByConversation) ); export const getConversationsStoppedForVerification = createSelector( getConversationByIdSelector, getConversationIdsStoppedForVerification, ( conversationSelector: (id: string) => undefined | ConversationType, conversationIds: ReadonlyArray ): Array => { const conversations = conversationIds .map(conversationId => conversationSelector(conversationId)) .filter(isNotNil); return sortByTitle(conversations); } ); export const getConversationIdsStoppingSend = createSelector( getConversationVerificationData, (pendingData): Array => { const result = new Set(); Object.values(pendingData).forEach(item => { if (item.type === ConversationVerificationState.PendingVerification) { item.conversationsNeedingVerification.forEach(conversationId => { result.add(conversationId); }); } }); return Array.from(result); } ); export const getConversationsStoppingSend = createSelector( getConversationByIdSelector, getConversationIdsStoppingSend, ( conversationSelector: (id: string) => undefined | ConversationType, conversationIds: ReadonlyArray ): Array => { const conversations = conversationIds .map(conversationId => conversationSelector(conversationId)) .filter(isNotNil); if (conversationIds.length !== conversations.length) { log.warn( `getConversationsStoppingSend: Started with ${conversationIds.length} items, ended up with ${conversations.length}.` ); } return sortByTitle(conversations); } );