# frozen_string_literal: true require 'singleton' class Formatter include Singleton include RoutingHelper include ActionView::Helpers::TextHelper include ActionView::Helpers::SanitizeHelper AUTOLINK_RE = /https?:\/\/([\S]+\.[!#$&-;=?-[\]_a-z~]|%[\w\d]{2}]+[\w])/i def format(status) return reformat(status.content) unless status.local? html = status.text html = encode(html) html = simple_format(html, {}, sanitize: false) html = html.gsub(/\n/, '') html = link_urls(html) html = link_mentions(html, status.mentions) html = link_hashtags(html) html.html_safe # rubocop:disable Rails/OutputSafety end def reformat(html) sanitize(html, tags: %w(a br p span), attributes: %w(href rel class)) end def simplified_format(account) return reformat(account.note) unless account.local? html = encode(account.note) html = link_urls(html) html = link_hashtags(html) html.html_safe # rubocop:disable Rails/OutputSafety end private def encode(html) HTMLEntities.new.encode(html) end def link_urls(html) Twitter::Autolink.auto_link_urls(html, url_target: '_blank', link_attribute_block: lambda { |_, a| a[:rel] << ' noopener' }, link_text_block: lambda { |_, text| link_html(text) }) end def link_mentions(html, mentions) html.gsub(Account::MENTION_RE) do |match| acct = Account::MENTION_RE.match(match)[1] mention = mentions.find { |item| item.account.acct.casecmp(acct).zero? } mention.nil? ? match : mention_html(match, mention.account) end end def link_hashtags(html) html.gsub(Tag::HASHTAG_RE) do |match| hashtag_html(match) end end def link_html(url) prefix = url.match(/\Ahttps?:\/\/(www\.)?/).to_s text = url[prefix.length, 30] suffix = url[prefix.length + 30..-1] cutoff = url[prefix.length..-1].length > 30 "#{prefix}#{text}#{suffix}" end def hashtag_html(match) prefix, affix = match.split('#') "#{prefix}##{affix}" end def mention_html(match, account) "#{match.split('@').first}@#{account.username}" end end