123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 |
- # 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.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 'envelope 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
- private
- def simplified_text(text)
- text.dup.tap do |new_text|
- URI.extract(new_text).each do |url|
- new_text.gsub!(url, '')
- end
- new_text.gsub!(Account::MENTION_RE, '')
- new_text.gsub!(Tag::HASHTAG_RE, '')
- new_text.gsub!(/\s+/, '')
- end
- end
- def embedded_view?
- params[:controller] == EMBEDDED_CONTROLLER && params[:action] == EMBEDDED_ACTION
- end
- def render_video_component(status, **options)
- video = status.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.media_attachments }
- end
- end
- def render_audio_component(status, **options)
- audio = status.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.media_attachments }
- end
- end
- def render_media_gallery_component(status, **options)
- component_params = {
- sensitive: sensitized?(status, current_account),
- autoplay: prefers_autoplay?,
- media: status.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.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
|