2016-06-02 11:05:42 +05:30
|
|
|
namespace :gemojione do
|
|
|
|
desc 'Generates Emoji SHA256 digests'
|
2017-08-17 22:00:37 +05:30
|
|
|
task digests: ['yarn:check', 'environment'] do
|
2016-06-02 11:05:42 +05:30
|
|
|
require 'digest/sha2'
|
|
|
|
require 'json'
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
# We don't have `node_modules` available in built versions of GitLab
|
|
|
|
FileUtils.cp_r(Rails.root.join('node_modules', 'emoji-unicode-version', 'emoji-unicode-version-map.json'), File.join(Rails.root, 'fixtures', 'emojis'))
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
dir = Gemojione.images_path
|
|
|
|
resultant_emoji_map = {}
|
|
|
|
|
|
|
|
Gitlab::Emoji.emojis.each do |name, emoji_hash|
|
|
|
|
# Ignore aliases
|
|
|
|
unless Gitlab::Emoji.emojis_aliases.key?(name)
|
|
|
|
fpath = File.join(dir, "#{emoji_hash['unicode']}.png")
|
|
|
|
hash_digest = Digest::SHA256.file(fpath).hexdigest
|
|
|
|
|
|
|
|
entry = {
|
|
|
|
category: emoji_hash['category'],
|
|
|
|
moji: emoji_hash['moji'],
|
|
|
|
description: emoji_hash['description'],
|
|
|
|
unicodeVersion: Gitlab::Emoji.emoji_unicode_version(name),
|
|
|
|
digest: hash_digest,
|
|
|
|
}
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
resultant_emoji_map[name] = entry
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
out = File.join(Rails.root, 'fixtures', 'emojis', 'digests.json')
|
|
|
|
|
|
|
|
File.open(out, 'w') do |handle|
|
2017-08-17 22:00:37 +05:30
|
|
|
handle.write(JSON.pretty_generate(resultant_emoji_map))
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# This task will generate a standard and Retina sprite of all of the current
|
|
|
|
# Gemojione Emojis, with the accompanying SCSS map.
|
|
|
|
#
|
|
|
|
# It will not appear in `rake -T` output, and the dependent gems are not
|
|
|
|
# included in the Gemfile by default, because this task will only be needed
|
|
|
|
# occasionally, such as when new Emojis are added to Gemojione.
|
|
|
|
task sprite: :environment do
|
|
|
|
begin
|
|
|
|
require 'sprite_factory'
|
|
|
|
require 'rmagick'
|
|
|
|
rescue LoadError
|
|
|
|
# noop
|
|
|
|
end
|
|
|
|
|
|
|
|
check_requirements!
|
|
|
|
|
|
|
|
SIZE = 20
|
|
|
|
RETINA = SIZE * 2
|
|
|
|
|
2016-08-24 12:49:21 +05:30
|
|
|
# Update these values to the width and height of the spritesheet when
|
|
|
|
# new emoji are added.
|
|
|
|
SPRITESHEET_WIDTH = 860
|
|
|
|
SPRITESHEET_HEIGHT = 840
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
# Setup a map to rename image files
|
|
|
|
emoji_unicode_string_to_name_map = {}
|
|
|
|
Gitlab::Emoji.emojis.each do |name, emoji_hash|
|
|
|
|
# Ignore aliases
|
|
|
|
unless Gitlab::Emoji.emojis_aliases.key?(name)
|
|
|
|
emoji_unicode_string_to_name_map[emoji_hash['unicode']] = name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Copy the Gemojione assets to the temporary folder for renaming
|
|
|
|
emoji_dir = "app/assets/images/emoji"
|
|
|
|
FileUtils.rm_rf(emoji_dir)
|
|
|
|
FileUtils.mkdir_p(emoji_dir, mode: 0700)
|
|
|
|
FileUtils.cp_r(File.join(Gemojione.images_path, '.'), emoji_dir)
|
|
|
|
Dir[File.join(emoji_dir, "**/*.png")].each do |png|
|
|
|
|
image_path = png
|
|
|
|
rename_to_named_emoji_image!(emoji_unicode_string_to_name_map, image_path)
|
|
|
|
end
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
Dir.mktmpdir do |tmpdir|
|
2017-08-17 22:00:37 +05:30
|
|
|
FileUtils.cp_r(File.join(emoji_dir, '.'), tmpdir)
|
2016-06-02 11:05:42 +05:30
|
|
|
|
|
|
|
Dir.chdir(tmpdir) do
|
|
|
|
Dir["**/*.png"].each do |png|
|
2017-08-17 22:00:37 +05:30
|
|
|
tmp_image_path = File.join(tmpdir, png)
|
|
|
|
resize!(tmp_image_path, SIZE)
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
style_path = Rails.root.join(*%w(app assets stylesheets framework emoji-sprites.scss))
|
2016-06-02 11:05:42 +05:30
|
|
|
|
|
|
|
# Combine the resized assets into a packed sprite and re-generate the SCSS
|
|
|
|
SpriteFactory.cssurl = "image-url('$IMAGE')"
|
2017-08-17 22:00:37 +05:30
|
|
|
SpriteFactory.run!(tmpdir, {
|
2016-06-02 11:05:42 +05:30
|
|
|
output_style: style_path,
|
|
|
|
output_image: "app/assets/images/emoji.png",
|
|
|
|
selector: '.emoji-',
|
|
|
|
style: :scss,
|
|
|
|
nocomments: true,
|
|
|
|
pngcrush: true,
|
|
|
|
layout: :packed
|
|
|
|
})
|
|
|
|
|
|
|
|
# SpriteFactory's SCSS is a bit too verbose for our purposes here, so
|
|
|
|
# let's simplify it
|
|
|
|
system(%Q(sed -i '' "s/width: #{SIZE}px; height: #{SIZE}px; background: image-url('emoji.png')/background-position:/" #{style_path}))
|
|
|
|
system(%Q(sed -i '' "s/ no-repeat//" #{style_path}))
|
2017-08-17 22:00:37 +05:30
|
|
|
system(%Q(sed -i '' "s/ 0px/ 0/g" #{style_path}))
|
2016-06-02 11:05:42 +05:30
|
|
|
|
|
|
|
# Append a generic rule that applies to all Emojis
|
|
|
|
File.open(style_path, 'a') do |f|
|
|
|
|
f.puts
|
|
|
|
f.puts <<-CSS.strip_heredoc
|
|
|
|
.emoji-icon {
|
|
|
|
background-image: image-url('emoji.png');
|
|
|
|
background-repeat: no-repeat;
|
2017-08-17 22:00:37 +05:30
|
|
|
color: transparent;
|
|
|
|
text-indent: -99em;
|
2016-06-02 11:05:42 +05:30
|
|
|
height: #{SIZE}px;
|
|
|
|
width: #{SIZE}px;
|
|
|
|
|
|
|
|
@media only screen and (-webkit-min-device-pixel-ratio: 2),
|
|
|
|
only screen and (min--moz-device-pixel-ratio: 2),
|
|
|
|
only screen and (-o-min-device-pixel-ratio: 2/1),
|
|
|
|
only screen and (min-device-pixel-ratio: 2),
|
|
|
|
only screen and (min-resolution: 192dpi),
|
|
|
|
only screen and (min-resolution: 2dppx) {
|
|
|
|
background-image: image-url('emoji@2x.png');
|
2016-08-24 12:49:21 +05:30
|
|
|
background-size: #{SPRITESHEET_WIDTH}px #{SPRITESHEET_HEIGHT}px;
|
2016-06-02 11:05:42 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
CSS
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Now do it again but for Retina
|
|
|
|
Dir.mktmpdir do |tmpdir|
|
|
|
|
# Copy the Gemojione assets to the temporary folder for resizing
|
2017-08-17 22:00:37 +05:30
|
|
|
FileUtils.cp_r(File.join(emoji_dir, '.'), tmpdir)
|
2016-06-02 11:05:42 +05:30
|
|
|
|
|
|
|
Dir.chdir(tmpdir) do
|
|
|
|
Dir["**/*.png"].each do |png|
|
2017-08-17 22:00:37 +05:30
|
|
|
tmp_image_path = File.join(tmpdir, png)
|
|
|
|
resize!(tmp_image_path, RETINA)
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Combine the resized assets into a packed sprite and re-generate the SCSS
|
2017-08-17 22:00:37 +05:30
|
|
|
SpriteFactory.run!(tmpdir, {
|
2016-06-02 11:05:42 +05:30
|
|
|
output_image: "app/assets/images/emoji@2x.png",
|
|
|
|
style: false,
|
|
|
|
nocomments: true,
|
|
|
|
pngcrush: true,
|
|
|
|
layout: :packed
|
|
|
|
})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def check_requirements!
|
|
|
|
return if defined?(SpriteFactory) && defined?(Magick)
|
|
|
|
|
|
|
|
puts <<-MSG.strip_heredoc
|
|
|
|
This task is disabled by default and should only be run when the Gemojione
|
|
|
|
gem is updated with new Emojis.
|
|
|
|
|
|
|
|
To enable this task, *temporarily* add the following lines to Gemfile and
|
|
|
|
re-bundle:
|
|
|
|
|
|
|
|
gem 'sprite-factory'
|
|
|
|
gem 'rmagick'
|
|
|
|
MSG
|
|
|
|
|
|
|
|
exit 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def resize!(image_path, size)
|
|
|
|
# Resize the image in-place, save it, and free the object
|
|
|
|
image = Magick::Image.read(image_path).first
|
|
|
|
image.resize!(size, size)
|
|
|
|
image.write(image_path) { self.quality = 100 }
|
|
|
|
image.destroy!
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
EMOJI_IMAGE_PATH_RE = /(.*?)(([0-9a-f]-?)+)\.png$/i
|
|
|
|
def rename_to_named_emoji_image!(emoji_unicode_string_to_name_map, image_path)
|
|
|
|
# Rename file from unicode to emoji name
|
|
|
|
matches = EMOJI_IMAGE_PATH_RE.match(image_path)
|
|
|
|
preceding_path = matches[1]
|
|
|
|
unicode_string = matches[2]
|
|
|
|
name = emoji_unicode_string_to_name_map[unicode_string]
|
|
|
|
if name
|
|
|
|
new_png_path = File.join(preceding_path, "#{name}.png")
|
|
|
|
FileUtils.mv(image_path, new_png_path)
|
|
|
|
new_png_path
|
|
|
|
else
|
|
|
|
puts "Warning: emoji_unicode_string_to_name_map missing entry for #{unicode_string}. Full path: #{image_path}"
|
|
|
|
end
|
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|