#-- encoding: UTF-8 #-- copyright # OpenProject is a project management system. # Copyright (C) 2012-2018 the OpenProject Foundation (OPF) # # 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 'uri' require 'cgi' require File.expand_path(File.join(File.dirname(__FILE__), '..', 'support', 'paths')) require File.expand_path(File.join(File.dirname(__FILE__), '..', 'support', 'selectors')) module WithinHelpers def press_key_on_element(key, element) page.find(element).native.send_keys(Selenium::WebDriver::Keys[key.to_sym]) end def right_click(elements) builder = page.driver.browser.action Array(elements).each do |e| builder.context_click(e.native) end builder.perform end def ctrl_click(elements) builder = page.driver.browser.action # Hold control key down builder.key_down(:control) # Note that you can retrieve the elements using capybara's # standard methods. When passing them to the builder # make sure to do .native Array(elements).each do |e| builder.click(e.native) end # Release control key builder.key_up(:control) # Do the action setup builder.perform end def with_scope(locator, options = {}) locator ? within(*selector_for(locator), options) { yield } : yield end end World(WithinHelpers) # Single-line step scoper When /^(.*) within "(.*[^:"])"$/ do |step_name, parent| with_scope(parent) { step step_name } end When /^(.*) \[i18n\]$/ do |actual_step| step translate(actual_step) end When(/^I ctrl\-click on "([^\"]+)"$/) do |text| # Click all elements that you want, in this case we click all as elements = page.all('a', text: text) ctrl_click(elements) end # Single-line step scoper When /^(.*) within_hidden (.*[^:])$/ do |step_name, parent| with_scope(parent, visible: false) { step step_name } end # Multi-line step scoper When /^(.*) within (.*[^:]):$/ do |step_name, parent, table_or_string| with_scope(parent) { step "#{step_name}:", table_or_string } end Given /^(?:|I )am on (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )go to (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )press "([^"]*)"$/ do |button| click_button(button) end When /^(?:|I )follow "([^"]*)"$/ do |link| click_link(link) end When /^(?:|I )fill in "([^"]*)" with "([^"]*)"$/ do |field, value| fill_in(field, with: value) end When /^(?:|I )fill in "([^"]*)" for "([^"]*)"$/ do |value, field| fill_in(field, with: value) end # Use this to fill in an entire form with data from a table. Example: # # When I fill in the following: # | Account Number | 5002 | # | Expiry date | 2009-11-01 | # | Note | Nice guy | # | Wants Email? | | # # TODO: Add support for checkbox and option # based on naming conventions. # When /^(?:|I )fill in the following:$/ do |fields| fields.rows_hash.each do |name, value| field = find_field(name) if field.tag_name == 'select' step(%{I select "#{value}" from "#{name}"}) else step(%{I fill in "#{name}" with "#{value}"}) end end end When (/^I do some ajax$/) do click_link('Apply') end When /^(?:|I )select "([^"]*)" from "([^"]*)"$/ do |value, field| begin select(value, from: field) rescue Capybara::ElementNotFound # find the label, get the parent and from there find the appropriate # select2 container. There are currently two dom structures in which # this can happen. xpath_selector = "//label[contains(., '#{field}')]/" + "parent::*/*[contains(@class, 'select2-container')] | " + "//label[contains(., '#{field}')]/" + "..//*[contains(@class, 'select2-container')]" container = find(:xpath, xpath_selector) container.find('.select2-choice').click if container['class'].include?('ui-select-container') # ui-select (Angular) find('ul.select2-result-single li', text: value).click else # classic select2 (jQuery) find(:xpath, "//*[@id='select2-drop']/descendant::li[contains(., '#{value}')]").click end end end When /^(?:|I )check "([^"]*)"$/ do |field| check(field) end When /^(?:|I )uncheck "([^"]*)"$/ do |field| uncheck(field) end When /^(?:|I )choose "([^"]*)"$/ do |field| choose(field) end When /^(?:|I )attach the file "([^"]*)" to "([^"]*)"$/ do |path, field| attach_file(field, File.expand_path(path)) end Then /^(?:|I )should see "([^"]*)"$/ do |text| regexp = Regexp.new(Regexp.escape(text), Regexp::IGNORECASE) page.should have_content(regexp) end Then /^(?:|I )should see \/([^\/]*)\/$/ do |regexp| regexp = Regexp.new(regexp) should have_content(regexp) end Then(/^(?:|I )should not see "([^"]*)" in the same table row as "([^"]*)"$/) do |text1, text2| within('table.generic-table tbody') do page.all(:xpath, "//tr/td[contains(.,'#{text2}')]/following-sibling::td").each do |td| expect(td).to have_no_content(text1) end end end Then /^(?:|I )should not see "([^"]*)"$/ do |text| regexp = Regexp.new(Regexp.escape(text), Regexp::IGNORECASE) page.should have_no_content(regexp) end Then /^(?:|I )should not see \/([^\/]*)\/$/ do |regexp| regexp = Regexp.new(regexp) should have_no_content(regexp) end Then /^the "([^"]*)" field(?: within (.*))? should contain "([^"]*)"$/ do |field, parent, value| with_scope(parent) do field = find_field(field) field.value.should =~ /#{value}/ end end Then /^the "([^"]*)" field(?: within (.*))? should not contain "([^"]*)"$/ do |field, parent, value| with_scope(parent) do field = find_field(field) field.value.should_not =~ /#{value}/ end end Then /^the "([^"]*)" field should have the error "([^"]*)"$/ do |field, error_message| element = find_field(field) classes = element.find(:xpath, '..')[:class].split(' ') form_for_input = element.find(:xpath, 'ancestor::form[1]') using_formtastic = form_for_input[:class].include?('formtastic') error_class = using_formtastic ? 'error' : 'field_with_errors' classes.should include(error_class) if using_formtastic error_paragraph = element.find(:xpath, '../*[@class="inline-errors"][1]') error_paragraph.should have_content(error_message) else page.should have_content("#{field.titlecase} #{error_message}") end end Then /^the "([^"]*)" field should have no error$/ do |field| element = find_field(field) classes = element.find(:xpath, '..')[:class].split(' ') classes.should_not include('field_with_errors') classes.should_not include('error') end Then /^the (hidden )?"([^"]*)" checkbox should be checked$/ do |hidden, label| field_checked = find_field(label, visible: hidden.nil?)['checked'] field_checked.should be_truthy end Then /^the (hidden )?"([^"]*)" checkbox should not be checked$/ do |hidden, label| field_checked = find_field(label, visible: hidden.nil?)['checked'] field_checked.should be_falsey end Then /^(?:|I )should be on (.+)$/ do |page_name| current_path = URI.parse(current_url).path CGI.unescape(current_path).should == CGI.unescape(path_to(page_name)) end Then /^(?:|I )should have the following query string:$/ do |expected_pairs| query = URI.parse(current_url).query actual_params = query ? CGI.parse(query) : {} expected_params = {} expected_pairs.rows_hash.each_pair { |k, v| expected_params[k] = v.split(',') } actual_params.should == expected_params end Then /^show me the page$/ do # save_and_open_page sleep 2 # sleep to ensure page has been fully loaded save_and_open_screenshot end # newly generated until here When /^I wait(?: (\d+) seconds)? for(?: the)? [Aa][Jj][Aa][Xx](?: requests?(?: to finish)?)?$/ do |timeout| ajax_done = lambda do is_done = false while !is_done is_done = page.evaluate_script(%{ (function (){ return !(window.jQuery && document.ajaxActive); }()) }.gsub("\n", '')) end end timeout = timeout.present? ? timeout.to_f : 5.0 wait_until(timeout, i_know_im_immoral: true) do ajax_done.call end end Then /^there should be a( disabled)? "(.+)" field( visible| invisible)?$/ do |disabled, fieldname, visible| # Checking for a disabled field will only work for field with labels where the label # has a correctly filled "for" attribute visibility = visible && !visible.include?('invisible') if disabled # disabled fields can not be found via find_field field_id = find('label', text: fieldname)['for'] should have_css("##{field_id}", visible: visibility) else should have_field(fieldname, visible: visibility) end end Then /^there should not be a "(.+)" field$/ do |fieldname| should_not have_field(fieldname) end Then /^there should be a "(.+)" button$/ do |button_label| page.should have_xpath("//input[@value='#{button_label}']") end Then /^the "([^\"]*)" select(?: within "([^\"]*)")? should have the following options:$/ do |field, selector, option_table| options_expected = option_table.raw.flatten with_scope(selector) do field = find_field(field) options_actual = field.all('option').map(&:text) options_actual.should =~ options_expected end end Then /^there should be the disabled "(.+)" element$/ do |element| page.find(element)[:disabled].should == 'true' end Then /^the element "(.+)" should be invalid$/ do |element| expect(page).to have_selector("#{element}:invalid") end # This needs an active js driver to work properly Given /^I (accept|dismiss) the alert dialog$/ do |method| if Capybara.current_driver.to_s.include?('selenium') page.driver.browser.switch_to.alert.send(method.to_s) end end Then(/^(.*) in the new window$/) do |step| new_window = windows.last page.within_window new_window do step(step) end end Then /^(.*) in the iframe "([^\"]+)"$/ do |step, iframe_name| browser = page.driver.browser browser.switch_to.frame(iframe_name) step(step) browser.switch_to.default_content end When /^(?:|I )click the toolbar button named "(.*?)"$/ do |action_name| find('.toolbar-container').click_button action_name end When /^(?:|I )choose "(.*?)" from the toolbar "(.*?)" dropdown$/ do |action_name, dropdown_id| find("button[has-dropdown-menu][target=#{dropdown_id}DropdownMenu]").click find("##{dropdown_id}Dropdown").click_link action_name end # that's capybara's old behaviour: clicking the first button that matches When /^(?:|I )click on the first button matching "([^"]*)"$/ do |button| first(:button, button).click end When /^(?:|I )follow the first link matching "([^"]*)"$/ do |link| first(:link, link).click end When /^(?:|I )click on the first anchor matching "([^"]*)"$/ do |anchor| find(:xpath, "(//a[text()='#{anchor}'])[1]").click end def find_lowest_containing_element(text, selector) elements = [] node_criteria = "[contains(normalize-space(.), \"#{text}\") and not(self::script) and not(child::*[contains(normalize-space(.), \"#{text}\")])]" if selector search_string = Nokogiri::CSS.xpath_for(selector).first + "//*#{node_criteria}" search_string += ' | ' + Nokogiri::CSS.xpath_for(selector).first + "#{node_criteria}" else search_string = "//*#{node_criteria}" end elements = all(:xpath, search_string) rescue Nokogiri::CSS::SyntaxError elements end require 'timeout' def wait_until(seconds = 5, options = {}, &block) unless options[:i_know_im_immoral] raise "You are immoral. I can't stand this. Goodbye. You really shouldn't use wait_until and wait for an element using Capybara instead, e.g. using page.should have_selector(...) See http://www.elabs.se/blog/53-why-wait_until-was-removed-from-capybara " end Timeout.timeout(seconds, &block) end When /^I confirm popups$/ do page.driver.browser.switch_to.alert.accept end # Needs Selenium! Then(/^I should( not )?see a(?:n) alert dialog$/) do |negative| negative = !!negative if Capybara.current_driver.to_s.include?('selenium') begin page.driver.browser.switch_to.alert expect(negative).to eq(false) rescue Selenium::WebDriver::Error::NoSuchAlertError expect(negative).to eq(true) end end end Then(/^I should see a confirm dialog$/) do page.should have_selector('#confirm_dialog') end Then /^I confirm the JS confirm dialog$/ do page.driver.browser.switch_to.alert.accept rescue Selenium::WebDriver::Error::NoAlertOpenError end Then /^I should see a JS confirm dialog$/ do page.driver.browser.switch_to.alert.text.should_not be_nil page.driver.browser.switch_to.alert.accept end