Add emoji from Twemoji 15.0 to the emoji picker/completion (#33395)
This commit is contained in:
		@@ -63,6 +63,7 @@ docker-compose.override.yml
 | 
			
		||||
 | 
			
		||||
# Ignore emoji map file
 | 
			
		||||
/app/javascript/mastodon/features/emoji/emoji_map.json
 | 
			
		||||
/app/javascript/mastodon/features/emoji/emoji_sheet.json
 | 
			
		||||
 | 
			
		||||
# Ignore locale files
 | 
			
		||||
/app/javascript/mastodon/locales/*.json
 | 
			
		||||
 
 | 
			
		||||
@@ -12,11 +12,14 @@ import Overlay from 'react-overlays/Overlay';
 | 
			
		||||
 | 
			
		||||
import MoodIcon from '@/material-icons/400-20px/mood.svg?react';
 | 
			
		||||
import { IconButton } from 'mastodon/components/icon_button';
 | 
			
		||||
import emojiCompressed from 'mastodon/features/emoji/emoji_compressed';
 | 
			
		||||
import { assetHost } from 'mastodon/utils/config';
 | 
			
		||||
 | 
			
		||||
import { buildCustomEmojis, categoriesFromEmojis } from '../../emoji/emoji';
 | 
			
		||||
import { EmojiPicker as EmojiPickerAsync } from '../../ui/util/async-components';
 | 
			
		||||
 | 
			
		||||
const nimblePickerData = emojiCompressed[5];
 | 
			
		||||
 | 
			
		||||
const messages = defineMessages({
 | 
			
		||||
  emoji: { id: 'emoji_button.label', defaultMessage: 'Insert emoji' },
 | 
			
		||||
  emoji_search: { id: 'emoji_button.search', defaultMessage: 'Search...' },
 | 
			
		||||
@@ -37,15 +40,18 @@ let EmojiPicker, Emoji; // load asynchronously
 | 
			
		||||
 | 
			
		||||
const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true;
 | 
			
		||||
 | 
			
		||||
const backgroundImageFn = () => `${assetHost}/emoji/sheet_13.png`;
 | 
			
		||||
const backgroundImageFn = () => `${assetHost}/emoji/sheet_15.png`;
 | 
			
		||||
 | 
			
		||||
const notFoundFn = () => (
 | 
			
		||||
  <div className='emoji-mart-no-results'>
 | 
			
		||||
    <Emoji
 | 
			
		||||
      data={nimblePickerData}
 | 
			
		||||
      emoji='sleuth_or_spy'
 | 
			
		||||
      set='twitter'
 | 
			
		||||
      size={32}
 | 
			
		||||
      sheetSize={32}
 | 
			
		||||
      sheetColumns={62}
 | 
			
		||||
      sheetRows={62}
 | 
			
		||||
      backgroundImageFn={backgroundImageFn}
 | 
			
		||||
    />
 | 
			
		||||
 | 
			
		||||
@@ -67,7 +73,7 @@ class ModifierPickerMenu extends PureComponent {
 | 
			
		||||
    this.props.onSelect(e.currentTarget.getAttribute('data-index') * 1);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  UNSAFE_componentWillReceiveProps (nextProps) {
 | 
			
		||||
  UNSAFE_componentWillReceiveProps(nextProps) {
 | 
			
		||||
    if (nextProps.active) {
 | 
			
		||||
      this.attachListeners();
 | 
			
		||||
    } else {
 | 
			
		||||
@@ -75,7 +81,7 @@ class ModifierPickerMenu extends PureComponent {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentWillUnmount () {
 | 
			
		||||
  componentWillUnmount() {
 | 
			
		||||
    this.removeListeners();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -85,12 +91,12 @@ class ModifierPickerMenu extends PureComponent {
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  attachListeners () {
 | 
			
		||||
  attachListeners() {
 | 
			
		||||
    document.addEventListener('click', this.handleDocumentClick, { capture: true });
 | 
			
		||||
    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  removeListeners () {
 | 
			
		||||
  removeListeners() {
 | 
			
		||||
    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
 | 
			
		||||
    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
 | 
			
		||||
  }
 | 
			
		||||
@@ -99,17 +105,17 @@ class ModifierPickerMenu extends PureComponent {
 | 
			
		||||
    this.node = c;
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
  render() {
 | 
			
		||||
    const { active } = this.props;
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className='emoji-picker-dropdown__modifiers__menu' style={{ display: active ? 'block' : 'none' }} ref={this.setRef}>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={1}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={1} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={2}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={2} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={3}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={3} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={4}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={4} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={5}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={5} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={6}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={6} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={1}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={1} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={2}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={2} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={3}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={3} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={4}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={4} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={5}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={5} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
        <button type='button' onClick={this.handleClick} data-index={6}><Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={6} backgroundImageFn={backgroundImageFn} /></button>
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
@@ -139,12 +145,12 @@ class ModifierPicker extends PureComponent {
 | 
			
		||||
    this.props.onClose();
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
  render() {
 | 
			
		||||
    const { active, modifier } = this.props;
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      <div className='emoji-picker-dropdown__modifiers'>
 | 
			
		||||
        <Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={modifier} onClick={this.handleClick} backgroundImageFn={backgroundImageFn} />
 | 
			
		||||
        <Emoji data={nimblePickerData} sheetColumns={62} sheetRows={62} emoji='fist' set='twitter' size={22} sheetSize={32} skin={modifier} onClick={this.handleClick} backgroundImageFn={backgroundImageFn} />
 | 
			
		||||
        <ModifierPickerMenu active={active} onSelect={this.handleSelect} onClose={this.props.onClose} />
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
@@ -184,7 +190,7 @@ class EmojiPickerMenuImpl extends PureComponent {
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  componentDidMount () {
 | 
			
		||||
  componentDidMount() {
 | 
			
		||||
    document.addEventListener('click', this.handleDocumentClick, { capture: true });
 | 
			
		||||
    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
 | 
			
		||||
 | 
			
		||||
@@ -199,7 +205,7 @@ class EmojiPickerMenuImpl extends PureComponent {
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  componentWillUnmount () {
 | 
			
		||||
  componentWillUnmount() {
 | 
			
		||||
    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
 | 
			
		||||
    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
 | 
			
		||||
  }
 | 
			
		||||
@@ -252,7 +258,7 @@ class EmojiPickerMenuImpl extends PureComponent {
 | 
			
		||||
    this.props.onSkinTone(modifier);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
  render() {
 | 
			
		||||
    const { loading, style, intl, custom_emojis, skinTone, frequentlyUsedEmojis } = this.props;
 | 
			
		||||
 | 
			
		||||
    if (loading) {
 | 
			
		||||
@@ -280,6 +286,9 @@ class EmojiPickerMenuImpl extends PureComponent {
 | 
			
		||||
    return (
 | 
			
		||||
      <div className={classNames('emoji-picker-dropdown__menu', { selecting: modifierOpen })} style={style} ref={this.setRef}>
 | 
			
		||||
        <EmojiPicker
 | 
			
		||||
          data={nimblePickerData}
 | 
			
		||||
          sheetColumns={62}
 | 
			
		||||
          sheetRows={62}
 | 
			
		||||
          perLine={8}
 | 
			
		||||
          emojiSize={22}
 | 
			
		||||
          sheetSize={32}
 | 
			
		||||
@@ -345,7 +354,7 @@ class EmojiPickerDropdown extends PureComponent {
 | 
			
		||||
 | 
			
		||||
      EmojiPickerAsync().then(EmojiMart => {
 | 
			
		||||
        EmojiPicker = EmojiMart.Picker;
 | 
			
		||||
        Emoji       = EmojiMart.Emoji;
 | 
			
		||||
        Emoji = EmojiMart.Emoji;
 | 
			
		||||
 | 
			
		||||
        this.setState({ loading: false });
 | 
			
		||||
      }).catch(() => {
 | 
			
		||||
@@ -386,7 +395,7 @@ class EmojiPickerDropdown extends PureComponent {
 | 
			
		||||
    this.setState({ placement: state.placement });
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  render () {
 | 
			
		||||
  render() {
 | 
			
		||||
    const { intl, onPickEmoji, onSkinTone, skinTone, frequentlyUsedEmojis } = this.props;
 | 
			
		||||
    const title = intl.formatMessage(messages.emoji);
 | 
			
		||||
    const { active, loading, placement } = this.state;
 | 
			
		||||
@@ -403,7 +412,7 @@ class EmojiPickerDropdown extends PureComponent {
 | 
			
		||||
        />
 | 
			
		||||
 | 
			
		||||
        <Overlay show={active} placement={placement} flip target={this.findTarget} popperConfig={{ strategy: 'fixed', onFirstUpdate: this.handleOverlayEnter }}>
 | 
			
		||||
          {({ props, placement })=> (
 | 
			
		||||
          {({ props, placement }) => (
 | 
			
		||||
            <div {...props} style={{ ...props.style }}>
 | 
			
		||||
              <div className={`dropdown-animation ${placement}`}>
 | 
			
		||||
                <EmojiPickerMenu
 | 
			
		||||
 
 | 
			
		||||
@@ -45,6 +45,7 @@ type EmojiCompressed = [
 | 
			
		||||
  Category[],
 | 
			
		||||
  Data['aliases'],
 | 
			
		||||
  EmojisWithoutShortCodes,
 | 
			
		||||
  Data,
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 
 | 
			
		||||
@@ -9,18 +9,91 @@
 | 
			
		||||
 | 
			
		||||
// This version comment should be bumped each time the emoji data is changed
 | 
			
		||||
// to ensure that the prevaled file is regenerated by Babel
 | 
			
		||||
// version: 2
 | 
			
		||||
// version: 3
 | 
			
		||||
 | 
			
		||||
const { emojiIndex } = require('emoji-mart');
 | 
			
		||||
let data = require('emoji-mart/data/all.json');
 | 
			
		||||
// This json file contains the names of the categories.
 | 
			
		||||
const emojiMart5LocalesData = require('@emoji-mart/data/i18n/en.json');
 | 
			
		||||
const emojiMart5Data = require('@emoji-mart/data/sets/15/all.json');
 | 
			
		||||
const { uncompress: emojiMartUncompress } = require('emoji-mart/dist/utils/data');
 | 
			
		||||
const _ = require('lodash');
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
const emojiMap = require('./emoji_map.json');
 | 
			
		||||
// This json file is downloaded from https://github.com/iamcal/emoji-data/
 | 
			
		||||
// and is used to correct the sheet coordinates since we're using that repo's sheet
 | 
			
		||||
const emojiSheetData = require('./emoji_sheet.json');
 | 
			
		||||
const { unicodeToFilename } = require('./unicode_to_filename');
 | 
			
		||||
const { unicodeToUnifiedName } = require('./unicode_to_unified_name');
 | 
			
		||||
 | 
			
		||||
if(data.compressed) {
 | 
			
		||||
  data = emojiMartUncompress(data);
 | 
			
		||||
// Grabbed from `emoji_utils` to avoid circular dependency
 | 
			
		||||
function unifiedToNative(unified) {
 | 
			
		||||
  let unicodes = unified.split('-'),
 | 
			
		||||
    codePoints = unicodes.map((u) => `0x${u}`);
 | 
			
		||||
 | 
			
		||||
  return String.fromCodePoint(...codePoints);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
let data = {
 | 
			
		||||
  compressed: true,
 | 
			
		||||
  categories: emojiMart5Data.categories.map(cat => {
 | 
			
		||||
    return {
 | 
			
		||||
      ...cat,
 | 
			
		||||
      name: emojiMart5LocalesData.categories[cat.id]
 | 
			
		||||
    };
 | 
			
		||||
  }),
 | 
			
		||||
  aliases: emojiMart5Data.aliases,
 | 
			
		||||
  emojis: _(emojiMart5Data.emojis).values().map(emoji => {
 | 
			
		||||
    let skin_variations = {};
 | 
			
		||||
    const unified = emoji.skins[0].unified.toUpperCase();
 | 
			
		||||
    const emojiFromRawData = emojiSheetData.find(e => e.unified === unified);
 | 
			
		||||
 | 
			
		||||
    if (!emojiFromRawData) {
 | 
			
		||||
      return undefined;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (emoji.skins.length > 1) {
 | 
			
		||||
      const [, ...nonDefaultSkins] = emoji.skins;
 | 
			
		||||
      nonDefaultSkins.forEach(skin => {
 | 
			
		||||
        const [matchingRawCodePoints,matchingRawEmoji] = Object.entries(emojiFromRawData.skin_variations).find((pair) => {
 | 
			
		||||
          const [, value] = pair;
 | 
			
		||||
          return value.unified.toLowerCase() === skin.unified;
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        if (matchingRawEmoji && matchingRawCodePoints) {
 | 
			
		||||
          // At the time of writing, the json from `@emoji-mart/data` doesn't have data
 | 
			
		||||
          // for emoji like `woman-heart-woman` with two different skin tones.
 | 
			
		||||
          const skinToneCode = matchingRawCodePoints.split('-')[0];
 | 
			
		||||
          skin_variations[skinToneCode] = {
 | 
			
		||||
            unified: matchingRawEmoji.unified.toUpperCase(),
 | 
			
		||||
            non_qualified: null,
 | 
			
		||||
            sheet_x: matchingRawEmoji.sheet_x,
 | 
			
		||||
            sheet_y: matchingRawEmoji.sheet_y,
 | 
			
		||||
            has_img_twitter: true,
 | 
			
		||||
            native: unifiedToNative(matchingRawEmoji.unified.toUpperCase())
 | 
			
		||||
          };
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return {
 | 
			
		||||
      a: emoji.name,
 | 
			
		||||
      b: unified,
 | 
			
		||||
      c: undefined,
 | 
			
		||||
      f: true,
 | 
			
		||||
      j: [emoji.id, ...emoji.keywords],
 | 
			
		||||
      k: [emojiFromRawData.sheet_x, emojiFromRawData.sheet_y],
 | 
			
		||||
      m: emoji.emoticons?.[0],
 | 
			
		||||
      l: emoji.emoticons,
 | 
			
		||||
      o: emoji.version,
 | 
			
		||||
      id: emoji.id,
 | 
			
		||||
      skin_variations,
 | 
			
		||||
      native: unifiedToNative(unified.toUpperCase())
 | 
			
		||||
    };
 | 
			
		||||
  }).compact().keyBy(e => e.id).mapValues(e => _.omit(e, 'id')).value()
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
if (data.compressed) {
 | 
			
		||||
  emojiMartUncompress(data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const emojiMartData = data;
 | 
			
		||||
@@ -32,15 +105,10 @@ const shortcodeMap   = {};
 | 
			
		||||
const shortCodesToEmojiData = {};
 | 
			
		||||
const emojisWithoutShortCodes = [];
 | 
			
		||||
 | 
			
		||||
Object.keys(emojiIndex.emojis).forEach(key => {
 | 
			
		||||
  let emoji = emojiIndex.emojis[key];
 | 
			
		||||
Object.keys(emojiMart5Data.emojis).forEach(key => {
 | 
			
		||||
  let emoji = emojiMart5Data.emojis[key];
 | 
			
		||||
 | 
			
		||||
  // Emojis with skin tone modifiers are stored like this
 | 
			
		||||
  if (Object.hasOwn(emoji, '1')) {
 | 
			
		||||
    emoji = emoji['1'];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  shortcodeMap[emoji.native] = emoji.id;
 | 
			
		||||
  shortcodeMap[emoji.skins[0].native] = emoji.id;
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const stripModifiers = unicode => {
 | 
			
		||||
@@ -84,13 +152,9 @@ Object.keys(emojiMap).forEach(key => {
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
Object.keys(emojiIndex.emojis).forEach(key => {
 | 
			
		||||
  let emoji = emojiIndex.emojis[key];
 | 
			
		||||
Object.keys(emojiMartData.emojis).forEach(key => {
 | 
			
		||||
  let emoji = emojiMartData.emojis[key];
 | 
			
		||||
 | 
			
		||||
  // Emojis with skin tone modifiers are stored like this
 | 
			
		||||
  if (Object.hasOwn(emoji, '1')) {
 | 
			
		||||
    emoji = emoji['1'];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const { native } = emoji;
 | 
			
		||||
  let { short_names, search, unified } = emojiMartData.emojis[key];
 | 
			
		||||
@@ -135,4 +199,5 @@ module.exports = JSON.parse(JSON.stringify([
 | 
			
		||||
  emojiMartData.categories,
 | 
			
		||||
  emojiMartData.aliases,
 | 
			
		||||
  emojisWithoutShortCodes,
 | 
			
		||||
  emojiMartData
 | 
			
		||||
]));
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
import Emoji from 'emoji-mart/dist-es/components/emoji/emoji';
 | 
			
		||||
import Picker from 'emoji-mart/dist-es/components/picker/picker';
 | 
			
		||||
import Emoji from 'emoji-mart/dist-es/components/emoji/nimble-emoji';
 | 
			
		||||
import Picker from 'emoji-mart/dist-es/components/picker/nimble-picker';
 | 
			
		||||
 | 
			
		||||
export {
 | 
			
		||||
  Picker,
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1
									
								
								app/javascript/mastodon/features/emoji/emoji_sheet.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								app/javascript/mastodon/features/emoji/emoji_sheet.json
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							@@ -48,7 +48,7 @@ module.exports = merge(sharedConfig, {
 | 
			
		||||
      logLevel: 'silent', // do not bother Webpacker, who runs with --json and parses stdout
 | 
			
		||||
    }),
 | 
			
		||||
    new InjectManifest({
 | 
			
		||||
      additionalManifestEntries: ['1f602.svg', 'sheet_13.png'].map((filename) => {
 | 
			
		||||
      additionalManifestEntries: ['1f602.svg', 'sheet_15.png'].map((filename) => {
 | 
			
		||||
        const path = resolve(root, 'public', 'emoji', filename);
 | 
			
		||||
        const body = readFileSync(path);
 | 
			
		||||
        const md5  = createHash('md5');
 | 
			
		||||
 
 | 
			
		||||
@@ -103,4 +103,36 @@ namespace :emojis do
 | 
			
		||||
      gen_border map[emoji], 'white'
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  desc 'Download the JSON sheet data of emojis'
 | 
			
		||||
  task :download_sheet_json do
 | 
			
		||||
    source = 'https://raw.githubusercontent.com/iamcal/emoji-data/refs/tags/v15.1.2/emoji.json'
 | 
			
		||||
    dest   = Rails.root.join('app', 'javascript', 'mastodon', 'features', 'emoji', 'emoji_sheet.json')
 | 
			
		||||
 | 
			
		||||
    puts "Downloading emoji data from source... (#{source})"
 | 
			
		||||
 | 
			
		||||
    res = HTTP.get(source).to_s
 | 
			
		||||
    data = JSON.parse(res)
 | 
			
		||||
 | 
			
		||||
    filtered_data = data.map do |emoji|
 | 
			
		||||
      filtered_item = {
 | 
			
		||||
        'unified' => emoji['unified'],
 | 
			
		||||
        'sheet_x' => emoji['sheet_x'],
 | 
			
		||||
        'sheet_y' => emoji['sheet_y'],
 | 
			
		||||
        'skin_variations' => {},
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      emoji['skin_variations']&.each do |key, variation|
 | 
			
		||||
        filtered_item['skin_variations'][key] = {
 | 
			
		||||
          'unified' => variation['unified'],
 | 
			
		||||
          'sheet_x' => variation['sheet_x'],
 | 
			
		||||
          'sheet_y' => variation['sheet_y'],
 | 
			
		||||
        }
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      filtered_item
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    File.write(dest, JSON.generate(filtered_data))
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -46,6 +46,7 @@
 | 
			
		||||
    "@dnd-kit/core": "^6.1.0",
 | 
			
		||||
    "@dnd-kit/sortable": "^10.0.0",
 | 
			
		||||
    "@dnd-kit/utilities": "^3.2.2",
 | 
			
		||||
    "@emoji-mart/data": "1.2.1",
 | 
			
		||||
    "@formatjs/intl-pluralrules": "^5.2.2",
 | 
			
		||||
    "@gamestdio/websocket": "^0.3.2",
 | 
			
		||||
    "@github/webauthn-json": "^2.1.1",
 | 
			
		||||
 
 | 
			
		||||
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 1.2 MiB  | 
							
								
								
									
										
											BIN
										
									
								
								public/emoji/sheet_15.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/emoji/sheet_15.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 1.3 MiB  | 
@@ -2016,6 +2016,13 @@ __metadata:
 | 
			
		||||
  languageName: node
 | 
			
		||||
  linkType: hard
 | 
			
		||||
 | 
			
		||||
"@emoji-mart/data@npm:1.2.1":
 | 
			
		||||
  version: 1.2.1
 | 
			
		||||
  resolution: "@emoji-mart/data@npm:1.2.1"
 | 
			
		||||
  checksum: 10c0/6784b97bf49a0d3ff110d8447bbd3b0449fcbc497294be3d1c3a6cb1609308776895c7520200be604cbecaa5e172c76927e47f34419c72ba8a76fd4e5a53674b
 | 
			
		||||
  languageName: node
 | 
			
		||||
  linkType: hard
 | 
			
		||||
 | 
			
		||||
"@emotion/babel-plugin@npm:^11.11.0":
 | 
			
		||||
  version: 11.11.0
 | 
			
		||||
  resolution: "@emotion/babel-plugin@npm:11.11.0"
 | 
			
		||||
@@ -2805,6 +2812,7 @@ __metadata:
 | 
			
		||||
    "@dnd-kit/core": "npm:^6.1.0"
 | 
			
		||||
    "@dnd-kit/sortable": "npm:^10.0.0"
 | 
			
		||||
    "@dnd-kit/utilities": "npm:^3.2.2"
 | 
			
		||||
    "@emoji-mart/data": "npm:1.2.1"
 | 
			
		||||
    "@formatjs/cli": "npm:^6.1.1"
 | 
			
		||||
    "@formatjs/intl-pluralrules": "npm:^5.2.2"
 | 
			
		||||
    "@gamestdio/websocket": "npm:^0.3.2"
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user