kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
317 lines
9.8 KiB
317 lines
9.8 KiB
#-- copyright
|
|
# OpenProject is an open source project management software.
|
|
# Copyright (C) 2012-2021 the OpenProject GmbH
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License version 3.
|
|
#
|
|
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
|
|
# Copyright (C) 2006-2013 Jean-Philippe Lang
|
|
# Copyright (C) 2010-2013 the ChiliProject Team
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public License
|
|
# as published by the Free Software Foundation; either version 2
|
|
# of the License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
#
|
|
# See docs/COPYRIGHT.rdoc for more details.
|
|
#++
|
|
|
|
require 'spec_helper'
|
|
|
|
describe 'Projects', type: :feature, js: true do
|
|
let(:current_user) { FactoryBot.create(:admin) }
|
|
let(:name_field) { ::FormFields::InputFormField.new :name }
|
|
let(:parent_field) { ::FormFields::SelectFormField.new :parent }
|
|
|
|
before do
|
|
allow(User).to receive(:current).and_return current_user
|
|
end
|
|
|
|
describe 'creation' do
|
|
shared_let(:project) { FactoryBot.create(:project, name: 'Foo project', identifier: 'foo-project') }
|
|
|
|
before do
|
|
visit projects_path
|
|
end
|
|
|
|
it 'can create a project' do
|
|
click_on 'New project'
|
|
|
|
name_field.set_value 'Foo bar'
|
|
click_button 'Save'
|
|
|
|
sleep 1
|
|
|
|
expect(page).to have_content 'Foo bar'
|
|
expect(page).to have_current_path /\/projects\/foo-bar\/?/
|
|
end
|
|
|
|
it 'can create a subproject' do
|
|
click_on project.name
|
|
SeleniumHubWaiter.wait
|
|
click_on 'Project settings'
|
|
SeleniumHubWaiter.wait
|
|
click_on 'New subproject'
|
|
|
|
name_field.set_value 'Foo child'
|
|
|
|
sleep 1
|
|
|
|
parent_field.expect_selected project.name
|
|
|
|
click_button 'Save'
|
|
|
|
sleep 1
|
|
|
|
expect(page).to have_current_path /\/projects\/foo-child\/?/
|
|
|
|
child = Project.last
|
|
expect(child.identifier).to eq 'foo-child'
|
|
expect(child.parent).to eq project
|
|
end
|
|
|
|
it 'does not create a project with an already existing identifier' do
|
|
click_on 'New project'
|
|
|
|
name_field.set_value 'Foo project'
|
|
click_on 'Save'
|
|
|
|
expect(page).to have_current_path /\/projects\/foo-project-1\/?/
|
|
|
|
project = Project.last
|
|
expect(project.identifier).to eq 'foo-project-1'
|
|
end
|
|
|
|
context 'with a multi-select custom field' do
|
|
let!(:list_custom_field) { FactoryBot.create(:list_project_custom_field, name: 'List CF', multi_value: true) }
|
|
let(:list_field) { ::FormFields::SelectFormField.new list_custom_field }
|
|
|
|
it 'can create a project' do
|
|
click_on 'New project'
|
|
|
|
name_field.set_value 'Foo bar'
|
|
|
|
find('.form--fieldset-legend a', text: 'ADVANCED SETTINGS').click
|
|
|
|
list_field.select_option 'A', 'B'
|
|
|
|
click_button 'Save'
|
|
|
|
expect(page).to have_current_path /\/projects\/foo-bar\/?/
|
|
expect(page).to have_content 'Foo bar'
|
|
|
|
project = Project.last
|
|
expect(project.name).to eq 'Foo bar'
|
|
cvs = project.custom_value_for(list_custom_field)
|
|
expect(cvs.count).to eq 2
|
|
expect(cvs.map(&:typed_value)).to contain_exactly 'A', 'B'
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'project types' do
|
|
let(:phase_type) { FactoryBot.create(:type, name: 'Phase', is_default: true) }
|
|
let(:milestone_type) { FactoryBot.create(:type, name: 'Milestone', is_default: false) }
|
|
let!(:project) { FactoryBot.create(:project, name: 'Foo project', types: [phase_type, milestone_type]) }
|
|
|
|
it "have the correct types checked for the project's types" do
|
|
visit projects_path
|
|
click_on 'Foo project'
|
|
click_on 'Project settings'
|
|
click_on 'Work package types'
|
|
|
|
field_checked = find_field('Phase', visible: false)['checked']
|
|
expect(field_checked).to be_truthy
|
|
field_checked = find_field('Milestone', visible: false)['checked']
|
|
expect(field_checked).to be_truthy
|
|
end
|
|
end
|
|
|
|
describe 'deletion' do
|
|
let(:project) { FactoryBot.create(:project) }
|
|
let(:projects_page) { Pages::Projects::Destroy.new(project) }
|
|
|
|
before do
|
|
projects_page.visit!
|
|
end
|
|
|
|
describe 'disable delete w/o confirm' do
|
|
it { expect(page).to have_css('.danger-zone .button[disabled]') }
|
|
end
|
|
|
|
describe 'disable delete with wrong input' do
|
|
let(:input) { find('.danger-zone input') }
|
|
it do
|
|
input.set 'Not the project name'
|
|
expect(page).to have_css('.danger-zone .button[disabled]')
|
|
end
|
|
end
|
|
|
|
describe 'enable delete with correct input' do
|
|
let(:input) { find('.danger-zone input') }
|
|
it do
|
|
input.set project.name
|
|
expect(page).to have_css('.danger-zone .button:not([disabled])')
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'identifier edit' do
|
|
let!(:project) { FactoryBot.create(:project, identifier: 'foo') }
|
|
|
|
it 'updates the project identifier' do
|
|
visit projects_path
|
|
click_on project.name
|
|
SeleniumHubWaiter.wait
|
|
click_on 'Project settings'
|
|
SeleniumHubWaiter.wait
|
|
click_on 'Change identifier'
|
|
|
|
expect(page).to have_content "CHANGE THE PROJECT'S IDENTIFIER"
|
|
expect(current_path).to eq '/projects/foo/identifier'
|
|
|
|
fill_in 'project[identifier]', with: 'foo-bar'
|
|
click_on 'Update'
|
|
|
|
expect(page).to have_content 'Successful update.'
|
|
expect(current_path).to match '/projects/foo-bar/settings/generic'
|
|
expect(Project.first.identifier).to eq 'foo-bar'
|
|
end
|
|
|
|
it 'displays error messages on invalid input' do
|
|
visit identifier_project_path(project)
|
|
|
|
fill_in 'project[identifier]', with: 'FOOO'
|
|
click_on 'Update'
|
|
|
|
expect(page).to have_content 'Identifier is invalid.'
|
|
expect(current_path).to eq '/projects/foo/identifier'
|
|
end
|
|
end
|
|
|
|
describe 'form' do
|
|
let(:project) { FactoryBot.build(:project, name: 'Foo project', identifier: 'foo-project') }
|
|
|
|
context 'when creating' do
|
|
it 'hides the active field and the identifier' do
|
|
visit new_project_path
|
|
|
|
find('.form--fieldset-legend a', text: 'ADVANCED SETTINGS').click
|
|
|
|
expect(page).to have_no_content 'Active'
|
|
expect(page).to have_no_content 'Identifier'
|
|
end
|
|
end
|
|
|
|
context 'when editing' do
|
|
it 'hides the active field' do
|
|
project.save!
|
|
|
|
visit settings_generic_project_path(project.id)
|
|
|
|
expect(page).to have_no_text :all, 'Active'
|
|
expect(page).to have_no_text :all, 'Identifier'
|
|
end
|
|
end
|
|
|
|
context 'with optional and required custom fields' do
|
|
let!(:optional_custom_field) do
|
|
FactoryBot.create(:custom_field, name: 'Optional Foo',
|
|
type: ProjectCustomField,
|
|
is_for_all: true)
|
|
end
|
|
let!(:required_custom_field) do
|
|
FactoryBot.create(:custom_field, name: 'Required Foo',
|
|
type: ProjectCustomField,
|
|
is_for_all: true,
|
|
is_required: true)
|
|
end
|
|
|
|
it 'seperates optional and required custom fields for new' do
|
|
visit new_project_path
|
|
|
|
expect(page).to have_content 'Required Foo'
|
|
|
|
click_on 'Advanced settings'
|
|
|
|
within('.form--fieldset') do
|
|
expect(page).to have_text 'Optional Foo'
|
|
expect(page).to have_no_text 'Required Foo'
|
|
end
|
|
end
|
|
|
|
it 'shows optional and required custom fields for edit without a separation' do
|
|
project.custom_field_values.last.value = 'FOO'
|
|
project.save!
|
|
|
|
visit settings_generic_project_path(project.id)
|
|
|
|
expect(page).to have_text 'Optional Foo'
|
|
expect(page).to have_text 'Required Foo'
|
|
end
|
|
end
|
|
|
|
context 'with a length restricted custom field' do
|
|
let(:project) { FactoryBot.create(:project, name: 'Foo project', identifier: 'foo-project') }
|
|
let!(:required_custom_field) do
|
|
FactoryBot.create(:string_project_custom_field,
|
|
name: 'Foo',
|
|
type: ProjectCustomField,
|
|
min_length: 1,
|
|
max_length: 2,
|
|
is_for_all: true)
|
|
end
|
|
let(:foo_field) { ::FormFields::InputFormField.new required_custom_field }
|
|
|
|
it 'shows the errors of that field when saving (Regression #33766)' do
|
|
visit settings_generic_project_path(project.id)
|
|
|
|
expect(page).to have_content 'Foo'
|
|
|
|
# Enter something too long
|
|
foo_field.set_value '1234'
|
|
|
|
# It should cut of that remaining value
|
|
foo_field.expect_value '12'
|
|
|
|
click_button 'Save'
|
|
|
|
expect(page).to have_text 'Successful update.'
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with a multi-select custom field' do
|
|
include_context 'ng-select-autocomplete helpers'
|
|
|
|
let(:project) { FactoryBot.create(:project, name: 'Foo project', identifier: 'foo-project') }
|
|
let!(:list_custom_field) { FactoryBot.create(:list_project_custom_field, name: 'List CF', multi_value: true) }
|
|
let(:form_field) { ::FormFields::SelectFormField.new list_custom_field }
|
|
|
|
it 'can create a project' do
|
|
visit settings_generic_project_path(project.id)
|
|
|
|
form_field.select_option 'A', 'B'
|
|
|
|
click_on 'Save'
|
|
|
|
expect(page).to have_content 'Successful update.'
|
|
|
|
form_field.expect_selected 'A', 'B'
|
|
|
|
cvs = project.reload.custom_value_for(list_custom_field)
|
|
expect(cvs.count).to eq 2
|
|
expect(cvs.map(&:typed_value)).to contain_exactly 'A', 'B'
|
|
end
|
|
end
|
|
end
|
|
|