// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { assert } from 'chai'; import { RefObject } from 'react'; import Delta from 'quill-delta'; import { matchMention } from '../../../quill/mentions/matchers'; import { MemberRepository } from '../../../quill/memberRepository'; import { ConversationType } from '../../../state/ducks/conversations'; class FakeTokenList extends Array { constructor(elements: Array) { super(); elements.forEach(element => this.push(element)); } contains(searchElement: T) { return this.includes(searchElement); } } const createMockElement = ( className: string, dataset: Record ): HTMLElement => (({ classList: new FakeTokenList([className]), dataset, } as unknown) as HTMLElement); const createMockAtMentionElement = ( dataset: Record ): HTMLElement => createMockElement('module-message-body__at-mention', dataset); const createMockMentionBlotElement = ( dataset: Record ): HTMLElement => createMockElement('mention-blot', dataset); const memberMahershala: ConversationType = { id: '555444', uuid: 'abcdefg', title: 'Mahershala Ali', firstName: 'Mahershala', profileName: 'Mahershala A.', type: 'direct', lastUpdated: Date.now(), markedUnread: false, }; const memberShia: ConversationType = { id: '333222', uuid: 'hijklmno', title: 'Shia LaBeouf', firstName: 'Shia', profileName: 'Shia L.', type: 'direct', lastUpdated: Date.now(), markedUnread: false, }; const members: Array = [memberMahershala, memberShia]; const memberRepositoryRef: RefObject = { current: new MemberRepository(members), }; const matcher = matchMention(memberRepositoryRef); interface Mention { uuid: string; title: string; } interface MentionInsert { mention: Mention; } const isMention = (insert?: unknown): insert is MentionInsert => { if (insert) { if (Object.getOwnPropertyNames(insert).includes('mention')) return true; } return false; }; const EMPTY_DELTA = new Delta(); describe('matchMention', () => { it('handles an AtMentionify from clipboard', () => { const result = matcher( createMockAtMentionElement({ id: memberMahershala.id, title: memberMahershala.title, }), EMPTY_DELTA ); const { ops } = result; assert.isNotEmpty(ops); const [op] = ops; const { insert } = op; if (isMention(insert)) { const { title, uuid } = insert.mention; assert.equal(title, memberMahershala.title); assert.equal(uuid, memberMahershala.uuid); } else { assert.fail('insert is invalid'); } }); it('handles an MentionBlot from clipboard', () => { const result = matcher( createMockMentionBlotElement({ uuid: memberMahershala.uuid || '', title: memberMahershala.title, }), EMPTY_DELTA ); const { ops } = result; assert.isNotEmpty(ops); const [op] = ops; const { insert } = op; if (isMention(insert)) { const { title, uuid } = insert.mention; assert.equal(title, memberMahershala.title); assert.equal(uuid, memberMahershala.uuid); } else { assert.fail('insert is invalid'); } }); it('converts a missing AtMentionify to string', () => { const result = matcher( createMockAtMentionElement({ id: 'florp', title: 'Nonexistent', }), EMPTY_DELTA ); const { ops } = result; assert.isNotEmpty(ops); const [op] = ops; const { insert } = op; if (isMention(insert)) { assert.fail('insert is invalid'); } else { assert.equal(insert, '@Nonexistent'); } }); it('converts a missing MentionBlot to string', () => { const result = matcher( createMockMentionBlotElement({ uuid: 'florp', title: 'Nonexistent', }), EMPTY_DELTA ); const { ops } = result; assert.isNotEmpty(ops); const [op] = ops; const { insert } = op; if (isMention(insert)) { assert.fail('insert is invalid'); } else { assert.equal(insert, '@Nonexistent'); } }); it('passes other clipboard elements through', () => { const result = matcher(createMockElement('ignore', {}), EMPTY_DELTA); assert.equal(result, EMPTY_DELTA); }); });