mirror of
				https://github.com/lunaisnotaboy/mastodon.git
				synced 2025-11-04 06:45:01 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			200 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# frozen_string_literal: true
 | 
						|
 | 
						|
module StatusesHelper
 | 
						|
  EMBEDDED_CONTROLLER = 'statuses'
 | 
						|
  EMBEDDED_ACTION = 'embed'
 | 
						|
 | 
						|
  def link_to_newer(url)
 | 
						|
    link_to t('statuses.show_newer'), url, class: 'load-more load-gap'
 | 
						|
  end
 | 
						|
 | 
						|
  def link_to_older(url)
 | 
						|
    link_to t('statuses.show_older'), url, class: 'load-more load-gap'
 | 
						|
  end
 | 
						|
 | 
						|
  def nothing_here(extra_classes = '')
 | 
						|
    content_tag(:div, class: "nothing-here #{extra_classes}") do
 | 
						|
      t('accounts.nothing_here')
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def media_summary(status)
 | 
						|
    attachments = { image: 0, video: 0, audio: 0 }
 | 
						|
 | 
						|
    status.ordered_media_attachments.each do |media|
 | 
						|
      if media.video?
 | 
						|
        attachments[:video] += 1
 | 
						|
      elsif media.audio?
 | 
						|
        attachments[:audio] += 1
 | 
						|
      else
 | 
						|
        attachments[:image] += 1
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    text = attachments.to_a.reject { |_, value| value.zero? }.map { |key, value| I18n.t("statuses.attached.#{key}", count: value) }.join(' · ')
 | 
						|
 | 
						|
    return if text.blank?
 | 
						|
 | 
						|
    I18n.t('statuses.attached.description', attached: text)
 | 
						|
  end
 | 
						|
 | 
						|
  def status_text_summary(status)
 | 
						|
    return if status.spoiler_text.blank?
 | 
						|
 | 
						|
    I18n.t('statuses.content_warning', warning: status.spoiler_text)
 | 
						|
  end
 | 
						|
 | 
						|
  def poll_summary(status)
 | 
						|
    return unless status.preloadable_poll
 | 
						|
 | 
						|
    status.preloadable_poll.options.map { |o| "[ ] #{o}" }.join("\n")
 | 
						|
  end
 | 
						|
 | 
						|
  def status_description(status)
 | 
						|
    components = [[media_summary(status), status_text_summary(status)].reject(&:blank?).join(' · ')]
 | 
						|
 | 
						|
    if status.spoiler_text.blank?
 | 
						|
      components << status.text
 | 
						|
      components << poll_summary(status)
 | 
						|
    end
 | 
						|
 | 
						|
    components.reject(&:blank?).join("\n\n")
 | 
						|
  end
 | 
						|
 | 
						|
  def stream_link_target
 | 
						|
    embedded_view? ? '_blank' : nil
 | 
						|
  end
 | 
						|
 | 
						|
  def style_classes(status, is_predecessor, is_successor, include_threads)
 | 
						|
    classes = ['entry']
 | 
						|
    classes << 'entry-predecessor' if is_predecessor
 | 
						|
    classes << 'entry-reblog' if status.reblog?
 | 
						|
    classes << 'entry-successor' if is_successor
 | 
						|
    classes << 'entry-center' if include_threads
 | 
						|
    classes.join(' ')
 | 
						|
  end
 | 
						|
 | 
						|
  def microformats_classes(status, is_direct_parent, is_direct_child)
 | 
						|
    classes = []
 | 
						|
    classes << 'p-in-reply-to' if is_direct_parent
 | 
						|
    classes << 'p-repost-of' if status.reblog? && is_direct_parent
 | 
						|
    classes << 'p-comment' if is_direct_child
 | 
						|
    classes.join(' ')
 | 
						|
  end
 | 
						|
 | 
						|
  def microformats_h_class(status, is_predecessor, is_successor, include_threads)
 | 
						|
    if is_predecessor || status.reblog? || is_successor
 | 
						|
      'h-cite'
 | 
						|
    elsif include_threads
 | 
						|
      ''
 | 
						|
    else
 | 
						|
      'h-entry'
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def fa_visibility_icon(status)
 | 
						|
    case status.visibility
 | 
						|
    when 'public'
 | 
						|
      fa_icon 'globe fw'
 | 
						|
    when 'unlisted'
 | 
						|
      fa_icon 'unlock fw'
 | 
						|
    when 'private'
 | 
						|
      fa_icon 'lock fw'
 | 
						|
    when 'direct'
 | 
						|
      fa_icon 'at fw'
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def sensitized?(status, account)
 | 
						|
    if !account.nil? && account.id == status.account_id
 | 
						|
      status.sensitive
 | 
						|
    else
 | 
						|
      status.account.sensitized? || status.sensitive
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def embedded_view?
 | 
						|
    params[:controller] == EMBEDDED_CONTROLLER && params[:action] == EMBEDDED_ACTION
 | 
						|
  end
 | 
						|
 | 
						|
  def render_video_component(status, **options)
 | 
						|
    video = status.ordered_media_attachments.first
 | 
						|
 | 
						|
    meta = video.file.meta || {}
 | 
						|
 | 
						|
    component_params = {
 | 
						|
      sensitive: sensitized?(status, current_account),
 | 
						|
      src: full_asset_url(video.file.url(:original)),
 | 
						|
      preview: full_asset_url(video.thumbnail.present? ? video.thumbnail.url : video.file.url(:small)),
 | 
						|
      alt: video.description,
 | 
						|
      blurhash: video.blurhash,
 | 
						|
      frameRate: meta.dig('original', 'frame_rate'),
 | 
						|
      inline: true,
 | 
						|
      media: [
 | 
						|
        ActiveModelSerializers::SerializableResource.new(video, serializer: REST::MediaAttachmentSerializer),
 | 
						|
      ].as_json,
 | 
						|
    }.merge(**options)
 | 
						|
 | 
						|
    react_component :video, component_params do
 | 
						|
      render partial: 'statuses/attachment_list', locals: { attachments: status.ordered_media_attachments }
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def render_audio_component(status, **options)
 | 
						|
    audio = status.ordered_media_attachments.first
 | 
						|
 | 
						|
    meta = audio.file.meta || {}
 | 
						|
 | 
						|
    component_params = {
 | 
						|
      src: full_asset_url(audio.file.url(:original)),
 | 
						|
      poster: full_asset_url(audio.thumbnail.present? ? audio.thumbnail.url : status.account.avatar_static_url),
 | 
						|
      alt: audio.description,
 | 
						|
      backgroundColor: meta.dig('colors', 'background'),
 | 
						|
      foregroundColor: meta.dig('colors', 'foreground'),
 | 
						|
      accentColor: meta.dig('colors', 'accent'),
 | 
						|
      duration: meta.dig('original', 'duration'),
 | 
						|
    }.merge(**options)
 | 
						|
 | 
						|
    react_component :audio, component_params do
 | 
						|
      render partial: 'statuses/attachment_list', locals: { attachments: status.ordered_media_attachments }
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def render_media_gallery_component(status, **options)
 | 
						|
    component_params = {
 | 
						|
      sensitive: sensitized?(status, current_account),
 | 
						|
      autoplay: prefers_autoplay?,
 | 
						|
      media: status.ordered_media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json },
 | 
						|
    }.merge(**options)
 | 
						|
 | 
						|
    react_component :media_gallery, component_params do
 | 
						|
      render partial: 'statuses/attachment_list', locals: { attachments: status.ordered_media_attachments }
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def render_card_component(status, **options)
 | 
						|
    component_params = {
 | 
						|
      sensitive: sensitized?(status, current_account),
 | 
						|
      maxDescription: 160,
 | 
						|
      card: ActiveModelSerializers::SerializableResource.new(status.preview_card, serializer: REST::PreviewCardSerializer).as_json,
 | 
						|
    }.merge(**options)
 | 
						|
 | 
						|
    react_component :card, component_params
 | 
						|
  end
 | 
						|
 | 
						|
  def render_poll_component(status, **options)
 | 
						|
    component_params = {
 | 
						|
      disabled: true,
 | 
						|
      poll: ActiveModelSerializers::SerializableResource.new(status.preloadable_poll, serializer: REST::PollSerializer, scope: current_user, scope_name: :current_user).as_json,
 | 
						|
    }.merge(**options)
 | 
						|
 | 
						|
    react_component :poll, component_params do
 | 
						|
      render partial: 'statuses/poll', locals: { status: status, poll: status.preloadable_poll, autoplay: prefers_autoplay? }
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def prefers_autoplay?
 | 
						|
    ActiveModel::Type::Boolean.new.cast(params[:autoplay]) || current_user&.setting_auto_play_gif
 | 
						|
  end
 | 
						|
end
 |