115 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
			
		
		
	
	
			115 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| class EmojiFormatter
 | |
|   include RoutingHelper
 | |
| 
 | |
|   DISALLOWED_BOUNDING_REGEX = /[[:alnum:]:]/.freeze
 | |
| 
 | |
|   attr_reader :html, :custom_emojis, :options
 | |
| 
 | |
|   # @param [ActiveSupport::SafeBuffer] html
 | |
|   # @param [Array<CustomEmoji>] custom_emojis
 | |
|   # @param [Hash] options
 | |
|   # @option options [Boolean] :animate
 | |
|   # @option options [String] :style
 | |
|   def initialize(html, custom_emojis, options = {})
 | |
|     raise ArgumentError unless html.html_safe?
 | |
| 
 | |
|     @html = html
 | |
|     @custom_emojis = custom_emojis
 | |
|     @options = options
 | |
|   end
 | |
| 
 | |
|   def to_s
 | |
|     return html if custom_emojis.empty? || html.blank?
 | |
| 
 | |
|     i                     = -1
 | |
|     tag_open_index        = nil
 | |
|     inside_shortname      = false
 | |
|     shortname_start_index = -1
 | |
|     invisible_depth       = 0
 | |
|     last_index            = 0
 | |
|     result                = ''.dup
 | |
| 
 | |
|     while i + 1 < html.size
 | |
|       i += 1
 | |
| 
 | |
|       if invisible_depth.zero? && inside_shortname && html[i] == ':'
 | |
|         inside_shortname = false
 | |
|         shortcode = html[shortname_start_index + 1..i - 1]
 | |
|         char_after = html[i + 1]
 | |
| 
 | |
|         next unless (char_after.nil? || !DISALLOWED_BOUNDING_REGEX.match?(char_after)) && (emoji = emoji_map[shortcode])
 | |
| 
 | |
|         result << html[last_index..shortname_start_index - 1] if shortname_start_index.positive?
 | |
|         result << image_for_emoji(shortcode, emoji)
 | |
|         last_index = i + 1
 | |
|       elsif tag_open_index && html[i] == '>'
 | |
|         tag = html[tag_open_index..i]
 | |
|         tag_open_index = nil
 | |
| 
 | |
|         if invisible_depth.positive?
 | |
|           invisible_depth += count_tag_nesting(tag)
 | |
|         elsif tag == '<span class="invisible">'
 | |
|           invisible_depth = 1
 | |
|         end
 | |
|       elsif html[i] == '<'
 | |
|         tag_open_index = i
 | |
|         inside_shortname = false
 | |
|       elsif !tag_open_index && html[i] == ':' && (i.zero? || !DISALLOWED_BOUNDING_REGEX.match?(html[i - 1]))
 | |
|         inside_shortname = true
 | |
|         shortname_start_index = i
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     result << html[last_index..-1]
 | |
| 
 | |
|     result.html_safe # rubocop:disable Rails/OutputSafety
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def emoji_map
 | |
|     @emoji_map ||= custom_emojis.each_with_object({}) { |e, h| h[e.shortcode] = [full_asset_url(e.image.url), full_asset_url(e.image.url(:static))] }
 | |
|   end
 | |
| 
 | |
|   def count_tag_nesting(tag)
 | |
|     if tag[1] == '/'
 | |
|       -1
 | |
|     elsif tag[-2] == '/'
 | |
|       0
 | |
|     else
 | |
|       1
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def image_for_emoji(shortcode, emoji)
 | |
|     original_url, static_url = emoji
 | |
| 
 | |
|     image_tag(
 | |
|       animate? ? original_url : static_url,
 | |
|       image_attributes.merge(alt: ":#{shortcode}:", title: ":#{shortcode}:", data: image_data_attributes(original_url, static_url))
 | |
|     )
 | |
|   end
 | |
| 
 | |
|   def image_attributes
 | |
|     { rel: 'emoji', draggable: false, width: 16, height: 16, class: image_class_names, style: image_style }
 | |
|   end
 | |
| 
 | |
|   def image_data_attributes(original_url, static_url)
 | |
|     { original: original_url, static: static_url } unless animate?
 | |
|   end
 | |
| 
 | |
|   def image_class_names
 | |
|     animate? ? 'emojione' : 'emojione custom-emoji'
 | |
|   end
 | |
| 
 | |
|   def image_style
 | |
|     @options[:style]
 | |
|   end
 | |
| 
 | |
|   def animate?
 | |
|     @options[:animate] || @options.key?(:style)
 | |
|   end
 | |
| end
 |