Files

82 lines
2.9 KiB
TypeScript
Raw Permalink Normal View History

import emojis from '../../../assets/emoji.json' with {type: 'json'};
2024-10-29 10:20:49 +01:00
import {GET} from '../modules/fetch.ts';
2026-03-07 21:37:37 +01:00
import {showErrorToast} from '../modules/toast.ts';
import {parseIssuePageInfo} from '../utils.ts';
import type {Issue, Mention} from '../types.ts';
2023-04-22 18:32:34 +03:00
const maxMatches = 6;
2024-10-29 10:20:49 +01:00
function sortAndReduce<T>(map: Map<T, number>): T[] {
2023-05-18 03:14:31 +02:00
const sortedMap = new Map(Array.from(map.entries()).sort((a, b) => a[1] - b[1]));
2023-04-22 18:32:34 +03:00
return Array.from(sortedMap.keys()).slice(0, maxMatches);
}
2024-08-28 18:32:38 +02:00
export function matchEmoji(queryText: string): string[] {
2023-04-22 18:32:34 +03:00
const query = queryText.toLowerCase().replaceAll('_', ' ');
if (!query) return emojis.slice(0, maxMatches).map((e) => e.aliases[0]);
// results is a map of weights, lower is better
2024-08-28 18:32:38 +02:00
const results = new Map<string, number>();
2023-04-22 18:32:34 +03:00
for (const {aliases} of emojis) {
const mainAlias = aliases[0];
for (const [aliasIndex, alias] of aliases.entries()) {
const index = alias.replaceAll('_', ' ').indexOf(query);
if (index === -1) continue;
const existing = results.get(mainAlias);
const rankedIndex = index + aliasIndex;
results.set(mainAlias, existing ? existing - rankedIndex : rankedIndex);
}
}
return sortAndReduce(results);
}
2026-03-07 21:37:37 +01:00
let cachedMentionsPromise: Promise<Mention[]> | undefined;
let cachedMentionsUrl: string;
export function fetchMentions(mentionsUrl: string): Promise<Mention[]> {
if (cachedMentionsPromise && cachedMentionsUrl === mentionsUrl) {
return cachedMentionsPromise;
}
cachedMentionsUrl = mentionsUrl;
cachedMentionsPromise = (async () => {
try {
const issueIndex = parseIssuePageInfo().issueNumber;
const query = issueIndex ? `?issue_index=${issueIndex}` : '';
const res = await GET(`${mentionsUrl}${query}`);
if (!res.ok) throw new Error(res.statusText);
return await res.json() as Mention[];
} catch (e) {
showErrorToast(`Failed to load mentions: ${e}`);
return [];
}
})();
return cachedMentionsPromise;
}
export async function matchMention(mentionsUrl: string, queryText: string): Promise<Mention[]> {
const values = await fetchMentions(mentionsUrl);
2023-04-22 18:32:34 +03:00
const query = queryText.toLowerCase();
// results is a map of weights, lower is better
2026-03-07 21:37:37 +01:00
const results = new Map<Mention, number>();
for (const obj of values) {
2023-04-22 18:32:34 +03:00
const index = obj.key.toLowerCase().indexOf(query);
if (index === -1) continue;
const existing = results.get(obj);
results.set(obj, existing ? existing - index : index);
}
return sortAndReduce(results);
}
2024-10-29 10:20:49 +01:00
export async function matchIssue(owner: string, repo: string, issueIndexStr: string, query: string): Promise<Issue[]> {
const res = await GET(`${window.config.appSubUrl}/${owner}/${repo}/issues/suggestions?q=${encodeURIComponent(query)}`);
const issues: Issue[] = await res.json();
2024-10-31 04:06:36 +08:00
const issueNumber = parseInt(issueIndexStr);
2024-10-29 10:20:49 +01:00
// filter out issue with same id
2024-10-31 04:06:36 +08:00
return issues.filter((i) => i.number !== issueNumber);
2024-10-29 10:20:49 +01:00
}