# frozen_string_literal: true # Redmine - project management software # Copyright (C) 2006-2021 Jean-Philippe Lang # # 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. require File.expand_path('../../test_helper', __FILE__) class ContextMenusControllerTest < Redmine::ControllerTest fixtures :projects, :trackers, :projects_trackers, :roles, :member_roles, :members, :enabled_modules, :workflows, :journals, :journal_details, :versions, :issues, :issue_statuses, :issue_categories, :users, :enumerations, :time_entries, :custom_fields, :custom_fields_trackers, :custom_fields_projects def test_context_menu_one_issue_should_link_to_issue_path @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_response :success assert_select 'a.icon-edit[href=?]', '/issues/1/edit', :text => 'Edit' assert_select 'a.icon-copy-link[data-clipboard-text=?]', 'http://test.host/issues/1', :text => 'Copy link' assert_select 'a.icon-copy[href=?]', '/projects/ecookbook/issues/1/copy', :text => 'Copy' assert_select 'a.icon-del[href=?]', '/issues?ids%5B%5D=1', :text => 'Delete' # Statuses assert_select 'a[href=?][data-method="patch"]', '/issues/1?ids%5B%5D=1&issue%5Bstatus_id%5D=5', :text => 'Closed' assert_select 'a[href=?][data-method="patch"]', '/issues/1?ids%5B%5D=1&issue%5Bpriority_id%5D=8', :text => 'Immediate' # No inactive priorities assert_select 'a', :text => /Inactive Priority/, :count => 0 # Versions assert_select 'a[href=?][data-method="patch"]', '/issues/1?ids%5B%5D=1&issue%5Bfixed_version_id%5D=3', :text => '2.0' assert_select 'a[href=?][data-method="patch"]', '/issues/1?ids%5B%5D=1&issue%5Bfixed_version_id%5D=4', :text => 'eCookbook Subproject 1 - 2.0' # Assignees assert_select 'a[href=?][data-method="patch"]', '/issues/1?ids%5B%5D=1&issue%5Bassigned_to_id%5D=3', :text => 'Dave Lopper' end def test_context_menu_multiple_issues_should_link_to_bulk_update_issues_path @request.session[:user_id] = 2 get :issues, :params => { :ids => [1, 2] } assert_response :success assert_select 'a.icon-edit[href=?]', '/issues/bulk_edit?ids%5B%5D=1&ids%5B%5D=2', :text => 'Bulk edit' assert_select 'a.icon-copy[href=?]', '/issues/bulk_edit?copy=1&ids%5B%5D=1&ids%5B%5D=2', :text => 'Copy' assert_select 'a.icon-del[href=?]', '/issues?ids%5B%5D=1&ids%5B%5D=2', :text => 'Delete' # Statuses assert_select 'a[href=?][data-method="patch"]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bstatus_id%5D=5', :text => 'Closed' assert_select 'a[href=?][data-method="patch"]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bpriority_id%5D=8', :text => 'Immediate' # No inactive priorities assert_select 'a', :text => /Inactive Priority/, :count => 0 # Versions assert_select 'a[href=?][data-method="patch"]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bfixed_version_id%5D=3', :text => '2.0' assert_select 'a[href=?][data-method="patch"]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bfixed_version_id%5D=4', :text => 'eCookbook Subproject 1 - 2.0' # Assignees assert_select 'a[href=?][data-method="patch"]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bassigned_to_id%5D=3', :text => 'Dave Lopper' end def test_context_menu_one_issue_by_anonymous with_settings :default_language => 'en' do get( :issues, :params => { :ids => [1] } ) assert_response :success assert_select 'a.icon-del.disabled[href="#"]', :text => 'Delete' end end def test_context_menu_multiple_issues_of_same_project @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2] } ) assert_response :success ids = [1, 2].map {|i| "ids%5B%5D=#{i}"}.join('&') assert_select 'a.icon-edit[href=?]', "/issues/bulk_edit?#{ids}", :text => 'Bulk edit' # issue_id: '1,2', set_filter: 1, status_id: '*' assert_select 'a.icon-copy-link[data-clipboard-text=?]', "http://test.host/projects/ecookbook/issues?issue_id=1%2C2&set_filter=1&status_id=%2A", :text => 'Copy link' assert_select 'a.icon-copy[href=?]', "/issues/bulk_edit?copy=1&#{ids}", :text => 'Copy' assert_select 'a.icon-del[href=?]', "/issues?#{ids}", :text => 'Delete' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bstatus_id%5D=5", :text => 'Closed' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bpriority_id%5D=8", :text => 'Immediate' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bassigned_to_id%5D=3", :text => 'Dave Lopper' end def test_context_menu_multiple_issues_of_different_projects @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2, 6] } ) assert_response :success ids = [1, 2, 6].map {|i| "ids%5B%5D=#{i}"}.join('&') assert_select 'a.icon-edit[href=?]', "/issues/bulk_edit?#{ids}", :text => 'Bulk edit' # issue_id: '1,2,6', set_filter: 1, status_id: '*' assert_select 'a.icon-copy-link[data-clipboard-text=?]', "http://test.host/issues?issue_id=1%2C2%2C6&set_filter=1&status_id=%2A", :text => 'Copy link' assert_select 'a.icon-del[href=?]', "/issues?#{ids}", :text => 'Delete' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bstatus_id%5D=5", :text => 'Closed' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bpriority_id%5D=8", :text => 'Immediate' assert_select 'a[href=?]', "/issues/bulk_update?#{ids}&issue%5Bassigned_to_id%5D=2", :text => 'John Smith' end def test_context_menu_should_include_list_custom_fields field = IssueCustomField.create!(:name => 'List', :field_format => 'list', :possible_values => ['Foo', 'Bar'], :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'List' assert_select 'ul' do assert_select 'a', 3 assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=Foo", :text => 'Foo' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_context_menu_multiple_issues_should_include_list_custom_fields field = IssueCustomField.create!(:name => 'List', :field_format => 'list', :possible_values => ['Foo', 'Bar'], :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'List' assert_select 'ul' do assert_select 'a', 3 assert_select 'a[href=?]', "/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=Foo", :text => 'Foo' assert_select 'a[href=?]', "/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_context_menu_should_not_include_null_value_for_required_custom_fields field = IssueCustomField.create!(:name => 'List', :is_required => true, :field_format => 'list', :possible_values => ['Foo', 'Bar'], :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'List' assert_select 'ul' do assert_select 'a', 2 assert_select 'a', :text => 'none', :count => 0 end end end def test_context_menu_on_single_issue_should_select_current_custom_field_value field = IssueCustomField.create!(:name => 'List', :field_format => 'list', :possible_values => ['Foo', 'Bar'], :is_for_all => true, :tracker_ids => [1, 2, 3]) issue = Issue.find(1) issue.custom_field_values = {field.id => 'Bar'} issue.save! @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'List' assert_select 'ul' do assert_select 'a', 3 assert_select 'a.icon.icon-checked', :text => 'Bar' end end end def test_context_menu_should_include_bool_custom_fields field = IssueCustomField.create!(:name => 'Bool', :field_format => 'bool', :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'Bool' assert_select 'ul' do assert_select 'a', 3 assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=0", :text => 'No' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=1", :text => 'Yes' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_context_menu_should_include_user_custom_fields field = IssueCustomField.create!(:name => 'User', :field_format => 'user', :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'User' assert_select 'ul' do assert_select 'a', Project.find(1).members.count + 2 # users + 'none' + 'me' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=2", :text => 'John Smith' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_context_menu_should_include_version_custom_fields field = IssueCustomField.create!(:name => 'Version', :field_format => 'version', :is_for_all => true, :tracker_ids => [1, 2, 3]) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => 'Version' assert_select 'ul' do assert_select 'a', Project.find(1).shared_versions.count + 1 assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=3", :text => '2.0' assert_select 'a[href=?]', "/issues/1?ids%5B%5D=1&issue%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_context_menu_should_show_enabled_custom_fields_for_the_role_only enabled_cf = IssueCustomField.generate!( :field_format => 'bool', :is_for_all => true, :tracker_ids => [1], :visible => false, :role_ids => [1, 2] ) disabled_cf = IssueCustomField.generate!( :field_format => 'bool', :is_for_all => true, :tracker_ids => [1], :visible => false, :role_ids => [2] ) issue = Issue.generate!(:project_id => 1, :tracker_id => 1) @request.session[:user_id] = 2 get( :issues, :params => { :ids => [issue.id] } ) assert_select "li.cf_#{enabled_cf.id}" assert_select "li.cf_#{disabled_cf.id}", 0 end def test_context_menu_by_assignable_user_should_include_assigned_to_me_link @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2] } ) assert_response :success assert_select 'a[href=?]', '/issues/bulk_update?ids%5B%5D=1&ids%5B%5D=2&issue%5Bassigned_to_id%5D=2', :text => / me / end def test_context_menu_should_propose_shared_versions_for_issues_from_different_projects @request.session[:user_id] = 2 version = Version.create!(:name => 'Shared', :sharing => 'system', :project_id => 1) get( :issues, :params => { :ids => [1, 4] } ) assert_response :success assert_select 'a', :text => 'eCookbook - Shared' end def test_context_menu_should_include_add_subtask_link @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1] } ) assert_response :success assert_select 'a.icon-add[href=?]', '/projects/ecookbook/issues/new?issue%5Bparent_issue_id%5D=1&issue%5Btracker_id%5D=1', :text => 'Add subtask' end def test_context_menu_with_closed_issue_should_not_include_add_subtask_link @request.session[:user_id] = 2 get( :issues, :params => { :ids => [8] } ) assert_response :success assert_select 'a.icon-add', :text => 'Add subtask', :count => 0 end def test_context_menu_multiple_issues_should_not_include_add_subtask_link @request.session[:user_id] = 2 get( :issues, :params => { :ids => [1, 2] } ) assert_response :success assert_select 'a.icon-add', :text => 'Add subtask', :count => 0 end def test_context_menu_with_issue_that_is_not_visible_should_fail get( :issues, :params => { :ids => [1, 4] # issue 4 is not visible } ) assert_response 302 end def test_should_respond_with_404_without_ids get :issues assert_response 404 end def test_time_entries_context_menu @request.session[:user_id] = 2 get( :time_entries, :params => { :ids => [1, 2] } ) assert_response :success assert_select 'a:not(.disabled)', :text => 'Bulk edit' end def test_context_menu_for_one_time_entry @request.session[:user_id] = 2 get( :time_entries, :params => { :ids => [1] } ) assert_response :success assert_select 'a:not(.disabled)', :text => 'Edit' end def test_time_entries_context_menu_should_include_custom_fields field = TimeEntryCustomField.generate!(:name => "Field", :field_format => "list", :possible_values => ["foo", "bar"]) @request.session[:user_id] = 2 get( :time_entries, :params => { :ids => [1, 2] } ) assert_response :success assert_select "li.cf_#{field.id}" do assert_select 'a[href="#"]', :text => "Field" assert_select 'ul' do assert_select 'a', 3 assert_select 'a[href=?]', "/time_entries/bulk_update?ids%5B%5D=1&ids%5B%5D=2&time_entry%5Bcustom_field_values%5D%5B#{field.id}%5D=foo", :text => 'foo' assert_select 'a[href=?]', "/time_entries/bulk_update?ids%5B%5D=1&ids%5B%5D=2&time_entry%5Bcustom_field_values%5D%5B#{field.id}%5D=bar", :text => 'bar' assert_select 'a[href=?]', "/time_entries/bulk_update?ids%5B%5D=1&ids%5B%5D=2&time_entry%5Bcustom_field_values%5D%5B#{field.id}%5D=__none__", :text => 'none' end end end def test_time_entries_context_menu_with_edit_own_time_entries_permission @request.session[:user_id] = 2 Role.find_by_name('Manager').remove_permission! :edit_time_entries Role.find_by_name('Manager').add_permission! :edit_own_time_entries ids = (0..1).map {TimeEntry.generate!(:user => User.find(2)).id} get( :time_entries, :params => { :ids => ids } ) assert_response :success assert_select 'a:not(.disabled)', :text => 'Bulk edit' end def test_time_entries_context_menu_without_edit_permission @request.session[:user_id] = 2 Role.find_by_name('Manager').remove_permission! :edit_time_entries get( :time_entries, :params => { :ids => [1, 2] } ) assert_response :success assert_select 'a.disabled', :text => 'Bulk edit' end end