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.

198 lines
6.3 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 fa_icon(icon, attributes = {})
  74. class_names = attributes[:class]&.split(' ') || []
  75. class_names << 'fa'
  76. class_names += icon.split(' ').map { |cl| "fa-#{cl}" }
  77. content_tag(:i, nil, attributes.merge(class: class_names.join(' ')))
  78. end
  79. def visibility_icon(status)
  80. if status.public_visibility?
  81. fa_icon('globe', title: I18n.t('statuses.visibilities.public'))
  82. elsif status.unlisted_visibility?
  83. fa_icon('unlock', title: I18n.t('statuses.visibilities.unlisted'))
  84. elsif status.private_visibility? || status.limited_visibility?
  85. fa_icon('lock', title: I18n.t('statuses.visibilities.private'))
  86. elsif status.direct_visibility?
  87. fa_icon('envelope', title: I18n.t('statuses.visibilities.direct'))
  88. end
  89. end
  90. def interrelationships_icon(relationships, account_id)
  91. if relationships.following[account_id] && relationships.followed_by[account_id]
  92. fa_icon('exchange', title: I18n.t('relationships.mutual'), class: 'fa-fw active passive')
  93. elsif relationships.following[account_id]
  94. fa_icon(locale_direction == 'ltr' ? 'arrow-right' : 'arrow-left', title: I18n.t('relationships.following'), class: 'fa-fw active')
  95. elsif relationships.followed_by[account_id]
  96. fa_icon(locale_direction == 'ltr' ? 'arrow-left' : 'arrow-right', title: I18n.t('relationships.followers'), class: 'fa-fw passive')
  97. end
  98. end
  99. def custom_emoji_tag(custom_emoji, animate = true)
  100. if animate
  101. image_tag(custom_emoji.image.url, class: 'emojione', alt: ":#{custom_emoji.shortcode}:")
  102. else
  103. 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)))
  104. end
  105. end
  106. def opengraph(property, content)
  107. tag(:meta, content: content, property: property)
  108. end
  109. def react_component(name, props = {}, &block)
  110. if block.nil?
  111. content_tag(:div, nil, data: { component: name.to_s.camelcase, props: Oj.dump(props) })
  112. else
  113. content_tag(:div, data: { component: name.to_s.camelcase, props: Oj.dump(props) }, &block)
  114. end
  115. end
  116. def body_classes
  117. output = (@body_classes || '').split(' ')
  118. output << "theme-#{current_theme.parameterize}"
  119. output << 'system-font' if current_account&.user&.setting_system_font_ui
  120. output << (current_account&.user&.setting_reduce_motion ? 'reduce-motion' : 'no-reduce-motion')
  121. output << 'rtl' if locale_direction == 'rtl'
  122. output.reject(&:blank?).join(' ')
  123. end
  124. def cdn_host
  125. Rails.configuration.action_controller.asset_host
  126. end
  127. def cdn_host?
  128. cdn_host.present?
  129. end
  130. def storage_host
  131. "https://#{ENV['S3_ALIAS_HOST'].presence || ENV['S3_CLOUDFRONT_HOST']}"
  132. end
  133. def storage_host?
  134. ENV['S3_ALIAS_HOST'].present? || ENV['S3_CLOUDFRONT_HOST'].present?
  135. end
  136. def quote_wrap(text, line_width: 80, break_sequence: "\n")
  137. text = word_wrap(text, line_width: line_width - 2, break_sequence: break_sequence)
  138. text.split("\n").map { |line| '> ' + line }.join("\n")
  139. end
  140. def render_initial_state
  141. state_params = {
  142. settings: {
  143. known_fediverse: Setting.show_known_fediverse_at_about_page,
  144. },
  145. text: [params[:title], params[:text], params[:url]].compact.join(' '),
  146. }
  147. permit_visibilities = %w(public unlisted private direct)
  148. default_privacy = current_account&.user&.setting_default_privacy
  149. permit_visibilities.shift(permit_visibilities.index(default_privacy) + 1) if default_privacy.present?
  150. state_params[:visibility] = params[:visibility] if permit_visibilities.include? params[:visibility]
  151. if user_signed_in?
  152. state_params[:settings] = state_params[:settings].merge(Web::Setting.find_by(user: current_user)&.data || {})
  153. state_params[:push_subscription] = current_account.user.web_push_subscription(current_session)
  154. state_params[:current_account] = current_account
  155. state_params[:token] = current_session.token
  156. state_params[:admin] = Account.find_local(Setting.site_contact_username.strip.gsub(/\A@/, ''))
  157. end
  158. json = ActiveModelSerializers::SerializableResource.new(InitialStatePresenter.new(state_params), serializer: InitialStateSerializer).to_json
  159. # rubocop:disable Rails/OutputSafety
  160. content_tag(:script, json_escape(json).html_safe, id: 'initial-state', type: 'application/json')
  161. # rubocop:enable Rails/OutputSafety
  162. end
  163. end