import {emojiKeys, emojiHTML, emojiString} from './emoji.ts'; import {html, htmlRaw} from '../utils/html.ts'; import {fetchMentions} from '../utils/match.ts'; import type {TributeCollection} from 'tributejs'; import type {Mention} from '../types.ts'; export async function attachTribute(element: HTMLElement) { const {default: Tribute} = await import('tributejs'); const mentionsUrl = element.closest('[data-mentions-url]')?.getAttribute('data-mentions-url'); const emojiCollection: TributeCollection = { // emojis trigger: ':', requireLeadingSpace: true, values: (query: string, cb: (matches: Array) => void) => { const matches = []; for (const name of emojiKeys) { if (name.includes(query)) { matches.push(name); if (matches.length > 5) break; } } cb(matches); }, lookup: (item) => item, selectTemplate: (item) => { if (item === undefined) return ''; return emojiString(item.original) ?? ''; }, menuItemTemplate: (item) => { return html`
${htmlRaw(emojiHTML(item.original))}${item.original}
`; }, }; const mentionCollection: TributeCollection = { values: async (_query: string, cb: (matches: Mention[]) => void) => { // eslint-disable-line @typescript-eslint/no-misused-promises cb(mentionsUrl ? await fetchMentions(mentionsUrl) : []); }, requireLeadingSpace: true, menuItemTemplate: (item) => { const fullNameHtml = item.original.fullname && item.original.fullname !== '' ? html`${item.original.fullname}` : ''; return html`
${item.original.name} ${htmlRaw(fullNameHtml)}
`; }, }; const tribute = new Tribute({ collection: [ emojiCollection as TributeCollection, mentionCollection as TributeCollection, ], noMatchTemplate: () => '', }); tribute.attach(element); return tribute; }