417 lines
13 KiB
Ruby
417 lines
13 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module IssuablesHelper
|
|
include GitlabRoutingHelper
|
|
include IssuablesDescriptionTemplatesHelper
|
|
|
|
def sidebar_gutter_toggle_icon
|
|
content_tag(:span, class: 'js-sidebar-toggle-container', data: { is_expanded: !sidebar_gutter_collapsed? }) do
|
|
sprite_icon('chevron-double-lg-left', css_class: "js-sidebar-expand #{'hidden' unless sidebar_gutter_collapsed?}") +
|
|
sprite_icon('chevron-double-lg-right', css_class: "js-sidebar-collapse #{'hidden' if sidebar_gutter_collapsed?}")
|
|
end
|
|
end
|
|
|
|
def sidebar_gutter_collapsed_class
|
|
"right-sidebar-#{sidebar_gutter_collapsed? ? 'collapsed' : 'expanded'}"
|
|
end
|
|
|
|
def sidebar_gutter_tooltip_text
|
|
sidebar_gutter_collapsed? ? _('Expand sidebar') : _('Collapse sidebar')
|
|
end
|
|
|
|
def assignees_label(issuable, include_value: true)
|
|
assignees = issuable.assignees
|
|
|
|
if include_value
|
|
sanitized_list = sanitize_name(issuable.assignee_list)
|
|
ns_('NotificationEmail|Assignee: %{users}', 'NotificationEmail|Assignees: %{users}', assignees.count) % { users: sanitized_list }
|
|
else
|
|
ns_('NotificationEmail|Assignee', 'NotificationEmail|Assignees', assignees.count)
|
|
end
|
|
end
|
|
|
|
def sidebar_milestone_tooltip_label(milestone)
|
|
return _('Milestone') unless milestone.present?
|
|
|
|
[escape_once(milestone[:title]), sidebar_milestone_remaining_days(milestone) || _('Milestone')].join('<br/>')
|
|
end
|
|
|
|
def sidebar_milestone_remaining_days(milestone)
|
|
due_date_with_remaining_days(milestone[:due_date], milestone[:start_date])
|
|
end
|
|
|
|
def sidebar_due_date_tooltip_label(due_date)
|
|
[_('Due date'), due_date_with_remaining_days(due_date)].compact.join('<br/>')
|
|
end
|
|
|
|
def due_date_with_remaining_days(due_date, start_date = nil)
|
|
return unless due_date
|
|
|
|
"#{due_date.to_s(:medium)} (#{remaining_days_in_words(due_date, start_date)})"
|
|
end
|
|
|
|
def multi_label_name(current_labels, default_label)
|
|
return default_label if current_labels.blank?
|
|
|
|
title = current_labels.first.try(:title) || current_labels.first[:title]
|
|
|
|
if current_labels.size > 1
|
|
"#{title} +#{current_labels.size - 1} more"
|
|
else
|
|
title
|
|
end
|
|
end
|
|
|
|
def serialize_issuable(issuable, opts = {})
|
|
serializer_klass = case issuable
|
|
when Issue
|
|
IssueSerializer
|
|
when MergeRequest
|
|
MergeRequestSerializer
|
|
end
|
|
|
|
serializer_klass
|
|
.new(current_user: current_user, project: issuable.project)
|
|
.represent(issuable, opts)
|
|
.to_json
|
|
end
|
|
|
|
def users_dropdown_label(selected_users)
|
|
case selected_users.length
|
|
when 0
|
|
"Unassigned"
|
|
when 1
|
|
selected_users[0].name
|
|
else
|
|
"#{selected_users[0].name} + #{selected_users.length - 1} more"
|
|
end
|
|
end
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def user_dropdown_label(user_id, default_label)
|
|
return default_label if user_id.nil?
|
|
return "Unassigned" if user_id == "0"
|
|
|
|
user = User.find_by(id: user_id)
|
|
|
|
if user
|
|
user.name
|
|
else
|
|
default_label
|
|
end
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def project_dropdown_label(project_id, default_label)
|
|
return default_label if project_id.nil?
|
|
return "Any project" if project_id == "0"
|
|
|
|
project = Project.find_by(id: project_id)
|
|
|
|
if project
|
|
project.full_name
|
|
else
|
|
default_label
|
|
end
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def group_dropdown_label(group_id, default_label)
|
|
return default_label if group_id.nil?
|
|
return "Any group" if group_id == "0"
|
|
|
|
group = ::Group.find_by(id: group_id)
|
|
|
|
if group
|
|
group.full_name
|
|
else
|
|
default_label
|
|
end
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
def milestone_dropdown_label(milestone_title, default_label = "Milestone")
|
|
title =
|
|
case milestone_title
|
|
when Milestone::Upcoming.name then Milestone::Upcoming.title
|
|
when Milestone::Started.name then Milestone::Started.title
|
|
else milestone_title.presence
|
|
end
|
|
|
|
h(title || default_label)
|
|
end
|
|
|
|
def issuable_meta_author_status(author)
|
|
return "" unless show_status_emoji?(author&.status) && status = user_status(author)
|
|
|
|
"#{status}".html_safe
|
|
end
|
|
|
|
def issuable_meta(issuable, project)
|
|
output = []
|
|
output << "Created #{time_ago_with_tooltip(issuable.created_at)} by ".html_safe
|
|
|
|
if issuable.is_a?(Issue) && issuable.service_desk_reply_to
|
|
output << "#{html_escape(issuable.service_desk_reply_to)} via "
|
|
end
|
|
|
|
output << content_tag(:strong) do
|
|
author_output = link_to_member(project, issuable.author, size: 24, mobile_classes: "d-none d-sm-inline")
|
|
author_output << link_to_member(project, issuable.author, size: 24, by_username: true, avatar: false, mobile_classes: "d-inline d-sm-none")
|
|
|
|
author_output << issuable_meta_author_slot(issuable.author, css_class: 'ml-1')
|
|
author_output << issuable_meta_author_status(issuable.author)
|
|
|
|
author_output
|
|
end
|
|
|
|
if access = project.team.human_max_access(issuable.author_id)
|
|
output << content_tag(:span, access, class: "user-access-role has-tooltip d-none d-xl-inline-block gl-ml-3 ", title: _("This user has the %{access} role in the %{name} project.") % { access: access.downcase, name: project.name })
|
|
elsif project.team.contributor?(issuable.author_id)
|
|
output << content_tag(:span, _("Contributor"), class: "user-access-role has-tooltip d-none d-xl-inline-block gl-ml-3", title: _("This user has previously committed to the %{name} project.") % { name: project.name })
|
|
end
|
|
|
|
output << content_tag(:span, (sprite_icon('first-contribution', css_class: 'gl-icon gl-vertical-align-middle') if issuable.first_contribution?), class: 'has-tooltip gl-ml-2', title: _('1st contribution!'))
|
|
|
|
output << content_tag(:span, (issuable.task_status if issuable.tasks?), id: "task_status", class: "d-none d-md-inline-block gl-ml-3")
|
|
output << content_tag(:span, (issuable.task_status_short if issuable.tasks?), id: "task_status_short", class: "d-md-none")
|
|
|
|
output.join.html_safe
|
|
end
|
|
|
|
# This is a dummy method, and has an override defined in ee
|
|
def issuable_meta_author_slot(author, css_class: nil)
|
|
nil
|
|
end
|
|
|
|
def issuables_state_counter_text(issuable_type, state, display_count)
|
|
titles = {
|
|
opened: "Open"
|
|
}
|
|
|
|
state_title = titles[state] || state.to_s.humanize
|
|
html = content_tag(:span, state_title)
|
|
|
|
return html.html_safe unless display_count
|
|
|
|
count = issuables_count_for_state(issuable_type, state)
|
|
|
|
if count != -1
|
|
html << " " << content_tag(:span, number_with_delimiter(count), class: 'badge badge-pill')
|
|
end
|
|
|
|
html.html_safe
|
|
end
|
|
|
|
def assigned_issuables_count(issuable_type)
|
|
case issuable_type
|
|
when :issues
|
|
current_user.assigned_open_issues_count
|
|
when :merge_requests
|
|
current_user.assigned_open_merge_requests_count
|
|
else
|
|
raise ArgumentError, "invalid issuable `#{issuable_type}`"
|
|
end
|
|
end
|
|
|
|
def issuable_reference(issuable)
|
|
@show_full_reference ? issuable.to_reference(full: true) : issuable.to_reference(@group || @project)
|
|
end
|
|
|
|
def issuable_initial_data(issuable)
|
|
data = {
|
|
endpoint: issuable_path(issuable),
|
|
updateEndpoint: "#{issuable_path(issuable)}.json",
|
|
canUpdate: can?(current_user, :"update_#{issuable.to_ability_name}", issuable),
|
|
canDestroy: can?(current_user, :"destroy_#{issuable.to_ability_name}", issuable),
|
|
issuableRef: issuable.to_reference,
|
|
markdownPreviewPath: preview_markdown_path(parent),
|
|
markdownDocsPath: help_page_path('user/markdown'),
|
|
lockVersion: issuable.lock_version,
|
|
issuableTemplateNamesPath: template_names_path(parent, issuable),
|
|
initialTitleHtml: markdown_field(issuable, :title),
|
|
initialTitleText: issuable.title,
|
|
initialDescriptionHtml: markdown_field(issuable, :description),
|
|
initialDescriptionText: issuable.description,
|
|
initialTaskStatus: issuable.task_status
|
|
}
|
|
data.merge!(issue_only_initial_data(issuable))
|
|
data.merge!(path_data(parent))
|
|
data.merge!(updated_at_by(issuable))
|
|
|
|
data
|
|
end
|
|
|
|
def issue_only_initial_data(issuable)
|
|
return {} unless issuable.is_a?(Issue)
|
|
|
|
{
|
|
hasClosingMergeRequest: issuable.merge_requests_count(current_user) != 0,
|
|
issueType: issuable.issue_type,
|
|
zoomMeetingUrl: ZoomMeeting.canonical_meeting_url(issuable),
|
|
sentryIssueIdentifier: SentryIssue.find_by(issue: issuable)&.sentry_issue_identifier, # rubocop:disable CodeReuse/ActiveRecord
|
|
iid: issuable.iid.to_s
|
|
}
|
|
end
|
|
|
|
def path_data(parent)
|
|
return { groupPath: parent.path } if parent.is_a?(Group)
|
|
|
|
{
|
|
projectPath: ref_project.path,
|
|
projectId: ref_project.id,
|
|
projectNamespace: ref_project.namespace.full_path
|
|
}
|
|
end
|
|
|
|
def updated_at_by(issuable)
|
|
return {} unless issuable.edited?
|
|
|
|
{
|
|
updatedAt: issuable.last_edited_at.to_time.iso8601,
|
|
updatedBy: {
|
|
name: issuable.last_edited_by.name,
|
|
path: user_path(issuable.last_edited_by)
|
|
}
|
|
}
|
|
end
|
|
|
|
def issuables_count_for_state(issuable_type, state)
|
|
Gitlab::IssuablesCountForState.new(finder)[state]
|
|
end
|
|
|
|
def close_issuable_path(issuable)
|
|
issuable_path(issuable, close_reopen_params(issuable, :close))
|
|
end
|
|
|
|
def reopen_issuable_path(issuable)
|
|
issuable_path(issuable, close_reopen_params(issuable, :reopen))
|
|
end
|
|
|
|
def issuable_path(issuable, *options)
|
|
polymorphic_path(issuable, *options)
|
|
end
|
|
|
|
def issuable_author_is_current_user(issuable)
|
|
issuable.author == current_user
|
|
end
|
|
|
|
def issuable_display_type(issuable)
|
|
case issuable
|
|
when Issue
|
|
issuable.issue_type.downcase
|
|
when MergeRequest
|
|
issuable.model_name.human.downcase
|
|
end
|
|
end
|
|
|
|
def has_filter_bar_param?
|
|
finder.class.scalar_params.any? { |p| params[p].present? }
|
|
end
|
|
|
|
def assignee_sidebar_data(assignee, merge_request: nil)
|
|
{ avatar_url: assignee.avatar_url, name: assignee.name, username: assignee.username }.tap do |data|
|
|
data[:can_merge] = merge_request.can_be_merged_by?(assignee) if merge_request
|
|
data[:availability] = assignee.status.availability if assignee.association(:status).loaded? && assignee.status&.availability
|
|
end
|
|
end
|
|
|
|
def reviewer_sidebar_data(reviewer, merge_request: nil)
|
|
{ avatar_url: reviewer.avatar_url, name: reviewer.name, username: reviewer.username }.tap do |data|
|
|
data[:can_merge] = merge_request.can_be_merged_by?(reviewer) if merge_request
|
|
end
|
|
end
|
|
|
|
def issuable_squash_option?(issuable, project)
|
|
if issuable.persisted?
|
|
issuable.squash
|
|
else
|
|
project.squash_enabled_by_default?
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def sidebar_gutter_collapsed?
|
|
cookies[:collapsed_gutter] == 'true'
|
|
end
|
|
|
|
def issuable_todo_button_data(issuable, is_collapsed)
|
|
{
|
|
todo_text: _('Add a to do'),
|
|
mark_text: _('Mark as done'),
|
|
todo_icon: sprite_icon('todo-add'),
|
|
mark_icon: sprite_icon('todo-done', css_class: 'todo-undone'),
|
|
issuable_id: issuable[:id],
|
|
issuable_type: issuable[:type],
|
|
create_path: issuable[:create_todo_path],
|
|
delete_path: issuable.dig(:current_user, :todo, :delete_path),
|
|
placement: is_collapsed ? 'left' : nil,
|
|
container: is_collapsed ? 'body' : nil,
|
|
boundary: 'viewport',
|
|
is_collapsed: is_collapsed,
|
|
track_label: "right_sidebar",
|
|
track_property: "update_todo",
|
|
track_event: "click_button",
|
|
track_value: ""
|
|
}
|
|
end
|
|
|
|
def close_reopen_params(issuable, action)
|
|
{
|
|
issuable.model_name.to_s.underscore => { state_event: action }
|
|
}.tap do |params|
|
|
params[:format] = :json if issuable.is_a?(Issue)
|
|
end
|
|
end
|
|
|
|
def labels_path
|
|
if @project
|
|
project_labels_path(@project)
|
|
elsif @group
|
|
group_labels_path(@group)
|
|
end
|
|
end
|
|
|
|
def issuable_sidebar_options(issuable)
|
|
{
|
|
endpoint: "#{issuable[:issuable_json_path]}?serializer=sidebar_extras",
|
|
toggleSubscriptionEndpoint: issuable[:toggle_subscription_path],
|
|
moveIssueEndpoint: issuable[:move_issue_path],
|
|
projectsAutocompleteEndpoint: issuable[:projects_autocomplete_path],
|
|
editable: issuable.dig(:current_user, :can_edit),
|
|
currentUser: issuable[:current_user],
|
|
rootPath: root_path,
|
|
fullPath: issuable[:project_full_path],
|
|
iid: issuable[:iid],
|
|
severity: issuable[:severity],
|
|
timeTrackingLimitToHours: Gitlab::CurrentSettings.time_tracking_limit_to_hours,
|
|
createNoteEmail: issuable[:create_note_email],
|
|
issuableType: issuable[:type],
|
|
projectMembersPath: project_project_members_path(@project, sort: :access_level_desc)
|
|
}
|
|
end
|
|
|
|
def sidebar_labels_data(issuable_sidebar, project)
|
|
{
|
|
allow_label_create: issuable_sidebar.dig(:current_user, :can_admin_label).to_s,
|
|
allow_scoped_labels: issuable_sidebar[:scoped_labels_available].to_s,
|
|
can_edit: issuable_sidebar.dig(:current_user, :can_edit).to_s,
|
|
iid: issuable_sidebar[:iid],
|
|
issuable_type: issuable_sidebar[:type],
|
|
labels_fetch_path: issuable_sidebar[:project_labels_path],
|
|
labels_manage_path: project_labels_path(project),
|
|
project_issues_path: issuable_sidebar[:project_issuables_path],
|
|
project_path: project.full_path,
|
|
selected_labels: issuable_sidebar[:labels].to_json
|
|
}
|
|
end
|
|
|
|
def parent
|
|
@project || @group
|
|
end
|
|
end
|
|
|
|
IssuablesHelper.prepend_if_ee('EE::IssuablesHelper')
|