Skip to content

Commit 61173a4

Browse files
committed
Use most recent username or display name when unwrapping mention tags
1 parent e9d189e commit 61173a4

File tree

3 files changed

+125
-12
lines changed

3 files changed

+125
-12
lines changed

src/sidebar/components/MarkdownEditor.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -585,8 +585,8 @@ export default function MarkdownEditor({
585585
const input = useRef<HTMLTextAreaElement>(null);
586586

587587
const textWithoutMentionTags = useMemo(
588-
() => unwrapMentions(text, mentionMode),
589-
[mentionMode, text],
588+
() => unwrapMentions({ text, mentionMode, mentions }),
589+
[mentionMode, mentions, text],
590590
);
591591

592592
useEffect(() => {

src/sidebar/helpers/mentions.ts

+37-8
Original file line numberDiff line numberDiff line change
@@ -96,27 +96,56 @@ export function wrapDisplayNameMentions(
9696
// some assumptions about the tag content in order to be able to get away
9797
// with it. Specifically we assume that these tags will not contain `<` or
9898
// `>` in attribute values or the HTML content of the tag.
99-
const MENTION_TAG_RE = /<a[^>]\bdata-hyp-mention\b[^>]*>([^<]+)<\/a>/g;
99+
const MENTION_TAG_RE = /<a[^>]\bdata-hyp-mention\b[^>]*>@([^<]+)<\/a>/g;
100+
101+
export type UnwrapMentionsOptions = {
102+
text: string;
103+
mentionMode: MentionMode;
104+
mentions?: Mention[];
105+
};
100106

101107
/**
102108
* Replace all mentions wrapped in the special `<a data-hyp-mention />` tag with
103109
* their plain-text representation.
110+
*
104111
* The plain-text representation depends on the mention mode:
105112
* - `username`: @username
106113
* - `display-name`: @[Display Name]
114+
*
115+
* If a list of mentions is provided, the tag's userid will be matched against
116+
* it, so that the plain-text version uses the most recent username or display
117+
* name, in case they have changed since the mention was created.
118+
* If a list is not provided or the mention is not found, the tag's content will
119+
* be used as username or display name.
107120
*/
108-
export function unwrapMentions(text: string, mentionMode: MentionMode) {
121+
export function unwrapMentions({
122+
text,
123+
mentionMode,
124+
mentions = [],
125+
}: UnwrapMentionsOptions) {
109126
// Use a regex rather than HTML parser to replace the mentions in order
110127
// to avoid modifying any of the content outside of the replaced tags. This
111128
// includes avoiding modifications such as encoding characters that will
112129
// happen when parsing and re-serializing HTML via eg. `innerHTML`.
113-
return text.replace(MENTION_TAG_RE, (match, mention) => {
114-
if (mentionMode === 'username') {
115-
return mention;
116-
}
130+
return text.replace(MENTION_TAG_RE, (match, tagContent) => {
131+
// Even though we need to capture the tag content via a regex for the
132+
// reasons explained above, we can use regular DOM APIs to get the userid
133+
const tempElement = document.createElement('div');
134+
tempElement.innerHTML = match;
135+
const userid = tempElement
136+
.querySelector('a[data-hyp-mention]')
137+
?.getAttribute('data-userid');
138+
139+
const mention = mentions.find(
140+
({ original_userid }) => original_userid === userid,
141+
);
117142

118-
const [atChar, ...rest] = mention;
119-
return `${atChar}[${rest.join('')}]`;
143+
return mentionMode === 'username'
144+
? `@${mention?.username ?? tagContent}`
145+
: // Using || rather than ?? for the display name, to avoid setting an
146+
// empty string if the user used to have a display name and has been
147+
// removed since the mention was created
148+
`@[${mention?.display_name || tagContent}]`;
120149
});
121150
}
122151

src/sidebar/helpers/test/mentions-test.js

+86-2
Original file line numberDiff line numberDiff line change
@@ -134,7 +134,10 @@ Hello ${mentionTag('jane.doe', 'example.com')}.`,
134134

135135
describe('unwrapMentions - `username` mode', () => {
136136
it('removes wrapping mention tags', () => {
137-
assert.equal(unwrapMentions(textWithTags, 'username'), text);
137+
assert.equal(
138+
unwrapMentions({ text: textWithTags, mentionMode: 'username' }),
139+
text,
140+
);
138141
});
139142
});
140143
});
@@ -211,7 +214,88 @@ Hello ${mentionTag('jane.doe', 'example.com')}.`,
211214

212215
describe('unwrapMentions - `display-name` mode', () => {
213216
it('removes wrapping mention tags', () => {
214-
assert.equal(unwrapMentions(textWithTags, 'display-name'), text);
217+
assert.equal(
218+
unwrapMentions({ text: textWithTags, mentionMode: 'display-name' }),
219+
text,
220+
);
221+
});
222+
});
223+
});
224+
225+
describe('unwrapMentions', () => {
226+
[
227+
// Mention not found. Tag content kept
228+
{
229+
mentionMode: 'username',
230+
mentions: [],
231+
text: `Hello ${mentionTag('jane_doe')}`,
232+
expectedResult: 'Hello @jane_doe',
233+
},
234+
{
235+
mentionMode: 'display-name',
236+
mentions: [],
237+
text: `Hello ${displayNameMentionTag('Jane Doe', 'jane_doe')}`,
238+
expectedResult: 'Hello @[Jane Doe]',
239+
},
240+
241+
// Mention found with a new username
242+
{
243+
mentionMode: 'username',
244+
mentions: [
245+
{
246+
original_userid: 'acct:[email protected]',
247+
username: 'jane_edited',
248+
},
249+
],
250+
text: `Hello ${mentionTag('jane_doe')}`,
251+
expectedResult: 'Hello @jane_edited',
252+
},
253+
254+
// Mention found with a new display name
255+
{
256+
mentionMode: 'display-name',
257+
mentions: [
258+
{
259+
original_userid: 'acct:[email protected]',
260+
display_name: 'My new name',
261+
},
262+
],
263+
text: `Hello ${displayNameMentionTag('Jane Doe', 'jane_doe')}`,
264+
expectedResult: 'Hello @[My new name]',
265+
},
266+
267+
// Mention found with a now empty display name
268+
{
269+
mentionMode: 'display-name',
270+
mentions: [
271+
{
272+
original_userid: 'acct:[email protected]',
273+
display_name: '',
274+
},
275+
],
276+
text: `Hello ${displayNameMentionTag('Jane Doe', 'jane_doe')}`,
277+
expectedResult: 'Hello @[Jane Doe]',
278+
},
279+
280+
// data-userid not present
281+
{
282+
mentionMode: 'username',
283+
mentions: [],
284+
text: 'Hello <a data-hyp-mention="">@user_id_missing</a>',
285+
expectedResult: 'Hello @user_id_missing',
286+
},
287+
{
288+
mentionMode: 'display-name',
289+
mentions: [],
290+
text: 'Hello <a data-hyp-mention="">@User ID Missing</a>',
291+
expectedResult: 'Hello @[User ID Missing]',
292+
},
293+
].forEach(({ text, mentionMode, mentions, expectedResult }) => {
294+
it('replaces mention tag with current username or display name', () => {
295+
assert.equal(
296+
unwrapMentions({ text, mentionMode, mentions }),
297+
expectedResult,
298+
);
215299
});
216300
});
217301
});

0 commit comments

Comments
 (0)