2011-10-08 17:36:38 -04:00
|
|
|
module IssuesHelper
|
2014-09-25 18:07:40 -04:00
|
|
|
def issue_css_classes(issue)
|
2012-06-21 01:29:53 -04:00
|
|
|
classes = "issue"
|
2013-02-18 04:10:58 -05:00
|
|
|
classes << " closed" if issue.closed?
|
2012-06-21 01:29:53 -04:00
|
|
|
classes << " today" if issue.today?
|
|
|
|
classes
|
|
|
|
end
|
2012-06-27 14:20:35 -04:00
|
|
|
|
2012-08-16 18:13:22 -04:00
|
|
|
# Returns an OpenStruct object suitable for use by <tt>options_from_collection_for_select</tt>
|
|
|
|
# to allow filtering issues by an unassigned User or Milestone
|
|
|
|
def unassigned_filter
|
|
|
|
# Milestone uses :title, Issue uses :name
|
2013-06-22 11:19:00 -04:00
|
|
|
OpenStruct.new(id: 0, title: 'None (backlog)', name: 'Unassigned')
|
2012-08-13 20:49:18 -04:00
|
|
|
end
|
2012-10-09 15:09:46 -04:00
|
|
|
|
2015-03-25 05:03:55 -04:00
|
|
|
def url_for_project_issues(project = @project, options = {})
|
2014-05-26 08:40:10 -04:00
|
|
|
return '' if project.nil?
|
2013-02-19 08:06:40 -05:00
|
|
|
|
2016-04-21 11:13:14 -04:00
|
|
|
url =
|
|
|
|
if options[:only_path]
|
|
|
|
project.issues_tracker.project_path
|
|
|
|
else
|
|
|
|
project.issues_tracker.project_url
|
|
|
|
end
|
|
|
|
|
|
|
|
# Ensure we return a valid URL to prevent possible XSS.
|
|
|
|
URI.parse(url).to_s
|
|
|
|
rescue URI::InvalidURIError
|
|
|
|
''
|
2013-02-19 08:06:40 -05:00
|
|
|
end
|
|
|
|
|
2015-03-25 05:03:55 -04:00
|
|
|
def url_for_new_issue(project = @project, options = {})
|
2014-05-26 08:40:10 -04:00
|
|
|
return '' if project.nil?
|
2013-03-26 04:27:34 -04:00
|
|
|
|
2016-04-21 11:13:14 -04:00
|
|
|
url =
|
|
|
|
if options[:only_path]
|
|
|
|
project.issues_tracker.new_issue_path
|
|
|
|
else
|
|
|
|
project.issues_tracker.new_issue_url
|
|
|
|
end
|
|
|
|
|
|
|
|
# Ensure we return a valid URL to prevent possible XSS.
|
|
|
|
URI.parse(url).to_s
|
|
|
|
rescue URI::InvalidURIError
|
|
|
|
''
|
2013-03-26 04:27:34 -04:00
|
|
|
end
|
|
|
|
|
2015-03-25 05:03:55 -04:00
|
|
|
def url_for_issue(issue_iid, project = @project, options = {})
|
2014-05-26 08:40:10 -04:00
|
|
|
return '' if project.nil?
|
2013-02-11 08:32:29 -05:00
|
|
|
|
2016-04-21 11:13:14 -04:00
|
|
|
url =
|
|
|
|
if options[:only_path]
|
|
|
|
project.issues_tracker.issue_path(issue_iid)
|
|
|
|
else
|
|
|
|
project.issues_tracker.issue_url(issue_iid)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Ensure we return a valid URL to prevent possible XSS.
|
|
|
|
URI.parse(url).to_s
|
|
|
|
rescue URI::InvalidURIError
|
|
|
|
''
|
2013-01-23 09:13:28 -05:00
|
|
|
end
|
|
|
|
|
2013-11-25 11:57:44 -05:00
|
|
|
def bulk_update_milestone_options
|
2016-01-23 17:31:46 -05:00
|
|
|
milestones = @project.milestones.active.reorder(due_date: :asc, title: :asc).to_a
|
2015-12-02 08:11:32 -05:00
|
|
|
milestones.unshift(Milestone::None)
|
|
|
|
|
|
|
|
options_from_collection_for_select(milestones, 'id', 'title', params[:milestone_id])
|
2013-11-25 11:57:44 -05:00
|
|
|
end
|
|
|
|
|
2014-09-25 18:07:40 -04:00
|
|
|
def milestone_options(object)
|
2016-01-23 17:31:46 -05:00
|
|
|
milestones = object.project.milestones.active.reorder(due_date: :asc, title: :asc).to_a
|
2016-04-06 10:02:22 -04:00
|
|
|
milestones.unshift(object.milestone) if object.milestone.present? && object.milestone.closed?
|
2015-12-02 08:20:43 -05:00
|
|
|
milestones.unshift(Milestone::None)
|
|
|
|
|
|
|
|
options_from_collection_for_select(milestones, 'id', 'title', object.milestone_id)
|
2013-12-19 12:54:57 -05:00
|
|
|
end
|
2014-02-18 15:05:44 -05:00
|
|
|
|
2016-03-15 10:01:26 -04:00
|
|
|
def project_options(issuable, current_user, ability: :read_project)
|
|
|
|
projects = current_user.authorized_projects
|
|
|
|
projects = projects.select do |project|
|
2016-03-16 05:24:44 -04:00
|
|
|
current_user.can?(ability, project)
|
2016-03-15 10:01:26 -04:00
|
|
|
end
|
|
|
|
|
2016-03-16 05:24:44 -04:00
|
|
|
no_project = OpenStruct.new(id: 0, name_with_namespace: 'No project')
|
|
|
|
projects.unshift(no_project)
|
|
|
|
projects.delete(issuable.project)
|
2016-03-15 10:01:26 -04:00
|
|
|
|
2016-03-16 05:24:44 -04:00
|
|
|
options_from_collection_for_select(projects, :id, :name_with_namespace)
|
2016-03-15 10:01:26 -04:00
|
|
|
end
|
|
|
|
|
2015-12-16 10:13:22 -05:00
|
|
|
def status_box_class(item)
|
2014-02-27 14:04:44 -05:00
|
|
|
if item.respond_to?(:expired?) && item.expired?
|
2015-12-16 10:13:22 -05:00
|
|
|
'status-box-expired'
|
2014-02-27 14:04:44 -05:00
|
|
|
elsif item.respond_to?(:merged?) && item.merged?
|
2015-12-16 10:13:22 -05:00
|
|
|
'status-box-merged'
|
2014-02-27 14:04:44 -05:00
|
|
|
elsif item.closed?
|
2015-12-16 10:13:22 -05:00
|
|
|
'status-box-closed'
|
2014-02-18 15:05:44 -05:00
|
|
|
else
|
2015-12-16 10:13:22 -05:00
|
|
|
'status-box-open'
|
2014-02-18 15:05:44 -05:00
|
|
|
end
|
|
|
|
end
|
2014-12-04 15:14:20 -05:00
|
|
|
|
2016-01-23 17:31:46 -05:00
|
|
|
def issue_button_visibility(issue, closed)
|
2015-12-21 13:06:09 -05:00
|
|
|
return 'hidden' if issue.closed? == closed
|
|
|
|
end
|
|
|
|
|
2015-10-12 06:04:20 -04:00
|
|
|
def merge_requests_sentence(merge_requests)
|
2015-11-30 15:09:36 -05:00
|
|
|
# Sorting based on the `!123` or `group/project!123` reference will sort
|
|
|
|
# local merge requests first.
|
|
|
|
merge_requests.map do |merge_request|
|
|
|
|
merge_request.to_reference(@project)
|
|
|
|
end.sort.to_sentence(last_word_connector: ', or ')
|
2015-10-12 06:04:20 -04:00
|
|
|
end
|
|
|
|
|
2016-03-17 17:00:03 -04:00
|
|
|
def confidential_icon(issue)
|
|
|
|
icon('eye-slash') if issue.confidential?
|
|
|
|
end
|
|
|
|
|
2016-04-07 19:40:22 -04:00
|
|
|
def emoji_icon(name, unicode = nil, aliases = [], sprite: true)
|
2016-01-04 07:06:48 -05:00
|
|
|
unicode ||= Emoji.emoji_filename(name) rescue ""
|
2015-12-17 07:29:55 -05:00
|
|
|
|
2016-04-07 19:40:22 -04:00
|
|
|
data = {
|
|
|
|
aliases: aliases.join(" "),
|
|
|
|
emoji: name,
|
|
|
|
unicode_name: unicode
|
|
|
|
}
|
|
|
|
|
|
|
|
if sprite
|
|
|
|
# Emoji icons for the emoji menu, these use a spritesheet.
|
|
|
|
content_tag :div, "",
|
|
|
|
class: "icon emoji-icon emoji-#{unicode}",
|
|
|
|
title: name,
|
|
|
|
data: data
|
2016-04-25 14:10:20 -04:00
|
|
|
else
|
2016-04-07 19:40:22 -04:00
|
|
|
# Emoji icons displayed separately, used for the awards already given
|
|
|
|
# to an issue or merge request.
|
|
|
|
content_tag :img, "",
|
|
|
|
class: "icon emoji",
|
|
|
|
title: name,
|
|
|
|
height: "20px",
|
|
|
|
width: "20px",
|
|
|
|
src: url_to_image("#{unicode}.png"),
|
|
|
|
data: data
|
|
|
|
end
|
2015-11-11 08:12:51 -05:00
|
|
|
end
|
|
|
|
|
2016-04-16 15:09:08 -04:00
|
|
|
def award_user_list(awards, current_user)
|
2016-04-25 14:10:20 -04:00
|
|
|
awards.map do |award|
|
|
|
|
award.user == current_user ? 'me' : award.user.name
|
|
|
|
end.join(', ')
|
2015-11-18 08:43:53 -05:00
|
|
|
end
|
|
|
|
|
2016-04-16 15:09:08 -04:00
|
|
|
def award_active_class(awards, current_user)
|
|
|
|
if current_user && awards.find { |a| a.user_id == current_user.id }
|
2015-11-19 06:20:09 -05:00
|
|
|
"active"
|
|
|
|
else
|
|
|
|
""
|
|
|
|
end
|
2015-11-18 17:59:07 -05:00
|
|
|
end
|
|
|
|
|
2015-12-25 05:08:53 -05:00
|
|
|
def awards_sort(awards)
|
|
|
|
awards.sort_by do |award, notes|
|
|
|
|
if award == "thumbsup"
|
|
|
|
0
|
|
|
|
elsif award == "thumbsdown"
|
|
|
|
1
|
|
|
|
else
|
|
|
|
2
|
|
|
|
end
|
|
|
|
end.to_h
|
|
|
|
end
|
|
|
|
|
2016-03-10 09:26:56 -05:00
|
|
|
def due_date_options
|
|
|
|
options = [
|
2016-04-19 07:03:28 -04:00
|
|
|
Issue::AnyDueDate,
|
|
|
|
Issue::NoDueDate,
|
|
|
|
Issue::DueThisWeek,
|
|
|
|
Issue::DueThisMonth,
|
|
|
|
Issue::Overdue
|
2016-03-10 09:26:56 -05:00
|
|
|
]
|
2016-03-18 13:17:01 -04:00
|
|
|
|
|
|
|
options_from_collection_for_select(options, 'name', 'title', params[:due_date])
|
2016-03-10 09:26:56 -05:00
|
|
|
end
|
|
|
|
|
2015-12-15 10:10:32 -05:00
|
|
|
# Required for Banzai::Filter::IssueReferenceFilter
|
2015-04-30 17:27:33 -04:00
|
|
|
module_function :url_for_issue
|
2011-10-08 17:36:38 -04:00
|
|
|
end
|