#-- 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 COPYRIGHT and LICENSE files for more details. #++ require 'support/components/ng_select_autocomplete_helpers' require 'support/pages/page' module Pages class Members < Page include ::Components::NgSelectAutocompleteHelpers attr_reader :project_identifier def initialize(project_identifier) @project_identifier = project_identifier end def visit! super expect(page).to have_selector('h2', text: I18n.t(:label_member_plural)) self end def path "/projects/#{project_identifier}/members" end def open_new_member! click_on 'Add member' end def open_filters! find('#filter-member-button').click end def search_for_name(name) fill_in 'name', with: name find('.simple-filters--controls input[type=submit]').click end ## # Adds the given user to this project. # # @param user_name [String] The full name of the user. # @param as [String] The role as which the user should be added. def add_user!(user_name, as:) click_on 'Add member' SeleniumHubWaiter.wait select_principal! user_name if user_name select_role! as if as click_on 'Add' end def remove_user!(user_name) find_user(user_name).find('a[data-method=delete]').click end def remove_group!(group_name) find_group(group_name).find('a[data-method=delete]').click end def has_added_user?(name, visible: true, css: "tr") has_text?("Added #{name} to the project") and ((not visible) or has_css?(css, text: user_name_to_text(name))) end def has_added_group?(name, visible: true) has_added_user? name, visible: visible, css: "tr.group" end ## # Checks if the members page lists the given user. # # @param name [String] The full name of the user. # @param roles [Array] Checks if the user has the given role. # @param group_membership [Boolean] True if the member is added through a group. # Such members cannot be removed separately which # is why there must be only an edit and no delete button. def has_user?(name, roles: nil, group_membership: nil, group: false) css = group ? "tr.group" : "tr" has_selector?(css, text: user_name_to_text(name)) && (roles.nil? || has_roles?(name, roles, group: group)) && (group_membership.nil? || group_membership == has_group_membership?(name)) end def has_group?(name, roles: nil) has_user?(name, roles: roles, group: true) end def find_user(name) find('tr', text: name) end def find_mail(mail) find('td.email', text: mail) end def find_group(name) find('tr.group', text: name) end ## # Get contents of all cells sorted def contents(column, raw: false) nodes = if raw all("td.#{column}") else all("td.#{column} a, td.#{column} span") end nodes.map(&:text) end def user_name_to_text(name) # the members table shows last name and first name separately # let's just look for the last name name.split(" ").last end def edit_user!(name, add_roles: [], remove_roles: []) user = find_user(name) user.find('a[title=Edit]').click Array(add_roles).each { |role| check role } Array(remove_roles).each { |role| uncheck role } click_on 'Change' end def has_group_membership?(user_name) user = find_user(user_name) user.has_selector?('a[title=Edit]') && user.has_no_selector?('a[title=Delete]') end def has_roles?(user_name, roles, group: false) user = group ? find_group(user_name) : find_user(user_name) Array(roles).all? { |role| user.has_text? role } end def select_principal!(principal_name) select_autocomplete page.find("members-autocompleter"), query: principal_name, results_selector: '.ng-dropdown-panel-items' end ## # Searches for a string in the 'New Member' dialogue's principal # selection and selects the given entry. # # @param query What to search for in the user search field. # @param selection The exact result to select. def search_and_select_principal!(query, selection) search_principal! query select_search_result! selection end def search_principal!(query) search_autocomplete page.find("members-autocompleter"), query: query, results_selector: '.ng-dropdown-panel-items' end def select_search_result!(value) find('.ng-option', text: value).click end def has_search_result?(value) page.has_selector?('.ng-option', text: value) end def has_no_search_results? page.has_selector?('.ng-option', text: 'No items found') end def sort_by(column) find('.generic-table--sort-header a', text: column.upcase).click end def expect_sorted_by(column, desc: false) page.within('.generic-table--sort-header', text: column.upcase) do if desc expect(page).to have_selector('.sort.desc') else expect(page).to have_selector('.sort.asc') end end end ## # Indicates whether the given principal has been selected as one # of the users to be added to the project in the 'New member' dialogue. def has_selected_new_principal?(name) has_selector? '.ng-value', text: name end def select_role!(role_name) select = find('select#member_role_ids') select.select role_name end def expect_role(role_name, present: true) expect(page).to have_conditional_selector(present, '#member_role_ids option', text: role_name) end def go_to_page!(number) find('.op-pagination--pages a', text: number.to_s).click end end end