signal-desktop/ts/components/conversationList/MessageSearchResult.tsx

210 lines
5.3 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2019 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { FunctionComponent, ReactNode } from 'react';
import React, { useCallback } from 'react';
import { noop } from 'lodash';
import { ContactName } from '../conversation/ContactName';
import type { BodyRangesForDisplayType } from '../../types/BodyRange';
import { processBodyRangesForSearchResult } from '../../types/BodyRange';
import type { LocalizerType, ThemeType } from '../../types/Util';
import { BaseConversationListItem } from './BaseConversationListItem';
2022-06-16 19:12:50 +00:00
import type {
ConversationType,
ShowConversationType,
} from '../../state/ducks/conversations';
2021-11-17 21:11:21 +00:00
import type { PreferredBadgeSelectorType } from '../../state/selectors/badges';
2024-05-15 21:48:02 +00:00
import { I18n } from '../I18n';
import {
MessageTextRenderer,
RenderLocation,
} from '../conversation/MessageTextRenderer';
const EMPTY_OBJECT = Object.freeze(Object.create(null));
export type PropsDataType = {
isSelected?: boolean;
isSearchingInConversation?: boolean;
id: string;
conversationId: string;
sentAt?: number;
snippet: string;
body: string;
bodyRanges: BodyRangesForDisplayType;
from: Pick<
ConversationType,
| 'acceptedMessageRequest'
2024-07-11 19:44:09 +00:00
| 'avatarUrl'
2021-11-17 21:11:21 +00:00
| 'badges'
| 'color'
| 'isMe'
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
| 'title'
| 'type'
2024-07-11 19:44:09 +00:00
| 'unblurredAvatarUrl'
>;
to: Pick<
ConversationType,
'isMe' | 'phoneNumber' | 'profileName' | 'title' | 'type'
>;
};
type PropsHousekeepingType = {
2021-11-17 21:11:21 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
i18n: LocalizerType;
2022-06-16 19:12:50 +00:00
showConversation: ShowConversationType;
2021-11-17 21:11:21 +00:00
theme: ThemeType;
};
export type PropsType = PropsDataType & PropsHousekeepingType;
const renderPerson = (
i18n: LocalizerType,
person: Readonly<{
isMe?: boolean;
title: string;
}>
2024-03-04 18:03:11 +00:00
): JSX.Element =>
person.isMe ? (
2024-05-15 21:48:02 +00:00
<I18n i18n={i18n} id="icu:you" />
2024-03-04 18:03:11 +00:00
) : (
<ContactName title={person.title} />
);
export const MessageSearchResult: FunctionComponent<PropsType> = React.memo(
2021-08-11 19:29:07 +00:00
function MessageSearchResult({
body,
bodyRanges,
conversationId,
from,
2021-11-17 21:11:21 +00:00
getPreferredBadge,
i18n,
id,
sentAt,
2022-06-16 19:12:50 +00:00
showConversation,
snippet,
2021-11-17 21:11:21 +00:00
theme,
to,
2021-08-11 19:29:07 +00:00
}) {
const onClickItem = useCallback(() => {
2022-06-16 19:12:50 +00:00
showConversation({ conversationId, messageId: id });
}, [showConversation, conversationId, id]);
if (!from || !to) {
// Note: mapStateToProps() may return null if the message is not found.
2021-08-11 16:23:21 +00:00
return <div />;
}
const isNoteToSelf = from.isMe && to.isMe;
let headerName: ReactNode;
if (isNoteToSelf) {
2023-03-30 00:03:25 +00:00
headerName = i18n('icu:noteToSelf');
} else if (from.isMe) {
if (to.type === 'group') {
headerName = (
<span>
2024-05-15 21:48:02 +00:00
<I18n
i18n={i18n}
id="icu:searchResultHeader--you-to-group"
components={{
receiverGroup: renderPerson(i18n, to),
}}
/>
</span>
);
} else {
headerName = (
<span>
2024-05-15 21:48:02 +00:00
<I18n
i18n={i18n}
id="icu:searchResultHeader--you-to-receiver"
components={{
receiverContact: renderPerson(i18n, to),
}}
/>
</span>
);
}
} else {
// eslint-disable-next-line no-lonely-if
if (to.type === 'group') {
headerName = (
<span>
2024-05-15 21:48:02 +00:00
<I18n
i18n={i18n}
id="icu:searchResultHeader--sender-to-group"
components={{
sender: renderPerson(i18n, from),
receiverGroup: renderPerson(i18n, to),
}}
/>
</span>
);
} else {
headerName = (
<span>
2024-05-15 21:48:02 +00:00
<I18n
i18n={i18n}
id="icu:searchResultHeader--sender-to-you"
components={{
sender: renderPerson(i18n, from),
}}
/>
</span>
);
}
}
const { cleanedSnippet, bodyRanges: displayBodyRanges } =
processBodyRangesForSearchResult({ snippet, body, bodyRanges });
const messageText = (
<MessageTextRenderer
messageText={cleanedSnippet}
bodyRanges={displayBodyRanges}
direction={undefined}
disableLinks
emojiSizeClass={undefined}
i18n={i18n}
isSpoilerExpanded={EMPTY_OBJECT}
onMentionTrigger={noop}
renderLocation={RenderLocation.SearchResult}
textLength={cleanedSnippet.length}
/>
);
return (
<BaseConversationListItem
acceptedMessageRequest={from.acceptedMessageRequest}
2024-07-11 19:44:09 +00:00
avatarUrl={from.avatarUrl}
2021-11-17 21:11:21 +00:00
badge={getPreferredBadge(from.badges)}
color={from.color}
conversationType="direct"
headerDate={sentAt}
headerName={headerName}
i18n={i18n}
id={id}
isNoteToSelf={isNoteToSelf}
isMe={from.isMe}
isSelected={false}
messageText={messageText}
onClick={onClickItem}
phoneNumber={from.phoneNumber}
profileName={from.profileName}
sharedGroupNames={from.sharedGroupNames}
2021-11-17 21:11:21 +00:00
theme={theme}
title={from.title}
2024-07-11 19:44:09 +00:00
unblurredAvatarUrl={from.unblurredAvatarUrl}
/>
);
}
);