Commit eaa84df8 authored by Heinrich Lee Yu's avatar Heinrich Lee Yu

Merge branch 'pl-work-item-prep-work' into 'master'

Move helper methods related to issue types into a helper module

See merge request gitlab-org/gitlab!73823
parents 9d34f894 d44bd660
# frozen_string_literal: true # frozen_string_literal: true
module IssuesHelper module IssuesHelper
include Issues::IssueTypeHelpers
def issue_css_classes(issue) def issue_css_classes(issue)
classes = ["issue"] classes = ["issue"]
classes << "closed" if issue.closed? classes << "closed" if issue.closed?
......
# frozen_string_literal: true
module Issues
module IssueTypeHelpers
# @param object [Issue, Project]
# @param issue_type [String, Symbol]
def create_issue_type_allowed?(object, issue_type)
WorkItem::Type.base_types.key?(issue_type.to_s) &&
can?(current_user, :"create_#{issue_type}", object)
end
end
end
...@@ -3,6 +3,7 @@ ...@@ -3,6 +3,7 @@
module Issues module Issues
class BaseService < ::IssuableBaseService class BaseService < ::IssuableBaseService
include IncidentManagement::UsageData include IncidentManagement::UsageData
include IssueTypeHelpers
def hook_data(issue, action, old_associations: {}) def hook_data(issue, action, old_associations: {})
hook_data = issue.to_hook_data(current_user, old_associations: old_associations) hook_data = issue.to_hook_data(current_user, old_associations: old_associations)
...@@ -44,7 +45,7 @@ module Issues ...@@ -44,7 +45,7 @@ module Issues
def filter_params(issue) def filter_params(issue)
super super
params.delete(:issue_type) unless issue_type_allowed?(issue) params.delete(:issue_type) unless create_issue_type_allowed?(issue, params[:issue_type])
filter_incident_label(issue) if params[:issue_type] filter_incident_label(issue) if params[:issue_type]
moved_issue = params.delete(:moved_issue) moved_issue = params.delete(:moved_issue)
...@@ -89,12 +90,6 @@ module Issues ...@@ -89,12 +90,6 @@ module Issues
Milestones::IssuesCountService.new(milestone).delete_cache Milestones::IssuesCountService.new(milestone).delete_cache
end end
# @param object [Issue, Project]
def issue_type_allowed?(object)
WorkItem::Type.base_types.key?(params[:issue_type]) &&
can?(current_user, :"create_#{params[:issue_type]}", object)
end
# @param issue [Issue] # @param issue [Issue]
def filter_incident_label(issue) def filter_incident_label(issue)
return unless add_incident_label?(issue) || remove_incident_label?(issue) return unless add_incident_label?(issue) || remove_incident_label?(issue)
......
...@@ -80,7 +80,7 @@ module Issues ...@@ -80,7 +80,7 @@ module Issues
] ]
allowed_params << :milestone_id if can?(current_user, :admin_issue, project) allowed_params << :milestone_id if can?(current_user, :admin_issue, project)
allowed_params << :issue_type if issue_type_allowed?(project) allowed_params << :issue_type if create_issue_type_allowed?(project, params[:issue_type])
params.slice(*allowed_params) params.slice(*allowed_params)
end end
......
...@@ -18,17 +18,19 @@ ...@@ -18,17 +18,19 @@
= sprite_icon('close', size: 16, css_class: 'dropdown-menu-close-icon') = sprite_icon('close', size: 16, css_class: 'dropdown-menu-close-icon')
.dropdown-content{ data: { testid: 'issue-type-select-dropdown' } } .dropdown-content{ data: { testid: 'issue-type-select-dropdown' } }
%ul %ul
%li.js-filter-issuable-type - if create_issue_type_allowed?(@project, :issue)
= link_to new_project_issue_path(@project), class: ("is-active" if issuable.issue?) do %li.js-filter-issuable-type
#{sprite_icon(work_item_type_icon(:issue), css_class: 'gl-icon')} #{_("Issue")} = link_to new_project_issue_path(@project), class: ("is-active" if issuable.issue?) do
%li.js-filter-issuable-type{ data: { track: { action: "select_issue_type_incident", label: "select_issue_type_incident_dropdown_option" } } } #{sprite_icon(work_item_type_icon(:issue), css_class: 'gl-icon')} #{_('Issue')}
= link_to new_project_issue_path(@project, { issuable_template: 'incident', issue: { issue_type: 'incident' } }), class: ("is-active" if issuable.incident?) do - if create_issue_type_allowed?(@project, :incident)
#{sprite_icon(work_item_type_icon(:incident), css_class: 'gl-icon')} #{_("Incident")} %li.js-filter-issuable-type{ data: { track: { action: "select_issue_type_incident", label: "select_issue_type_incident_dropdown_option" } } }
= link_to new_project_issue_path(@project, { issuable_template: 'incident', issue: { issue_type: 'incident' } }), class: ("is-active" if issuable.incident?) do
#{sprite_icon(work_item_type_icon(:incident), css_class: 'gl-icon')} #{_('Incident')}
#js-type-popover #js-type-popover
- if issuable.incident? - if issuable.incident?
%p.form-text.text-muted %p.form-text.text-muted
- incident_docs_url = help_page_path('operations/incident_management/incidents.md') - incident_docs_url = help_page_path('operations/incident_management/incidents.md')
- incident_docs_start = '<a href="%{url}" target="_blank" rel="noopener noreferrer">'.html_safe % { url: incident_docs_url } - incident_docs_start = format('<a href="%{url}" target="_blank" rel="noopener noreferrer">', url: incident_docs_url)
= _('A %{incident_docs_start}modified issue%{incident_docs_end} to guide the resolution of incidents.').html_safe % { incident_docs_start: incident_docs_start, incident_docs_end: '</a>'.html_safe } = format(_('A %{incident_docs_start}modified issue%{incident_docs_end} to guide the resolution of incidents.'), incident_docs_start: incident_docs_start, incident_docs_end: '</a>').html_safe
...@@ -4,25 +4,29 @@ require 'spec_helper' ...@@ -4,25 +4,29 @@ require 'spec_helper'
RSpec.describe 'New/edit issue', :js do RSpec.describe 'New/edit issue', :js do
include ActionView::Helpers::JavaScriptHelper include ActionView::Helpers::JavaScriptHelper
include FormHelper
let_it_be(:project) { create(:project) } let_it_be(:project) { create(:project) }
let_it_be(:user) { create(:user)} let_it_be(:user) { create(:user) }
let_it_be(:user2) { create(:user)} let_it_be(:user2) { create(:user) }
let_it_be(:milestone) { create(:milestone, project: project) } let_it_be(:milestone) { create(:milestone, project: project) }
let_it_be(:label) { create(:label, project: project) } let_it_be(:label) { create(:label, project: project) }
let_it_be(:label2) { create(:label, project: project) } let_it_be(:label2) { create(:label, project: project) }
let_it_be(:issue) { create(:issue, project: project, assignees: [user], milestone: milestone) } let_it_be(:issue) { create(:issue, project: project, assignees: [user], milestone: milestone) }
before do let(:current_user) { user }
stub_licensed_features(multiple_issue_assignees: false, issue_weights: false)
before_all do
project.add_maintainer(user) project.add_maintainer(user)
project.add_maintainer(user2) project.add_maintainer(user2)
sign_in(user)
end end
context 'new issue' do before do
stub_licensed_features(multiple_issue_assignees: false, issue_weights: false)
sign_in(current_user)
end
describe 'new issue' do
before do before do
visit new_project_issue_path(project) visit new_project_issue_path(project)
end end
...@@ -235,29 +239,42 @@ RSpec.describe 'New/edit issue', :js do ...@@ -235,29 +239,42 @@ RSpec.describe 'New/edit issue', :js do
end end
describe 'displays issue type options in the dropdown' do describe 'displays issue type options in the dropdown' do
shared_examples 'type option is visible' do |label:, identifier:|
it "shows #{identifier} option", :aggregate_failures do
page.within('[data-testid="issue-type-select-dropdown"]') do
expect(page).to have_selector(%([data-testid="issue-type-#{identifier}-icon"]))
expect(page).to have_content(label)
end
end
end
before do before do
page.within('.issue-form') do page.within('.issue-form') do
click_button 'Issue' click_button 'Issue'
end end
end end
it 'correctly displays the Issue type option with an icon', :aggregate_failures do it_behaves_like 'type option is visible', label: 'Issue', identifier: :issue
page.within('[data-testid="issue-type-select-dropdown"]') do it_behaves_like 'type option is visible', label: 'Incident', identifier: :incident
expect(page).to have_selector('[data-testid="issue-type-issue-icon"]')
expect(page).to have_content('Issue')
end
end
it 'correctly displays the Incident type option with an icon', :aggregate_failures do context 'when user is guest' do
page.within('[data-testid="issue-type-select-dropdown"]') do let_it_be(:guest) { create(:user) }
expect(page).to have_selector('[data-testid="issue-type-incident-icon"]')
expect(page).to have_content('Incident') let(:current_user) { guest }
before_all do
project.add_guest(guest)
end end
it_behaves_like 'type option is visible', label: 'Issue', identifier: :issue
it_behaves_like 'type option is visible', label: 'Incident', identifier: :incident
end end
end end
describe 'milestone' do describe 'milestone' do
let!(:milestone) { create(:milestone, title: '">&lt;img src=x onerror=alert(document.domain)&gt;', project: project) } let!(:milestone) do
create(:milestone, title: '">&lt;img src=x onerror=alert(document.domain)&gt;', project: project)
end
it 'escapes milestone' do it 'escapes milestone' do
click_button 'Milestone' click_button 'Milestone'
...@@ -274,7 +291,7 @@ RSpec.describe 'New/edit issue', :js do ...@@ -274,7 +291,7 @@ RSpec.describe 'New/edit issue', :js do
end end
end end
context 'edit issue' do describe 'edit issue' do
before do before do
visit edit_project_issue_path(project, issue) visit edit_project_issue_path(project, issue)
end end
...@@ -329,7 +346,7 @@ RSpec.describe 'New/edit issue', :js do ...@@ -329,7 +346,7 @@ RSpec.describe 'New/edit issue', :js do
end end
end end
context 'inline edit' do describe 'inline edit' do
before do before do
visit project_issue_path(project, issue) visit project_issue_path(project, issue)
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment