// Copyright 2019 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only // Camelcase disabled due to emoji-datasource using snake_case /* eslint-disable camelcase */ import untypedData from 'emoji-datasource'; import emojiRegex from 'emoji-regex'; import { compact, flatMap, get, groupBy, isNumber, keyBy, map, mapValues, sortBy, take, } from 'lodash'; import Fuse from 'fuse.js'; import PQueue from 'p-queue'; import { getOwn } from '../../util/getOwn'; import * as log from '../../logging/log'; import { MINUTE } from '../../util/durations'; import { drop } from '../../util/drop'; import type { LocaleEmojiType } from '../../types/emoji'; export const skinTones = ['1F3FB', '1F3FC', '1F3FD', '1F3FE', '1F3FF']; export type SkinToneKey = '1F3FB' | '1F3FC' | '1F3FD' | '1F3FE' | '1F3FF'; export type SizeClassType = | '' | 'small' | 'medium' | 'large' | 'extra-large' | 'max'; export type EmojiSkinVariation = { unified: string; non_qualified: null; image: string; sheet_x: number; sheet_y: number; added_in: string; has_img_apple: boolean; has_img_google: boolean; has_img_twitter: boolean; has_img_emojione: boolean; has_img_facebook: boolean; has_img_messenger: boolean; }; export type EmojiData = { name: string; unified: string; non_qualified: string | null; docomo: string | null; au: string | null; softbank: string | null; google: string | null; image: string; sheet_x: number; sheet_y: number; short_name: string; short_names: Array; text: string | null; texts: Array | null; category: string; sort_order: number; added_in: string; has_img_apple: boolean; has_img_google: boolean; has_img_twitter: boolean; has_img_facebook: boolean; skin_variations?: { [key: string]: EmojiSkinVariation; }; }; const data = (untypedData as Array) .filter(emoji => emoji.has_img_apple) .map(emoji => // Why this weird map? // the emoji dataset has two separate categories for Emotions and People // yet in our UI we display these as a single merged category. In order // for the emojis to be sorted properly we're manually incrementing the // sort_order for the People & Body emojis so that they fall below the // Smiley & Emotions category. emoji.category === 'People & Body' ? { ...emoji, sort_order: emoji.sort_order + 1000 } : emoji ); const ROOT_PATH = get( typeof window !== 'undefined' ? window : null, 'ROOT_PATH', '' ); const makeImagePath = (src: string) => { return `${ROOT_PATH}node_modules/emoji-datasource-apple/img/apple/64/${src}`; }; const imageQueue = new PQueue({ concurrency: 10, timeout: MINUTE * 30, throwOnTimeout: true, }); const images = new Set(); export const preloadImages = async (): Promise => { // Preload images const preload = async (src: string) => new Promise((resolve, reject) => { const img = new Image(); img.onload = resolve; img.onerror = reject; img.src = src; images.add(img); setTimeout(reject, 5000); }); log.info('Preloading emoji images'); const start = Date.now(); data.forEach(emoji => { drop(imageQueue.add(() => preload(makeImagePath(emoji.image)))); if (emoji.skin_variations) { Object.values(emoji.skin_variations).forEach(variation => { drop(imageQueue.add(() => preload(makeImagePath(variation.image)))); }); } }); await imageQueue.onEmpty(); const end = Date.now(); log.info(`Done preloading emoji images in ${end - start}ms`); }; const dataByShortName = keyBy(data, 'short_name'); const imageByEmoji: { [key: string]: string } = {}; const dataByEmoji: { [key: string]: EmojiData } = {}; export const dataByCategory = mapValues( groupBy(data, ({ category }) => { if (category === 'Activities') { return 'activity'; } if (category === 'Animals & Nature') { return 'animal'; } if (category === 'Flags') { return 'flag'; } if (category === 'Food & Drink') { return 'food'; } if (category === 'Objects') { return 'object'; } if (category === 'Travel & Places') { return 'travel'; } if (category === 'Smileys & Emotion') { return 'emoji'; } if (category === 'People & Body') { return 'emoji'; } if (category === 'Symbols') { return 'symbol'; } return 'misc'; }), arr => sortBy(arr, 'sort_order') ); export function getEmojiData( shortName: keyof typeof dataByShortName, skinTone?: SkinToneKey | number ): EmojiData | EmojiSkinVariation { const base = dataByShortName[shortName]; if (skinTone && base.skin_variations) { const variation = isNumber(skinTone) ? skinTones[skinTone - 1] : skinTone; if (base.skin_variations[variation]) { return base.skin_variations[variation]; } // For emojis that have two people in them which can have diff skin tones // the Map is of SkinTone-SkinTone. If we don't find the correct skin tone // in the list of variations then we assume it is one of those double skin // emojis and we default to both people having same skin. return base.skin_variations[`${variation}-${variation}`]; } return base; } export function getImagePath( shortName: keyof typeof dataByShortName, skinTone?: SkinToneKey | number ): string { const emojiData = getEmojiData(shortName, skinTone); return makeImagePath(emojiData.image); } export type SearchFnType = (query: string, count?: number) => Array; export type SearchEmojiListType = ReadonlyArray< Pick >; type CachedSearchFnType = Readonly<{ localeEmoji: SearchEmojiListType; fn: SearchFnType; }>; let cachedSearchFn: CachedSearchFnType | undefined; export function createSearch(localeEmoji: SearchEmojiListType): SearchFnType { if (cachedSearchFn && cachedSearchFn.localeEmoji === localeEmoji) { return cachedSearchFn.fn; } const knownSet = new Set(); const knownEmoji = localeEmoji.filter(({ shortName }) => { knownSet.add(shortName); return dataByShortName[shortName] != null; }); for (const entry of data) { if (!knownSet.has(entry.short_name)) { knownEmoji.push({ shortName: entry.short_name, rank: 0, tags: entry.short_names, }); } } let maxShortNameLength = 0; for (const { shortName } of knownEmoji) { maxShortNameLength = Math.max(maxShortNameLength, shortName.length); } const fuse = new Fuse(knownEmoji, { shouldSort: false, threshold: 0.2, minMatchCharLength: 1, keys: ['shortName', 'tags'], includeScore: true, }); const fuseExactPrefix = new Fuse(knownEmoji, { // We re-rank and sort manually below shouldSort: false, threshold: 0, // effectively a prefix search minMatchCharLength: 2, keys: ['shortName', 'tags'], includeScore: true, }); const fn = (query: string, count = 0): Array => { // when we only have 2 characters, do an exact prefix match // to avoid matching on emoticon, like :-P const fuseIndex = query.length === 2 ? fuseExactPrefix : fuse; const rawResults = fuseIndex.search(query.substr(0, 32)); const rankedResults = rawResults.map(entry => { const rank = entry.item.rank || 1e9; // Rank exact prefix matches in [0,1] range if (entry.item.shortName.startsWith(query)) { return { score: entry.item.shortName.length / maxShortNameLength, item: entry.item, }; } // Other matches in [1,], ordered by score and rank return { score: 1 + (entry.score ?? 0) + rank / knownEmoji.length, item: entry.item, }; }); const results = rankedResults .sort((a, b) => a.score - b.score) .map(result => result.item.shortName); if (count) { return take(results, count); } return results; }; cachedSearchFn = { localeEmoji, fn }; return fn; } const shortNames = new Set([ ...map(data, 'short_name'), ...compact(flatMap(data, 'short_names')), ]); export function isShortName(name: string): boolean { return shortNames.has(name); } export function unifiedToEmoji(unified: string): string { return unified .split('-') .map(c => String.fromCodePoint(parseInt(c, 16))) .join(''); } export function convertShortNameToData( shortName: string, skinTone: number | SkinToneKey = 0 ): EmojiData | undefined { const base = dataByShortName[shortName]; if (!base) { return undefined; } const toneKey = isNumber(skinTone) ? skinTones[skinTone - 1] : skinTone; if (skinTone && base.skin_variations) { const variation = base.skin_variations[toneKey]; if (variation) { return { ...base, ...variation, }; } } return base; } export function convertShortName( shortName: string, skinTone: number | SkinToneKey = 0 ): string { const emojiData = convertShortNameToData(shortName, skinTone); if (!emojiData) { return ''; } return unifiedToEmoji(emojiData.unified); } export function emojiToImage(emoji: string): string | undefined { return getOwn(imageByEmoji, emoji); } export function emojiToData(emoji: string): EmojiData | undefined { return getOwn(dataByEmoji, emoji); } export function getEmojiCount(str: string): number { const regex = emojiRegex(); let match = regex.exec(str); let count = 0; if (!regex.global) { return match ? 1 : 0; } while (match) { count += 1; match = regex.exec(str); } return count; } export function hasNonEmojiText(str: string): boolean { return str.replace(emojiRegex(), '').trim().length > 0; } export function getSizeClass(str: string): SizeClassType { // Do we have non-emoji characters? if (hasNonEmojiText(str)) { return ''; } const emojiCount = getEmojiCount(str); if (emojiCount === 1) { return 'max'; } if (emojiCount === 2) { return 'extra-large'; } if (emojiCount === 3) { return 'large'; } if (emojiCount === 4) { return 'medium'; } if (emojiCount === 5) { return 'small'; } return ''; } data.forEach(emoji => { const { short_name, short_names, skin_variations, image } = emoji; if (short_names) { short_names.forEach(name => { dataByShortName[name] = emoji; }); } imageByEmoji[convertShortName(short_name)] = makeImagePath(image); dataByEmoji[convertShortName(short_name)] = emoji; if (skin_variations) { Object.entries(skin_variations).forEach(([tone, variation]) => { imageByEmoji[convertShortName(short_name, tone as SkinToneKey)] = makeImagePath(variation.image); dataByEmoji[convertShortName(short_name, tone as SkinToneKey)] = emoji; }); } });