debian-mirror-gitlab/app/helpers/application_helper.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

490 lines
14 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'uri'
module ApplicationHelper
2021-01-03 14:25:43 +05:30
# See https://docs.gitlab.com/ee/development/ee_features.html#code-in-appviews
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2021-01-03 14:25:43 +05:30
# We allow partial to be nil so that collection views can be passed in
# `render partial: 'some/view', collection: @some_collection`
def render_if_exists(partial = nil, **options)
return unless partial_exists?(partial || options[:partial])
if partial.nil?
render(**options)
else
render(partial, options)
end
2018-12-05 23:21:45 +05:30
end
2022-05-07 20:08:51 +05:30
def dispensable_render(...)
render(...)
2022-08-27 11:52:29 +05:30
rescue StandardError => e
2023-03-17 16:20:25 +05:30
Gitlab::ErrorTracking.track_and_raise_for_dev_exception(e)
nil
2022-05-07 20:08:51 +05:30
end
def dispensable_render_if_exists(...)
render_if_exists(...)
2022-08-27 11:52:29 +05:30
rescue StandardError => e
2023-03-17 16:20:25 +05:30
Gitlab::ErrorTracking.track_and_raise_for_dev_exception(e)
nil
2022-05-07 20:08:51 +05:30
end
2018-12-05 23:21:45 +05:30
def partial_exists?(partial)
lookup_context.exists?(partial, [], true)
2018-11-08 19:23:39 +05:30
end
2018-12-05 23:21:45 +05:30
def template_exists?(template)
lookup_context.exists?(template, [], false)
end
# rubocop: enable CodeReuse/ActiveRecord
2014-09-02 18:07:02 +05:30
# Check if a particular controller is the current one
#
2018-12-05 23:21:45 +05:30
# args - One or more controller names to check (using path notation when inside namespaces)
2014-09-02 18:07:02 +05:30
#
# Examples
#
# # On TreeController
# current_controller?(:tree) # => true
# current_controller?(:commits) # => false
# current_controller?(:commits, :tree) # => true
2018-12-05 23:21:45 +05:30
#
# # On Admin::ApplicationController
# current_controller?(:application) # => true
# current_controller?('admin/application') # => true
# current_controller?('gitlab/application') # => false
2014-09-02 18:07:02 +05:30
def current_controller?(*args)
2015-09-25 12:07:36 +05:30
args.any? do |v|
2021-03-08 18:12:59 +05:30
Gitlab::Utils.safe_downcase!(v.to_s) == controller.controller_name || Gitlab::Utils.safe_downcase!(v.to_s) == controller.controller_path
2015-09-25 12:07:36 +05:30
end
2014-09-02 18:07:02 +05:30
end
# Check if a particular action is the current one
#
# args - One or more action names to check
#
# Examples
#
# # On Projects#new
# current_action?(:new) # => true
# current_action?(:create) # => false
# current_action?(:new, :create) # => true
def current_action?(*args)
2021-03-08 18:12:59 +05:30
args.any? { |v| Gitlab::Utils.safe_downcase!(v.to_s) == action_name }
2014-09-02 18:07:02 +05:30
end
2020-05-24 23:13:21 +05:30
def admin_section?
controller.class.ancestors.include?(Admin::ApplicationController)
end
2014-09-02 18:07:02 +05:30
def last_commit(project)
if project.repo_exists?
time_ago_with_tooltip(project.repository.commit.committed_date)
else
2015-04-26 12:48:37 +05:30
'Never'
2014-09-02 18:07:02 +05:30
end
2021-06-08 01:23:25 +05:30
rescue StandardError
2015-04-26 12:48:37 +05:30
'Never'
2014-09-02 18:07:02 +05:30
end
# Define whenever show last push event
# with suggestion to create MR
2018-12-05 23:21:45 +05:30
# rubocop: disable CodeReuse/ActiveRecord
2014-09-02 18:07:02 +05:30
def show_last_push_widget?(event)
# Skip if event is not about added or modified non-master branch
return false unless event && event.last_push_to_non_root? && !event.rm_ref?
project = event.project
# Skip if project repo is empty or MR disabled
2016-09-29 09:46:39 +05:30
return false unless project && !project.empty_repo? && project.feature_available?(:merge_requests, current_user)
2014-09-02 18:07:02 +05:30
# Skip if user already created appropriate MR
return false if project.merge_requests.where(source_branch: event.branch_name).opened.any?
# Skip if user removed branch right after that
2016-06-22 15:30:34 +05:30
return false unless project.repository.branch_exists?(event.branch_name)
2014-09-02 18:07:02 +05:30
true
end
2018-12-05 23:21:45 +05:30
# rubocop: enable CodeReuse/ActiveRecord
2014-09-02 18:07:02 +05:30
def hexdigest(string)
Digest::SHA1.hexdigest string
end
def simple_sanitize(str)
2023-03-04 22:38:38 +05:30
sanitize(str, tags: %w[a span])
2014-09-02 18:07:02 +05:30
end
2020-01-01 13:55:28 +05:30
def body_data
{
page: body_data_page,
page_type_id: controller.params[:id],
find_file: find_file_path,
2020-06-23 00:09:42 +05:30
group: @group&.path
2020-01-01 13:55:28 +05:30
}.merge(project_data)
end
def project_data
return {} unless @project
{
project_id: @project.id,
project: @project.path,
2020-06-23 00:09:42 +05:30
group: @project.group&.path,
2020-01-01 13:55:28 +05:30
namespace_id: @project.namespace&.id
}
end
2014-09-02 18:07:02 +05:30
def body_data_page
2017-09-10 17:25:29 +05:30
[*controller.controller_path.split('/'), controller.action_name].compact.join(':')
2014-09-02 18:07:02 +05:30
end
# shortcut for gitlab config
def gitlab_config
Gitlab.config.gitlab
end
# shortcut for gitlab extra config
def extra_config
Gitlab.config.extra
end
2019-12-21 20:55:43 +05:30
# shortcut for gitlab registry config
def registry_config
Gitlab.config.registry
end
2015-09-11 14:41:01 +05:30
# Render a `time` element with Javascript-based relative date and tooltip
#
# time - Time object
# placement - Tooltip placement String (default: "top")
# html_class - Custom class for `time` element (default: "time_ago")
#
# By default also includes a `script` element with Javascript necessary to
# initialize the `timeago` jQuery extension. If this method is called many
# times, for example rendering hundreds of commits, it's advisable to disable
# this behavior using the `skip_js` argument and re-initializing `timeago`
# manually once all of the elements have been rendered.
#
# A `js-timeago` class is always added to the element, even when a custom
# `html_class` argument is provided.
#
# Returns an HTML-safe String
2017-08-17 22:00:37 +05:30
def time_ago_with_tooltip(time, placement: 'top', html_class: '', short_format: false)
2023-03-17 16:20:25 +05:30
return "" if time.nil?
2018-12-05 23:21:45 +05:30
css_classes = [short_format ? 'js-short-timeago' : 'js-timeago']
css_classes << html_class unless html_class.blank?
2016-09-13 17:45:13 +05:30
2021-04-29 21:17:54 +05:30
content_tag :time, l(time, format: "%b %d, %Y"),
2023-05-27 22:25:52 +05:30
class: css_classes.join(' '),
title: l(time.to_time.in_time_zone, format: :timeago_tooltip),
datetime: time.to_time.getutc.iso8601,
data: {
toggle: 'tooltip',
placement: placement,
container: 'body'
}
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', exclude_author: false)
2018-03-17 18:26:18 +05:30
return unless object.edited?
2016-06-02 11:05:42 +05:30
2017-08-17 22:00:37 +05:30
content_tag :small, class: 'edited-text' do
output = content_tag(:span, 'Edited ')
output << time_ago_with_tooltip(object.last_edited_at, placement: placement, html_class: html_class)
2016-06-02 11:05:42 +05:30
2017-08-17 22:00:37 +05:30
if !exclude_author && object.last_edited_by
output << content_tag(:span, ' by ')
2023-05-27 22:25:52 +05:30
output << link_to_member(object.project, object.last_edited_by, avatar: false, extra_class: 'gl-hover-text-decoration-underline', author_class: nil)
2016-06-02 11:05:42 +05:30
end
output
end
end
2021-04-29 21:17:54 +05:30
# This needs to be used outside of Rails
def self.promo_host
2015-04-26 12:48:37 +05:30
'about.gitlab.com'
end
2021-04-29 21:17:54 +05:30
# Convenient method for Rails helper
def promo_host
ApplicationHelper.promo_host
end
2022-08-27 11:52:29 +05:30
# This needs to be used outside of Rails
def self.community_forum
'https://forum.gitlab.com'
end
# Convenient method for Rails helper
def community_forum
ApplicationHelper.community_forum
end
2015-04-26 12:48:37 +05:30
def promo_url
2023-03-04 22:38:38 +05:30
"https://#{promo_host}"
2015-04-26 12:48:37 +05:30
end
2017-09-10 17:25:29 +05:30
def support_url
2023-04-23 21:23:45 +05:30
Gitlab::CurrentSettings.current_application_settings.help_page_support_url.presence || "#{promo_url}/get-help/"
2017-09-10 17:25:29 +05:30
end
2021-01-03 14:25:43 +05:30
def instance_review_permitted?
2023-03-17 16:20:25 +05:30
::Gitlab::CurrentSettings.instance_review_permitted? && current_user&.can_read_all_resources?
2021-01-03 14:25:43 +05:30
end
2019-12-04 20:38:33 +05:30
def static_objects_external_storage_enabled?
Gitlab::CurrentSettings.static_objects_external_storage_enabled?
end
def external_storage_url_or_path(path, project = @project)
2020-03-13 15:44:24 +05:30
return path if @snippet || !static_objects_external_storage_enabled?
2019-12-04 20:38:33 +05:30
uri = URI(Gitlab::CurrentSettings.static_objects_external_storage_url)
path = URI(path) # `path` could have query parameters, so we need to split query and path apart
query = Rack::Utils.parse_nested_query(path.query)
query['token'] = current_user.static_object_token unless project.public?
uri.path = path.path
uri.query = query.to_query unless query.empty?
uri.to_s
end
2015-04-26 12:48:37 +05:30
def page_filter_path(options = {})
without = options.delete(:without)
2019-02-15 15:39:39 +05:30
options = request.query_parameters.merge(options)
2015-04-26 12:48:37 +05:30
if without.present?
without.each do |key|
options.delete(key)
2014-09-02 18:07:02 +05:30
end
end
2019-02-15 15:39:39 +05:30
"#{request.path}?#{options.compact.to_param}"
2015-04-26 12:48:37 +05:30
end
2020-10-24 23:57:45 +05:30
def stylesheet_link_tag_defer(path)
2023-03-04 22:38:38 +05:30
if startup_css_enabled?
stylesheet_link_tag(path, media: "print", crossorigin: ActionController::Base.asset_host ? 'anonymous' : nil)
else
2023-07-09 08:55:56 +05:30
stylesheet_link_tag(path, media: "all", crossorigin: ActionController::Base.asset_host ? 'anonymous' : nil)
2023-03-04 22:38:38 +05:30
end
end
def startup_css_enabled?
2023-06-20 00:43:36 +05:30
!Feature.enabled?(:remove_startup_css) && !params.has_key?(:no_startup_css)
end
def sign_in_with_redirect?
current_page?(new_user_session_path) && session[:user_return_to].present?
2023-03-04 22:38:38 +05:30
end
2015-04-26 12:48:37 +05:30
def outdated_browser?
2020-04-08 14:13:33 +05:30
browser.ie?
2015-04-26 12:48:37 +05:30
end
def path_to_key(key, admin = false)
if admin
admin_user_key_path(@user, key)
else
profile_key_path(key)
end
end
2015-10-24 18:46:33 +05:30
def truncate_first_line(message, length = 50)
truncate(message.each_line.first.chomp, length: length) if message
end
2016-08-24 12:49:21 +05:30
# While similarly named to Rails's `link_to_if`, this method behaves quite differently.
# If `condition` is truthy, a link will be returned with the result of the block
# as its body. If `condition` is falsy, only the result of the block will be returned.
def conditional_link_to(condition, options, html_options = {}, &block)
if condition
link_to options, html_options, &block
else
capture(&block)
end
end
2016-09-13 17:45:13 +05:30
def page_class
2017-09-10 17:25:29 +05:30
class_names = []
2021-03-11 19:13:27 +05:30
class_names << 'issue-boards-page gl-overflow-auto' if current_controller?(:boards)
2021-09-30 23:02:18 +05:30
class_names << 'epic-boards-page gl-overflow-auto' if current_controller?(:epic_boards)
2017-09-10 17:25:29 +05:30
class_names << 'with-performance-bar' if performance_bar_enabled?
2023-06-20 00:43:36 +05:30
class_names << 'with-top-bar' if show_super_sidebar? && !@hide_top_bar
2019-07-07 11:18:12 +05:30
class_names << system_message_class
2023-01-13 00:05:48 +05:30
class_names << 'logged-out-marketing-header' if !current_user && ::Gitlab.com?
2017-09-10 17:25:29 +05:30
class_names
2016-09-13 17:45:13 +05:30
end
2017-08-17 22:00:37 +05:30
2018-05-09 12:01:36 +05:30
def system_message_class
2019-07-07 11:18:12 +05:30
class_names = []
return class_names unless appearance
class_names << 'with-system-header' if appearance.show_header?
class_names << 'with-system-footer' if appearance.show_footer?
class_names
2018-05-09 12:01:36 +05:30
end
2017-08-17 22:00:37 +05:30
# Returns active css class when condition returns true
# otherwise returns nil.
#
# Example:
# %li{ class: active_when(params[:filter] == '1') }
def active_when(condition)
'active' if condition
end
2017-09-10 17:25:29 +05:30
def show_callout?(name)
cookies[name] != 'true'
end
def linkedin_url(user)
name = user.linkedin
2018-05-09 12:01:36 +05:30
if name =~ %r{\Ahttps?://(www\.)?linkedin\.com/in/}
2017-09-10 17:25:29 +05:30
name
else
"https://www.linkedin.com/in/#{name}"
end
end
def twitter_url(user)
name = user.twitter
2018-05-09 12:01:36 +05:30
if name =~ %r{\Ahttps?://(www\.)?twitter\.com/}
2017-09-10 17:25:29 +05:30
name
else
2018-05-09 12:01:36 +05:30
"https://twitter.com/#{name}"
2017-09-10 17:25:29 +05:30
end
end
2023-04-23 21:23:45 +05:30
def discord_url(user)
return '' if user.discord.blank?
"https://discord.com/users/#{user.discord}"
end
2017-09-10 17:25:29 +05:30
def collapsed_sidebar?
cookies["sidebar_collapsed"] == "true"
end
2023-05-27 22:25:52 +05:30
def collapsed_super_sidebar?
2023-07-09 08:55:56 +05:30
return false if @force_desktop_expanded_sidebar
2023-05-27 22:25:52 +05:30
cookies["super_sidebar_collapsed"] == "true"
end
2018-03-17 18:26:18 +05:30
def locale_path
asset_path("locale/#{Gitlab::I18n.locale}/app.js")
end
2018-05-09 12:01:36 +05:30
# Overridden in EE
def read_only_message
return unless Gitlab::Database.read_only?
_('You are on a read-only GitLab instance.')
end
2018-10-15 14:42:47 +05:30
2019-03-02 22:35:43 +05:30
def client_class_list
2023-01-13 00:05:48 +05:30
"gl-browser-#{browser_id} gl-platform-#{platform_id}"
2019-03-02 22:35:43 +05:30
end
def client_js_flags
{
2023-01-13 00:05:48 +05:30
"is#{browser_id.titlecase}": true,
"is#{platform_id.titlecase}": true
2019-03-02 22:35:43 +05:30
}
end
2021-02-22 17:27:13 +05:30
def add_page_specific_style(path, defer: true)
2021-01-03 14:25:43 +05:30
content_for :page_specific_styles do
2021-02-22 17:27:13 +05:30
if defer
stylesheet_link_tag_defer path
else
stylesheet_link_tag path
end
2021-01-03 14:25:43 +05:30
end
end
2020-07-28 23:09:34 +05:30
def page_startup_api_calls
@api_startup_calls
end
def add_page_startup_api_call(api_path, options: {})
@api_startup_calls ||= {}
@api_startup_calls[api_path] = options
end
2018-10-15 14:42:47 +05:30
def autocomplete_data_sources(object, noteable_type)
return {} unless object && noteable_type
2021-06-08 01:23:25 +05:30
if object.is_a?(Group)
{
members: members_group_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
issues: issues_group_autocomplete_sources_path(object),
mergeRequests: merge_requests_group_autocomplete_sources_path(object),
labels: labels_group_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
milestones: milestones_group_autocomplete_sources_path(object),
commands: commands_group_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id])
}
else
{
members: members_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
issues: issues_project_autocomplete_sources_path(object),
mergeRequests: merge_requests_project_autocomplete_sources_path(object),
labels: labels_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
milestones: milestones_project_autocomplete_sources_path(object),
commands: commands_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
2022-04-04 11:22:00 +05:30
snippets: snippets_project_autocomplete_sources_path(object),
2022-11-25 23:54:43 +05:30
contacts: contacts_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id])
2021-06-08 01:23:25 +05:30
}
end
2018-10-15 14:42:47 +05:30
end
2019-07-07 11:18:12 +05:30
2019-12-26 22:10:19 +05:30
def asset_to_string(name)
app = Rails.application
if Rails.configuration.assets.compile
app.assets.find_asset(name).to_s
else
controller.view_context.render(file: Rails.root.join('public/assets', app.assets_manifest.assets[name]).to_s)
end
end
2021-10-27 15:23:28 +05:30
def gitlab_ui_form_for(record, *args, &block)
options = args.extract_options!
form_for(record, *(args << options.merge({ builder: ::Gitlab::FormBuilders::GitlabUiFormBuilder })), &block)
end
2022-11-25 23:54:43 +05:30
def gitlab_ui_form_with(**args, &block)
form_with(**args.merge({ builder: ::Gitlab::FormBuilders::GitlabUiFormBuilder }), &block)
end
2019-07-07 11:18:12 +05:30
private
2023-01-13 00:05:48 +05:30
def browser_id
browser.unknown? ? 'generic' : browser.id.to_s
2019-07-07 11:18:12 +05:30
end
2022-03-02 08:16:31 +05:30
2023-01-13 00:05:48 +05:30
def platform_id
browser.platform.unknown? ? 'other' : browser.platform.id.to_s
end
2022-03-02 08:16:31 +05:30
2023-01-13 00:05:48 +05:30
def appearance
::Appearance.current
2022-03-02 08:16:31 +05:30
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
2021-06-08 01:23:25 +05:30
ApplicationHelper.prepend_mod