230 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# frozen_string_literal: true
 | 
						||
 | 
						||
def gen_border(codepoint, color)
 | 
						||
  input = Rails.public_path.join('emoji', "#{codepoint}.svg")
 | 
						||
  dest = Rails.public_path.join('emoji', "#{codepoint}_border.svg")
 | 
						||
  doc = File.open(input) { |f| Nokogiri::XML(f) }
 | 
						||
  svg = doc.at_css('svg')
 | 
						||
  if svg.key?('viewBox')
 | 
						||
    view_box = svg['viewBox'].split.map(&:to_i)
 | 
						||
    view_box[0] -= 2
 | 
						||
    view_box[1] -= 2
 | 
						||
    view_box[2] += 4
 | 
						||
    view_box[3] += 4
 | 
						||
    svg['viewBox'] = view_box.join(' ')
 | 
						||
  end
 | 
						||
  g = doc.create_element('g')
 | 
						||
  doc.css('svg > *').each do |elem|
 | 
						||
    border_elem = elem.dup
 | 
						||
 | 
						||
    border_elem.delete('fill')
 | 
						||
 | 
						||
    border_elem['stroke'] = color
 | 
						||
    border_elem['stroke-linejoin'] = 'round'
 | 
						||
    border_elem['stroke-width'] = '4px'
 | 
						||
 | 
						||
    g.add_child(border_elem)
 | 
						||
  end
 | 
						||
  svg.prepend_child(g)
 | 
						||
  File.write(dest, doc.to_xml)
 | 
						||
  puts "Wrote bordered #{codepoint}.svg to #{dest}!"
 | 
						||
end
 | 
						||
 | 
						||
def codepoints_to_filename(codepoints)
 | 
						||
  codepoints.downcase.gsub(/\A0+/, '').tr(' ', '-')
 | 
						||
end
 | 
						||
 | 
						||
def codepoints_to_unicode(codepoints)
 | 
						||
  if codepoints.include?(' ')
 | 
						||
    codepoints.split.map(&:hex).pack('U*')
 | 
						||
  else
 | 
						||
    [codepoints.hex].pack('U')
 | 
						||
  end
 | 
						||
end
 | 
						||
 | 
						||
def get_image(row, emoji_base, fallback, compressed)
 | 
						||
  path = emoji_base.join("#{row[compressed ? 'b' : 'unified'].downcase}.svg")
 | 
						||
  path = emoji_base.join("#{row[compressed ? 'c' : 'non_qualified'].downcase.sub(/^00/, '')}.svg") if !path.exist? && row[compressed ? 'c' : 'non_qualified']
 | 
						||
  if path.exist?
 | 
						||
    Vips::Image.new_from_file(path.to_s, dpi: 64)
 | 
						||
  else
 | 
						||
    fallback
 | 
						||
  end
 | 
						||
end
 | 
						||
 | 
						||
