2019-10-12 21:52:04 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
require 'spec_helper'
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
RSpec.describe 'Issue Boards', :js do
|
2018-03-17 18:26:18 +05:30
|
|
|
include BoardHelpers
|
2019-07-31 22:56:46 +05:30
|
|
|
include FilteredSearchHelpers
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
let(:user) { create(:user) }
|
|
|
|
let(:user2) { create(:user) }
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:project) { create(:project, :public) }
|
2017-08-17 22:00:37 +05:30
|
|
|
let!(:milestone) { create(:milestone, project: project) }
|
|
|
|
let!(:development) { create(:label, project: project, name: 'Development') }
|
|
|
|
let!(:bug) { create(:label, project: project, name: 'Bug') }
|
|
|
|
let!(:regression) { create(:label, project: project, name: 'Regression') }
|
|
|
|
let!(:stretch) { create(:label, project: project, name: 'Stretch') }
|
|
|
|
let!(:issue1) { create(:labeled_issue, project: project, assignees: [user], milestone: milestone, labels: [development], relative_position: 2) }
|
|
|
|
let!(:issue2) { create(:labeled_issue, project: project, labels: [development, stretch], relative_position: 1) }
|
|
|
|
let(:board) { create(:board, project: project) }
|
|
|
|
let!(:list) { create(:list, board: board, label: development, position: 0) }
|
2019-09-30 21:07:59 +05:30
|
|
|
let(:card) { find('.board:nth-child(2)').first('.board-card') }
|
|
|
|
|
|
|
|
let(:application_settings) { {} }
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
around do |example|
|
2020-11-24 15:15:51 +05:30
|
|
|
freeze_time { example.run }
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
before do
|
2018-11-18 11:00:15 +05:30
|
|
|
project.add_maintainer(user)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
sign_in(user)
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
stub_application_setting(application_settings)
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
visit project_board_path(project, board)
|
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows sidebar when clicking issue' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
expect(page).to have_selector('.issue-boards-sidebar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'closes sidebar when clicking issue' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
expect(page).to have_selector('.issue-boards-sidebar')
|
|
|
|
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
expect(page).not_to have_selector('.issue-boards-sidebar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'closes sidebar when clicking close button' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
expect(page).to have_selector('.issue-boards-sidebar')
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
find('.gutter-toggle').click
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(page).not_to have_selector('.issue-boards-sidebar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows issue details when sidebar is open' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.issue-boards-sidebar') do
|
|
|
|
expect(page).to have_content(issue2.title)
|
|
|
|
expect(page).to have_content(issue2.to_reference)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'assignee' do
|
|
|
|
it 'updates the issues assignee' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.assignee') do
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
assignee = first('.gl-avatar-labeled').find('.gl-avatar-labeled-label').text
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
page.within('.dropdown-menu-user') do
|
|
|
|
first('.gl-avatar-labeled').click
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
|
|
|
wait_for_requests
|
|
|
|
|
|
|
|
expect(page).to have_content(assignee)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
expect(card).to have_selector('.avatar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'removes the assignee' do
|
2018-11-08 19:23:39 +05:30
|
|
|
card_two = find('.board:nth-child(2)').find('.board-card:nth-child(2)')
|
2017-08-17 22:00:37 +05:30
|
|
|
click_card(card_two)
|
|
|
|
|
|
|
|
page.within('.assignee') do
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
page.within('.dropdown-menu-user') do
|
2021-03-11 19:13:27 +05:30
|
|
|
find('[data-testid="unassign"]').click
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
expect(page).to have_content('None')
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
expect(card_two).not_to have_selector('.avatar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'assignees to current user' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within(find('.assignee')) do
|
2019-07-31 22:56:46 +05:30
|
|
|
expect(page).to have_content('None')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
click_button 'assign yourself'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(page).to have_content(user.name)
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(card).to have_selector('.avatar')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates assignee dropdown' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.assignee') do
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
assignee = first('.gl-avatar-labeled').find('.gl-avatar-labeled-label').text
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
page.within('.dropdown-menu-user') do
|
|
|
|
first('.gl-avatar-labeled').click
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
|
|
|
wait_for_requests
|
|
|
|
|
|
|
|
expect(page).to have_content(assignee)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
page.within(find('.board:nth-child(2)')) do
|
2018-11-08 19:23:39 +05:30
|
|
|
find('.board-card:nth-child(2)').click
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
page.within('.assignee') do
|
2021-03-11 19:13:27 +05:30
|
|
|
click_button('Edit')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(find('.dropdown-menu')).to have_selector('.gl-new-dropdown-item-check-icon')
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'milestone' do
|
|
|
|
it 'adds a milestone' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.milestone') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
click_link milestone.title
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
page.within('.value') do
|
|
|
|
expect(page).to have_content(milestone.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'removes a milestone' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.milestone') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
click_link "No milestone"
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
page.within('.value') do
|
|
|
|
expect(page).not_to have_content(milestone.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
context 'time tracking' do
|
2021-01-03 14:25:43 +05:30
|
|
|
let(:compare_meter_tooltip) { find('.time-tracking .time-tracking-content .compare-meter')['title'] }
|
2019-09-30 21:07:59 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
before do
|
|
|
|
issue2.timelogs.create(time_spent: 14400, user: user)
|
2019-09-30 21:07:59 +05:30
|
|
|
issue2.update!(time_estimate: 128800)
|
|
|
|
|
|
|
|
click_card(card)
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'shows time tracking progress bar' do
|
2019-09-30 21:07:59 +05:30
|
|
|
expect(compare_meter_tooltip).to eq('Time remaining: 3d 7h 46m')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when time_tracking_limit_to_hours is true' do
|
|
|
|
let(:application_settings) { { time_tracking_limit_to_hours: true } }
|
2019-07-31 22:56:46 +05:30
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
it 'shows time tracking progress bar' do
|
|
|
|
expect(compare_meter_tooltip).to eq('Time remaining: 31h 46m')
|
2019-07-31 22:56:46 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
context 'due date' do
|
|
|
|
it 'updates due date' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.due_date') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
|
|
|
click_button Date.today.day
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(page).to have_content(Date.today.to_s(:medium))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'labels' do
|
2018-10-15 14:42:47 +05:30
|
|
|
it 'shows current labels when editing' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
|
|
|
wait_for_requests
|
|
|
|
|
|
|
|
page.within('.value') do
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(page).to have_selector('.gl-label-text', count: 2)
|
2018-10-15 14:42:47 +05:30
|
|
|
expect(page).to have_content(development.title)
|
|
|
|
expect(page).to have_content(stretch.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
it 'adds a single label' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
click_link bug.title
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
find('.dropdown-menu-close-icon').click
|
|
|
|
|
|
|
|
page.within('.value') do
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(page).to have_selector('.gl-label-text', count: 3)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(page).to have_content(bug.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# 'Development' label does not show since the card is in a 'Development' list label
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(card).to have_selector('.gl-label', count: 2)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(card).to have_content(bug.title)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'adds a multiple labels' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
click_link bug.title
|
2020-03-13 15:44:24 +05:30
|
|
|
|
|
|
|
wait_for_requests
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
click_link regression.title
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
find('.dropdown-menu-close-icon').click
|
|
|
|
|
|
|
|
page.within('.value') do
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(page).to have_selector('.gl-label-text', count: 4)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(page).to have_content(bug.title)
|
|
|
|
expect(page).to have_content(regression.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# 'Development' label does not show since the card is in a 'Development' list label
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(card).to have_selector('.gl-label', count: 3)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(card).to have_content(bug.title)
|
|
|
|
expect(card).to have_content(regression.title)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'removes a label' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
within('.dropdown-menu-labels') do
|
|
|
|
click_link stretch.title
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
find('.dropdown-menu-close-icon').click
|
|
|
|
|
|
|
|
page.within('.value') do
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(page).to have_selector('.gl-label-text', count: 1)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(page).not_to have_content(stretch.title)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# 'Development' label does not show since the card is in a 'Development' list label
|
2020-04-08 14:13:33 +05:30
|
|
|
expect(card).to have_selector('.gl-label-text', count: 0)
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(card).not_to have_content(stretch.title)
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-05-09 12:01:36 +05:30
|
|
|
it 'creates project label' do
|
2018-03-17 18:26:18 +05:30
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
2019-09-04 21:01:54 +05:30
|
|
|
wait_for_requests
|
|
|
|
|
2018-05-09 12:01:36 +05:30
|
|
|
click_link 'Create project label'
|
2018-03-17 18:26:18 +05:30
|
|
|
fill_in 'new_label_name', with: 'test label'
|
|
|
|
first('.suggest-colors-dropdown a').click
|
|
|
|
click_button 'Create'
|
|
|
|
wait_for_requests
|
|
|
|
|
|
|
|
expect(page).to have_link 'test label'
|
|
|
|
end
|
2019-07-07 11:18:12 +05:30
|
|
|
expect(page).to have_selector('.board', count: 3)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'creates project label and list' do
|
|
|
|
click_card(card)
|
|
|
|
|
|
|
|
page.within('.labels') do
|
|
|
|
click_link 'Edit'
|
2019-09-04 21:01:54 +05:30
|
|
|
wait_for_requests
|
|
|
|
|
2019-07-07 11:18:12 +05:30
|
|
|
click_link 'Create project label'
|
|
|
|
fill_in 'new_label_name', with: 'test label'
|
|
|
|
first('.suggest-colors-dropdown a').click
|
|
|
|
first('.js-add-list').click
|
|
|
|
click_button 'Create'
|
|
|
|
wait_for_requests
|
|
|
|
|
|
|
|
expect(page).to have_link 'test label'
|
|
|
|
end
|
|
|
|
expect(page).to have_selector('.board', count: 4)
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'subscription' do
|
|
|
|
it 'changes issue subscription' do
|
|
|
|
click_card(card)
|
2018-03-17 18:26:18 +05:30
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
page.within('.subscriptions') do
|
2021-03-11 19:13:27 +05:30
|
|
|
find('[data-testid="subscription-toggle"] button:not(.is-checked)').click
|
2017-09-10 17:25:29 +05:30
|
|
|
wait_for_requests
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(page).to have_css('[data-testid="subscription-toggle"] button.is-checked')
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'has checked subscription toggle when already subscribed' do
|
|
|
|
create(:subscription, user: user, project: project, subscribable: issue2, subscribed: true)
|
|
|
|
visit project_board_path(project, board)
|
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
click_card(card)
|
|
|
|
wait_for_requests
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
page.within('.subscriptions') do
|
2021-03-11 19:13:27 +05:30
|
|
|
find('[data-testid="subscription-toggle"] button.is-checked').click
|
2018-03-17 18:26:18 +05:30
|
|
|
wait_for_requests
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(page).to have_css('[data-testid="subscription-toggle"] button:not(.is-checked)')
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|