You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

203 lines
6.4 KiB

8 years ago
3 years ago
8 years ago
  1. # frozen_string_literal: true
  2. module ApplicationHelper
  3. DANGEROUS_SCOPES = %w(
  4. read
  5. write
  6. follow
  7. ).freeze
  8. RTL_LOCALES = %i(
  9. ar
  10. fa
  11. he
  12. ku
  13. ).freeze
  14. def friendly_number_to_human(number, **options)
  15. # By default, the number of precision digits used by number_to_human
  16. # is looked up from the locales definition, and rails-i18n comes with
  17. # values that don't seem to make much sense for many languages, so
  18. # override these values with a default of 3 digits of precision.
  19. options[:precision] = 3
  20. options[:strip_insignificant_zeros] = true
  21. number_to_human(number, **options)
  22. end
  23. def active_nav_class(*paths)
  24. paths.any? { |path| current_page?(path) } ? 'active' : ''
  25. end
  26. def active_link_to(label, path, **options)
  27. link_to label, path, options.merge(class: active_nav_class(path))
  28. end
  29. def show_landing_strip?
  30. !user_signed_in? && !single_user_mode?
  31. end
  32. def open_registrations?
  33. Setting.registrations_mode == 'open'
  34. end
  35. def approved_registrations?
  36. Setting.registrations_mode == 'approved'
  37. end
  38. def closed_registrations?
  39. Setting.registrations_mode == 'none'
  40. end
  41. def available_sign_up_path
  42. if closed_registrations?
  43. #'https://joinmastodon.org/#getting-started'
  44. 'https://closed.social'
  45. else
  46. new_user_registration_path
  47. end
  48. end
  49. def open_deletion?
  50. Setting.open_deletion
  51. end
  52. def locale_direction
  53. if RTL_LOCALES.include?(I18n.locale)
  54. 'rtl'
  55. else
  56. 'ltr'
  57. end
  58. end
  59. def favicon_path
  60. env_suffix = Rails.env.production? ? '' : '-dev'
  61. "/favicon#{env_suffix}.ico"
  62. end
  63. def title
  64. Rails.env.production? ? site_title : "#{site_title} (Dev)"
  65. end
  66. def class_for_scope(scope)
  67. 'scope-danger' if DANGEROUS_SCOPES.include?(scope.to_s)
  68. end
  69. def can?(action, record)
  70. return false if record.nil?
  71. policy(record).public_send("#{action}?")
  72. end
  73. def masked_email(email)
  74. email_username = email.split('@').first
  75. "#{email_username[0]}***#{email_username[-1]}@#{email.split('@').last}"
  76. end
  77. def fa_icon(icon, attributes = {})
  78. class_names = attributes[:class]&.split(' ') || []
  79. class_names << 'fa'
  80. class_names += icon.split(' ').map { |cl| "fa-#{cl}" }
  81. content_tag(:i, nil, attributes.merge(class: class_names.join(' ')))
  82. end
  83. def visibility_icon(status)
  84. if status.public_visibility?
  85. fa_icon('globe', title: I18n.t('statuses.visibilities.public'))
  86. elsif status.unlisted_visibility?
  87. fa_icon('unlock', title: I18n.t('statuses.visibilities.unlisted'))
  88. elsif status.private_visibility? || status.limited_visibility?
  89. fa_icon('lock', title: I18n.t('statuses.visibilities.private'))
  90. elsif status.direct_visibility?
  91. fa_icon('envelope', title: I18n.t('statuses.visibilities.direct'))
  92. end
  93. end
  94. def interrelationships_icon(relationships, account_id)
  95. if relationships.following[account_id] && relationships.followed_by[account_id]
  96. fa_icon('exchange', title: I18n.t('relationships.mutual'), class: 'fa-fw active passive')
  97. elsif relationships.following[account_id]
  98. fa_icon(locale_direction == 'ltr' ? 'arrow-right' : 'arrow-left', title: I18n.t('relationships.following'), class: 'fa-fw active')
  99. elsif relationships.followed_by[account_id]
  100. fa_icon(locale_direction == 'ltr' ? 'arrow-left' : 'arrow-right', title: I18n.t('relationships.followers'), class: 'fa-fw passive')
  101. end
  102. end
  103. def custom_emoji_tag(custom_emoji, animate = true)
  104. if animate
  105. image_tag(custom_emoji.image.url, class: 'emojione', alt: ":#{custom_emoji.shortcode}:")
  106. else
  107. image_tag(custom_emoji.image.url(:static), class: 'emojione custom-emoji', alt: ":#{custom_emoji.shortcode}", 'data-original' => full_asset_url(custom_emoji.image.url), 'data-static' => full_asset_url(custom_emoji.image.url(:static)))
  108. end
  109. end
  110. def opengraph(property, content)
  111. tag(:meta, content: content, property: property)
  112. end
  113. def react_component(name, props = {}, &block)
  114. if block.nil?
  115. content_tag(:div, nil, data: { component: name.to_s.camelcase, props: Oj.dump(props) })
  116. else
  117. content_tag(:div, data: { component: name.to_s.camelcase, props: Oj.dump(props) }, &block)
  118. end
  119. end
  120. def body_classes
  121. output = (@body_classes || '').split(' ')
  122. output << "theme-#{current_theme.parameterize}"
  123. output << 'system-font' if current_account&.user&.setting_system_font_ui
  124. output << (current_account&.user&.setting_reduce_motion ? 'reduce-motion' : 'no-reduce-motion')
  125. output << 'rtl' if locale_direction == 'rtl'
  126. output.reject(&:blank?).join(' ')
  127. end
  128. def cdn_host
  129. Rails.configuration.action_controller.asset_host
  130. end
  131. def cdn_host?
  132. cdn_host.present?
  133. end
  134. def storage_host
  135. "https://#{ENV['S3_ALIAS_HOST'].presence || ENV['S3_CLOUDFRONT_HOST']}"
  136. end
  137. def storage_host?
  138. ENV['S3_ALIAS_HOST'].present? || ENV['S3_CLOUDFRONT_HOST'].present?
  139. end
  140. def quote_wrap(text, line_width: 80, break_sequence: "\n")
  141. text = word_wrap(text, line_width: line_width - 2, break_sequence: break_sequence)
  142. text.split("\n").map { |line| '> ' + line }.join("\n")
  143. end
  144. def render_initial_state
  145. state_params = {
  146. settings: {
  147. known_fediverse: Setting.show_known_fediverse_at_about_page,
  148. },
  149. text: [params[:title], params[:text], params[:url]].compact.join(' '),
  150. }
  151. permit_visibilities = %w(public unlisted private direct)
  152. default_privacy = current_account&.user&.setting_default_privacy
  153. permit_visibilities.shift(permit_visibilities.index(default_privacy) + 1) if default_privacy.present?
  154. state_params[:visibility] = params[:visibility] if permit_visibilities.include? params[:visibility]
  155. if user_signed_in?
  156. state_params[:settings] = state_params[:settings].merge(Web::Setting.find_by(user: current_user)&.data || {})
  157. state_params[:push_subscription] = current_account.user.web_push_subscription(current_session)
  158. state_params[:current_account] = current_account
  159. state_params[:token] = current_session.token
  160. state_params[:admin] = Account.find_local(Setting.site_contact_username.strip.gsub(/\A@/, ''))
  161. end
  162. json = ActiveModelSerializers::SerializableResource.new(InitialStatePresenter.new(state_params), serializer: InitialStateSerializer).to_json
  163. # rubocop:disable Rails/OutputSafety
  164. content_tag(:script, json_escape(json).html_safe, id: 'initial-state', type: 'application/json')
  165. # rubocop:enable Rails/OutputSafety
  166. end
  167. end