import { createSelector } from 'reselect'; import { StateType } from '../reducer'; import { ConversationLookupType, ConversationsStateType, MentionsMembersType, MessageModelPropsWithConvoProps, MessageModelPropsWithoutConvoProps, MessagePropsDetails, ReduxConversationType, SortedMessageModelProps, } from '../ducks/conversations'; import { getIntl } from './user'; import { BlockedNumberController } from '../../util'; import { ConversationModel } from '../../models/conversation'; import { LocalizerType } from '../../types/Util'; import { ConversationHeaderTitleProps } from '../../components/conversation/ConversationHeader'; import { ReplyingToMessageProps } from '../../components/conversation/composition/CompositionBox'; import { MessageAttachmentSelectorProps } from '../../components/conversation/message/message-content/MessageAttachment'; import { MessageAuthorSelectorProps } from '../../components/conversation/message/message-content/MessageAuthorText'; import { MessageAvatarSelectorProps } from '../../components/conversation/message/message-content/MessageAvatar'; import { MessageContentSelectorProps } from '../../components/conversation/message/message-content/MessageContent'; import { MessageContentWithStatusSelectorProps } from '../../components/conversation/message/message-content/MessageContentWithStatus'; import { MessageContextMenuSelectorProps } from '../../components/conversation/message/message-content/MessageContextMenu'; import { MessagePreviewSelectorProps } from '../../components/conversation/message/message-content/MessagePreview'; import { MessageQuoteSelectorProps } from '../../components/conversation/message/message-content/MessageQuote'; import { MessageStatusSelectorProps } from '../../components/conversation/message/message-content/MessageStatus'; import { MessageTextSelectorProps } from '../../components/conversation/message/message-content/MessageText'; import { GenericReadableMessageSelectorProps } from '../../components/conversation/message/message-item/GenericReadableMessage'; import { LightBoxOptions } from '../../components/conversation/SessionConversation'; import { getConversationController } from '../../session/conversations'; import { UserUtils } from '../../session/utils'; import { Storage } from '../../util/storage'; import { ConversationTypeEnum } from '../../models/conversationAttributes'; import { MessageReactsSelectorProps } from '../../components/conversation/message/message-content/MessageReactions'; import { filter, isEmpty, pick, sortBy } from 'lodash'; export const getConversations = (state: StateType): ConversationsStateType => state.conversations; export const getConversationLookup = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationLookup; } ); export const getConversationsCount = createSelector(getConversationLookup, (state): number => { return Object.values(state).length; }); export const getSelectedConversationKey = createSelector( getConversations, (state: ConversationsStateType): string | undefined => { return state.selectedConversation; } ); export const getSelectedConversation = createSelector( getConversations, (state: ConversationsStateType): ReduxConversationType | undefined => { return state.selectedConversation ? state.conversationLookup[state.selectedConversation] : undefined; } ); export const getSelectedConversationIsPublic = createSelector( getSelectedConversation, (state: ReduxConversationType | undefined): boolean => { return state?.isPublic || false; } ); export const getIsTypingEnabled = createSelector( getConversations, getSelectedConversationKey, (state: ConversationsStateType, selectedConvoPubkey?: string): boolean => { if (!selectedConvoPubkey) { return false; } const selectedConvo = state.conversationLookup[selectedConvoPubkey]; if (!selectedConvo) { return false; } const { isBlocked, isKickedFromGroup, left } = selectedConvo; return !(isBlocked || isKickedFromGroup || left); } ); /** * Returns true if the current conversation selected is a group conversation. * Returns false if the current conversation selected is not a group conversation, or none are selected */ export const isGroupConversation = createSelector( getSelectedConversation, (state: ReduxConversationType | undefined): boolean => { return state?.type === 'group' || false; } ); /** * Returns true if the current conversation selected is a closed group and false otherwise. */ export const isClosedGroupConversation = createSelector( getSelectedConversation, (state: ReduxConversationType | undefined): boolean => { return (state?.type === 'group' && !state.isPublic) || false; } ); /** * Returns true if the current conversation selected is a public group and false otherwise. */ export const isPublicGroupConversation = createSelector( getSelectedConversation, (state: ReduxConversationType | undefined): boolean => { return (state?.type === 'group' && state.isPublic) || false; } ); export const getOurPrimaryConversation = createSelector( getConversations, (state: ConversationsStateType): ReduxConversationType => state.conversationLookup[Storage.get('primaryDevicePubKey') as string] ); const getMessagesOfSelectedConversation = createSelector( getConversations, (state: ConversationsStateType): Array => state.messages ); // Redux recommends to do filtered and deriving state in a selector rather than ourself export const getSortedMessagesOfSelectedConversation = createSelector( getMessagesOfSelectedConversation, (messages: Array): Array => { if (messages.length === 0) { return []; } const convoId = messages[0].propsForMessage.convoId; const convo = getConversationController().get(convoId); if (!convo) { return []; } const isPublic = convo.isPublic() || false; const sortedMessage = sortMessages(messages, isPublic); return updateFirstMessageOfSeries(sortedMessage); } ); export const hasSelectedConversationIncomingMessages = createSelector( getSortedMessagesOfSelectedConversation, (messages: Array): boolean => { if (messages.length === 0) { return false; } return Boolean(messages.filter(m => m.propsForMessage.direction === 'incoming').length); } ); export const getFirstUnreadMessageId = createSelector( getConversations, (state: ConversationsStateType): string | undefined => { return state.firstUnreadMessageId; } ); export const getConversationHasUnread = createSelector(getFirstUnreadMessageId, unreadId => { return Boolean(unreadId); }); export type MessagePropsType = | 'group-notification' | 'group-invitation' | 'data-extraction' | 'message-request-response' | 'timer-notification' | 'regular-message' | 'unread-indicator' | 'call-notification'; export const getSortedMessagesTypesOfSelectedConversation = createSelector( getSortedMessagesOfSelectedConversation, getFirstUnreadMessageId, (sortedMessages, firstUnreadId) => { const maxMessagesBetweenTwoDateBreaks = 5; // we want to show the date break if there is a large jump in time // remember that messages are sorted from the most recent to the oldest return sortedMessages.map((msg, index) => { const isFirstUnread = Boolean(firstUnreadId === msg.propsForMessage.id); const messageTimestamp = msg.propsForMessage.serverTimestamp || msg.propsForMessage.timestamp; // do not show the date break if we are the oldest message (no previous) // this is to smooth a bit the loading of older message (to avoid a jump once new messages are rendered) const previousMessageTimestamp = index + 1 >= sortedMessages.length ? Number.MAX_SAFE_INTEGER : sortedMessages[index + 1].propsForMessage.serverTimestamp || sortedMessages[index + 1].propsForMessage.timestamp; const showDateBreak = messageTimestamp - previousMessageTimestamp > maxMessagesBetweenTwoDateBreaks * 60 * 1000 ? messageTimestamp : undefined; const common = { showUnreadIndicator: isFirstUnread, showDateBreak }; if (msg.propsForDataExtractionNotification) { return { ...common, message: { messageType: 'data-extraction', props: { ...msg.propsForDataExtractionNotification, messageId: msg.propsForMessage.id }, }, }; } if (msg.propsForMessageRequestResponse) { return { ...common, message: { messageType: 'message-request-response', props: { ...msg.propsForMessageRequestResponse, messageId: msg.propsForMessage.id }, }, }; } if (msg.propsForGroupInvitation) { return { ...common, message: { messageType: 'group-invitation', props: { ...msg.propsForGroupInvitation, messageId: msg.propsForMessage.id }, }, }; } if (msg.propsForGroupUpdateMessage) { return { ...common, message: { messageType: 'group-notification', props: { ...msg.propsForGroupUpdateMessage, messageId: msg.propsForMessage.id }, }, }; } if (msg.propsForTimerNotification) { return { ...common, message: { messageType: 'timer-notification', props: { ...msg.propsForTimerNotification, messageId: msg.propsForMessage.id }, }, }; } if (msg.propsForCallNotification) { return { ...common, message: { messageType: 'call-notification', props: { ...msg.propsForCallNotification, messageId: msg.propsForMessage.id, }, }, }; } return { showUnreadIndicator: isFirstUnread, showDateBreak, message: { messageType: 'regular-message', props: { messageId: msg.propsForMessage.id }, }, }; }); } ); function getConversationTitle( conversation: ReduxConversationType, testingi18n?: LocalizerType ): string { if (conversation.displayNameInProfile) { return conversation.displayNameInProfile; } if (conversation.type === 'group') { return (testingi18n || window.i18n)('unknown'); } return conversation.id; } const collator = new Intl.Collator(); export const _getConversationComparator = (testingi18n?: LocalizerType) => { return (left: ReduxConversationType, right: ReduxConversationType): number => { // Pin is the first criteria to check if (left.isPinned && !right.isPinned) { return -1; } if (!left.isPinned && right.isPinned) { return 1; } // Then if none is pinned, check other criteria const leftActiveAt = left.activeAt; const rightActiveAt = right.activeAt; if (leftActiveAt && !rightActiveAt) { return -1; } if (rightActiveAt && !leftActiveAt) { return 1; } if (leftActiveAt && rightActiveAt && leftActiveAt !== rightActiveAt) { return rightActiveAt - leftActiveAt; } const leftTitle = getConversationTitle(left, testingi18n).toLowerCase(); const rightTitle = getConversationTitle(right, testingi18n).toLowerCase(); return collator.compare(leftTitle, rightTitle); }; }; export const getConversationComparator = createSelector(getIntl, _getConversationComparator); // export only because we use it in some of our tests // tslint:disable-next-line: cyclomatic-complexity export const _getLeftPaneLists = ( sortedConversations: Array ): { conversations: Array; contacts: Array; unreadCount: number; } => { const conversations: Array = []; const directConversations: Array = []; let unreadCount = 0; for (const conversation of sortedConversations) { if ( conversation.activeAt !== undefined && conversation.type === ConversationTypeEnum.PRIVATE && conversation.isApproved && !conversation.isBlocked ) { directConversations.push(conversation); } if (!conversation.isApproved && conversation.isPrivate) { // dont increase unread counter, don't push to convo list. continue; } if (conversation.isBlocked) { continue; } if ( unreadCount < 9 && conversation.unreadCount && conversation.unreadCount > 0 && conversation.currentNotificationSetting !== 'disabled' ) { unreadCount += conversation.unreadCount; } conversations.push(conversation); } return { conversations, contacts: directConversations, unreadCount, }; }; export const _getSortedConversations = ( lookup: ConversationLookupType, comparator: (left: ReduxConversationType, right: ReduxConversationType) => number, selectedConversation?: string ): Array => { const values = Object.values(lookup); const sorted = values.sort(comparator); const sortedConversations: Array = []; for (let conversation of sorted) { if (selectedConversation === conversation.id) { conversation = { ...conversation, isSelected: true, }; } const isBlocked = BlockedNumberController.isBlocked(conversation.id) || BlockedNumberController.isGroupBlocked(conversation.id); if (isBlocked) { conversation = { ...conversation, isBlocked: true, }; } // Remove all invalid conversations and conversatons of devices associated // with cancelled attempted links if (!conversation.isPublic && !conversation.activeAt) { continue; } sortedConversations.push(conversation); } return sortedConversations; }; export const getSortedConversations = createSelector( getConversationLookup, getConversationComparator, getSelectedConversationKey, _getSortedConversations ); /** * * @param sortedConversations List of conversations that are valid for both requests and regular conversation inbox * @returns A list of message request conversations. */ const _getConversationRequests = ( sortedConversations: Array ): Array => { return filter(sortedConversations, conversation => { const { isApproved, isBlocked, isPrivate, isMe, activeAt } = conversation; const isRequest = ConversationModel.hasValidIncomingRequestValues({ isApproved, isBlocked, isPrivate, isMe, activeAt, }); return isRequest; }); }; export const getConversationRequests = createSelector( getSortedConversations, _getConversationRequests ); const _getUnreadConversationRequests = ( sortedConversationRequests: Array ): Array => { return filter(sortedConversationRequests, conversation => { return conversation && conversation.unreadCount && conversation.unreadCount > 0; }); }; export const getUnreadConversationRequests = createSelector( getConversationRequests, _getUnreadConversationRequests ); const _getPrivateContactsPubkeys = ( sortedConversations: Array ): Array => { return filter(sortedConversations, conversation => { return ( conversation.isPrivate && !conversation.isBlocked && !conversation.isMe && conversation.didApproveMe && conversation.isApproved && Boolean(conversation.activeAt) ); }).map(convo => convo.id); }; /** * Returns all the conversation ids of private conversations which are * - private * - not me * - not blocked * - approved (or message requests are disabled) * - active_at is set to something truthy */ export const getPrivateContactsPubkeys = createSelector( getSortedConversations, _getPrivateContactsPubkeys ); export const getLeftPaneLists = createSelector(getSortedConversations, _getLeftPaneLists); export const getDirectContacts = createSelector( getLeftPaneLists, (state: { conversations: Array; contacts: Array; unreadCount: number; }) => state.contacts ); export const getDirectContactsCount = createSelector( getDirectContacts, (contacts: Array) => contacts.length ); export type DirectContactsByNameType = { displayName?: string; id: string; }; // make sure that createSelector is called here so this function is memoized export const getDirectContactsByName = createSelector( getDirectContacts, (contacts: Array): Array => { const extractedContacts = contacts .filter(m => m.id !== UserUtils.getOurPubKeyStrFromCache()) .map(m => { return { id: m.id, displayName: m.nickname || m.displayNameInProfile, }; }); const extractedContactsNoDisplayName = sortBy( extractedContacts.filter(m => !m.displayName), 'id' ); const extractedContactsWithDisplayName = sortBy( extractedContacts.filter(m => Boolean(m.displayName)), 'displayName' ); return [...extractedContactsWithDisplayName, ...extractedContactsNoDisplayName]; } ); export const getUnreadMessageCount = createSelector(getLeftPaneLists, (state): number => { return state.unreadCount; }); export const getConversationHeaderTitleProps = createSelector(getSelectedConversation, (state): | ConversationHeaderTitleProps | undefined => { if (!state) { return undefined; } return { isKickedFromGroup: !!state.isKickedFromGroup, conversationKey: state.id, isMe: !!state.isMe, members: state.members || [], isPublic: !!state.isPublic, subscriberCount: state.subscriberCount, isGroup: state.type === 'group', currentNotificationSetting: state.currentNotificationSetting, }; }); /** * Returns the formatted text for notification setting. */ export const getCurrentNotificationSettingText = createSelector(getSelectedConversation, (state): | string | undefined => { if (!state) { return undefined; } switch (state.currentNotificationSetting) { case 'all': return window.i18n('notificationForConvo_all'); case 'mentions_only': return window.i18n('notificationForConvo_mentions_only'); case 'disabled': return window.i18n('notificationForConvo_disabled'); default: return window.i18n('notificationForConvo_all'); } }); export const getIsSelectedPrivate = createSelector( getSelectedConversation, (selectedProps): boolean => { return selectedProps?.isPrivate || false; } ); export const getIsSelectedBlocked = createSelector( getSelectedConversation, (selectedProps): boolean => { return selectedProps?.isBlocked || false; } ); /** * Returns true if the currently selected conversation is active (has an active_at field > 0) */ export const getIsSelectedActive = createSelector( getSelectedConversation, (selectedProps): boolean => { return Boolean(selectedProps?.activeAt); } ); export const getIsSelectedNoteToSelf = createSelector( getSelectedConversation, (selectedProps): boolean => { return selectedProps?.isMe || false; } ); export const getNumberOfPinnedConversations = createSelector(getConversations, (state): number => { const values = Object.values(state.conversationLookup); return values.filter(conversation => conversation.isPinned).length; }); export const isMessageDetailView = createSelector( getConversations, (state: ConversationsStateType): boolean => state.messageDetailProps !== undefined ); export const getMessageDetailsViewProps = createSelector( getConversations, (state: ConversationsStateType): MessagePropsDetails | undefined => state.messageDetailProps ); export const isRightPanelShowing = createSelector( getConversations, (state: ConversationsStateType): boolean => state.showRightPanel ); export const isMessageSelectionMode = createSelector( getConversations, (state: ConversationsStateType): boolean => Boolean(state.selectedMessageIds.length > 0) ); export const getSelectedMessageIds = createSelector( getConversations, (state: ConversationsStateType): Array => state.selectedMessageIds ); export const getLightBoxOptions = createSelector( getConversations, (state: ConversationsStateType): LightBoxOptions | undefined => state.lightBox ); export const getQuotedMessage = createSelector( getConversations, (state: ConversationsStateType): ReplyingToMessageProps | undefined => state.quotedMessage ); export const areMoreMessagesBeingFetched = createSelector( getConversations, (state: ConversationsStateType): boolean => state.areMoreMessagesBeingFetched || false ); export const getShowScrollButton = createSelector( getConversations, (state: ConversationsStateType): boolean => state.showScrollButton || false ); export const getQuotedMessageToAnimate = createSelector( getConversations, (state: ConversationsStateType): string | undefined => state.animateQuotedMessageId || undefined ); export const getShouldHighlightMessage = createSelector( getConversations, (state: ConversationsStateType): boolean => Boolean(state.animateQuotedMessageId && state.shouldHighlightMessage) ); export const getNextMessageToPlayId = createSelector( getConversations, (state: ConversationsStateType): string | undefined => state.nextMessageToPlayId || undefined ); export const getMentionsInput = createSelector( getConversations, (state: ConversationsStateType): MentionsMembersType => state.mentionMembers ); /// Those calls are just related to ordering messages in the redux store. function updateFirstMessageOfSeries( messageModelsProps: Array ): Array { // messages are got from the more recent to the oldest, so we need to check if // the next messages in the list is still the same author. // The message is the first of the series if the next message is not from the same author const sortedMessageProps: Array = []; for (let i = 0; i < messageModelsProps.length; i++) { const currentSender = messageModelsProps[i].propsForMessage?.sender; // most recent message is at index 0, so the previous message sender is 1+index const previousSender = i < messageModelsProps.length - 1 ? messageModelsProps[i + 1].propsForMessage?.sender : undefined; const nextSender = i > 0 ? messageModelsProps[i - 1].propsForMessage?.sender : undefined; // Handle firstMessageOfSeries for conditional avatar rendering sortedMessageProps.push({ ...messageModelsProps[i], firstMessageOfSeries: !(i >= 0 && currentSender === previousSender), lastMessageOfSeries: currentSender !== nextSender, }); } return sortedMessageProps; } function sortMessages( messages: Array, isPublic: boolean ): Array { // we order by serverTimestamp for public convos // be sure to update the sorting order to fetch messages from the DB too at getMessagesByConversation if (isPublic) { return messages.slice().sort((a, b) => { return (b.propsForMessage.serverTimestamp || 0) - (a.propsForMessage.serverTimestamp || 0); }); } if (messages.some(n => !n.propsForMessage.timestamp && !n.propsForMessage.receivedAt)) { throw new Error('Found some messages without any timestamp set'); } // for non public convos, we order by sent_at or received_at timestamp. // we assume that a message has either a sent_at or a received_at field set. const messagesSorted = messages .slice() .sort( (a, b) => (b.propsForMessage.timestamp || b.propsForMessage.receivedAt || 0) - (a.propsForMessage.timestamp || a.propsForMessage.receivedAt || 0) ); return messagesSorted; } /** * This returns the most recent message id in the database. This is not the most recent message shown, * but the most recent one, which could still not be loaded. */ export const getMostRecentMessageId = createSelector( getConversations, (state: ConversationsStateType): string | null => { return state.mostRecentMessageId; } ); export const getOldestMessageId = createSelector( getSortedMessagesOfSelectedConversation, (messages: Array): string | undefined => { const oldest = messages.length > 0 ? messages[messages.length - 1].propsForMessage.id : undefined; return oldest; } ); export const getYoungestMessageId = createSelector( getSortedMessagesOfSelectedConversation, (messages: Array): string | undefined => { const youngest = messages.length > 0 ? messages[0].propsForMessage.id : undefined; return youngest; } ); export const getLoadedMessagesLength = createSelector( getConversations, (state: ConversationsStateType): number => { return state.messages.length || 0; } ); export const isFirstUnreadMessageIdAbove = createSelector( getConversations, (state: ConversationsStateType): boolean => { if (!state.firstUnreadMessageId) { return false; } const isNotPresent = !state.messages.some( m => m.propsForMessage.id === state.firstUnreadMessageId ); return isNotPresent; } ); const getMessageId = (_whatever: any, id: string) => id; export const getMessagePropsByMessageId = createSelector( getConversations, getSortedMessagesOfSelectedConversation, getConversationLookup, getMessageId, // tslint:disable-next-line: cyclomatic-complexity ( _convoState, messages: Array, conversations, id ): MessageModelPropsWithConvoProps | undefined => { const foundMessageProps: SortedMessageModelProps | undefined = messages?.find( m => m?.propsForMessage?.id === id ); if (!foundMessageProps || !foundMessageProps.propsForMessage.convoId) { return undefined; } const sender = foundMessageProps?.propsForMessage?.sender; const foundMessageConversation = conversations[foundMessageProps.propsForMessage.convoId]; if (!foundMessageConversation || !sender) { return undefined; } const foundSenderConversation = conversations[sender]; if (!foundSenderConversation) { return undefined; } const ourPubkey = UserUtils.getOurPubKeyStrFromCache(); const isGroup = !foundMessageConversation.isPrivate; const isPublic = foundMessageConversation.isPublic; const groupAdmins = (isGroup && foundMessageConversation.groupAdmins) || []; const weAreAdmin = groupAdmins.includes(ourPubkey) || false; const groupModerators = (isGroup && foundMessageConversation.groupModerators) || []; const weAreModerator = groupModerators.includes(ourPubkey) || false; // A message is deletable if // either we sent it, // or the convo is not a public one (in this case, we will only be able to delete for us) // or the convo is public and we are an admin or moderator const isDeletable = sender === ourPubkey || !isPublic || (isPublic && (weAreAdmin || weAreModerator)); // A message is deletable for everyone if // either we sent it no matter what the conversation type, // or the convo is public and we are an admin or moderator const isDeletableForEveryone = sender === ourPubkey || (isPublic && (weAreAdmin || weAreModerator)) || false; const isSenderAdmin = groupAdmins.includes(sender); const senderIsUs = sender === ourPubkey; const authorName = foundSenderConversation.nickname || foundSenderConversation.displayNameInProfile || null; const authorProfileName = senderIsUs ? window.i18n('you') : foundSenderConversation.nickname || foundSenderConversation.displayNameInProfile || window.i18n('anonymous'); const messageProps: MessageModelPropsWithConvoProps = { ...foundMessageProps, propsForMessage: { ...foundMessageProps.propsForMessage, isBlocked: !!foundMessageConversation.isBlocked, isPublic: !!isPublic, isOpenGroupV2: !!isPublic, isSenderAdmin, isDeletable, isDeletableForEveryone, weAreAdmin, conversationType: foundMessageConversation.type, sender, authorAvatarPath: foundSenderConversation.avatarPath || null, isKickedFromGroup: foundMessageConversation.isKickedFromGroup || false, authorProfileName: authorProfileName || 'Unknown', authorName, }, }; return messageProps; } ); export const getMessageAvatarProps = createSelector(getMessagePropsByMessageId, (props): | MessageAvatarSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const messageAvatarProps: MessageAvatarSelectorProps = { lastMessageOfSeries: props.lastMessageOfSeries, ...pick(props.propsForMessage, [ 'authorAvatarPath', 'authorName', 'sender', 'authorProfileName', 'conversationType', 'direction', 'isPublic', 'isSenderAdmin', ]), }; return messageAvatarProps; }); export const getMessageReactsProps = createSelector(getMessagePropsByMessageId, (props): | MessageReactsSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageReactsSelectorProps = pick(props.propsForMessage, [ 'convoId', 'conversationType', 'isPublic', 'reacts', 'serverId', ]); if (msgProps.reacts) { // NOTE we don't want to render reactions that have 'senders' as an object this is a deprecated type used during development 25/08/2022 const oldReactions = Object.values(msgProps.reacts).filter( reaction => !Array.isArray(reaction.senders) ); if (oldReactions.length > 0) { msgProps.reacts = undefined; return msgProps; } const sortedReacts = Object.entries(msgProps.reacts).sort((a, b) => { return a[1].index < b[1].index ? -1 : a[1].index > b[1].index ? 1 : 0; }); msgProps.sortedReacts = sortedReacts; } return msgProps; }); export const getMessagePreviewProps = createSelector(getMessagePropsByMessageId, (props): | MessagePreviewSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessagePreviewSelectorProps = pick(props.propsForMessage, [ 'attachments', 'previews', ]); return msgProps; }); export const getMessageQuoteProps = createSelector(getMessagePropsByMessageId, (props): | MessageQuoteSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageQuoteSelectorProps = pick(props.propsForMessage, ['direction', 'quote']); return msgProps; }); export const getMessageStatusProps = createSelector(getMessagePropsByMessageId, (props): | MessageStatusSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageStatusSelectorProps = pick(props.propsForMessage, ['direction', 'status']); return msgProps; }); export const getMessageTextProps = createSelector(getMessagePropsByMessageId, (props): | MessageTextSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageTextSelectorProps = pick(props.propsForMessage, [ 'direction', 'status', 'text', 'isDeleted', 'conversationType', ]); return msgProps; }); export const getMessageContextMenuProps = createSelector(getMessagePropsByMessageId, (props): | MessageContextMenuSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageContextMenuSelectorProps = pick(props.propsForMessage, [ 'attachments', 'sender', 'convoId', 'direction', 'status', 'isDeletable', 'isPublic', 'isOpenGroupV2', 'weAreAdmin', 'isSenderAdmin', 'text', 'serverTimestamp', 'timestamp', 'isBlocked', 'isDeletableForEveryone', ]); return msgProps; }); export const getMessageAuthorProps = createSelector(getMessagePropsByMessageId, (props): | MessageAuthorSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageAuthorSelectorProps = { firstMessageOfSeries: props.firstMessageOfSeries, ...pick(props.propsForMessage, ['authorName', 'sender', 'authorProfileName', 'direction']), }; return msgProps; }); export const getMessageIsDeletable = createSelector( getMessagePropsByMessageId, (props): boolean => { if (!props || isEmpty(props)) { return false; } return props.propsForMessage.isDeletable; } ); export const getMessageAttachmentProps = createSelector(getMessagePropsByMessageId, (props): | MessageAttachmentSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageAttachmentSelectorProps = { attachments: props.propsForMessage.attachments || [], ...pick(props.propsForMessage, [ 'direction', 'isTrustedForAttachmentDownload', 'timestamp', 'serverTimestamp', 'sender', 'convoId', ]), }; return msgProps; }); export const getIsMessageSelected = createSelector( getMessagePropsByMessageId, getSelectedMessageIds, (props, selectedIds): boolean => { if (!props || isEmpty(props)) { return false; } const { id } = props.propsForMessage; return selectedIds.includes(id); } ); export const getMessageContentSelectorProps = createSelector(getMessagePropsByMessageId, (props): | MessageContentSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageContentSelectorProps = { ...pick(props.propsForMessage, [ 'direction', 'serverTimestamp', 'text', 'timestamp', 'previews', 'quote', 'attachments', ]), }; return msgProps; }); export const getMessageContentWithStatusesSelectorProps = createSelector( getMessagePropsByMessageId, (props): MessageContentWithStatusSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: MessageContentWithStatusSelectorProps = { hasAttachments: Boolean(props.propsForMessage.attachments?.length) || false, ...pick(props.propsForMessage, ['direction', 'isDeleted', 'isTrustedForAttachmentDownload']), }; return msgProps; } ); export const getGenericReadableMessageSelectorProps = createSelector( getMessagePropsByMessageId, (props): GenericReadableMessageSelectorProps | undefined => { if (!props || isEmpty(props)) { return undefined; } const msgProps: GenericReadableMessageSelectorProps = pick(props.propsForMessage, [ 'convoId', 'direction', 'conversationType', 'expirationLength', 'expirationTimestamp', 'isExpired', 'isUnread', 'receivedAt', 'isKickedFromGroup', 'isDeleted', ]); return msgProps; } ); export const getOldTopMessageId = createSelector( getConversations, (state: ConversationsStateType): string | null => state.oldTopMessageId || null ); export const getOldBottomMessageId = createSelector( getConversations, (state: ConversationsStateType): string | null => state.oldBottomMessageId || null ); export const getIsSelectedConvoInitialLoadingInProgress = createSelector( getSelectedConversation, (convo: ReduxConversationType | undefined): boolean => Boolean(convo?.isInitialFetchingInProgress) );