teamcapybara--capybara/lib/capybara/node/actions.rb

416 lines
19 KiB
Ruby
Raw Normal View History

2016-03-08 00:52:19 +00:00
# frozen_string_literal: true
2018-01-08 20:23:54 +00:00
module Capybara
module Node
module Actions
# @!macro waiting_behavior
# If the driver is capable of executing JavaScript, this method will wait for a set amount of time
# and continuously retry finding the element until either the element is found or the time
# expires. The length of time this method will wait is controlled through {Capybara.configure default_max_wait_time}.
#
# @option options [false, true, Numeric] wait
# Maximum time to wait for matching element to appear. Defaults to {Capybara.configure default_max_wait_time}.
##
#
# Finds a button or link and clicks it. See {#click_button} and
# {#click_link} for what locator will match against for each type of element.
#
# @overload click_link_or_button([locator], **options)
# @macro waiting_behavior
# @param [String] locator See {#click_button} and {#click_link}
#
# @return [Capybara::Node::Element] The element clicked
#
2018-01-09 22:05:50 +00:00
def click_link_or_button(locator = nil, **options)
2013-02-24 15:28:48 +00:00
find(:link_or_button, locator, options).click
end
2010-10-29 11:41:49 +00:00
alias_method :click_on, :click_link_or_button
##
#
# Finds a link by id, {Capybara.configure test_id} attribute, text or title and clicks it. Also looks at image
# alt text inside the link.
#
# @overload click_link([locator], **options)
# @macro waiting_behavior
# @param [String] locator text, id, {Capybara.configure test_id} attribute, title or nested image's alt attribute
# @param [Hash] options See {Capybara::Node::Finders#find_link}
#
# @return [Capybara::Node::Element] The element clicked
2018-01-09 22:05:50 +00:00
def click_link(locator = nil, **options)
2013-02-19 20:12:54 +00:00
find(:link, locator, options).click
end
##
#
# Finds a button on the page and clicks it.
# This can be any `<input>` element of type submit, reset, image, button or it can be a
# `<button>` element. All buttons can be found by their id, name, {Capybara.configure test_id} attribute, value, or title. `<button>` elements can also be found
# by their text content, and image `<input>` elements by their alt attribute.
#
2018-05-17 21:45:53 +00:00
# @overload click_button([locator], **options)
# @macro waiting_behavior
# @param [String] locator Which button to find
# @param [Hash] options See {Capybara::Node::Finders#find_button}
# @return [Capybara::Node::Element] The element clicked
2018-01-09 22:05:50 +00:00
def click_button(locator = nil, **options)
2013-02-24 15:36:08 +00:00
find(:button, locator, options).click
end
##
#
# Locate a text field or text area and fill it in with the given text.
2019-05-31 19:21:40 +00:00
# The field can be found via its name, id, {Capybara.configure test_id} attribute, placeholder, or label text.
# If no locator is provided this will operate on self or a descendant.
#
2018-09-10 22:34:12 +00:00
# # will fill in a descendant fillable field with name, id, or label text matching 'Name'
2016-10-04 18:10:29 +00:00
# page.fill_in 'Name', with: 'Bob'
#
2018-09-10 22:34:12 +00:00
# # will fill in `el` if it's a fillable field
# el.fill_in with: 'Tom'
#
#
2018-05-17 21:45:53 +00:00
# @overload fill_in([locator], with:, **options)
# @param [String] locator Which field to fill in
# @param [Hash] options
# @param with: [String] The value to fill in
# @macro waiting_behavior
2018-05-17 21:45:53 +00:00
# @option options [String] currently_with The current value property of the field to fill in
# @option options [Boolean] multiple Match fields that can have multiple values?
2018-10-28 19:00:34 +00:00
# @option options [String, Regexp] id Match fields that match the id attribute
2018-05-17 21:45:53 +00:00
# @option options [String] name Match fields that match the name attribute
# @option options [String] placeholder Match fields that match the placeholder attribute
2018-10-28 19:00:34 +00:00
# @option options [String, Array<String>, Regexp] class Match fields that match the class(es) provided
# @option options [Hash] fill_options Driver specific options regarding how to fill fields (Defaults come from {Capybara.configure default_set_options})
#
# @return [Capybara::Node::Element] The element filled in
2018-07-04 17:02:02 +00:00
def fill_in(locator = nil, with:, currently_with: nil, fill_options: {}, **find_options)
find_options[:with] = currently_with if currently_with
find_options[:allow_self] = true if locator.nil?
2018-07-04 17:02:02 +00:00
find(:fillable_field, locator, find_options).set(with, fill_options)
end
# @!macro label_click
# @option options [Boolean] allow_label_click
# Attempt to click the label to toggle state if element is non-visible. Defaults to {Capybara.configure automatic_label_click}.
##
#
# Find a descendant radio button and mark it as checked. The radio button can be found
2019-05-31 19:21:40 +00:00
# via name, id, {Capybara.configure test_id} attribute or label text. If no locator is
# provided this will match against self or a descendant.
#
2018-09-10 22:34:12 +00:00
# # will choose a descendant radio button with a name, id, or label text matching 'Male'
# page.choose('Male')
#
2018-09-10 22:34:12 +00:00
# # will choose `el` if it's a radio button element
# el.choose()
#
2018-05-17 21:45:53 +00:00
# @overload choose([locator], **options)
# @param [String] locator Which radio button to choose
#
2018-05-17 21:45:53 +00:00
# @option options [String] option Value of the radio_button to choose
2018-10-28 19:00:34 +00:00
# @option options [String, Regexp] id Match fields that match the id attribute
2018-05-17 21:45:53 +00:00
# @option options [String] name Match fields that match the name attribute
2018-10-28 19:00:34 +00:00
# @option options [String, Array<String>, Regexp] class Match fields that match the class(es) provided
# @macro waiting_behavior
# @macro label_click
#
# @return [Capybara::Node::Element] The element chosen or the label clicked
2018-01-09 22:05:50 +00:00
def choose(locator = nil, **options)
_check_with_label(:radio_button, true, locator, options)
end
##
#
# Find a descendant check box and mark it as checked. The check box can be found
2019-05-31 19:21:40 +00:00
# via name, id, {Capybara.configure test_id} attribute, or label text. If no locator
# is provided this will match against self or a descendant.
#
2018-09-10 22:34:12 +00:00
# # will check a descendant checkbox with a name, id, or label text matching 'German'
# page.check('German')
#
2018-09-10 22:34:12 +00:00
# # will check `el` if it's a checkbox element
# el.check()
#
#
2018-05-17 21:45:53 +00:00
# @overload check([locator], **options)
# @param [String] locator Which check box to check
#
2018-05-17 21:45:53 +00:00
# @option options [String] option Value of the checkbox to select
2018-10-28 19:00:34 +00:00
# @option options [String, Regexp] id Match fields that match the id attribute
2016-04-27 05:44:48 +00:00
# @option options [String] name Match fields that match the name attribute
2018-10-28 19:00:34 +00:00
# @option options [String, Array<String>, Regexp] class Match fields that match the class(es) provided
# @macro label_click
# @macro waiting_behavior
#
# @return [Capybara::Node::Element] The element checked or the label clicked
2018-04-27 16:39:34 +00:00
def check(locator = nil, **options)
_check_with_label(:checkbox, true, locator, options)
end
##
#
2018-09-10 22:34:12 +00:00
# Find a descendant check box and uncheck it. The check box can be found
2019-05-31 19:21:40 +00:00
# via name, id, {Capybara.configure test_id} attribute, or label text. If
# no locator is provided this will match against self or a descendant.
#
2018-09-10 22:34:12 +00:00
# # will uncheck a descendant checkbox with a name, id, or label text matching 'German'
# page.uncheck('German')
#
2018-09-10 22:34:12 +00:00
# # will uncheck `el` if it's a checkbox element
# el.uncheck()
#
#
2018-05-17 21:45:53 +00:00
# @overload uncheck([locator], **options)
# @param [String] locator Which check box to uncheck
#
2018-05-17 21:45:53 +00:00
# @option options [String] option Value of the checkbox to deselect
2018-10-28 19:00:34 +00:00
# @option options [String, Regexp] id Match fields that match the id attribute
2016-04-27 05:44:48 +00:00
# @option options [String] name Match fields that match the name attribute
2018-10-28 19:00:34 +00:00
# @option options [String, Array<String>, Regexp] class Match fields that match the class(es) provided
# @macro label_click
# @macro waiting_behavior
#
# @return [Capybara::Node::Element] The element unchecked or the label clicked
2018-01-09 22:05:50 +00:00
def uncheck(locator = nil, **options)
_check_with_label(:checkbox, false, locator, options)
end
##
#
# If `from` option is present, {#select} finds a select box, or text input with associated datalist,
# on the page and selects a particular option from it.
2014-08-24 13:20:45 +00:00
# Otherwise it finds an option inside current scope and selects it.
# If the select box is a multiple select, {#select} can be called multiple times to select more than
2014-08-24 13:20:45 +00:00
# one option.
2019-05-31 19:21:40 +00:00
# The select box can be found via its name, id, {Capybara.configure test_id} attribute, or label text.
# The option can be found by its text.
#
2016-10-04 18:10:29 +00:00
# page.select 'March', from: 'Month'
#
# @overload select(value = nil, from: nil, **options)
# @macro waiting_behavior
#
# @param value [String] Which option to select
# @param from [String] The id, {Capybara.configure test_id} attribute, name or label of the select box
#
# @return [Capybara::Node::Element] The option element selected
2018-01-09 22:05:50 +00:00
def select(value = nil, from: nil, **options)
raise ArgumentError, 'The :from option does not take an element' if from.is_a? Capybara::Node::Element
2018-05-15 16:17:09 +00:00
el = from ? find_select_or_datalist_input(from, options) : self
2018-07-10 21:18:39 +00:00
if el.respond_to?(:tag_name) && (el.tag_name == 'input')
2018-05-15 16:17:09 +00:00
select_datalist_option(el, value)
else
2018-05-15 16:17:09 +00:00
el.find(:option, value, options).select_option
end
end
##
#
2011-04-07 19:43:30 +00:00
# Find a select box on the page and unselect a particular option from it. If the select
# box is a multiple select, {#unselect} can be called multiple times to unselect more than
2019-05-31 19:21:40 +00:00
# one option. The select box can be found via its name, id, {Capybara.configure test_id} attribute,
# or label text.
#
2016-10-04 18:10:29 +00:00
# page.unselect 'March', from: 'Month'
#
# @overload unselect(value = nil, from: nil, **options)
# @macro waiting_behavior
#
# @param value [String] Which option to unselect
# @param from [String] The id, {Capybara.configure test_id} attribute, name or label of the select box
#
#
# @return [Capybara::Node::Element] The option element unselected
2018-01-09 22:05:50 +00:00
def unselect(value = nil, from: nil, **options)
raise ArgumentError, 'The :from option does not take an element' if from.is_a? Capybara::Node::Element
2018-01-13 00:57:41 +00:00
scope = from ? find(:select, from, options) : self
scope.find(:option, value, options).unselect_option
end
##
#
2019-02-25 19:10:34 +00:00
# Find a descendant file field on the page and attach a file given its path. There are two ways to use
2019-05-31 19:21:40 +00:00
# {#attach_file}, in the first method the file field can be found via its name, id,
# {Capybara.configure test_id} attribute, or label text. In the case of the file field being hidden for
2018-07-12 16:26:51 +00:00
# styling reasons the `make_visible` option can be used to temporarily change the CSS of
# the file field, attach the file, and then revert the CSS back to original. If no locator is
# passed this will match self or a descendant.
2019-02-25 19:10:34 +00:00
# The second method, which is currently in beta and may be changed/removed, involves passing a block
# which performs whatever actions would trigger the file chooser to appear.
#
2018-09-10 22:34:12 +00:00
# # will attach file to a descendant file input element that has a name, id, or label_text matching 'My File'
# page.attach_file('My File', '/path/to/file.png')
#
# # will attach file to el if it's a file input element
# el.attach_file('/path/to/file.png')
#
2019-02-25 19:10:34 +00:00
# # will attach file to whatever file input is triggered by the block
# page.attach_file('/path/to/file.png') do
# page.find('#upload_button').click
# end
#
# @overload attach_file([locator], paths, **options)
2018-05-17 21:45:53 +00:00
# @macro waiting_behavior
#
2018-08-17 20:57:12 +00:00
# @param [String] locator Which field to attach the file to
# @param [String, Array<String>] paths The path(s) of the file(s) that will be attached
#
# @option options [Symbol] match
# The matching strategy to use (:one, :first, :prefer_exact, :smart). Defaults to {Capybara.configure match}.
# @option options [Boolean] exact
# Match the exact label name/contents or accept a partial match. Defaults to {Capybara.configure exact}.
2018-05-17 21:45:53 +00:00
# @option options [Boolean] multiple Match field which allows multiple file selection
2018-10-28 19:00:34 +00:00
# @option options [String, Regexp] id Match fields that match the id attribute
2018-05-17 21:45:53 +00:00
# @option options [String] name Match fields that match the name attribute
2018-10-28 19:00:34 +00:00
# @option options [String, Array<String>, Regexp] class Match fields that match the class(es) provided
# @option options [true, Hash] make_visible
# A Hash of CSS styles to change before attempting to attach the file, if `true`, `{ opacity: 1, display: 'block', visibility: 'visible' }` is used (may not be supported by all drivers).
2019-02-25 19:10:34 +00:00
# @overload attach_file(paths, &blk)
# @param [String, Array<String>] paths The path(s) of the file(s) that will be attached
# @yield Block whose actions will trigger the system file chooser to be shown
# @return [Capybara::Node::Element] The file field element
def attach_file(locator = nil, paths, make_visible: nil, **options) # rubocop:disable Style/OptionalArguments
2019-02-25 19:10:34 +00:00
raise ArgumentError, '``#attach_file` does not support passing both a locator and a block' if locator && block_given?
Array(paths).each do |path|
raise Capybara::FileNotFound, "cannot attach file, #{path} does not exist" unless File.exist?(path.to_s)
end
options[:allow_self] = true if locator.nil?
2019-02-25 19:10:34 +00:00
if block_given?
begin
execute_script CAPTURE_FILE_ELEMENT_SCRIPT
yield
file_field = evaluate_script 'window._capybara_clicked_file_input'
rescue ::Capybara::NotSupportedByDriverError
warn 'Block mode of `#attach_file` is not supported by the current driver - ignoring.'
end
end
# Allow user to update the CSS style of the file input since they are so often hidden on a page
2017-05-02 01:39:08 +00:00
if make_visible
2019-02-25 19:10:34 +00:00
ff = file_field || find(:file_field, locator, options.merge(visible: :all))
while_visible(ff, make_visible) { |el| el.set(paths) }
else
2019-02-25 19:10:34 +00:00
(file_field || find(:file_field, locator, options)).set(paths)
end
end
2016-11-22 18:45:15 +00:00
private
2018-01-09 22:05:50 +00:00
2018-05-15 16:17:09 +00:00
def find_select_or_datalist_input(from, options)
synchronize(Capybara::Queries::BaseQuery.wait(options, session_options.default_max_wait_time)) do
begin
find(:select, from, options)
2019-06-25 20:46:33 +00:00
rescue Capybara::ElementNotFound => select_error # rubocop:disable Naming/RescuedExceptionsVariableName
2018-05-15 16:17:09 +00:00
raise if %i[selected with_selected multiple].any? { |option| options.key?(option) }
2018-09-24 16:43:46 +00:00
2018-05-15 16:17:09 +00:00
begin
find(:datalist_input, from, options)
2019-06-25 20:46:33 +00:00
rescue Capybara::ElementNotFound => dlinput_error # rubocop:disable Naming/RescuedExceptionsVariableName
2018-05-15 16:17:09 +00:00
raise Capybara::ElementNotFound, "#{select_error.message} and #{dlinput_error.message}"
end
end
end
end
def select_datalist_option(input, value)
datalist_options = input.evaluate_script(DATALIST_OPTIONS_SCRIPT)
option = datalist_options.find { |opt| opt.values_at('value', 'label').include?(value) }
2018-07-10 21:18:39 +00:00
raise ::Capybara::ElementNotFound, %(Unable to find datalist option "#{value}") unless option
2018-09-24 16:43:46 +00:00
2018-07-10 21:18:39 +00:00
input.set(option['value'])
2018-05-15 16:17:09 +00:00
rescue ::Capybara::NotSupportedByDriverError
# Implement for drivers that don't support JS
datalist = find(:xpath, XPath.descendant(:datalist)[XPath.attr(:id) == input[:list]], visible: false)
option = datalist.find(:datalist_option, value, disabled: false)
input.set(option.value)
end
2018-01-12 00:45:50 +00:00
def while_visible(element, visible_css)
if visible_css == true
visible_css = { opacity: 1, display: 'block', visibility: 'visible', width: 'auto', height: 'auto' }
end
2018-01-12 00:45:50 +00:00
_update_style(element, visible_css)
2018-07-10 21:18:39 +00:00
raise ExpectationNotMet, 'The style changes in :make_visible did not make the file input visible' unless element.visible?
2018-09-24 16:43:46 +00:00
begin
2018-01-12 00:45:50 +00:00
yield element
ensure
_reset_style(element)
end
end
2016-11-22 18:45:15 +00:00
2018-01-12 00:45:50 +00:00
def _update_style(element, style)
element.execute_script(UPDATE_STYLE_SCRIPT, style)
2018-01-12 00:45:50 +00:00
rescue Capybara::NotSupportedByDriverError
2018-07-10 21:18:39 +00:00
warn 'The :make_visible option is not supported by the current driver - ignoring'
2018-01-12 00:45:50 +00:00
end
2017-01-09 20:50:56 +00:00
def _reset_style(element)
element.execute_script(RESET_STYLE_SCRIPT)
2018-05-14 21:30:34 +00:00
rescue StandardError # rubocop:disable Lint/HandleExceptions swallow extra errors
2017-01-09 20:50:56 +00:00
end
def _check_with_label(selector, checked, locator,
allow_label_click: session_options.automatic_label_click, **options)
options[:allow_self] = true if locator.nil?
2018-01-09 22:05:50 +00:00
synchronize(Capybara::Queries::BaseQuery.wait(options, session_options.default_max_wait_time)) do
2016-11-22 18:45:15 +00:00
begin
el = find(selector, locator, options)
el.set(checked)
2019-04-04 19:03:01 +00:00
rescue StandardError => e
raise unless allow_label_click && catch_error?(e)
2018-09-24 16:43:46 +00:00
2016-11-22 18:45:15 +00:00
begin
el ||= find(selector, locator, options.merge(visible: :all))
el.session.find(:label, for: el, visible: true).click unless el.checked? == checked
2018-05-14 21:30:34 +00:00
rescue StandardError # swallow extra errors - raise original
2019-04-04 19:03:01 +00:00
raise e
2016-11-22 18:45:15 +00:00
end
end
end
end
2018-01-12 00:45:50 +00:00
2018-05-16 19:47:08 +00:00
UPDATE_STYLE_SCRIPT = <<~'JS'
this.capybara_style_cache = this.style.cssText;
var css = arguments[0];
2018-01-12 00:45:50 +00:00
for (var prop in css){
if (css.hasOwnProperty(prop)) {
this.style.setProperty(prop, css[prop], "important");
2018-01-12 00:45:50 +00:00
}
}
JS
2018-05-16 19:47:08 +00:00
RESET_STYLE_SCRIPT = <<~'JS'
if (this.hasOwnProperty('capybara_style_cache')) {
this.style.cssText = this.capybara_style_cache;
delete this.capybara_style_cache;
2018-01-12 00:45:50 +00:00
}
JS
2018-05-16 19:47:08 +00:00
DATALIST_OPTIONS_SCRIPT = <<~'JS'
Array.prototype.slice.call((this.list||{}).options || []).
filter(function(el){ return !el.disabled }).
map(function(el){ return { "value": el.value, "label": el.label} })
JS
2019-02-25 19:10:34 +00:00
CAPTURE_FILE_ELEMENT_SCRIPT = <<~'JS'
document.addEventListener('click', function file_catcher(e){
if (e.target.matches("input[type='file']")) {
2019-02-25 19:10:34 +00:00
window._capybara_clicked_file_input = e.target;
this.removeEventListener('click', file_catcher);
2019-02-25 19:10:34 +00:00
e.preventDefault();
}
})
JS
end
end
end