summaryrefslogtreecommitdiff
path: root/packages/frontend/src/components/MkAutocomplete.vue
diff options
context:
space:
mode:
author1Step621 <86859447+1STEP621@users.noreply.github.com>2024-02-23 17:01:42 +0900
committerGitHub <noreply@github.com>2024-02-23 17:01:42 +0900
commita85fccaeea93d610d8cdd52def77851166a9391c (patch)
tree6a254421d72d598ad02b593c8b816fb918ea5900 /packages/frontend/src/components/MkAutocomplete.vue
parentfix(test): Chromaticが落ちているのを一部修正? (#13435) (diff)
downloadmisskey-a85fccaeea93d610d8cdd52def77851166a9391c.tar.gz
misskey-a85fccaeea93d610d8cdd52def77851166a9391c.tar.bz2
misskey-a85fccaeea93d610d8cdd52def77851166a9391c.zip
Fix(frontend): 絵文字オートコンプリートの優先順位がおかしいのを修正 (#13423)
* 絵文字オートコンプリートの優先順位がおかしいのを修正 * update CHANGELOG.md * テストを追加 * lint fix
Diffstat (limited to 'packages/frontend/src/components/MkAutocomplete.vue')
-rw-r--r--packages/frontend/src/components/MkAutocomplete.vue96
1 files changed, 2 insertions, 94 deletions
diff --git a/packages/frontend/src/components/MkAutocomplete.vue b/packages/frontend/src/components/MkAutocomplete.vue
index 412325bfee..cae6bc7111 100644
--- a/packages/frontend/src/components/MkAutocomplete.vue
+++ b/packages/frontend/src/components/MkAutocomplete.vue
@@ -57,18 +57,7 @@ import { i18n } from '@/i18n.js';
import { miLocalStorage } from '@/local-storage.js';
import { customEmojis } from '@/custom-emojis.js';
import { MFM_TAGS, MFM_PARAMS } from '@/const.js';
-
-type EmojiDef = {
- emoji: string;
- name: string;
- url: string;
- aliasOf?: string;
-} | {
- emoji: string;
- name: string;
- aliasOf?: string;
- isCustomEmoji?: true;
-};
+import { searchEmoji, EmojiDef } from '@/scripts/search-emoji.js';
const lib = emojilist.filter(x => x.category !== 'flags');
@@ -249,7 +238,7 @@ function exec() {
return;
}
- emojis.value = emojiAutoComplete(props.q, emojiDb.value);
+ emojis.value = searchEmoji(props.q, emojiDb.value);
} else if (props.type === 'mfmTag') {
if (!props.q || props.q === '') {
mfmTags.value = MFM_TAGS;
@@ -267,87 +256,6 @@ function exec() {
}
}
-type EmojiScore = { emoji: EmojiDef, score: number };
-
-function emojiAutoComplete(query: string | null, emojiDb: EmojiDef[], max = 30): EmojiDef[] {
- if (!query) {
- return [];
- }
-
- const matched = new Map<string, EmojiScore>();
- // 完全一致(エイリアス込み)
- emojiDb.some(x => {
- if (x.name === query && !matched.has(x.aliasOf ?? x.name)) {
- matched.set(x.aliasOf ?? x.name, { emoji: x, score: query.length + 2 });
- }
- return matched.size === max;
- });
-
- // 前方一致(エイリアスなし)
- if (matched.size < max) {
- emojiDb.some(x => {
- if (x.name.startsWith(query) && !x.aliasOf) {
- matched.set(x.name, { emoji: x, score: query.length + 1 });
- }
- return matched.size === max;
- });
- }
-
- // 前方一致(エイリアス込み)
- if (matched.size < max) {
- emojiDb.some(x => {
- if (x.name.startsWith(query) && !matched.has(x.aliasOf ?? x.name)) {
- matched.set(x.aliasOf ?? x.name, { emoji: x, score: query.length });
- }
- return matched.size === max;
- });
- }
-
- // 部分一致(エイリアス込み)
- if (matched.size < max) {
- emojiDb.some(x => {
- if (x.name.includes(query) && !matched.has(x.aliasOf ?? x.name)) {
- matched.set(x.aliasOf ?? x.name, { emoji: x, score: query.length - 1 });
- }
- return matched.size === max;
- });
- }
-
- // 簡易あいまい検索(3文字以上)
- if (matched.size < max && query.length > 3) {
- const queryChars = [...query];
- const hitEmojis = new Map<string, EmojiScore>();
-
- for (const x of emojiDb) {
- // 文字列の位置を進めながら、クエリの文字を順番に探す
-
- let pos = 0;
- let hit = 0;
- for (const c of queryChars) {
- pos = x.name.indexOf(c, pos);
- if (pos <= -1) break;
- hit++;
- }
-
- // 半分以上の文字が含まれていればヒットとする
- if (hit > Math.ceil(queryChars.length / 2) && hit - 2 > (matched.get(x.aliasOf ?? x.name)?.score ?? 0)) {
- hitEmojis.set(x.aliasOf ?? x.name, { emoji: x, score: hit - 2 });
- }
- }
-
- // ヒットしたものを全部追加すると雑多になるので、先頭の6件程度だけにしておく(6件=オートコンプリートのポップアップのサイズ分)
- [...hitEmojis.values()]
- .sort((x, y) => y.score - x.score)
- .slice(0, 6)
- .forEach(it => matched.set(it.emoji.name, it));
- }
-
- return [...matched.values()]
- .sort((x, y) => y.score - x.score)
- .slice(0, max)
- .map(it => it.emoji);
-}
-
function onMousedown(event: Event) {
if (!contains(rootEl.value, event.target) && (rootEl.value !== event.target)) props.close();
}