signal-desktop/ts/components/conversation/AtMentionify.tsx

117 lines
3.3 KiB
TypeScript
Raw Normal View History

2020-10-30 20:34:04 +00:00
// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
2020-09-16 22:42:48 +00:00
import React from 'react';
2021-03-19 20:37:06 +00:00
import { sortBy } from 'lodash';
2020-09-16 22:42:48 +00:00
import { Emojify } from './Emojify';
2022-11-10 04:59:36 +00:00
import type {
BodyRangesType,
HydratedBodyRangeType,
HydratedBodyRangesType,
} from '../../types/Util';
2020-09-16 22:42:48 +00:00
export type Props = {
2022-11-10 04:59:36 +00:00
bodyRanges?: HydratedBodyRangesType;
2020-09-16 22:42:48 +00:00
direction?: 'incoming' | 'outgoing';
openConversation?: (conversationId: string, messageId?: string) => void;
text: string;
};
2022-11-18 00:45:19 +00:00
export function AtMentionify({
2020-09-16 22:42:48 +00:00
bodyRanges,
direction,
openConversation,
text,
2022-11-18 00:45:19 +00:00
}: Props): JSX.Element {
2020-09-16 22:42:48 +00:00
if (!bodyRanges) {
return <>{text}</>;
}
const MENTIONS_REGEX = /(\uFFFC@(\d+))/g;
let match = MENTIONS_REGEX.exec(text);
let last = 0;
2022-11-10 04:59:36 +00:00
const rangeStarts = new Map<number, HydratedBodyRangeType>();
2020-09-16 22:42:48 +00:00
bodyRanges.forEach(range => {
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(
<span
2021-10-20 20:46:42 +00:00
className={`MessageBody__at-mention MessageBody__at-mention--${direction}`}
2020-09-16 22:42:48 +00:00
key={range.start}
onClick={() => {
2022-11-10 04:59:36 +00:00
if (openConversation) {
2020-09-16 22:42:48 +00:00
openConversation(range.conversationID);
}
}}
onKeyUp={e => {
if (
e.target === e.currentTarget &&
e.keyCode === 13 &&
2022-11-10 04:59:36 +00:00
openConversation
2020-09-16 22:42:48 +00:00
) {
openConversation(range.conversationID);
}
}}
tabIndex={0}
role="link"
2020-11-03 01:19:52 +00:00
data-id={range.conversationID}
data-title={range.replacementText}
2020-09-16 22:42:48 +00:00
>
2020-11-03 01:19:52 +00:00
<bdi>
@
<Emojify text={range.replacementText} />
</bdi>
2020-09-16 22:42:48 +00:00
</span>
);
}
last = MENTIONS_REGEX.lastIndex;
match = MENTIONS_REGEX.exec(text);
}
if (last < text.length) {
results.push(text.slice(last));
}
return <>{results}</>;
2022-11-18 00:45:19 +00:00
}
2020-09-16 22:42:48 +00: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.
AtMentionify.preprocessMentions = (
text: string,
bodyRanges?: BodyRangesType
): string => {
if (!bodyRanges || !bodyRanges.length) {
return text;
}
2021-03-19 20:37:06 +00:00
// Sorting by the start index to ensure that we always replace last -> first.
return sortBy(bodyRanges, 'start').reduceRight((str, range) => {
2020-09-16 22:42:48 +00: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);
};