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.
240 lines
7.8 KiB
240 lines
7.8 KiB
#-- copyright
|
|
# OpenProject is an open source project management software.
|
|
# Copyright (C) 2012-2020 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-2017 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 do
|
|
let(:current_user) { FactoryBot.create(:admin) }
|
|
|
|
before do
|
|
allow(User).to receive(:current).and_return current_user
|
|
end
|
|
|
|
describe 'creation', js: true do
|
|
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'
|
|
|
|
fill_in 'project[name]', with: 'Foo bar'
|
|
click_on 'Advanced settings'
|
|
fill_in 'project[identifier]', with: 'foo'
|
|
sleep 1
|
|
|
|
click_on 'Create'
|
|
|
|
expect(page).to have_content 'Successful creation.'
|
|
expect(page).to have_content 'Foo bar'
|
|
expect(current_path).to eq '/projects/foo/work_packages'
|
|
end
|
|
|
|
context 'work_packages module disabled',
|
|
with_settings: { default_projects_modules: %q(wiki) } do
|
|
it 'creates a project and redirects to settings' do
|
|
click_on 'New project'
|
|
|
|
fill_in 'project[name]', with: 'Foo bar'
|
|
click_on 'Advanced settings'
|
|
fill_in 'project[identifier]', with: 'foo'
|
|
sleep 1
|
|
click_on 'Create'
|
|
|
|
expect(page).to have_content 'Successful creation.'
|
|
expect(page).to have_content 'Foo bar'
|
|
expect(current_path).to eq '/projects/foo/'
|
|
end
|
|
end
|
|
|
|
it 'can create a subproject' do
|
|
click_on 'Foo project'
|
|
click_on 'Project settings'
|
|
click_on 'New subproject'
|
|
|
|
fill_in 'project[name]', with: 'Foo child'
|
|
click_on 'Create'
|
|
|
|
expect(page).to have_content 'Successful creation.'
|
|
expect(current_path).to eq '/projects/foo-child/work_packages'
|
|
end
|
|
|
|
it 'does not create a project with an already existing identifier' do
|
|
click_on 'New project'
|
|
|
|
fill_in 'project[name]', with: 'Foo project'
|
|
click_on 'Create'
|
|
|
|
expect(page).to have_content 'Identifier has already been taken'
|
|
expect(current_path).to eq '/projects'
|
|
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', js: true 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', js: true do
|
|
let!(:project) { FactoryBot.create(:project, identifier: 'foo') }
|
|
|
|
it 'updates the project identifier' do
|
|
visit projects_path
|
|
click_on project.name
|
|
click_on 'Project settings'
|
|
click_on 'Edit'
|
|
|
|
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 eq '/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', js: true do
|
|
let(:project) { FactoryBot.build(:project, name: 'Foo project', identifier: 'foo-project') }
|
|
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('#advanced-settings') do
|
|
expect(page).to have_content 'Optional Foo'
|
|
expect(page).not_to have_content 'Required Foo'
|
|
end
|
|
end
|
|
|
|
it 'shows optional and required custom fields for edit without an seperation' do
|
|
project.custom_field_values.last.value = 'FOO'
|
|
project.save!
|
|
|
|
visit settings_generic_project_path(project.id)
|
|
|
|
expect(page).to have_content 'Required Foo'
|
|
expect(page).to have_content 'Optional Foo'
|
|
end
|
|
|
|
context 'with a 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
|
|
|
|
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
|
|
fill_in 'Foo', with: '1234'
|
|
|
|
click_on 'Save'
|
|
expect(page).to have_selector('#errorExplanation', text: 'Foo is too long (maximum is 2 characters)')
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|