# frozen_string_literal: true module LabelsHelper extend self def show_label_issuables_link?(label, issuables_type, current_user: nil) return true unless label.project_label? label.project.feature_available?(issuables_type, current_user) end # Link to a Label # # label - LabelPresenter object to link to # type - The type of item the link will point to (:issue or # :merge_request). If omitted, defaults to :issue. # block - An optional block that will be passed to `link_to`, forming the # body of the link element. If omitted, defaults to # `render_colored_label`. # # Examples: # # # Allow the generated link to use the label's own subject # link_to_label(label) # # # Force the generated link to use a provided group # link_to_label(label, subject: Group.last) # # # Force the generated link to use a provided project # link_to_label(label, subject: Project.last) # # # Force the generated link to point to merge requests instead of issues # link_to_label(label, type: :merge_request) # # # Customize link body with a block # link_to_label(label) { "My Custom Label Text" } # # Returns a String def link_to_label(label, type: :issue, tooltip: true, small: false, css_class: nil, &block) link = label.filter_path(type: type) if block link_to link, class: css_class, &block else render_label(label, link: link, tooltip: tooltip, small: small) end end def render_label(label, link: nil, tooltip: true, dataset: nil, small: false) html = render_colored_label(label) if link title = label_tooltip_title(label) if tooltip html = render_label_link(html, link: link, title: title, dataset: dataset) end wrap_label_html(html, small: small, label: label) end def render_colored_label(label, suffix: '') render_label_text( label.name, suffix: suffix, css_class: "gl-label-text #{label.text_color_class}", bg_color: label.color ) end # We need the `label` argument here for EE def wrap_label_html(label_html, small:, label:) wrapper_classes = %w(gl-label) wrapper_classes << 'gl-label-sm' if small %(#{label_html}).html_safe end def label_tooltip_title(label) Sanitize.clean(label.description) end def suggested_colors { '#009966' => s_('SuggestedColors|Green-cyan'), '#8fbc8f' => s_('SuggestedColors|Dark sea green'), '#3cb371' => s_('SuggestedColors|Medium sea green'), '#00b140' => s_('SuggestedColors|Green screen'), '#013220' => s_('SuggestedColors|Dark green'), '#6699cc' => s_('SuggestedColors|Blue-gray'), '#0000ff' => s_('SuggestedColors|Blue'), '#e6e6fa' => s_('SuggestedColors|Lavender'), '#9400d3' => s_('SuggestedColors|Dark violet'), '#330066' => s_('SuggestedColors|Deep violet'), '#808080' => s_('SuggestedColors|Gray'), '#36454f' => s_('SuggestedColors|Charcoal grey'), '#f7e7ce' => s_('SuggestedColors|Champagne'), '#c21e56' => s_('SuggestedColors|Rose red'), '#cc338b' => s_('SuggestedColors|Magenta-pink'), '#dc143c' => s_('SuggestedColors|Crimson'), '#ff0000' => s_('SuggestedColors|Red'), '#cd5b45' => s_('SuggestedColors|Dark coral'), '#eee600' => s_('SuggestedColors|Titanium yellow'), '#ed9121' => s_('SuggestedColors|Carrot orange'), '#c39953' => s_('SuggestedColors|Aztec Gold') } end def render_suggested_colors colors_html = suggested_colors.map do |color_hex_value, color_name| link_to('', '#', class: "has-tooltip", style: "background-color: #{color_hex_value}", data: { color: color_hex_value }, title: color_name) end content_tag(:div, class: 'suggest-colors') do colors_html.join.html_safe end end def text_color_for_bg(bg_color) ::Gitlab::Color.of(bg_color).contrast end def labels_filter_path_with_defaults(only_group_labels: false, include_ancestor_groups: true, include_descendant_groups: false) options = {} options[:include_ancestor_groups] = include_ancestor_groups if include_ancestor_groups options[:include_descendant_groups] = include_descendant_groups if include_descendant_groups options[:only_group_labels] = only_group_labels if only_group_labels && @group options[:format] = :json labels_filter_path(options) end def labels_filter_path(options = {}) project = @target_project || @project format = options.delete(:format) if project project_labels_path(project, format, options) elsif @group group_labels_path(@group, format, options) else dashboard_labels_path(format, options) end end def label_subscription_status(label, project) return 'group-level' if label.subscribed?(current_user) return 'project-level' if label.subscribed?(current_user, project) 'unsubscribed' end def toggle_subscription_label_path(label, project) return toggle_subscription_group_label_path(label.group, label) unless project case label_subscription_status(label, project) when 'group-level' then toggle_subscription_group_label_path(label.group, label) when 'project-level' then toggle_subscription_project_label_path(project, label) when 'unsubscribed' then toggle_subscription_project_label_path(project, label) end end def label_subscription_toggle_button_text(label, project = nil) label.subscribed?(current_user, project) ? 'Unsubscribe' : 'Subscribe' end def create_label_title(subject) case subject when Group _('Create group label') when Project _('Create project label') else _('Create new label') end end def manage_labels_title(subject) case subject when Group _('Manage group labels') when Project _('Manage project labels') else _('Manage labels') end end def view_labels_title(subject) case subject when Group _('View group labels') when Project _('View project labels') else _('View labels') end end def label_status_tooltip(label, status) type = label.project_label? ? 'project' : 'group' level = status.unsubscribed? ? type : status.sub('-level', '') action = status.unsubscribed? ? 'Subscribe' : 'Unsubscribe' "#{action} at #{level} level" end def presented_labels_sorted_by_title(labels, subject) labels.sort_by(&:title).map { |label| label.present(issuable_subject: subject) } end def label_dropdown_data(project, opts = {}) { toggle: "dropdown", field_name: opts[:field_name] || "label_name[]", show_no: "true", show_any: "true", project_id: project&.try(:id), namespace_path: project&.try(:namespace)&.try(:full_path), project_path: project&.try(:path) }.merge(opts) end def issuable_types ['issues', 'merge requests'] end def show_labels_full_path?(project, group) project || group&.subgroup? end private def render_label_link(label_html, link:, title:, dataset:) classes = %w(gl-link gl-label-link) dataset ||= {} if title.present? classes << 'has-tooltip' dataset.merge!(html: true, title: title) end link_to(label_html, link, class: classes.join(' '), data: dataset) end def render_label_text(name, suffix: '', css_class: nil, bg_color: nil) <<~HTML.chomp.html_safe #{ERB::Util.html_escape_once(name)}#{suffix} HTML end end LabelsHelper.prepend_mod_with('LabelsHelper')