2021-02-23 20:34:28 +00:00
|
|
|
// Copyright 2019-2021 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-08-09 00:46:49 +00:00
|
|
|
import memoizee from 'memoizee';
|
2019-01-14 21:49:58 +00:00
|
|
|
import { createSelector } from 'reselect';
|
2021-02-23 20:34:28 +00:00
|
|
|
|
|
|
|
import { deconstructLookup } from '../../util/deconstructLookup';
|
2019-01-14 21:49:58 +00:00
|
|
|
|
|
|
|
import { StateType } from '../reducer';
|
|
|
|
|
|
|
|
import {
|
2019-08-09 00:46:49 +00:00
|
|
|
MessageSearchResultLookupType,
|
|
|
|
MessageSearchResultType,
|
|
|
|
SearchStateType,
|
|
|
|
} from '../ducks/search';
|
|
|
|
import {
|
|
|
|
ConversationLookupType,
|
|
|
|
ConversationType,
|
|
|
|
} from '../ducks/conversations';
|
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
import { LeftPaneSearchPropsType } from '../../components/leftPane/LeftPaneSearchHelper';
|
|
|
|
import { PropsDataType as MessageSearchResultPropsDataType } from '../../components/conversationList/MessageSearchResult';
|
2019-08-09 00:46:49 +00:00
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
import { getUserConversationId } from './user';
|
2019-08-09 00:46:49 +00:00
|
|
|
import {
|
|
|
|
GetConversationByIdType,
|
2019-01-14 21:49:58 +00:00
|
|
|
getConversationLookup,
|
2019-08-09 00:46:49 +00:00
|
|
|
getConversationSelector,
|
2019-01-14 21:49:58 +00:00
|
|
|
} from './conversations';
|
2019-03-12 00:20:16 +00:00
|
|
|
|
2021-03-05 17:57:09 +00:00
|
|
|
import { BodyRangeType } from '../../types/Util';
|
2021-09-17 18:27:53 +00:00
|
|
|
import * as log from '../../logging/log';
|
2021-03-05 17:57:09 +00:00
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
export const getSearch = (state: StateType): SearchStateType => state.search;
|
|
|
|
|
|
|
|
export const getQuery = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType): string => state.query
|
|
|
|
);
|
|
|
|
|
|
|
|
export const getSelectedMessage = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType): string | undefined => state.selectedMessage
|
|
|
|
);
|
|
|
|
|
2019-08-09 23:12:29 +00:00
|
|
|
export const getSearchConversationId = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType): string | undefined => state.searchConversationId
|
|
|
|
);
|
|
|
|
|
|
|
|
export const getSearchConversationName = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType): string | undefined => state.searchConversationName
|
|
|
|
);
|
|
|
|
|
2019-11-07 21:36:16 +00:00
|
|
|
export const getStartSearchCounter = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType): number => state.startSearchCounter
|
|
|
|
);
|
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
export const isSearching = createSelector(
|
2021-03-18 20:36:47 +00:00
|
|
|
getQuery,
|
2021-04-14 10:52:37 +00:00
|
|
|
(query: string): boolean => query.trim().length > 0
|
2019-01-14 21:49:58 +00:00
|
|
|
);
|
|
|
|
|
2019-08-09 00:46:49 +00:00
|
|
|
export const getMessageSearchResultLookup = createSelector(
|
|
|
|
getSearch,
|
|
|
|
(state: SearchStateType) => state.messageLookup
|
|
|
|
);
|
2021-02-23 20:34:28 +00:00
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
export const getSearchResults = createSelector(
|
2021-02-23 20:34:28 +00:00
|
|
|
[getSearch, getConversationLookup],
|
2019-01-14 21:49:58 +00:00
|
|
|
(
|
|
|
|
state: SearchStateType,
|
2021-02-23 20:34:28 +00:00
|
|
|
conversationLookup: ConversationLookupType
|
2021-05-19 16:14:35 +00:00
|
|
|
): Omit<LeftPaneSearchPropsType, 'primarySendsSms'> => {
|
2019-08-09 23:12:29 +00:00
|
|
|
const {
|
2021-02-23 20:34:28 +00:00
|
|
|
contactIds,
|
|
|
|
conversationIds,
|
2019-09-04 14:46:28 +00:00
|
|
|
discussionsLoading,
|
2019-08-09 23:12:29 +00:00
|
|
|
messageIds,
|
2021-02-23 20:34:28 +00:00
|
|
|
messageLookup,
|
2019-09-04 14:46:28 +00:00
|
|
|
messagesLoading,
|
2019-08-09 23:12:29 +00:00
|
|
|
searchConversationName,
|
|
|
|
} = state;
|
2019-08-09 00:46:49 +00:00
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
return {
|
|
|
|
conversationResults: discussionsLoading
|
|
|
|
? { isLoading: true }
|
|
|
|
: {
|
|
|
|
isLoading: false,
|
|
|
|
results: deconstructLookup(conversationLookup, conversationIds),
|
2019-08-09 00:46:49 +00:00
|
|
|
},
|
2021-02-23 20:34:28 +00:00
|
|
|
contactResults: discussionsLoading
|
|
|
|
? { isLoading: true }
|
|
|
|
: {
|
|
|
|
isLoading: false,
|
|
|
|
results: deconstructLookup(conversationLookup, contactIds),
|
|
|
|
},
|
|
|
|
messageResults: messagesLoading
|
|
|
|
? { isLoading: true }
|
|
|
|
: {
|
|
|
|
isLoading: false,
|
|
|
|
results: deconstructLookup(messageLookup, messageIds),
|
2019-08-09 00:46:49 +00:00
|
|
|
},
|
2019-08-09 23:12:29 +00:00
|
|
|
searchConversationName,
|
2019-01-14 21:49:58 +00:00
|
|
|
searchTerm: state.query,
|
2019-08-09 00:46:49 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// A little optimization to reset our selector cache whenever high-level application data
|
|
|
|
// changes: regionCode and userNumber.
|
|
|
|
type CachedMessageSearchResultSelectorType = (
|
|
|
|
message: MessageSearchResultType,
|
2021-01-06 15:41:43 +00:00
|
|
|
from: ConversationType,
|
|
|
|
to: ConversationType,
|
2019-08-09 23:12:29 +00:00
|
|
|
searchConversationId?: string,
|
2019-08-09 00:46:49 +00:00
|
|
|
selectedMessageId?: string
|
|
|
|
) => MessageSearchResultPropsDataType;
|
2021-03-05 17:57:09 +00:00
|
|
|
|
2019-08-09 00:46:49 +00:00
|
|
|
export const getCachedSelectorForMessageSearchResult = createSelector(
|
2021-01-06 15:41:43 +00:00
|
|
|
getUserConversationId,
|
2021-03-05 17:57:09 +00:00
|
|
|
getConversationSelector,
|
|
|
|
(
|
|
|
|
_,
|
|
|
|
conversationSelector: GetConversationByIdType
|
|
|
|
): CachedMessageSearchResultSelectorType => {
|
2019-08-09 00:46:49 +00:00
|
|
|
// Note: memoizee will check all parameters provided, and only run our selector
|
|
|
|
// if any of them have changed.
|
2021-03-05 17:57:09 +00:00
|
|
|
return memoizee(
|
|
|
|
(
|
|
|
|
message: MessageSearchResultType,
|
|
|
|
from: ConversationType,
|
|
|
|
to: ConversationType,
|
|
|
|
searchConversationId?: string,
|
|
|
|
selectedMessageId?: string
|
|
|
|
) => {
|
|
|
|
const bodyRanges = message.bodyRanges || [];
|
|
|
|
return {
|
|
|
|
from,
|
|
|
|
to,
|
|
|
|
|
|
|
|
id: message.id,
|
|
|
|
conversationId: message.conversationId,
|
|
|
|
sentAt: message.sent_at,
|
2021-06-17 17:15:10 +00:00
|
|
|
snippet: message.snippet || '',
|
2021-03-05 17:57:09 +00:00
|
|
|
bodyRanges: bodyRanges.map((bodyRange: BodyRangeType) => {
|
|
|
|
const conversation = conversationSelector(bodyRange.mentionUuid);
|
|
|
|
|
|
|
|
return {
|
|
|
|
...bodyRange,
|
|
|
|
replacementText: conversation.title,
|
|
|
|
};
|
|
|
|
}),
|
2021-06-17 17:15:10 +00:00
|
|
|
body: message.body || '',
|
2021-03-05 17:57:09 +00:00
|
|
|
|
|
|
|
isSelected: Boolean(
|
|
|
|
selectedMessageId && message.id === selectedMessageId
|
|
|
|
),
|
|
|
|
isSearchingInConversation: Boolean(searchConversationId),
|
|
|
|
};
|
|
|
|
},
|
|
|
|
{ max: 500 }
|
|
|
|
);
|
2019-08-09 00:46:49 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
type GetMessageSearchResultByIdType = (
|
|
|
|
id: string
|
|
|
|
) => MessageSearchResultPropsDataType | undefined;
|
2021-03-05 17:57:09 +00:00
|
|
|
|
2019-08-09 00:46:49 +00:00
|
|
|
export const getMessageSearchResultSelector = createSelector(
|
|
|
|
getCachedSelectorForMessageSearchResult,
|
|
|
|
getMessageSearchResultLookup,
|
|
|
|
getSelectedMessage,
|
|
|
|
getConversationSelector,
|
2019-08-09 23:12:29 +00:00
|
|
|
getSearchConversationId,
|
2021-01-06 15:41:43 +00:00
|
|
|
getUserConversationId,
|
2019-08-09 00:46:49 +00:00
|
|
|
(
|
|
|
|
messageSearchResultSelector: CachedMessageSearchResultSelectorType,
|
|
|
|
messageSearchResultLookup: MessageSearchResultLookupType,
|
2021-01-06 15:41:43 +00:00
|
|
|
selectedMessageId: string | undefined,
|
2019-08-09 00:46:49 +00:00
|
|
|
conversationSelector: GetConversationByIdType,
|
2019-08-09 23:12:29 +00:00
|
|
|
searchConversationId: string | undefined,
|
2021-01-06 15:41:43 +00:00
|
|
|
ourConversationId: string
|
2019-08-09 00:46:49 +00:00
|
|
|
): GetMessageSearchResultByIdType => {
|
|
|
|
return (id: string) => {
|
|
|
|
const message = messageSearchResultLookup[id];
|
|
|
|
if (!message) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-01-06 15:41:43 +00:00
|
|
|
`getMessageSearchResultSelector: messageSearchResultLookup was missing id ${id}`
|
|
|
|
);
|
2020-09-14 21:56:35 +00:00
|
|
|
return undefined;
|
2019-08-09 00:46:49 +00:00
|
|
|
}
|
|
|
|
|
2021-01-06 15:41:43 +00:00
|
|
|
const { conversationId, source, sourceUuid, type } = message;
|
|
|
|
let from: ConversationType;
|
|
|
|
let to: ConversationType;
|
2019-08-09 00:46:49 +00:00
|
|
|
|
|
|
|
if (type === 'incoming') {
|
2021-01-06 15:41:43 +00:00
|
|
|
from = conversationSelector(sourceUuid || source);
|
|
|
|
to = conversationSelector(conversationId);
|
2021-04-08 00:27:27 +00:00
|
|
|
if (from === to) {
|
|
|
|
to = conversationSelector(ourConversationId);
|
|
|
|
}
|
2019-08-09 00:46:49 +00:00
|
|
|
} else if (type === 'outgoing') {
|
2021-01-06 15:41:43 +00:00
|
|
|
from = conversationSelector(ourConversationId);
|
|
|
|
to = conversationSelector(conversationId);
|
|
|
|
} else {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(`getMessageSearchResultSelector: Got unexpected type ${type}`);
|
2021-01-06 15:41:43 +00:00
|
|
|
return undefined;
|
2019-08-09 00:46:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return messageSearchResultSelector(
|
|
|
|
message,
|
2021-01-06 15:41:43 +00:00
|
|
|
from,
|
|
|
|
to,
|
2019-08-09 23:12:29 +00:00
|
|
|
searchConversationId,
|
2021-01-06 15:41:43 +00:00
|
|
|
selectedMessageId
|
2019-08-09 00:46:49 +00:00
|
|
|
);
|
2019-01-14 21:49:58 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
);
|