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