2018-08-18 07:19:57 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2011-10-08 17:36:38 -04:00
|
|
|
require 'digest/md5'
|
2012-12-06 15:44:22 -05:00
|
|
|
require 'uri'
|
2012-09-26 15:06:07 -04:00
|
|
|
|
2011-10-08 17:36:38 -04:00
|
|
|
module ApplicationHelper
|
2018-05-23 08:56:11 -04:00
|
|
|
# See https://docs.gitlab.com/ee/development/ee_features.html#code-in-app-views
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-05-23 08:56:11 -04:00
|
|
|
def render_if_exists(partial, locals = {})
|
2018-09-11 11:24:34 -04:00
|
|
|
render(partial, locals) if partial_exists?(partial)
|
|
|
|
end
|
|
|
|
|
|
|
|
def partial_exists?(partial)
|
|
|
|
lookup_context.exists?(partial, [], true)
|
|
|
|
end
|
|
|
|
|
2018-09-18 04:43:19 -04:00
|
|
|
def template_exists?(template)
|
|
|
|
lookup_context.exists?(template, [], false)
|
2018-05-23 08:56:11 -04:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2018-05-23 08:56:11 -04:00
|
|
|
|
2012-09-25 19:22:44 -04:00
|
|
|
# Check if a particular controller is the current one
|
|
|
|
#
|
2018-09-18 12:12:37 -04:00
|
|
|
# args - One or more controller names to check (using path notation when inside namespaces)
|
2012-09-25 21:18:00 -04:00
|
|
|
#
|
2012-09-25 19:22:44 -04:00
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# # On TreeController
|
2012-09-25 21:18:00 -04:00
|
|
|
# current_controller?(:tree) # => true
|
|
|
|
# current_controller?(:commits) # => false
|
|
|
|
# current_controller?(:commits, :tree) # => true
|
2018-09-18 12:12:37 -04:00
|
|
|
#
|
|
|
|
# # On Admin::ApplicationController
|
|
|
|
# current_controller?(:application) # => true
|
|
|
|
# current_controller?('admin/application') # => true
|
|
|
|
# current_controller?('gitlab/application') # => false
|
2012-09-25 21:18:00 -04:00
|
|
|
def current_controller?(*args)
|
2015-09-17 08:22:43 -04:00
|
|
|
args.any? do |v|
|
|
|
|
v.to_s.downcase == controller.controller_name || v.to_s.downcase == controller.controller_path
|
|
|
|
end
|
2012-09-25 19:22:44 -04:00
|
|
|
end
|
|
|
|
|
2013-07-29 06:46:00 -04:00
|
|
|
# Check if a particular action is the current one
|
2012-09-26 15:06:07 -04:00
|
|
|
#
|
|
|
|
# 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)
|
|
|
|
args.any? { |v| v.to_s.downcase == action_name }
|
|
|
|
end
|
|
|
|
|
2011-10-08 17:36:38 -04:00
|
|
|
def last_commit(project)
|
2011-10-26 09:46:25 -04:00
|
|
|
if project.repo_exists?
|
2013-12-30 07:38:42 -05:00
|
|
|
time_ago_with_tooltip(project.repository.commit.committed_date)
|
2011-10-26 09:46:25 -04:00
|
|
|
else
|
2015-01-19 15:37:20 -05:00
|
|
|
'Never'
|
2011-10-08 17:36:38 -04:00
|
|
|
end
|
2011-11-15 03:34:30 -05:00
|
|
|
rescue
|
2015-01-19 15:37:20 -05:00
|
|
|
'Never'
|
2011-10-08 17:36:38 -04:00
|
|
|
end
|
|
|
|
|
2013-04-19 14:19:16 -04:00
|
|
|
# Define whenever show last push event
|
|
|
|
# with suggestion to create MR
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2012-06-21 11:41:22 -04:00
|
|
|
def show_last_push_widget?(event)
|
2013-04-19 14:19:16 -04:00
|
|
|
# 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-08-01 18:31:21 -04:00
|
|
|
return false unless project && !project.empty_repo? && project.feature_available?(:merge_requests, current_user)
|
2013-04-19 14:19:16 -04:00
|
|
|
|
|
|
|
# Skip if user already created appropriate MR
|
|
|
|
return false if project.merge_requests.where(source_branch: event.branch_name).opened.any?
|
|
|
|
|
2013-11-13 07:17:03 -05:00
|
|
|
# Skip if user removed branch right after that
|
2016-06-20 09:38:54 -04:00
|
|
|
return false unless project.repository.branch_exists?(event.branch_name)
|
2013-11-13 07:17:03 -05:00
|
|
|
|
2013-04-19 14:19:16 -04:00
|
|
|
true
|
2012-06-21 11:41:22 -04:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2012-07-03 13:52:48 -04:00
|
|
|
|
2012-07-10 16:12:38 -04:00
|
|
|
def hexdigest(string)
|
|
|
|
Digest::SHA1.hexdigest string
|
|
|
|
end
|
2012-08-30 01:13:36 -04:00
|
|
|
|
2013-10-19 23:57:34 -04:00
|
|
|
def simple_sanitize(str)
|
2013-02-18 02:28:18 -05:00
|
|
|
sanitize(str, tags: %w(a span))
|
|
|
|
end
|
|
|
|
|
2019-12-02 19:06:28 -05:00
|
|
|
def body_data
|
|
|
|
{
|
|
|
|
page: body_data_page,
|
|
|
|
page_type_id: controller.params[:id],
|
|
|
|
find_file: find_file_path,
|
|
|
|
group: "#{@group&.path}"
|
|
|
|
}.merge(project_data)
|
|
|
|
end
|
|
|
|
|
|
|
|
def project_data
|
|
|
|
return {} unless @project
|
|
|
|
|
|
|
|
{
|
|
|
|
project_id: @project.id,
|
|
|
|
project: @project.path,
|
|
|
|
namespace_id: @project.namespace&.id
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2013-05-02 04:18:29 -04:00
|
|
|
def body_data_page
|
2017-06-13 18:12:31 -04:00
|
|
|
[*controller.controller_path.split('/'), controller.action_name].compact.join(':')
|
2013-05-02 04:18:29 -04:00
|
|
|
end
|
2013-05-08 14:03:14 -04:00
|
|
|
|
|
|
|
# shortcut for gitlab config
|
|
|
|
def gitlab_config
|
|
|
|
Gitlab.config.gitlab
|
|
|
|
end
|
|
|
|
|
|
|
|
# shortcut for gitlab extra config
|
|
|
|
def extra_config
|
|
|
|
Gitlab.config.extra
|
|
|
|
end
|
2013-06-06 07:02:29 -04:00
|
|
|
|
2019-10-16 08:06:32 -04:00
|
|
|
# shortcut for gitlab registry config
|
|
|
|
def registry_config
|
|
|
|
Gitlab.config.registry
|
|
|
|
end
|
|
|
|
|
2015-06-17 14:04:14 -04:00
|
|
|
# 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
|
2016-09-08 14:57:24 -04:00
|
|
|
def time_ago_with_tooltip(time, placement: 'top', html_class: '', short_format: false)
|
2018-08-18 07:19:57 -04:00
|
|
|
css_classes = [short_format ? 'js-short-timeago' : 'js-timeago']
|
|
|
|
css_classes << html_class unless html_class.blank?
|
2016-07-12 18:52:36 -04:00
|
|
|
|
2017-06-09 10:38:49 -04:00
|
|
|
element = content_tag :time, l(time, format: "%b %d, %Y"),
|
2018-08-18 07:19:57 -04:00
|
|
|
class: css_classes.join(' '),
|
2017-06-12 04:35:36 -04:00
|
|
|
title: l(time.to_time.in_time_zone, format: :timeago_tooltip),
|
2016-09-08 14:57:24 -04:00
|
|
|
datetime: time.to_time.getutc.iso8601,
|
|
|
|
data: {
|
|
|
|
toggle: 'tooltip',
|
|
|
|
placement: placement,
|
|
|
|
container: 'body'
|
|
|
|
}
|
2015-06-17 14:04:14 -04:00
|
|
|
|
|
|
|
element
|
2013-12-13 16:31:23 -05:00
|
|
|
end
|
2014-01-08 09:32:03 -05:00
|
|
|
|
2017-05-03 07:46:10 -04:00
|
|
|
def edited_time_ago_with_tooltip(object, placement: 'top', html_class: 'time_ago', exclude_author: false)
|
2017-08-24 12:26:24 -04:00
|
|
|
return unless object.edited?
|
2016-03-02 06:33:55 -05:00
|
|
|
|
2017-05-03 01:32:21 -04:00
|
|
|
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-03-02 06:33:55 -05:00
|
|
|
|
2017-05-03 07:46:10 -04:00
|
|
|
if !exclude_author && object.last_edited_by
|
2017-05-03 01:32:21 -04:00
|
|
|
output << content_tag(:span, ' by ')
|
|
|
|
output << link_to_member(object.project, object.last_edited_by, avatar: false, author_class: nil)
|
2016-03-02 06:33:55 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
output
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-10-07 07:16:19 -04:00
|
|
|
def promo_host
|
|
|
|
'about.gitlab.com'
|
|
|
|
end
|
|
|
|
|
|
|
|
def promo_url
|
|
|
|
'https://' + promo_host
|
|
|
|
end
|
2014-12-23 11:49:39 -05:00
|
|
|
|
2017-06-13 12:46:02 -04:00
|
|
|
def support_url
|
2017-08-31 05:47:03 -04:00
|
|
|
Gitlab::CurrentSettings.current_application_settings.help_page_support_url.presence || promo_url + '/getting-help/'
|
2017-06-13 12:46:02 -04:00
|
|
|
end
|
|
|
|
|
2019-07-22 10:56:40 -04:00
|
|
|
def static_objects_external_storage_enabled?
|
|
|
|
Gitlab::CurrentSettings.static_objects_external_storage_enabled?
|
|
|
|
end
|
|
|
|
|
|
|
|
def external_storage_url_or_path(path, project = @project)
|
2020-01-06 16:07:43 -05:00
|
|
|
return path if @snippet || !static_objects_external_storage_enabled?
|
2019-07-22 10:56:40 -04:00
|
|
|
|
|
|
|
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-03-26 22:13:49 -04:00
|
|
|
def page_filter_path(options = {})
|
|
|
|
without = options.delete(:without)
|
|
|
|
|
2018-11-09 09:29:45 -05:00
|
|
|
options = request.query_parameters.merge(options)
|
2014-12-23 11:49:39 -05:00
|
|
|
|
2015-03-26 22:13:49 -04:00
|
|
|
if without.present?
|
|
|
|
without.each do |key|
|
|
|
|
options.delete(key)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-13 23:13:51 -05:00
|
|
|
"#{request.path}?#{options.compact.to_param}"
|
2014-12-23 11:49:39 -05:00
|
|
|
end
|
2014-12-15 05:11:38 -05:00
|
|
|
|
|
|
|
def outdated_browser?
|
2020-03-10 14:08:17 -04:00
|
|
|
browser.ie?
|
2014-12-15 05:11:38 -05:00
|
|
|
end
|
2014-12-30 05:01:30 -05:00
|
|
|
|
|
|
|
def path_to_key(key, admin = false)
|
|
|
|
if admin
|
|
|
|
admin_user_key_path(@user, key)
|
|
|
|
else
|
|
|
|
profile_key_path(key)
|
|
|
|
end
|
|
|
|
end
|
2016-02-04 11:10:11 -05:00
|
|
|
|
2015-10-06 14:15:06 -04:00
|
|
|
def truncate_first_line(message, length = 50)
|
|
|
|
truncate(message.each_line.first.chomp, length: length) if message
|
|
|
|
end
|
2016-06-20 12:57:10 -04:00
|
|
|
|
|
|
|
# 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-07-22 05:59:26 -04:00
|
|
|
|
|
|
|
def page_class
|
2017-07-31 04:07:51 -04:00
|
|
|
class_names = []
|
|
|
|
class_names << 'issue-boards-page' if current_controller?(:boards)
|
|
|
|
class_names << 'with-performance-bar' if performance_bar_enabled?
|
2018-04-02 10:26:59 -04:00
|
|
|
class_names << system_message_class
|
2017-07-31 04:07:51 -04:00
|
|
|
class_names
|
2016-07-22 05:59:26 -04:00
|
|
|
end
|
2017-02-12 03:02:26 -05:00
|
|
|
|
2018-04-02 10:27:12 -04:00
|
|
|
def system_message_class
|
2018-04-02 10:26:59 -04:00
|
|
|
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-04-02 10:27:12 -04:00
|
|
|
end
|
|
|
|
|
2017-02-12 03:02:26 -05:00
|
|
|
# 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-03-24 08:41:42 -04:00
|
|
|
|
2017-05-24 06:25:44 -04:00
|
|
|
def show_callout?(name)
|
|
|
|
cookies[name] != 'true'
|
2017-03-24 08:41:42 -04:00
|
|
|
end
|
2017-05-08 09:23:22 -04:00
|
|
|
|
|
|
|
def linkedin_url(user)
|
|
|
|
name = user.linkedin
|
2018-03-19 13:53:20 -04:00
|
|
|
if name =~ %r{\Ahttps?://(www\.)?linkedin\.com/in/}
|
2017-05-08 09:23:22 -04:00
|
|
|
name
|
|
|
|
else
|
|
|
|
"https://www.linkedin.com/in/#{name}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-05-08 10:45:59 -04:00
|
|
|
def twitter_url(user)
|
2017-05-08 09:23:22 -04:00
|
|
|
name = user.twitter
|
2018-03-19 13:53:20 -04:00
|
|
|
if name =~ %r{\Ahttps?://(www\.)?twitter\.com/}
|
2017-05-08 09:23:22 -04:00
|
|
|
name
|
|
|
|
else
|
2018-03-19 13:53:20 -04:00
|
|
|
"https://twitter.com/#{name}"
|
2017-05-08 09:23:22 -04:00
|
|
|
end
|
|
|
|
end
|
2017-06-16 06:31:39 -04:00
|
|
|
|
2017-08-07 13:24:52 -04:00
|
|
|
def collapsed_sidebar?
|
|
|
|
cookies["sidebar_collapsed"] == "true"
|
|
|
|
end
|
2017-08-07 23:44:47 -04:00
|
|
|
|
2017-10-18 04:29:18 -04:00
|
|
|
def locale_path
|
|
|
|
asset_path("locale/#{Gitlab::I18n.locale}/app.js")
|
|
|
|
end
|
2018-03-16 08:19:46 -04:00
|
|
|
|
|
|
|
# Overridden in EE
|
|
|
|
def read_only_message
|
|
|
|
return unless Gitlab::Database.read_only?
|
|
|
|
|
|
|
|
_('You are on a read-only GitLab instance.')
|
|
|
|
end
|
2018-04-26 15:53:13 -04:00
|
|
|
|
2018-12-24 05:25:21 -05:00
|
|
|
def client_class_list
|
|
|
|
"gl-browser-#{browser.id} gl-platform-#{browser.platform.id}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def client_js_flags
|
|
|
|
{
|
|
|
|
"is#{browser.id.to_s.titlecase}": true,
|
|
|
|
"is#{browser.platform.id.to_s.titlecase}": true
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2018-04-26 15:53:13 -04:00
|
|
|
def autocomplete_data_sources(object, noteable_type)
|
|
|
|
return {} unless object && noteable_type
|
|
|
|
|
|
|
|
{
|
|
|
|
members: members_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
|
|
|
|
issues: issues_project_autocomplete_sources_path(object),
|
2018-06-27 07:01:56 -04:00
|
|
|
mergeRequests: merge_requests_project_autocomplete_sources_path(object),
|
2018-04-26 15:53:13 -04:00
|
|
|
labels: labels_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
|
|
|
|
milestones: milestones_project_autocomplete_sources_path(object),
|
2018-10-05 05:42:38 -04:00
|
|
|
commands: commands_project_autocomplete_sources_path(object, type: noteable_type, type_id: params[:id]),
|
|
|
|
snippets: snippets_project_autocomplete_sources_path(object)
|
2018-04-26 15:53:13 -04:00
|
|
|
}
|
|
|
|
end
|
2018-04-02 10:26:59 -04:00
|
|
|
|
2019-11-12 16:06:30 -05:00
|
|
|
def asset_to_string(name)
|
|
|
|
app = Rails.application
|
|
|
|
if Rails.configuration.assets.compile
|
|
|
|
app.assets.find_asset(name).to_s
|
|
|
|
else
|
2019-11-13 16:06:45 -05:00
|
|
|
controller.view_context.render(file: Rails.root.join('public/assets', app.assets_manifest.assets[name]).to_s)
|
2019-11-12 16:06:30 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-04-02 10:26:59 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def appearance
|
|
|
|
::Appearance.current
|
|
|
|
end
|
2011-10-08 17:36:38 -04:00
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
|
|
|
ApplicationHelper.prepend_if_ee('EE::ApplicationHelper')
|