| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2018 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'; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  | import classNames from 'classnames'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   BaseConversationListItem, | 
					
						
							| 
									
										
										
										
											2021-10-14 10:48:48 -05:00
										 |  |  |   HEADER_NAME_CLASS_NAME, | 
					
						
							|  |  |  |   HEADER_CONTACT_NAME_CLASS_NAME, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |   MESSAGE_TEXT_CLASS_NAME, | 
					
						
							|  |  |  | } from './BaseConversationListItem'; | 
					
						
							|  |  |  | import { MessageBody } from '../conversation/MessageBody'; | 
					
						
							|  |  |  | import { ContactName } from '../conversation/ContactName'; | 
					
						
							|  |  |  | import { TypingAnimation } from '../conversation/TypingAnimation'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  | import type { LocalizerType, ThemeType } from '../../types/Util'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ConversationType } from '../../state/ducks/conversations'; | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  | import type { BadgeType } from '../../badges/types'; | 
					
						
							| 
									
										
										
										
											2022-11-08 21:38:19 -05:00
										 |  |  | import { isSignalConversation } from '../../util/isSignalConversation'; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | const MESSAGE_STATUS_ICON_CLASS_NAME = `${MESSAGE_TEXT_CLASS_NAME}__status-icon`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const MessageStatuses = [ | 
					
						
							|  |  |  |   'sending', | 
					
						
							|  |  |  |   'sent', | 
					
						
							|  |  |  |   'delivered', | 
					
						
							|  |  |  |   'read', | 
					
						
							| 
									
										
										
										
											2021-05-05 17:09:29 -07:00
										 |  |  |   'paused', | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |   'error', | 
					
						
							|  |  |  |   'partial-sent', | 
					
						
							|  |  |  | ] as const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type MessageStatusType = typeof MessageStatuses[number]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  | export type PropsData = Pick< | 
					
						
							|  |  |  |   ConversationType, | 
					
						
							|  |  |  |   | 'acceptedMessageRequest' | 
					
						
							|  |  |  |   | 'avatarPath' | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |   | 'badges' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'color' | 
					
						
							|  |  |  |   | 'draftPreview' | 
					
						
							| 
									
										
										
										
											2023-01-12 19:24:59 -05:00
										 |  |  |   | 'groupId' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'id' | 
					
						
							|  |  |  |   | 'isMe' | 
					
						
							| 
									
										
										
										
											2022-01-31 14:01:34 -08:00
										 |  |  |   // NOTE: Passed for CI, not used for rendering
 | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'isPinned' | 
					
						
							|  |  |  |   | 'isSelected' | 
					
						
							|  |  |  |   | 'lastMessage' | 
					
						
							|  |  |  |   | 'lastUpdated' | 
					
						
							|  |  |  |   | 'markedUnread' | 
					
						
							|  |  |  |   | 'muteExpiresAt' | 
					
						
							|  |  |  |   | 'phoneNumber' | 
					
						
							|  |  |  |   | 'profileName' | 
					
						
							|  |  |  |   | 'sharedGroupNames' | 
					
						
							|  |  |  |   | 'shouldShowDraft' | 
					
						
							|  |  |  |   | 'title' | 
					
						
							|  |  |  |   | 'type' | 
					
						
							| 
									
										
										
										
											2021-11-15 14:01:58 -06:00
										 |  |  |   | 'typingContactId' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'unblurredAvatarPath' | 
					
						
							|  |  |  |   | 'unreadCount' | 
					
						
							| 
									
										
										
										
											2022-11-08 21:38:19 -05:00
										 |  |  |   | 'uuid' | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  | > & { | 
					
						
							|  |  |  |   badge?: BadgeType; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | type PropsHousekeeping = { | 
					
						
							|  |  |  |   i18n: LocalizerType; | 
					
						
							|  |  |  |   onClick: (id: string) => void; | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |   theme: ThemeType; | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type Props = PropsData & PropsHousekeeping; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const ConversationListItem: FunctionComponent<Props> = React.memo( | 
					
						
							| 
									
										
										
										
											2021-08-11 12:29:07 -07:00
										 |  |  |   function ConversationListItem({ | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     acceptedMessageRequest, | 
					
						
							|  |  |  |     avatarPath, | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |     badge, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     color, | 
					
						
							|  |  |  |     draftPreview, | 
					
						
							| 
									
										
										
										
											2023-01-12 19:24:59 -05:00
										 |  |  |     groupId, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     i18n, | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     isMe, | 
					
						
							|  |  |  |     isSelected, | 
					
						
							|  |  |  |     lastMessage, | 
					
						
							|  |  |  |     lastUpdated, | 
					
						
							|  |  |  |     markedUnread, | 
					
						
							|  |  |  |     muteExpiresAt, | 
					
						
							|  |  |  |     onClick, | 
					
						
							|  |  |  |     phoneNumber, | 
					
						
							|  |  |  |     profileName, | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |     sharedGroupNames, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     shouldShowDraft, | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |     theme, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     title, | 
					
						
							|  |  |  |     type, | 
					
						
							| 
									
										
										
										
											2021-11-15 14:01:58 -06:00
										 |  |  |     typingContactId, | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |     unblurredAvatarPath, | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     unreadCount, | 
					
						
							| 
									
										
										
										
											2022-11-08 21:38:19 -05:00
										 |  |  |     uuid, | 
					
						
							| 
									
										
										
										
											2021-08-11 12:29:07 -07:00
										 |  |  |   }) { | 
					
						
							| 
									
										
										
										
											2021-10-14 10:48:48 -05:00
										 |  |  |     const isMuted = Boolean(muteExpiresAt && Date.now() < muteExpiresAt); | 
					
						
							|  |  |  |     const headerName = ( | 
					
						
							|  |  |  |       <> | 
					
						
							|  |  |  |         {isMe ? ( | 
					
						
							|  |  |  |           <span className={HEADER_CONTACT_NAME_CLASS_NAME}> | 
					
						
							|  |  |  |             {i18n('noteToSelf')} | 
					
						
							|  |  |  |           </span> | 
					
						
							|  |  |  |         ) : ( | 
					
						
							| 
									
										
										
										
											2022-11-08 21:38:19 -05:00
										 |  |  |           <ContactName | 
					
						
							|  |  |  |             module={HEADER_CONTACT_NAME_CLASS_NAME} | 
					
						
							|  |  |  |             isSignalConversation={isSignalConversation({ id, uuid })} | 
					
						
							|  |  |  |             title={title} | 
					
						
							|  |  |  |           /> | 
					
						
							| 
									
										
										
										
											2021-10-14 10:48:48 -05:00
										 |  |  |         )} | 
					
						
							|  |  |  |         {isMuted && <div className={`${HEADER_NAME_CLASS_NAME}__mute-icon`} />} | 
					
						
							|  |  |  |       </> | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let messageText: ReactNode = null; | 
					
						
							|  |  |  |     let messageStatusIcon: ReactNode = null; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-20 14:20:53 -05:00
										 |  |  |     if (!acceptedMessageRequest) { | 
					
						
							|  |  |  |       messageText = ( | 
					
						
							|  |  |  |         <span className={`${MESSAGE_TEXT_CLASS_NAME}__message-request`}> | 
					
						
							|  |  |  |           {i18n('ConversationListItem--message-request')} | 
					
						
							|  |  |  |         </span> | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-11-15 14:01:58 -06:00
										 |  |  |     } else if (typingContactId) { | 
					
						
							| 
									
										
										
										
											2021-09-20 14:20:53 -05:00
										 |  |  |       messageText = <TypingAnimation i18n={i18n} />; | 
					
						
							|  |  |  |     } else if (shouldShowDraft && draftPreview) { | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |       messageText = ( | 
					
						
							|  |  |  |         <> | 
					
						
							| 
									
										
										
										
											2021-09-20 14:20:53 -05:00
										 |  |  |           <span className={`${MESSAGE_TEXT_CLASS_NAME}__draft-prefix`}> | 
					
						
							|  |  |  |             {i18n('ConversationListItem--draft-prefix')} | 
					
						
							|  |  |  |           </span> | 
					
						
							|  |  |  |           <MessageBody | 
					
						
							|  |  |  |             text={truncateMessageText(draftPreview)} | 
					
						
							|  |  |  |             disableJumbomoji | 
					
						
							|  |  |  |             disableLinks | 
					
						
							|  |  |  |             i18n={i18n} | 
					
						
							|  |  |  |           /> | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         </> | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-09-20 14:20:53 -05:00
										 |  |  |     } else if (lastMessage?.deletedForEveryone) { | 
					
						
							|  |  |  |       messageText = ( | 
					
						
							|  |  |  |         <span className={`${MESSAGE_TEXT_CLASS_NAME}__deleted-for-everyone`}> | 
					
						
							|  |  |  |           {i18n('message--deletedForEveryone')} | 
					
						
							|  |  |  |         </span> | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } else if (lastMessage) { | 
					
						
							|  |  |  |       messageText = ( | 
					
						
							|  |  |  |         <MessageBody | 
					
						
							|  |  |  |           text={truncateMessageText(lastMessage.text)} | 
					
						
							| 
									
										
										
										
											2022-08-25 10:16:37 -06:00
										 |  |  |           author={type === 'group' ? lastMessage.author : undefined} | 
					
						
							| 
									
										
										
										
											2021-09-20 14:20:53 -05:00
										 |  |  |           disableJumbomoji | 
					
						
							|  |  |  |           disableLinks | 
					
						
							|  |  |  |           i18n={i18n} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |       if (lastMessage.status) { | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         messageStatusIcon = ( | 
					
						
							|  |  |  |           <div | 
					
						
							|  |  |  |             className={classNames( | 
					
						
							|  |  |  |               MESSAGE_STATUS_ICON_CLASS_NAME, | 
					
						
							|  |  |  |               `${MESSAGE_STATUS_ICON_CLASS_NAME}--${lastMessage.status}` | 
					
						
							|  |  |  |             )} | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const onClickItem = useCallback(() => onClick(id), [onClick, id]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |       <BaseConversationListItem | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |         acceptedMessageRequest={acceptedMessageRequest} | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         avatarPath={avatarPath} | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |         badge={badge} | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         color={color} | 
					
						
							|  |  |  |         conversationType={type} | 
					
						
							| 
									
										
										
										
											2023-01-12 19:24:59 -05:00
										 |  |  |         groupId={groupId} | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         headerDate={lastUpdated} | 
					
						
							|  |  |  |         headerName={headerName} | 
					
						
							|  |  |  |         i18n={i18n} | 
					
						
							|  |  |  |         id={id} | 
					
						
							|  |  |  |         isMe={isMe} | 
					
						
							|  |  |  |         isSelected={Boolean(isSelected)} | 
					
						
							|  |  |  |         markedUnread={markedUnread} | 
					
						
							|  |  |  |         messageStatusIcon={messageStatusIcon} | 
					
						
							|  |  |  |         messageText={messageText} | 
					
						
							| 
									
										
										
										
											2021-10-12 18:59:08 -05:00
										 |  |  |         messageTextIsAlwaysFullSize | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         onClick={onClickItem} | 
					
						
							|  |  |  |         phoneNumber={phoneNumber} | 
					
						
							|  |  |  |         profileName={profileName} | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |         sharedGroupNames={sharedGroupNames} | 
					
						
							| 
									
										
										
										
											2021-11-02 18:01:13 -05:00
										 |  |  |         theme={theme} | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         title={title} | 
					
						
							|  |  |  |         unreadCount={unreadCount} | 
					
						
							| 
									
										
										
										
											2021-04-30 14:40:25 -05:00
										 |  |  |         unblurredAvatarPath={unblurredAvatarPath} | 
					
						
							| 
									
										
										
										
											2023-01-12 19:24:59 -05:00
										 |  |  |         uuid={uuid} | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2021-08-27 15:49:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-07 15:38:37 -05:00
										 |  |  | // This takes `unknown` because, sometimes, values from the database don't match our
 | 
					
						
							|  |  |  | //   types. In the long term, we should fix that. In the short term, this smooths over the
 | 
					
						
							|  |  |  | //   problem.
 | 
					
						
							|  |  |  | function truncateMessageText(text: unknown): string { | 
					
						
							|  |  |  |   if (typeof text !== 'string') { | 
					
						
							|  |  |  |     return ''; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-10-12 18:59:08 -05:00
										 |  |  |   return text.replace(/(?:\r?\n)+/g, ' '); | 
					
						
							| 
									
										
										
										
											2021-08-27 15:49:45 -05:00
										 |  |  | } |