import { createSelector } from '@reduxjs/toolkit'; import { compact, isEmpty } from 'lodash'; import { StateType } from '../reducer'; import { ConversationLookupType } from '../ducks/conversations'; import { SearchStateType } from '../ducks/search'; import { getConversationLookup } from './conversations'; import { MessageResultProps } from '../../components/search/MessageSearchResults'; export const getSearch = (state: StateType): SearchStateType => state.search; export const getQuery = (state: StateType): string => getSearch(state).query; export const isSearching = (state: StateType) => { return !!getSearch(state)?.query?.trim(); }; const getSearchResults = createSelector( [getSearch, getConversationLookup], (searchState: SearchStateType, lookup: ConversationLookupType) => { return { contactsAndGroups: compact( searchState.contactsAndGroups .filter(id => { const value = lookup[id]; // on some edges cases, we have an id but no corresponding convo because it matches a query but the conversation was removed. // Don't return anything when activeAt is unset (i.e. no current conversations with this user) if (!value || value.activeAt === undefined || value.activeAt === 0) { // activeAt can be 0 when linking device return false; } return true; }) .map(id => lookup[id]) ), messages: compact(searchState.messages), searchTerm: searchState.query, }; } ); export const getSearchTerm = createSelector([getSearchResults], searchResult => { return searchResult.searchTerm; }); export const getSearchResultsIdsOnly = createSelector([getSearchResults], searchState => { return { ...searchState, contactsAndGroupsIds: searchState.contactsAndGroups.map(m => m.id), }; }); export const getHasSearchResults = createSelector([getSearchResults], searchState => { return !isEmpty(searchState.contactsAndGroups) || !isEmpty(searchState.messages); }); export const getSearchResultsContactOnly = createSelector([getSearchResults], searchState => { return searchState.contactsAndGroups.filter(m => m.isPrivate).map(m => m.id); }); /** * * When type is string, we render a sectionHeader. * When type just has a conversationId field, we render a ConversationListItem. * When type is MessageResultProps we render a MessageSearchResult */ export type SearchResultsMergedListItem = string | { contactConvoId: string } | MessageResultProps; export const getSearchResultsList = createSelector([getSearchResults], searchState => { const { contactsAndGroups, messages } = searchState; const builtList: Array = []; if (contactsAndGroups.length) { builtList.push(window.i18n('conversationsHeader', [`${contactsAndGroups.length}`])); builtList.push(...contactsAndGroups.map(m => ({ contactConvoId: m.id }))); } if (messages.length) { builtList.push(window.i18n('searchMessagesHeader', [`${messages.length}`])); builtList.push(...messages); } return builtList; });