// This code is largely borrowed from: // https://github.com/missive/emoji-mart/blob/5f2ffcc/src/utils/emoji-index.js import data from './emoji-mart-data-light'; import { getData, getSanitizedData, uniq, intersect } from './emoji-utils'; const originalPool = {}; let index = {}; const emojisList = {}; const emoticonsList = {}; let customEmojisList = []; for (const emoji in data.emojis) { const emojiData = data.emojis[emoji]; const { short_names, emoticons } = emojiData; const id = short_names[0]; if (emoticons) { emoticons.forEach(emoticon => { if (emoticonsList[emoticon]) { return; } emoticonsList[emoticon] = id; }); } emojisList[id] = getSanitizedData(id); originalPool[id] = emojiData; } function clearCustomEmojis(pool) { customEmojisList.forEach((emoji) => { const emojiId = emoji.id || emoji.short_names[0]; delete pool[emojiId]; delete emojisList[emojiId]; }); } export function addCustomToPool(custom, pool = originalPool) { if (customEmojisList.length) clearCustomEmojis(pool); custom.forEach((emoji) => { const emojiId = emoji.id || emoji.short_names[0]; if (emojiId && !pool[emojiId]) { pool[emojiId] = getData(emoji); emojisList[emojiId] = getSanitizedData(emoji); } }); customEmojisList = custom; index = {}; } export function search(value, { emojisToShowFilter, maxResults, include, exclude, custom } = {}) { if (custom !== undefined) { if (customEmojisList !== custom) addCustomToPool(custom, originalPool); } else { custom = []; } maxResults = maxResults || 75; include = include || []; exclude = exclude || []; let results = null, pool = originalPool; if (value.length) { if (value === '-' || value === '-1') { return [emojisList['-1']]; } let values = value.toLowerCase().split(/[\s|,|\-|_]+/), allResults = []; if (values.length > 2) { values = [values[0], values[1]]; } if (include.length || exclude.length) { pool = {}; data.categories.forEach(category => { const isIncluded = include && include.length ? include.includes(category.name.toLowerCase()) : true; const isExcluded = exclude && exclude.length ? exclude.includes(category.name.toLowerCase()) : false; if (!isIncluded || isExcluded) { return; } category.emojis.forEach(emojiId => pool[emojiId] = data.emojis[emojiId]); }); if (custom.length) { const customIsIncluded = include && include.length ? include.includes('custom') : true; const customIsExcluded = exclude && exclude.length ? exclude.includes('custom') : false; if (customIsIncluded && !customIsExcluded) { addCustomToPool(custom, pool); } } } const searchValue = (value) => { let aPool = pool, aIndex = index, length = 0; for (let charIndex = 0; charIndex < value.length; charIndex++) { const char = value[charIndex]; length++; aIndex[char] = aIndex[char] || {}; aIndex = aIndex[char]; if (!aIndex.results) { const scores = {}; aIndex.results = []; aIndex.pool = {}; for (const id in aPool) { const emoji = aPool[id], { search } = emoji, sub = value.substr(0, length), subIndex = search.indexOf(sub); if (subIndex !== -1) { let score = subIndex + 1; if (sub === id) score = 0; aIndex.results.push(emojisList[id]); aIndex.pool[id] = emoji; scores[id] = score; } } aIndex.results.sort((a, b) => { const aScore = scores[a.id], bScore = scores[b.id]; return aScore - bScore; }); } aPool = aIndex.pool; } return aIndex.results; }; if (values.length > 1) { results = searchValue(value); } else { results = []; } allResults = values.map(searchValue).filter(a => a); if (allResults.length > 1) { allResults = intersect.apply(null, allResults); } else if (allResults.length) { allResults = allResults[0]; } results = uniq(results.concat(allResults)); } if (results) { if (emojisToShowFilter) { results = results.filter((result) => emojisToShowFilter(data.emojis[result.id])); } if (results && results.length > maxResults) { results = results.slice(0, maxResults); } } return results; }