259 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			259 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // This code is largely borrowed from:
 | |
| // https://github.com/missive/emoji-mart/blob/5f2ffcc/src/utils/index.js
 | |
| 
 | |
| import * as data from './emoji_mart_data_light';
 | |
| 
 | |
| const buildSearch = (data) => {
 | |
|   const search = [];
 | |
| 
 | |
|   let addToSearch = (strings, split) => {
 | |
|     if (!strings) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     (Array.isArray(strings) ? strings : [strings]).forEach((string) => {
 | |
|       (split ? string.split(/[-|_|\s]+/) : [string]).forEach((s) => {
 | |
|         s = s.toLowerCase();
 | |
| 
 | |
|         if (search.indexOf(s) === -1) {
 | |
|           search.push(s);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   };
 | |
| 
 | |
|   addToSearch(data.short_names, true);
 | |
|   addToSearch(data.name, true);
 | |
|   addToSearch(data.keywords, false);
 | |
|   addToSearch(data.emoticons, false);
 | |
| 
 | |
|   return search.join(',');
 | |
| };
 | |
| 
 | |
| const _String = String;
 | |
| 
 | |
| const stringFromCodePoint = _String.fromCodePoint || function () {
 | |
|   let MAX_SIZE = 0x4000;
 | |
|   let codeUnits = [];
 | |
|   let highSurrogate;
 | |
|   let lowSurrogate;
 | |
|   let index = -1;
 | |
|   let length = arguments.length;
 | |
|   if (!length) {
 | |
|     return '';
 | |
|   }
 | |
|   let result = '';
 | |
|   while (++index < length) {
 | |
|     let codePoint = Number(arguments[index]);
 | |
|     if (
 | |
|       !isFinite(codePoint) ||       // `NaN`, `+Infinity`, or `-Infinity`
 | |
|       codePoint < 0 ||              // not a valid Unicode code point
 | |
|       codePoint > 0x10FFFF ||       // not a valid Unicode code point
 | |
|       Math.floor(codePoint) !== codePoint // not an integer
 | |
|     ) {
 | |
|       throw RangeError('Invalid code point: ' + codePoint);
 | |
|     }
 | |
|     if (codePoint <= 0xFFFF) { // BMP code point
 | |
|       codeUnits.push(codePoint);
 | |
|     } else { // Astral code point; split in surrogate halves
 | |
|       // http://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae
 | |
|       codePoint -= 0x10000;
 | |
|       highSurrogate = (codePoint >> 10) + 0xD800;
 | |
|       lowSurrogate = (codePoint % 0x400) + 0xDC00;
 | |
|       codeUnits.push(highSurrogate, lowSurrogate);
 | |
|     }
 | |
|     if (index + 1 === length || codeUnits.length > MAX_SIZE) {
 | |
|       result += String.fromCharCode.apply(null, codeUnits);
 | |
|       codeUnits.length = 0;
 | |
|     }
 | |
|   }
 | |
|   return result;
 | |
| };
 | |
| 
 | |
| 
 | |
| const _JSON = JSON;
 | |
| 
 | |
| const COLONS_REGEX = /^(?::([^:]+):)(?::skin-tone-(\d):)?$/;
 | |
| const SKINS = [
 | |
|   '1F3FA', '1F3FB', '1F3FC',
 | |
|   '1F3FD', '1F3FE', '1F3FF',
 | |
| ];
 | |
| 
 | |
| function unifiedToNative(unified) {
 | |
|   let unicodes = unified.split('-'),
 | |
|     codePoints = unicodes.map((u) => `0x${u}`);
 | |
| 
 | |
|   return stringFromCodePoint.apply(null, codePoints);
 | |
| }
 | |
| 
 | |
| function sanitize(emoji) {
 | |
|   let { name, short_names, skin_tone, skin_variations, emoticons, unified, custom, imageUrl } = emoji,
 | |
|     id = emoji.id || short_names[0],
 | |
|     colons = `:${id}:`;
 | |
| 
 | |
|   if (custom) {
 | |
|     return {
 | |
|       id,
 | |
|       name,
 | |
|       colons,
 | |
|       emoticons,
 | |
|       custom,
 | |
|       imageUrl,
 | |
|     };
 | |
|   }
 | |
| 
 | |
|   if (skin_tone) {
 | |
|     colons += `:skin-tone-${skin_tone}:`;
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     id,
 | |
|     name,
 | |
|     colons,
 | |
|     emoticons,
 | |
|     unified: unified.toLowerCase(),
 | |
|     skin: skin_tone || (skin_variations ? 1 : null),
 | |
|     native: unifiedToNative(unified),
 | |
|   };
 | |
| }
 | |
| 
 | |
| function getSanitizedData() {
 | |
|   return sanitize(getData(...arguments));
 | |
| }
 | |
| 
 | |
| function getData(emoji, skin, set) {
 | |
|   let emojiData = {};
 | |
| 
 | |
|   if (typeof emoji === 'string') {
 | |
|     let matches = emoji.match(COLONS_REGEX);
 | |
| 
 | |
|     if (matches) {
 | |
|       emoji = matches[1];
 | |
| 
 | |
|       if (matches[2]) {
 | |
|         skin = parseInt(matches[2]);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (Object.hasOwn(data.short_names, emoji)) {
 | |
|       emoji = data.short_names[emoji];
 | |
|     }
 | |
| 
 | |
|     if (Object.hasOwn(data.emojis, emoji)) {
 | |
|       emojiData = data.emojis[emoji];
 | |
|     }
 | |
|   } else if (emoji.id) {
 | |
|     if (Object.hasOwn(data.short_names, emoji.id)) {
 | |
|       emoji.id = data.short_names[emoji.id];
 | |
|     }
 | |
| 
 | |
|     if (Object.hasOwn(data.emojis, emoji.id)) {
 | |
|       emojiData = data.emojis[emoji.id];
 | |
|       skin = skin || emoji.skin;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!Object.keys(emojiData).length) {
 | |
|     emojiData = emoji;
 | |
|     emojiData.custom = true;
 | |
| 
 | |
|     if (!emojiData.search) {
 | |
|       emojiData.search = buildSearch(emoji);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   emojiData.emoticons = emojiData.emoticons || [];
 | |
|   emojiData.variations = emojiData.variations || [];
 | |
| 
 | |
|   if (emojiData.skin_variations && skin > 1 && set) {
 | |
|     emojiData = JSON.parse(_JSON.stringify(emojiData));
 | |
| 
 | |
|     let skinKey = SKINS[skin - 1],
 | |
|       variationData = emojiData.skin_variations[skinKey];
 | |
| 
 | |
|     if (!variationData.variations && emojiData.variations) {
 | |
|       delete emojiData.variations;
 | |
|     }
 | |
| 
 | |
|     if (variationData[`has_img_${set}`]) {
 | |
|       emojiData.skin_tone = skin;
 | |
| 
 | |
|       for (let k in variationData) {
 | |
|         let v = variationData[k];
 | |
|         emojiData[k] = v;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (emojiData.variations && emojiData.variations.length) {
 | |
|     emojiData = JSON.parse(_JSON.stringify(emojiData));
 | |
|     emojiData.unified = emojiData.variations.shift();
 | |
|   }
 | |
| 
 | |
|   return emojiData;
 | |
| }
 | |
| 
 | |
| function uniq(arr) {
 | |
|   return arr.reduce((acc, item) => {
 | |
|     if (acc.indexOf(item) === -1) {
 | |
|       acc.push(item);
 | |
|     }
 | |
|     return acc;
 | |
|   }, []);
 | |
| }
 | |
| 
 | |
| function intersect(a, b) {
 | |
|   const uniqA = uniq(a);
 | |
|   const uniqB = uniq(b);
 | |
| 
 | |
|   return uniqA.filter(item => uniqB.indexOf(item) >= 0);
 | |
| }
 | |
| 
 | |
| function deepMerge(a, b) {
 | |
|   let o = {};
 | |
| 
 | |
|   for (let key in a) {
 | |
|     let originalValue = a[key],
 | |
|       value = originalValue;
 | |
| 
 | |
|     if (Object.hasOwn(b, key)) {
 | |
|       value = b[key];
 | |
|     }
 | |
| 
 | |
|     if (typeof value === 'object') {
 | |
|       value = deepMerge(originalValue, value);
 | |
|     }
 | |
| 
 | |
|     o[key] = value;
 | |
|   }
 | |
| 
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| // https://github.com/sonicdoe/measure-scrollbar
 | |
| function measureScrollbar() {
 | |
|   const div = document.createElement('div');
 | |
| 
 | |
|   div.style.width = '100px';
 | |
|   div.style.height = '100px';
 | |
|   div.style.overflow = 'scroll';
 | |
|   div.style.position = 'absolute';
 | |
|   div.style.top = '-9999px';
 | |
| 
 | |
|   document.body.appendChild(div);
 | |
|   const scrollbarWidth = div.offsetWidth - div.clientWidth;
 | |
|   document.body.removeChild(div);
 | |
| 
 | |
|   return scrollbarWidth;
 | |
| }
 | |
| 
 | |
| export {
 | |
|   getData,
 | |
|   getSanitizedData,
 | |
|   uniq,
 | |
|   intersect,
 | |
|   deepMerge,
 | |
|   unifiedToNative,
 | |
|   measureScrollbar,
 | |
| };
 |