def titleize(string)
 | 
						||
  string.humanize.gsub(/\b(?<!['’`()])(?!(and|the|or|with|a)\b)[a-z]/, &:capitalize)
 | 
						||
end
 | 
						||
 | 
						||
namespace :emojis do
 | 
						||
  desc 'Generate a unicode to filename mapping'
 | 
						||
  task :generate do
 | 
						||
    source = 'http://www.unicode.org/Public/emoji/15.1/emoji-test.txt'
 | 
						||
    codes  = []
 | 
						||
    dest   = Rails.root.join('app', 'javascript', 'mastodon', 'features', 'emoji', 'emoji_map.json')
 | 
						||
 | 
						||
    puts "Downloading emojos from source... (#{source})"
 | 
						||
 | 
						||
    HTTP.get(source).to_s.split("\n").each do |line|
 | 
						||
      next if line.start_with? '#'
 | 
						||
 | 
						||
      parts = line.split(';').map(&:strip)
 | 
						||
      next if parts.size < 2
 | 
						||
 | 
						||
      codes << [parts[0], parts[1].start_with?('fully-qualified')]
 | 
						||
    end
 | 
						||
 | 
						||
    grouped_codes = codes.reduce([]) do |agg, current|
 | 
						||
      if current[1]
 | 
						||
        agg << [current[0]]
 | 
						||
      else
 | 
						||
        agg.last << current[0]
 | 
						||
        agg
 | 
						||
      end
 | 
						||
    end
 | 
						||
 | 
						||
    existence_maps = grouped_codes.map { |c| c.index_with { |cc| Rails.public_path.join('emoji', "#{codepoints_to_filename(cc)}.svg").exist? } }
 | 
						||
    map = {}
 | 
						||
 | 
						||
    existence_maps.each do |group|
 | 
						||
      existing_one = group.key(true)
 | 
						||
 | 
						||
      next if existing_one.nil?
 | 
						||
 | 
						||
      group.each_key do |key|
 | 
						||
        map[codepoints_to_unicode(key)] = codepoints_to_filename(existing_one)
 | 
						||
      end
 | 
						||
    end
 | 
						||
 | 
						||
    map = map.sort { |a, b| a[0].size <=> b[0].size }.to_h
 | 
						||
 | 
						||
    File.write(dest, Oj.dump(map))
 | 
						||
    puts "Wrote emojo to destination! (#{dest})"
 | 
						||
  end
 | 
						||
 | 
						||
  desc 'Generate emoji variants with white borders'
 | 
						||
  task :generate_borders do
 | 
						||
    src = Rails.root.join('app', 'javascript', 'mastodon', 'features', 'emoji', 'emoji_map.json')
 | 
						||
    emojis_light = '👽⚾🐔☁️💨🕊️👀🍥👻🐐❕❔⛸️🌩️🔊🔇📃🌧️🐏🍚🍙🐓🐑💀☠️🌨️🔉🔈💬💭🏐🏳️⚪⬜◽◻️▫️🪽🪿'
 | 
						||
    emojis_dark = '🎱🐜⚫🖤⬛◼️◾◼️✒️▪️💣🎳📷📸♣️🕶️✴️🔌💂♀️📽️🍳🦍💂🔪🕳️🕹️🕋🖊️🖋️💂♂️🎤🎓🎥🎼♠️🎩🦃📼📹🎮🐃🏴🐞🕺📱📲🚲🪮🐦⬛'
 | 
						||
 | 
						||
    map = Oj.load(File.read(src))
 | 
						||
 | 
						||
    emojis_light.each_grapheme_cluster do |emoji|
 | 
						||
      gen_border map[emoji], 'black'
 | 
						||
    end
 | 
						||
    emojis_dark.each_grapheme_cluster do |emoji|
 | 
						||
      gen_border map[emoji], 'white'
 | 
						||
    end
 | 
						||
  end
 | 
						||
 | 
						||
  desc 'Generate the JSON emoji data'
 | 
						||
  task :generate_json do
 | 
						||
    data_source = 'https://raw.githubusercontent.com/iamcal/emoji-data/refs/tags/v15.1.2/emoji.json'
 | 
						||
    keyword_source = 'https://raw.githubusercontent.com/muan/emojilib/refs/tags/v3.0.12/dist/emoji-en-US.json'
 | 
						||
    data_dest = Rails.root.join('app', 'javascript', 'mastodon', 'features', 'emoji', 'emoji_data.json')
 | 
						||
 | 
						||
    puts "Downloading emoji data from source... (#{data_source})"
 | 
						||
    res = HTTP.get(data_source).to_s
 | 
						||
    data = JSON.parse(res)
 | 
						||
 | 
						||
    puts "Downloading keyword data from source... (#{keyword_source})"
 | 
						||
    res = HTTP.get(keyword_source).to_s
 | 
						||
    keywords = JSON.parse(res)
 | 
						||
 | 
						||
    puts 'Generating JSON emoji data...'
 | 
						||
 | 
						||
    emoji_data = {
 | 
						||
      compressed: true,
 | 
						||
      categories: [
 | 
						||
        { id: 'smileys', name: 'Smileys & Emotion', emojis: [] },
 | 
						||
        { id: 'people', name: 'People & Body', emojis: [] },
 | 
						||
        { id: 'nature', name: 'Animals & Nature', emojis: [] },
 | 
						||
        { id: 'foods', name: 'Food & Drink', emojis: [] },
 | 
						||
        { id: 'activity', name: 'Activities', emojis: [] },
 | 
						||
        { id: 'places', name: 'Travel & Places', emojis: [] },
 | 
						||
        { id: 'objects', name: 'Objects', emojis: [] },
 | 
						||
        { id: 'symbols', name: 'Symbols', emojis: [] },
 | 
						||
        { id: 'flags', name: 'Flags', emojis: [] },
 | 
						||
      ],
 | 
						||
      emojis: {},
 | 
						||
      aliases: {},
 | 
						||
    }
 | 
						||
 | 
						||
    sorted = data.sort { |a, b| (a['sort_order'] || a['short_name']) - (b['sort_order'] || b['sort_name']) }
 | 
						||
    category_map = emoji_data[:categories].each_with_index.to_h { |c, i| [c[:name], i] }
 | 
						||
 | 
						||
    sorted.each do |emoji|
 | 
						||
      emoji_keywords = keywords[codepoints_to_unicode(emoji['unified'].downcase)]
 | 
						||
 | 
						||
      single_emoji = {
 | 
						||
        a: titleize(emoji['name']), # name
 | 
						||
        b: emoji['unified'], # unified
 | 
						||
        f: true, # has_img_twitter
 | 
						||
        k: [emoji['sheet_x'], emoji['sheet_y']], # sheet
 | 
						||
      }
 | 
						||
 | 
						||
      single_emoji[:c] = emoji['non_qualified'] unless emoji['non_qualified'].nil? # non_qualified
 | 
						||
      single_emoji[:j] = emoji_keywords.filter { |k| k != emoji['short_name'] } if emoji_keywords.present? # keywords
 | 
						||
      single_emoji[:l] = emoji['texts'] if emoji['texts'].present? # emoticons
 | 
						||
      single_emoji[:m] = emoji['text'] if emoji['text'].present? # text
 | 
						||
      single_emoji[:skin_variations] = emoji['skin_variations'] if emoji['skin_variations'].present?
 | 
						||
 | 
						||
      emoji_data[:emojis][emoji['short_name']] = single_emoji
 | 
						||
      emoji_data[:categories][category_map[emoji['category']]][:emojis].push(emoji['short_name']) if emoji['category'] != 'Component'
 | 
						||
 | 
						||
      emoji['short_names'].each do |name|
 | 
						||
        emoji_data[:aliases][name] = emoji['short_name'] unless name == emoji['short_name']
 | 
						||
      end
 | 
						||
    end
 | 
						||
 | 
						||
    smileys = emoji_data[:categories][0]
 | 
						||
    people = emoji_data[:categories][1]
 | 
						||
    smileys_and_people = { id: 'people', name: 'Smileys & People', emojis: [*smileys[:emojis][..128], *people[:emojis], *smileys[:emojis][129..]] }
 | 
						||
    emoji_data[:categories].unshift(smileys_and_people)
 | 
						||
    emoji_data[:categories] -= emoji_data[:categories][1, 2]
 | 
						||
 | 
						||
    File.write(data_dest, JSON.generate(emoji_data))
 | 
						||
  end
 | 
						||
 | 
						||
  desc 'Generate a spritesheet of emojis'
 | 
						||
  task :generate_emoji_sheet do
 | 
						||
    require 'vips'
 | 
						||
 | 
						||
    src = Rails.root.join('app', 'javascript', 'mastodon', 'features', 'emoji', 'emoji_data.json')
 | 
						||
    sheet = Oj.load(File.read(src))
 | 
						||
 | 
						||
    max = 0
 | 
						||
    sheet['emojis'].each_value do |row|
 | 
						||
      max = [max, row['k'][0], row['k'][1]].max
 | 
						||
      next if row['skin_variations'].blank?
 | 
						||
 | 
						||
      row['skin_variations'].each_value do |variation|
 | 
						||
        max = [max, variation['sheet_x'], variation['sheet_y']].max
 | 
						||
      end
 | 
						||
    end
 | 
						||
 | 
						||
    size = max + 1
 | 
						||
 | 
						||
    puts 'Generating spritesheet...'
 | 
						||
 | 
						||
    emoji_base = Rails.public_path.join('emoji')
 | 
						||
    fallback = Vips::Image.new_from_file(emoji_base.join('2753.svg').to_s, dpi: 64)
 | 
						||
    comp = Array.new(size) do
 | 
						||
      Array.new(size, 0)
 | 
						||
    end
 | 
						||
 | 
						||
    sheet['emojis'].each_value do |row|
 | 
						||
      comp[row['k'][1]][row['k'][0]] = get_image(row, emoji_base, fallback, true)
 | 
						||
      next if row['skin_variations'].blank?
 | 
						||
 | 
						||
      row['skin_variations'].each_value do |variation|
 | 
						||
        comp[variation['sheet_y']][variation['sheet_x']] = get_image(variation, emoji_base, fallback, false)
 | 
						||
      end
 | 
						||
    end
 | 
						||
 | 
						||
    joined = Vips::Image.arrayjoin(comp.flatten, across: size, hspacing: 34, halign: :centre, vspacing: 34, valign: :centre)
 | 
						||
    joined.write_to_file(emoji_base.join('sheet_15_1.png').to_s, palette: true, dither: 0, Q: 100)
 | 
						||
  end
 | 
						||
end
 |