| 
									
										
										
										
											2020-10-30 15:34:04 -05:00
										 |  |  | // Copyright 2020 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  | import React from 'react'; | 
					
						
							| 
									
										
										
										
											2021-03-19 16:37:06 -04:00
										 |  |  | import { sortBy } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2022-11-09 20:59:36 -08:00
										 |  |  | import type { | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   HydratedBodyRangeMention, | 
					
						
							|  |  |  |   BodyRange, | 
					
						
							|  |  |  | } from '../../types/BodyRange'; | 
					
						
							|  |  |  | import { AtMention } from './AtMention'; | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | export type Props = { | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   mentions?: ReadonlyArray<HydratedBodyRangeMention>; | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |   direction?: 'incoming' | 'outgoing'; | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |   showConversation?: (options: { | 
					
						
							|  |  |  |     conversationId: string; | 
					
						
							|  |  |  |     messageId?: string; | 
					
						
							|  |  |  |   }) => unknown; | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |   text: string; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function AtMentionify({ | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   mentions, | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |   direction, | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |   showConversation, | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |   text, | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }: Props): JSX.Element { | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   if (!mentions) { | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |     return <>{text}</>; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const MENTIONS_REGEX = /(\uFFFC@(\d+))/g; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let match = MENTIONS_REGEX.exec(text); | 
					
						
							|  |  |  |   let last = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   const rangeStarts = new Map<number, HydratedBodyRangeMention>(); | 
					
						
							|  |  |  |   mentions.forEach(range => { | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |     rangeStarts.set(range.start, range); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const results = []; | 
					
						
							|  |  |  |   while (match) { | 
					
						
							|  |  |  |     if (last < match.index) { | 
					
						
							|  |  |  |       const textWithNoMentions = text.slice(last, match.index); | 
					
						
							|  |  |  |       results.push(textWithNoMentions); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const rangeStart = Number(match[2]); | 
					
						
							|  |  |  |     const range = rangeStarts.get(rangeStart); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (range) { | 
					
						
							|  |  |  |       results.push( | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |         <AtMention | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |           key={range.start} | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |           direction={direction} | 
					
						
							|  |  |  |           isInvisible={false} | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |           onClick={() => { | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |             if (showConversation) { | 
					
						
							|  |  |  |               showConversation({ conversationId: range.conversationID }); | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |           }} | 
					
						
							|  |  |  |           onKeyUp={e => { | 
					
						
							|  |  |  |             if ( | 
					
						
							|  |  |  |               e.target === e.currentTarget && | 
					
						
							|  |  |  |               e.keyCode === 13 && | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |               showConversation | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |             ) { | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |               showConversation({ conversationId: range.conversationID }); | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |           }} | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |           id={range.conversationID} | 
					
						
							|  |  |  |           name={range.replacementText} | 
					
						
							|  |  |  |         /> | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     last = MENTIONS_REGEX.lastIndex; | 
					
						
							|  |  |  |     match = MENTIONS_REGEX.exec(text); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (last < text.length) { | 
					
						
							|  |  |  |     results.push(text.slice(last)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return <>{results}</>; | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | // At-mentions need to be pre-processed before being pushed through the
 | 
					
						
							|  |  |  | // AtMentionify component, this is due to bodyRanges containing start+length
 | 
					
						
							|  |  |  | // values that operate on the raw string. The text has to be passed through
 | 
					
						
							|  |  |  | // other components before being rendered in the <MessageBody />, components
 | 
					
						
							|  |  |  | // such as Linkify, and Emojify. These components receive the text prop as a
 | 
					
						
							|  |  |  | // string, therefore we're unable to mark it up with DOM nodes prior to handing
 | 
					
						
							|  |  |  | // it off to them. This function will encode the "start" position into the text
 | 
					
						
							|  |  |  | // string so we can later pull it off when rendering the @mention.
 | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  | AtMentionify.preprocessMentions = <T extends BodyRange.Mention>( | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |   text: string, | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   mentions?: ReadonlyArray<BodyRange<T>> | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  | ): string => { | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   if (!mentions || !mentions.length) { | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |     return text; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 16:37:06 -04:00
										 |  |  |   // Sorting by the start index to ensure that we always replace last -> first.
 | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   return sortBy(mentions, 'start').reduceRight((str, range) => { | 
					
						
							| 
									
										
										
										
											2020-09-16 18:42:48 -04:00
										 |  |  |     const textBegin = str.substr(0, range.start); | 
					
						
							|  |  |  |     const encodedMention = `\uFFFC@${range.start}`; | 
					
						
							|  |  |  |     const textEnd = str.substr(range.start + range.length, str.length); | 
					
						
							|  |  |  |     return `${textBegin}${encodedMention}${textEnd}`; | 
					
						
							|  |  |  |   }, text); | 
					
						
							|  |  |  | }; |