Commit d9d288bf authored by Simon Knox's avatar Simon Knox

Merge branches '2518-saved-configuration-for-issue-board' and...

Merge branches '2518-saved-configuration-for-issue-board' and '2518-saved-configuration-for-issue-board' of gitlab.com:gitlab-org/gitlab-ee into 2518-saved-configuration-for-issue-board
parents 5fb3b644 d5ab9895
......@@ -82,13 +82,12 @@ $(() => {
},
},
created () {
const updateFilterPath = (key, value, tokenName) => {
const updateFilterPath = (key, value) => {
if (!value) return;
const querystring = `${key}=${value}`;
Store.filter.path = [querystring].concat(
Store.filter.path.split('&').filter(param => param.match(new RegExp(`^${key}=(.*)$`, 'g')) === null)
).join('&');
this.cantEdit.push(tokenName);
};
if (this.milestoneId !== -1) {
......@@ -96,7 +95,8 @@ $(() => {
if (this.milestoneId === 0) {
milestoneTitle = 'No+Milestone';
}
updateFilterPath('milestone_title', milestoneTitle, 'milestone');
updateFilterPath('milestone_title', milestoneTitle);
this.cantEdit.push('milestone');
}
let weight = this.weight;
......@@ -104,9 +104,13 @@ $(() => {
if (weight === 0) {
weight = 'No+Weight';
}
updateFilterPath('weight', weight, 'weight');
updateFilterPath('weight', weight);
this.cantEdit.push('weight');
}
updateFilterPath('assignee_username', this.assigneeUsername);
if (this.assigneeUsername) {
this.cantEdit.push('assignee');
}
updateFilterPath('assignee_username', this.assigneeUsername, 'assignee');
const filterPath = gl.issueBoards.BoardsStore.filter.path.split('&');
this.labels.forEach((label) => {
......
......@@ -45,6 +45,7 @@ export default class FilteredSearchBoards extends gl.FilteredSearchManager {
canEdit(tokenName, tokenValue) {
if (this.cantEdit.includes(tokenName)) return false;
return this.cantEditWithValue.findIndex(t => t.name === tokenName && t.value === tokenValue) === -1;
return this.cantEditWithValue.findIndex(token => token.name === tokenName &&
token.value === tokenValue) === -1;
}
}
......@@ -181,7 +181,7 @@ describe 'issue board config', :js do
end
it 'can filter by additional labels' do
label_title = issue.labels.first.title
label_2_title = issue_2.labels.first.title
visit project_boards_path(project)
......@@ -268,9 +268,6 @@ describe 'issue board config', :js do
context 'add issue' do
it 'adds assignee' do
visit boards_path(filtered_board)
end
end
......
require 'rails_helper'
describe 'Scoped issue boards', :js do
include FilteredSearchHelpers
let(:user) { create(:user) }
let(:project) { create(:project, :public) }
let(:group) { create(:group, :public) }
let(:project) { create(:project, :public, namespace: group) }
let(:project_2) { create(:project, :public, namespace: group) }
let!(:project_label) { create(:label, project: project, name: 'Planning') }
let!(:group_label) { create(:group_label, group: group, name: 'Group Label') }
let!(:milestone) { create(:milestone, project: project) }
let!(:issue) { create(:closed_issue, project: project) }
let!(:issue_with_milestone) { create(:closed_issue, project: project, milestone: milestone) }
let!(:label) { create(:label, project: project) }
let!(:issue_with_label) { create(:labeled_issue, :closed, project: project, labels: [label]) }
let!(:issue_with_assignee) { create(:closed_issue, project: project, assignees: [user]) }
let!(:issue_with_weight) { create(:labeled_issue, :closed, project: project, weight: 5) }
let!(:board) { create(:board, project: project, name: 'Project board') }
let!(:group_board) { create(:board, group: group, name: 'Group board') }
let!(:filtered_board) { create(:board, project: project_2, name: 'Filtered board', milestone: milestone, assignee: user, weight: 2) }
let!(:issue) { create(:issue, project: project) }
let!(:issue_milestone) { create(:closed_issue, project: project, milestone: milestone) }
let!(:assigned_issue) { create(:issue, project: project, assignees: [user]) }
let(:edit_board) { find('.btn', text: 'Edit board') }
let(:view_scope) { find('.btn', text: 'View scope') }
let(:board_title) { find('.boards-selector-wrapper .dropdown-menu-toggle') }
before do
allow_any_instance_of(ApplicationHelper).to receive(:collapsed_sidebar?).and_return(true)
project.team << [user, :master]
sign_in(user)
stub_licensed_features(multiple_issue_boards: true)
stub_licensed_features(group_issue_boards: true)
stub_licensed_features(scoped_issue_boards: true)
end
context 'with the feature enabled' do
context 'user with edit permissions' do
before do
stub_licensed_features(scoped_issue_board: true)
project.add_master(user)
group.add_master(user)
login_as(user)
visit project_boards_path(project)
wait_for_requests
end
context 'new board' do
before do
visit project_boards_path(project)
end
context 'milestone' do
it 'creates board filtering by milestone' do
create_board_milestone(milestone.title)
it 'creates board with milestone' do
create_board_with_milestone
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(milestone.title)
expect(page).to have_selector('.card', count: 1)
end
expect(find('.tokens-container')).to have_content(milestone.title)
wait_for_requests
find('.card', match: :first)
it 'creates board to filtering by Any Milestone' do
create_board_milestone('Any Milestone')
expect(all('.board').last).to have_selector('.card', count: 1)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content("")
expect(page).to have_selector('.card', count: 3)
end
end
it 'creates board with labels' do
create_board_with_labels
context 'labels' do
let!(:label_1) { create(:label, project: project, name: 'Label 1') }
let!(:label_2) { create(:label, project: project, name: 'Label 2') }
let!(:issue) { create(:labeled_issue, project: project, labels: [label_1]) }
let!(:issue_2) { create(:labeled_issue, project: project, labels: [label_2]) }
let!(:issue_3) { create(:labeled_issue, project: project, labels: [label_1, label_2]) }
expect(find('.tokens-container')).to have_content(label.title)
wait_for_requests
find('.card', match: :first)
it 'creates board filtering by one label' do
create_board_label(label_1.title)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(label_1.title)
expect(page).to have_selector('.card', count: 2)
end
it 'creates board filtering by multiple labels' do
create_board_label([label_1.title, label_2.title])
expect(all('.board').last).to have_selector('.card', count: 1)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(label_1.title)
expect(find('.tokens-container')).to have_content(label_2.title)
expect(page).to have_selector('.card', count: 1)
end
xit 'only shows group labels in list on group boards' do
visit group_boards_path(group)
wait_for_requests
expect(page).to have_css('#js-multiple-boards-switcher')
page.within '#js-multiple-boards-switcher' do
find('.dropdown-menu-toggle').click
click_link 'Create new board'
end
page.within('.labels') do
click_link 'Edit'
page.within('.dropdown') do
expect(page).to have_content(group_label.title)
expect(page).not_to have_content(project_label.title)
end
end
end
end
it 'creates board with assignee' do
create_board_with_assignee
context 'assignee' do
it 'creates board filtering by assignee' do
create_board_assignee(user.name)
expect(find('.tokens-container')).to have_content(user.name)
wait_for_requests
find('.card', match: :first)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(user.name)
expect(page).to have_selector('.card', count: 1)
# Does not display assignee in search hint
filtered_search.click
expect(all('.board').last).to have_selector('.card', count: 1)
page.within('#js-dropdown-hint') do
expect(page).to have_content('label')
expect(page).not_to have_content('assignee')
end
end
it 'creates board filtering by "Any assignee"' do
create_board_assignee('Any assignee')
expect(page).not_to have_css('.js-visual-token')
expect(page).to have_selector('.card', count: 3)
end
end
it 'creates board with weight' do
create_board_with_weight
context 'weight' do
let!(:issue_weight_1) { create(:issue, project: project, weight: 1) }
expect(find('.tokens-container')).to have_content(issue_with_weight.weight)
wait_for_requests
find('.card', match: :first)
it 'creates board filtering by weight' do
create_board_weight(1)
expect(page).to have_selector('.card', count: 1)
expect(find('.card-title').text).to have_content(issue_weight_1.title)
# Does not display assignee in search hint
filtered_search.click
expect(all('.board').last).to have_selector('.card', count: 1)
page.within('#js-dropdown-hint') do
expect(page).to have_content('label')
expect(page).not_to have_content('weight')
end
end
it 'creates board filtering by "Any weight"' do
create_board_weight('Any Weight')
expect(page).to have_selector('.card', count: 4)
end
end
end
xcontext 'update board' do
context 'edit board' do
let!(:milestone_two) { create(:milestone, project: project) }
let!(:board) { create(:board, project: project, milestone: milestone) }
before do
visit project_boards_path(project)
end
it 'defaults milestone filter' do
page.within '#js-multiple-boards-switcher' do
find('.dropdown-menu-toggle').click
it 'edits board name' do
edit_board.click
wait_for_requests
page.within('.popup-dialog') do
fill_in 'board-new-name', with: 'Testing'
click_link board.name
click_button 'Save'
end
expect(find('.tokens-container')).to have_content(milestone.title)
expect(board_title).to have_content('Testing')
expect(board.reload.name).to eq('Testing')
end
find('.card', match: :first)
it 'prefills fields' do
visit project_boards_path(project_2)
expect(all('.board').last).to have_selector('.card', count: 1)
edit_board.click
expect(find('.milestone .value')).to have_content(milestone.title)
expect(find('.assignee .value')).to have_content(user.name)
expect(find('.weight .value')).to have_content(2)
end
it 'sets board to any milestone' do
update_board_milestone('Any Milestone')
context 'milestone' do
it 'sets board milestone' do
update_board_milestone(milestone.title)
expect(page).not_to have_css('.js-visual-token')
expect(find('.tokens-container')).not_to have_content(milestone.title)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(milestone.title)
find('.card', match: :first)
expect(page).to have_selector('.card', count: 1)
end
expect(page).to have_selector('.board', count: 3)
expect(all('.board').last).to have_selector('.card', count: 2)
end
it 'sets board to any milestone' do
update_board_milestone('Any Milestone')
it 'sets board to upcoming milestone' do
update_board_milestone('Upcoming')
expect(page).not_to have_css('.js-visual-token')
expect(find('.tokens-container')).not_to have_content(milestone.title)
expect(find('.tokens-container')).not_to have_content(milestone.title)
find('.card', match: :first)
find('.board', match: :first)
expect(page).to have_selector('.board', count: 3)
expect(all('.board').first).to have_selector('.card', count: 2)
expect(all('.board').last).to have_selector('.card', count: 1)
end
expect(all('.board')[1]).to have_selector('.card', count: 0)
end
it 'sets board to upcoming milestone' do
update_board_milestone('Upcoming')
it 'does not allow milestone in filter to be editted' do
find('.filtered-search').native.send_keys(:backspace)
expect(find('.tokens-container')).not_to have_content(milestone.title)
page.within('.tokens-container') do
expect(page).to have_selector('.value')
find('.board', match: :first)
expect(all('.board')[1]).to have_selector('.card', count: 0)
end
end
it 'does not render milestone in hint dropdown' do
find('.filtered-search').click
it 'does not display milestone in search hint' do
update_board_milestone(milestone.title)
filtered_search.click
page.within('#js-dropdown-hint') do
expect(page).not_to have_button('Milestone')
page.within('#js-dropdown-hint') do
expect(page).to have_content('label')
expect(page).not_to have_content('milestone')
end
end
end
end
xcontext 'removing issue from board' do
let(:label) { create(:label, project: project) }
let!(:issue) { create(:labeled_issue, project: project, labels: [label], milestone: milestone) }
let!(:board) { create(:board, project: project, milestone: milestone) }
let!(:list) { create(:list, board: board, label: label, position: 0) }
context 'labels' do
let!(:label_1) { create(:label, project: project, name: 'Label 1') }
let!(:label_2) { create(:label, project: project, name: 'Label 2') }
let!(:issue) { create(:labeled_issue, project: project, labels: [label_1]) }
let!(:issue_2) { create(:labeled_issue, project: project, labels: [label_2]) }
let!(:issue_3) { create(:labeled_issue, project: project, labels: [label_1, label_2]) }
before do
visit project_boards_path(project)
end
it 'adds label to board' do
label_title = issue.labels.first.title
visit project_boards_path(project)
it 'removes issues milestone when removing from the board' do
wait_for_requests
update_board_label(label_title)
first('.card .card-number').click
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(label_title)
click_button('Remove from board')
wait_for_requests
expect(page).to have_selector('.card', count: 2)
end
expect(issue.reload.milestone).to be_nil
it 'adds multiple labels to board' do
label_title = issue.labels.first.title
label_2_title = issue_2.labels.first.title
visit project_boards_path(project)
update_board_label(label_title)
update_board_label(label_2_title)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(label_title)
expect(find('.tokens-container')).to have_content(label_2_title)
expect(page).to have_selector('.card', count: 1)
end
it 'can filter by additional labels' do
label_title = issue.labels.first.title
label_2_title = issue_2.labels.first.title
visit project_boards_path(project)
update_board_label(label_title)
input_filtered_search("label:~#{label_2_title}")
expect(page).to have_selector('.card', count: 0)
end
context 'group board' do
it 'only shows group labels in list' do
visit group_boards_path(group)
edit_board.click
page.within(".labels") do
click_link 'Edit'
page.within('.dropdown') do
expect(page).to have_content(group_label.title)
expect(page).not_to have_content(project_label.title)
end
end
end
end
end
end
xcontext 'new issues' do
let(:label) { create(:label, project: project) }
let!(:list1) { create(:list, board: board, label: label, position: 0) }
let!(:board) { create(:board, project: project, milestone: milestone) }
let!(:issue) { create(:issue, project: project) }
context 'assignee' do
it 'sets board assignee' do
update_board_assignee(user.name)
before do
visit project_boards_path(project)
expect(page).to have_css('.js-visual-token')
expect(find('.tokens-container')).to have_content(user.name)
expect(page).to have_selector('.card', count: 1)
end
it 'sets board to Any assignee' do
update_board_assignee('Any assignee')
expect(page).not_to have_css('.js-visual-token')
expect(page).to have_selector('.card', count: 3)
end
it 'does not display assignee in search hint' do
update_board_assignee(user.name)
filtered_search.click
page.within('#js-dropdown-hint') do
expect(page).to have_content('label')
expect(page).not_to have_content('assignee')
end
end
end
it 'creates new issue with boards milestone' do
wait_for_requests
context 'weight' do
let!(:issue_weight_1) { create(:issue, project: project, weight: 1) }
page.within(first('.board')) do
find('.btn-default').click
it 'sets board weight' do
update_board_weight(1)
find('.form-control').set('testing new issue with milestone')
expect(page).to have_selector('.card', count: 1)
expect(find('.card-title').text).to have_content(issue_weight_1.title)
end
click_button('Submit issue')
it 'sets board to Any weight' do
update_board_weight('Any weight')
wait_for_requests
expect(page).to have_selector('.card', count: 4)
end
click_link('testing new issue with milestone')
it 'does not display weight in search hint' do
update_board_weight(1)
filtered_search.click
page.within('#js-dropdown-hint') do
expect(page).to have_content('label')
expect(page).not_to have_content('weight')
end
end
end
end
context 'add issue' do
it 'adds assignee' do
visit boards_path(filtered_board)
expect(page).to have_content(milestone.title)
end
end
context 'remove issue' do
let!(:issue) { create(:labeled_issue, project: project, labels: [project_label], milestone: milestone, assignees: [user]) }
let!(:list) { create(:list, board: board, label: project_label, position: 0) }
it 'updates issue with milestone from add issues modal' do
it 'removes issues milestone when removing from the board' do
visit project_boards_path(project)
wait_for_requests
click_button 'Add issues'
find(".card[data-issue-id='#{issue.id}']").click
page.within('.add-issues-modal') do
card = find('.card', :first)
expect(page).to have_selector('.card', count: 1)
click_button 'Remove from board'
wait_for_requests
card.click
expect(issue.reload.milestone).to be_nil
expect(issue.reload.assignees).to be_empty
end
end
end
click_button 'Add 1 issue'
end
context 'user without edit permissions' do
before do
visit project_boards_path(project)
wait_for_requests
end
click_link(issue.title)
it 'can view board scope' do
view_scope.click
expect(page).to have_content(milestone.title)
page.within('.popup-dialog') do
expect(find('.modal-header')).to have_content('Board scope')
expect(page).not_to have_content('Board name')
expect(page).not_to have_link('Edit')
expect(page).not_to have_button('Edit')
expect(page).not_to have_button('Save')
expect(page).not_to have_button('Cancel')
end
end
end
xcontext 'with the feature disabled' do
context 'with scoped_issue_boards feature disabled' do
before do
stub_licensed_features(issue_board_milestone: false)
stub_licensed_features(scoped_issue_boards: false)
project.team << [user, :master]
login_as(user)
visit project_boards_path(project)
wait_for_requests
end
it "doesn't show the input when creating a board" do
......@@ -217,83 +409,88 @@ describe 'Scoped issue boards', :js do
click_link 'Create new board'
# To make sure the form is shown
expect(page).to have_selector('#board-new-name')
expect(page).to have_field('board-new-name')
expect(page).not_to have_button('Milestone')
expect(page).not_to have_button('Toggle')
end
end
it "doesn't show the option to edit the milestone" do
page.within '#js-multiple-boards-switcher' do
find('.dropdown-menu-toggle').click
# To make sure the dropdown is open
expect(page).to have_link('Edit board name')
expect(page).not_to have_link('Edit board milestone')
end
it "doesn't show the button to edit scope" do
expect(page).not_to have_button('View Scope')
end
end
def create_board_with_milestone
creating_new_board do
find('.edit-link', match: :first).trigger('click')
find('a', text: milestone.title).trigger('click')
# Create board helper methods
#
def create_board_milestone(milestone_title)
create_board_scope('milestone', milestone_title)
end
click_button 'Create'
end
def create_board_label(label_title)
create_board_scope('labels', label_title)
end
def create_board_with_labels
creating_new_board do
page.all('.edit-link')[1].trigger('click')
find('a', text: label.title).trigger('click')
def create_board_weight(weight)
create_board_scope('weight', weight)
end
click_button 'Create'
end
def create_board_assignee(assignee_name)
create_board_scope('assignee', assignee_name)
end
def create_board_with_assignee
creating_new_board do
page.all('.edit-link')[2].trigger('click')
find('a', text: user.name).trigger('click')
# Update board helper methods
#
def update_board_milestone(milestone_title)
update_board_scope('milestone', milestone_title)
end
click_button 'Create'
end
def update_board_label(label_title)
update_board_scope('labels', label_title)
end
def create_board_with_weight
creating_new_board do
page.all('.edit-link')[3].trigger('click')
find('a', text: issue_with_weight.weight).trigger('click')
def update_board_assignee(assignee_name)
update_board_scope('assignee', assignee_name)
end
click_button 'Create'
end
def update_board_weight(weight)
update_board_scope('weight', weight)
end
def creating_new_board
def create_board_scope(filter, value)
page.within '#js-multiple-boards-switcher' do
find('.dropdown-menu-toggle').click
end
click_link 'Create new board'
click_link 'Create new board'
find('#board-new-name').set 'test'
find('#board-new-name').set 'test'
click_button 'Expand'
click_button 'Expand'
yield
page.within(".#{filter}") do
click_link 'Edit'
if value.is_a?(Array)
value.each { |value| click_link value }
else
click_link value
end
end
end
def update_board_milestone(milestone_title)
page.within '#js-multiple-boards-switcher' do
find('.dropdown-menu-toggle').click
click_link 'Edit board milestone'
click_button 'Create'
expect(page).to have_selector('.board-list-loading')
expect(page).not_to have_selector('.board-list-loading')
end
click_link milestone_title
def update_board_scope(filter, value)
edit_board.click
click_button 'Save'
page.within(".#{filter}") do
click_link 'Edit'
click_link value
end
click_button 'Save'
expect(page).to have_selector('.board-list-loading')
expect(page).not_to have_selector('.board-list-loading')
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