module IssuesHelper def issue_css_classes(issue) classes = "issue" classes << " closed" if issue.closed? classes << " today" if issue.today? classes end # Returns an OpenStruct object suitable for use by options_from_collection_for_select # to allow filtering issues by an unassigned User or Milestone def unassigned_filter # Milestone uses :title, Issue uses :name OpenStruct.new(id: 0, title: 'None (backlog)', name: 'Unassigned') end def url_for_issue(issue_iid, project = @project, options = {}) return '' if project.nil? 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 '' end def bulk_update_milestone_options milestones = @project.milestones.active.reorder(due_date: :asc, title: :asc).to_a milestones.unshift(Milestone::None) options_from_collection_for_select(milestones, 'id', 'title', params[:milestone_id]) end def milestone_options(object) milestones = object.project.milestones.active.reorder(due_date: :asc, title: :asc).to_a milestones.unshift(object.milestone) if object.milestone.present? && object.milestone.closed? milestones.unshift(Milestone::None) options_from_collection_for_select(milestones, 'id', 'title', object.milestone_id) end def project_options(issuable, current_user, ability: :read_project) projects = current_user.authorized_projects projects = projects.select do |project| current_user.can?(ability, project) end no_project = OpenStruct.new(id: 0, name_with_namespace: 'No project') projects.unshift(no_project) projects.delete(issuable.project) options_from_collection_for_select(projects, :id, :name_with_namespace) end def status_box_class(item) if item.respond_to?(:expired?) && item.expired? 'status-box-expired' elsif item.respond_to?(:merged?) && item.merged? 'status-box-merged' elsif item.closed? 'status-box-closed' elsif item.respond_to?(:upcoming?) && item.upcoming? 'status-box-upcoming' else 'status-box-open' end end def issue_button_visibility(issue, closed) return 'hidden' if issue.closed? == closed end def merge_requests_sentence(merge_requests) # 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 ') end def confidential_icon(issue) icon('eye-slash') if issue.confidential? end def emoji_icon(name, unicode = nil, aliases = [], sprite: true) unicode ||= Gitlab::Emoji.emoji_filename(name) rescue "" 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 else # 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 end def award_user_list(awards, current_user, limit: 10) names = awards.map do |award| award.user == current_user ? 'You' : award.user.name end current_user_name = names.delete('You') names = names.insert(0, current_user_name).compact.first(limit) names << "#{awards.size - names.size} more." if awards.size > names.size names.to_sentence end def award_active_class(awards, current_user) if current_user && awards.find { |a| a.user_id == current_user.id } "active" else "" end end 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 def due_date_options options = [ Issue::AnyDueDate, Issue::NoDueDate, Issue::DueThisWeek, Issue::DueThisMonth, Issue::Overdue ] options_from_collection_for_select(options, 'name', 'title', params[:due_date]) end # Required for Banzai::Filter::IssueReferenceFilter module_function :url_for_issue end