import memoizee from 'memoizee'; import { fromPairs, isNumber } from 'lodash'; import { createSelector } from 'reselect'; import { StateType } from '../reducer'; import { ConversationLookupType, ConversationMessageType, ConversationsStateType, ConversationType, MessageLookupType, MessagesByConversationType, MessageType, } from '../ducks/conversations'; import { getBubbleProps } from '../../shims/Whisper'; import { PropsDataType as TimelinePropsType } from '../../components/conversation/Timeline'; import { TimelineItemType } from '../../components/conversation/TimelineItem'; import { getInteractionMode, getIntl, getRegionCode, getUserConversationId, getUserNumber, } from './user'; export const getConversations = (state: StateType): ConversationsStateType => state.conversations; export const getConversationLookup = createSelector( getConversations, (state: ConversationsStateType): ConversationLookupType => { return state.conversationLookup; } ); export const getSelectedConversation = createSelector( getConversations, (state: ConversationsStateType): string | undefined => { return state.selectedConversation; } ); 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 getShowArchived = createSelector( getConversations, (state: ConversationsStateType): boolean => { return Boolean(state.showArchived); } ); 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 ): { conversations: Array; archivedConversations: Array; } => { const conversations: Array = []; const archivedConversations: Array = []; const values = Object.values(lookup); const max = values.length; for (let i = 0; i < max; i += 1) { let conversation = values[i]; if (conversation.activeAt) { if (selectedConversation === conversation.id) { conversation = { ...conversation, isSelected: true, }; } if (conversation.isArchived) { archivedConversations.push(conversation); } else { conversations.push(conversation); } } } conversations.sort(comparator); archivedConversations.sort(comparator); return { conversations, archivedConversations }; }; export const getLeftPaneLists = createSelector( getConversationLookup, getConversationComparator, getSelectedConversation, _getLeftPaneLists ); export const getMe = createSelector( [getConversationLookup, getUserConversationId], ( lookup: ConversationLookupType, ourConversationId: string ): ConversationType => { return lookup[ourConversationId]; } ); // 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 { return conversation; } // 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 | undefined; export const getConversationSelector = createSelector( getCachedSelectorForConversation, getConversationLookup, ( selector: CachedConversationSelectorType, lookup: ConversationLookupType ): GetConversationByIdType => { return (id: string) => { const conversation = lookup[id]; if (!conversation) { return undefined; } return selector(conversation); }; } ); // For now we use a shim, as selector logic is still happening in the Backbone Model. // What needs to happen to pull that selector logic here? // 1) translate ~500 lines of selector logic into TypeScript // 2) other places still rely on that prop-gen code - need to put these under Roots: // - quote compose // - message details export function _messageSelector( message: MessageType, _ourNumber: string, _regionCode: string, interactionMode: 'mouse' | 'keyboard', _conversation?: ConversationType, _author?: ConversationType, _quoted?: ConversationType, selectedMessageId?: string, selectedMessageCounter?: number ): TimelineItemType { // Note: We don't use all of those parameters here, but the shim we call does. // We want to call this function again if any of those parameters change. const props = getBubbleProps(message); if (selectedMessageId === message.id) { return { ...props, data: { ...props.data, interactionMode, isSelected: true, isSelectedCounter: selectedMessageCounter, }, }; } return { ...props, data: { ...props.data, interactionMode, }, }; } // A little optimization to reset our selector cache whenever high-level application data // changes: regionCode and userNumber. type CachedMessageSelectorType = ( message: MessageType, ourNumber: string, regionCode: string, interactionMode: 'mouse' | 'keyboard', conversation?: ConversationType, author?: ConversationType, quoted?: ConversationType, selectedMessageId?: string, selectedMessageCounter?: number ) => TimelineItemType; export const getCachedSelectorForMessage = createSelector( getRegionCode, getUserNumber, (): CachedMessageSelectorType => { // Note: memoizee will check all parameters provided, and only run our selector // if any of them have changed. return memoizee(_messageSelector, { max: 2000 }); } ); type GetMessageByIdType = (id: string) => TimelineItemType | undefined; export const getMessageSelector = createSelector( getCachedSelectorForMessage, getMessages, getSelectedMessage, getConversationSelector, getRegionCode, getUserNumber, getInteractionMode, ( messageSelector: CachedMessageSelectorType, messageLookup: MessageLookupType, selectedMessage: SelectedMessageType | undefined, conversationSelector: GetConversationByIdType, regionCode: string, ourNumber: string, interactionMode: 'keyboard' | 'mouse' ): GetMessageByIdType => { return (id: string) => { const message = messageLookup[id]; if (!message) { return undefined; } const { conversationId, source, type, quote } = message; const conversation = conversationSelector(conversationId); let author: ConversationType | undefined; let quoted: ConversationType | undefined; if (type === 'incoming') { author = conversationSelector(source); } else if (type === 'outgoing') { author = conversationSelector(ourNumber); } if (quote) { quoted = conversationSelector(quote.author); } return messageSelector( message, ourNumber, regionCode, interactionMode, conversation, author, quoted, selectedMessage ? selectedMessage.id : undefined, selectedMessage ? selectedMessage.counter : undefined ); }; } ); export function _conversationMessagesSelector( conversation: ConversationMessageType ): TimelinePropsType { const { heightChangeMessageIds, isLoadingMessages, isNearBottom, loadCountdownStart, messageIds, metrics, resetCounter, scrollToMessageId, scrollToMessageCounter, } = conversation; const firstId = messageIds[0]; const lastId = messageIds.length === 0 ? undefined : messageIds[messageIds.length - 1]; const { oldestUnread } = metrics; const haveNewest = !metrics.newest || !lastId || lastId === metrics.newest.id; const haveOldest = !metrics.oldest || !firstId || firstId === metrics.oldest.id; const items = messageIds; const messageHeightChangeLookup = heightChangeMessageIds && heightChangeMessageIds.length ? fromPairs(heightChangeMessageIds.map(id => [id, true])) : null; const messageHeightChangeIndex = messageHeightChangeLookup ? messageIds.findIndex(id => messageHeightChangeLookup[id]) : undefined; const oldestUnreadIndex = oldestUnread ? messageIds.findIndex(id => id === oldestUnread.id) : undefined; const scrollToIndex = scrollToMessageId ? messageIds.findIndex(id => id === scrollToMessageId) : undefined; const { totalUnread } = metrics; return { haveNewest, haveOldest, isLoadingMessages, loadCountdownStart, items, isNearBottom, messageHeightChangeIndex: isNumber(messageHeightChangeIndex) && messageHeightChangeIndex >= 0 ? messageHeightChangeIndex : undefined, oldestUnreadIndex: isNumber(oldestUnreadIndex) && oldestUnreadIndex >= 0 ? oldestUnreadIndex : undefined, resetCounter, scrollToIndex: isNumber(scrollToIndex) && scrollToIndex >= 0 ? scrollToIndex : undefined, scrollToIndexCounter: scrollToMessageCounter, totalUnread, }; } 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 | undefined => { const conversation = messagesByConversation[id]; if (!conversation) { return undefined; } return conversationMessagesSelector(conversation); }; } );