459 lines
13 KiB
TypeScript
459 lines
13 KiB
TypeScript
// Copyright 2019-2020 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
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<ConversationType>;
|
|
archivedConversations: Array<ConversationType>;
|
|
pinnedConversations: Array<ConversationType>;
|
|
} => {
|
|
const conversations: Array<ConversationType> = [];
|
|
const archivedConversations: Array<ConversationType> = [];
|
|
const pinnedConversations: Array<ConversationType> = [];
|
|
|
|
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 if (conversation.isPinned) {
|
|
pinnedConversations.push(conversation);
|
|
} else {
|
|
conversations.push(conversation);
|
|
}
|
|
}
|
|
}
|
|
|
|
conversations.sort(comparator);
|
|
archivedConversations.sort(comparator);
|
|
|
|
const pinnedConversationIds = window.storage.get<Array<string>>(
|
|
'pinnedConversationIds',
|
|
[]
|
|
);
|
|
pinnedConversations.sort(
|
|
(a, b) =>
|
|
pinnedConversationIds.indexOf(a.id) - pinnedConversationIds.indexOf(b.id)
|
|
);
|
|
|
|
return { conversations, archivedConversations, pinnedConversations };
|
|
};
|
|
|
|
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);
|
|
};
|
|
}
|
|
);
|