capybara 3.16.1 → 3.33.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.yardopts +1 -0
- data/History.md +321 -0
- data/README.md +51 -60
- data/lib/capybara.rb +71 -114
- data/lib/capybara/config.rb +8 -5
- data/lib/capybara/cucumber.rb +1 -1
- data/lib/capybara/driver/node.rb +15 -3
- data/lib/capybara/dsl.rb +10 -2
- data/lib/capybara/helpers.rb +5 -3
- data/lib/capybara/minitest.rb +242 -141
- data/lib/capybara/minitest/spec.rb +159 -90
- data/lib/capybara/node/actions.rb +85 -74
- data/lib/capybara/node/base.rb +4 -4
- data/lib/capybara/node/document.rb +2 -2
- data/lib/capybara/node/document_matchers.rb +3 -3
- data/lib/capybara/node/element.rb +216 -117
- data/lib/capybara/node/finders.rb +65 -65
- data/lib/capybara/node/matchers.rb +228 -126
- data/lib/capybara/node/simple.rb +9 -4
- data/lib/capybara/queries/ancestor_query.rb +5 -7
- data/lib/capybara/queries/base_query.rb +2 -1
- data/lib/capybara/queries/current_path_query.rb +1 -1
- data/lib/capybara/queries/selector_query.rb +296 -30
- data/lib/capybara/queries/sibling_query.rb +5 -4
- data/lib/capybara/queries/style_query.rb +2 -2
- data/lib/capybara/queries/text_query.rb +13 -1
- data/lib/capybara/queries/title_query.rb +1 -1
- data/lib/capybara/rack_test/browser.rb +7 -2
- data/lib/capybara/rack_test/driver.rb +1 -1
- data/lib/capybara/rack_test/form.rb +1 -1
- data/lib/capybara/rack_test/node.rb +43 -7
- data/lib/capybara/registration_container.rb +44 -0
- data/lib/capybara/registrations/drivers.rb +36 -0
- data/lib/capybara/registrations/patches/puma_ssl.rb +27 -0
- data/lib/capybara/registrations/servers.rb +44 -0
- data/lib/capybara/result.rb +36 -8
- data/lib/capybara/rspec/matcher_proxies.rb +6 -4
- data/lib/capybara/rspec/matchers.rb +100 -63
- data/lib/capybara/rspec/matchers/base.rb +23 -10
- data/lib/capybara/rspec/matchers/count_sugar.rb +37 -0
- data/lib/capybara/rspec/matchers/have_ancestor.rb +28 -0
- data/lib/capybara/rspec/matchers/have_current_path.rb +2 -2
- data/lib/capybara/rspec/matchers/have_selector.rb +16 -8
- data/lib/capybara/rspec/matchers/have_sibling.rb +27 -0
- data/lib/capybara/rspec/matchers/have_text.rb +4 -4
- data/lib/capybara/rspec/matchers/have_title.rb +2 -2
- data/lib/capybara/rspec/matchers/match_selector.rb +3 -3
- data/lib/capybara/rspec/matchers/match_style.rb +2 -2
- data/lib/capybara/rspec/matchers/spatial_sugar.rb +39 -0
- data/lib/capybara/selector.rb +219 -588
- data/lib/capybara/selector/builders/css_builder.rb +10 -6
- data/lib/capybara/selector/builders/xpath_builder.rb +1 -1
- data/lib/capybara/selector/css.rb +4 -2
- data/lib/capybara/selector/definition.rb +277 -0
- data/lib/capybara/selector/definition/button.rb +52 -0
- data/lib/capybara/selector/definition/checkbox.rb +26 -0
- data/lib/capybara/selector/definition/css.rb +10 -0
- data/lib/capybara/selector/definition/datalist_input.rb +35 -0
- data/lib/capybara/selector/definition/datalist_option.rb +25 -0
- data/lib/capybara/selector/definition/element.rb +27 -0
- data/lib/capybara/selector/definition/field.rb +40 -0
- data/lib/capybara/selector/definition/fieldset.rb +14 -0
- data/lib/capybara/selector/definition/file_field.rb +13 -0
- data/lib/capybara/selector/definition/fillable_field.rb +33 -0
- data/lib/capybara/selector/definition/frame.rb +17 -0
- data/lib/capybara/selector/definition/id.rb +6 -0
- data/lib/capybara/selector/definition/label.rb +62 -0
- data/lib/capybara/selector/definition/link.rb +54 -0
- data/lib/capybara/selector/definition/link_or_button.rb +16 -0
- data/lib/capybara/selector/definition/option.rb +27 -0
- data/lib/capybara/selector/definition/radio_button.rb +27 -0
- data/lib/capybara/selector/definition/select.rb +81 -0
- data/lib/capybara/selector/definition/table.rb +109 -0
- data/lib/capybara/selector/definition/table_row.rb +21 -0
- data/lib/capybara/selector/definition/xpath.rb +5 -0
- data/lib/capybara/selector/filter_set.rb +13 -9
- data/lib/capybara/selector/filters/base.rb +11 -2
- data/lib/capybara/selector/filters/locator_filter.rb +13 -3
- data/lib/capybara/selector/regexp_disassembler.rb +9 -2
- data/lib/capybara/selector/selector.rb +43 -448
- data/lib/capybara/selenium/atoms/getAttribute.min.js +1 -0
- data/lib/capybara/selenium/atoms/isDisplayed.min.js +1 -0
- data/lib/capybara/selenium/atoms/src/getAttribute.js +161 -0
- data/lib/capybara/selenium/atoms/src/isDisplayed.js +454 -0
- data/lib/capybara/selenium/driver.rb +125 -56
- data/lib/capybara/selenium/driver_specializations/chrome_driver.rb +73 -17
- data/lib/capybara/selenium/driver_specializations/edge_driver.rb +124 -0
- data/lib/capybara/selenium/driver_specializations/firefox_driver.rb +41 -2
- data/lib/capybara/selenium/driver_specializations/internet_explorer_driver.rb +14 -1
- data/lib/capybara/selenium/driver_specializations/safari_driver.rb +14 -5
- data/lib/capybara/selenium/extensions/file_input_click_emulation.rb +34 -0
- data/lib/capybara/selenium/extensions/find.rb +67 -45
- data/lib/capybara/selenium/extensions/html5_drag.rb +152 -36
- data/lib/capybara/selenium/extensions/modifier_keys_stack.rb +28 -0
- data/lib/capybara/selenium/logger_suppressor.rb +34 -0
- data/lib/capybara/selenium/node.rb +227 -56
- data/lib/capybara/selenium/nodes/chrome_node.rb +93 -8
- data/lib/capybara/selenium/nodes/edge_node.rb +104 -0
- data/lib/capybara/selenium/nodes/firefox_node.rb +37 -59
- data/lib/capybara/selenium/nodes/ie_node.rb +22 -0
- data/lib/capybara/selenium/nodes/safari_node.rb +27 -54
- data/lib/capybara/selenium/patches/action_pauser.rb +26 -0
- data/lib/capybara/selenium/patches/atoms.rb +18 -0
- data/lib/capybara/selenium/patches/is_displayed.rb +16 -0
- data/lib/capybara/selenium/patches/logs.rb +45 -0
- data/lib/capybara/server.rb +19 -3
- data/lib/capybara/server/animation_disabler.rb +2 -2
- data/lib/capybara/server/checker.rb +6 -2
- data/lib/capybara/server/middleware.rb +23 -13
- data/lib/capybara/session.rb +124 -106
- data/lib/capybara/session/config.rb +12 -10
- data/lib/capybara/session/matchers.rb +6 -6
- data/lib/capybara/spec/public/offset.js +6 -0
- data/lib/capybara/spec/public/test.js +94 -5
- data/lib/capybara/spec/session/all_spec.rb +84 -6
- data/lib/capybara/spec/session/ancestor_spec.rb +5 -0
- data/lib/capybara/spec/session/assert_current_path_spec.rb +5 -2
- data/lib/capybara/spec/session/assert_text_spec.rb +9 -5
- data/lib/capybara/spec/session/attach_file_spec.rb +14 -6
- data/lib/capybara/spec/session/check_spec.rb +10 -4
- data/lib/capybara/spec/session/choose_spec.rb +8 -2
- data/lib/capybara/spec/session/click_button_spec.rb +44 -1
- data/lib/capybara/spec/session/click_link_spec.rb +11 -0
- data/lib/capybara/spec/session/evaluate_script_spec.rb +12 -0
- data/lib/capybara/spec/session/fill_in_spec.rb +37 -2
- data/lib/capybara/spec/session/find_spec.rb +60 -6
- data/lib/capybara/spec/session/first_spec.rb +1 -1
- data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +14 -1
- data/lib/capybara/spec/session/frame/within_frame_spec.rb +12 -1
- data/lib/capybara/spec/session/has_ancestor_spec.rb +46 -0
- data/lib/capybara/spec/session/has_button_spec.rb +16 -0
- data/lib/capybara/spec/session/has_css_spec.rb +35 -6
- data/lib/capybara/spec/session/has_current_path_spec.rb +6 -4
- data/lib/capybara/spec/session/has_field_spec.rb +34 -0
- data/lib/capybara/spec/session/has_select_spec.rb +32 -4
- data/lib/capybara/spec/session/has_selector_spec.rb +4 -4
- data/lib/capybara/spec/session/has_sibling_spec.rb +50 -0
- data/lib/capybara/spec/session/has_table_spec.rb +51 -5
- data/lib/capybara/spec/session/has_text_spec.rb +47 -0
- data/lib/capybara/spec/session/matches_style_spec.rb +2 -2
- data/lib/capybara/spec/session/node_spec.rb +574 -16
- data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +2 -2
- data/lib/capybara/spec/session/save_screenshot_spec.rb +4 -4
- data/lib/capybara/spec/session/scroll_spec.rb +1 -1
- data/lib/capybara/spec/session/select_spec.rb +5 -10
- data/lib/capybara/spec/session/selectors_spec.rb +24 -3
- data/lib/capybara/spec/session/uncheck_spec.rb +2 -2
- data/lib/capybara/spec/session/unselect_spec.rb +1 -1
- data/lib/capybara/spec/session/window/window_spec.rb +10 -9
- data/lib/capybara/spec/spec_helper.rb +7 -2
- data/lib/capybara/spec/test_app.rb +26 -21
- data/lib/capybara/spec/views/animated.erb +49 -0
- data/lib/capybara/spec/views/form.erb +25 -4
- data/lib/capybara/spec/views/frame_child.erb +2 -1
- data/lib/capybara/spec/views/frame_one.erb +1 -0
- data/lib/capybara/spec/views/obscured.erb +9 -9
- data/lib/capybara/spec/views/offset.erb +32 -0
- data/lib/capybara/spec/views/react.erb +45 -0
- data/lib/capybara/spec/views/spatial.erb +31 -0
- data/lib/capybara/spec/views/with_animation.erb +29 -1
- data/lib/capybara/spec/views/with_dragula.erb +24 -0
- data/lib/capybara/spec/views/with_html.erb +28 -2
- data/lib/capybara/spec/views/with_js.erb +2 -1
- data/lib/capybara/spec/views/with_jstree.erb +26 -0
- data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
- data/lib/capybara/version.rb +1 -1
- data/lib/capybara/window.rb +10 -10
- data/spec/basic_node_spec.rb +6 -6
- data/spec/capybara_spec.rb +28 -28
- data/spec/dsl_spec.rb +16 -3
- data/spec/filter_set_spec.rb +5 -5
- data/spec/fixtures/selenium_driver_rspec_failure.rb +1 -1
- data/spec/fixtures/selenium_driver_rspec_success.rb +1 -1
- data/spec/minitest_spec.rb +12 -2
- data/spec/minitest_spec_spec.rb +56 -45
- data/spec/rack_test_spec.rb +25 -12
- data/spec/regexp_dissassembler_spec.rb +53 -39
- data/spec/result_spec.rb +50 -54
- data/spec/rspec/features_spec.rb +1 -0
- data/spec/rspec/shared_spec_matchers.rb +78 -62
- data/spec/rspec_spec.rb +5 -5
- data/spec/sauce_spec_chrome.rb +1 -0
- data/spec/selector_spec.rb +26 -16
- data/spec/selenium_spec_chrome.rb +84 -5
- data/spec/selenium_spec_chrome_remote.rb +23 -8
- data/spec/selenium_spec_edge.rb +23 -8
- data/spec/selenium_spec_firefox.rb +16 -21
- data/spec/selenium_spec_firefox_remote.rb +4 -13
- data/spec/selenium_spec_ie.rb +23 -15
- data/spec/selenium_spec_safari.rb +17 -17
- data/spec/server_spec.rb +87 -42
- data/spec/session_spec.rb +11 -4
- data/spec/shared_selenium_node.rb +83 -0
- data/spec/shared_selenium_session.rb +62 -72
- data/spec/spec_helper.rb +43 -5
- metadata +114 -16
@@ -3,6 +3,30 @@
|
|
3
3
|
require 'capybara/selenium/nodes/firefox_node'
|
4
4
|
|
5
5
|
module Capybara::Selenium::Driver::FirefoxDriver
|
6
|
+
def self.extended(driver)
|
7
|
+
driver.extend Capybara::Selenium::Driver::W3CFirefoxDriver if w3c?(driver)
|
8
|
+
bridge = driver.send(:bridge)
|
9
|
+
bridge.extend Capybara::Selenium::IsDisplayed unless bridge.commands(:is_element_displayed)
|
10
|
+
end
|
11
|
+
|
12
|
+
def self.w3c?(driver)
|
13
|
+
(defined?(Selenium::WebDriver::VERSION) && (Selenium::WebDriver::VERSION.to_f >= 4)) ||
|
14
|
+
driver.browser.capabilities.is_a?(::Selenium::WebDriver::Remote::W3C::Capabilities)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
module Capybara::Selenium::Driver::W3CFirefoxDriver
|
19
|
+
class << self
|
20
|
+
def extended(driver)
|
21
|
+
require 'capybara/selenium/patches/pause_duration_fix' if pause_broken?(driver.browser)
|
22
|
+
driver.options[:native_displayed] = false if driver.options[:native_displayed].nil?
|
23
|
+
end
|
24
|
+
|
25
|
+
def pause_broken?(sel_driver)
|
26
|
+
sel_driver.capabilities['moz:geckodriverVersion']&.start_with?('0.22.')
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
6
30
|
def resize_window_to(handle, width, height)
|
7
31
|
within_given_window(handle) do
|
8
32
|
# Don't set the size if already set - See https://github.com/mozilla/geckodriver/issues/643
|
@@ -18,6 +42,15 @@ module Capybara::Selenium::Driver::FirefoxDriver
|
|
18
42
|
# Use instance variable directly so we avoid starting the browser just to reset the session
|
19
43
|
return unless @browser
|
20
44
|
|
45
|
+
if browser_version >= 68
|
46
|
+
begin
|
47
|
+
# Firefox 68 hangs if we try to switch windows while a modal is visible
|
48
|
+
browser.switch_to.alert&.dismiss
|
49
|
+
rescue Selenium::WebDriver::Error::NoSuchAlertError
|
50
|
+
# Swallow
|
51
|
+
end
|
52
|
+
end
|
53
|
+
|
21
54
|
switch_to_window(window_handles.first)
|
22
55
|
window_handles.slice(1..-1).each { |win| close_window(win) }
|
23
56
|
super
|
@@ -28,7 +61,7 @@ module Capybara::Selenium::Driver::FirefoxDriver
|
|
28
61
|
accept_modal :confirm, wait: 0.1 do
|
29
62
|
super
|
30
63
|
end
|
31
|
-
rescue Capybara::ModalNotFound
|
64
|
+
rescue Capybara::ModalNotFound
|
32
65
|
# No modal was opened - page has refreshed - ignore
|
33
66
|
end
|
34
67
|
|
@@ -39,7 +72,7 @@ module Capybara::Selenium::Driver::FirefoxDriver
|
|
39
72
|
# so we have to move to the default_content and iterate back through the frames
|
40
73
|
handles = @frame_handles[current_window_handle]
|
41
74
|
browser.switch_to.default_content
|
42
|
-
handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh) }
|
75
|
+
handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh.native) }
|
43
76
|
end
|
44
77
|
|
45
78
|
private
|
@@ -47,4 +80,10 @@ private
|
|
47
80
|
def build_node(native_node, initial_cache = {})
|
48
81
|
::Capybara::Selenium::FirefoxNode.new(self, native_node, initial_cache)
|
49
82
|
end
|
83
|
+
|
84
|
+
def browser_version
|
85
|
+
browser.capabilities[:browser_version].to_f
|
86
|
+
end
|
50
87
|
end
|
88
|
+
|
89
|
+
Capybara::Selenium::Driver.register_specialization :firefox, Capybara::Selenium::Driver::FirefoxDriver
|
@@ -1,5 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require 'capybara/selenium/nodes/ie_node'
|
4
|
+
|
3
5
|
module Capybara::Selenium::Driver::InternetExplorerDriver
|
4
6
|
def switch_to_frame(frame)
|
5
7
|
return super unless frame == :parent
|
@@ -8,6 +10,17 @@ module Capybara::Selenium::Driver::InternetExplorerDriver
|
|
8
10
|
# so we have to move to the default_content and iterate back through the frames
|
9
11
|
handles = @frame_handles[current_window_handle]
|
10
12
|
browser.switch_to.default_content
|
11
|
-
handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh) }
|
13
|
+
handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh.native) }
|
14
|
+
end
|
15
|
+
|
16
|
+
private
|
17
|
+
|
18
|
+
def build_node(native_node, initial_cache = {})
|
19
|
+
::Capybara::Selenium::IENode.new(self, native_node, initial_cache)
|
12
20
|
end
|
13
21
|
end
|
22
|
+
|
23
|
+
module Capybara::Selenium
|
24
|
+
Driver.register_specialization :ie, Driver::InternetExplorerDriver
|
25
|
+
Driver.register_specialization :internet_explorer, Driver::InternetExplorerDriver
|
26
|
+
end
|
@@ -3,13 +3,22 @@
|
|
3
3
|
require 'capybara/selenium/nodes/safari_node'
|
4
4
|
|
5
5
|
module Capybara::Selenium::Driver::SafariDriver
|
6
|
-
|
6
|
+
def switch_to_frame(frame)
|
7
|
+
return super unless frame == :parent
|
7
8
|
|
8
|
-
|
9
|
-
|
9
|
+
# safaridriver/safari has an issue where switch_to_frame(:parent)
|
10
|
+
# behaves like switch_to_frame(:top)
|
11
|
+
handles = @frame_handles[current_window_handle]
|
12
|
+
browser.switch_to.default_content
|
13
|
+
handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh.native) }
|
10
14
|
end
|
11
15
|
|
12
|
-
|
13
|
-
|
16
|
+
private
|
17
|
+
|
18
|
+
def build_node(native_node, initial_cache = {})
|
19
|
+
::Capybara::Selenium::SafariNode.new(self, native_node, initial_cache)
|
14
20
|
end
|
15
21
|
end
|
22
|
+
|
23
|
+
Capybara::Selenium::Driver.register_specialization(/^(safari|Safari_Technology_Preview)$/,
|
24
|
+
Capybara::Selenium::Driver::SafariDriver)
|
@@ -0,0 +1,34 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Capybara::Selenium::Node
|
4
|
+
module FileInputClickEmulation
|
5
|
+
def click(keys = [], **options)
|
6
|
+
super
|
7
|
+
rescue Selenium::WebDriver::Error::InvalidArgumentError
|
8
|
+
return emulate_click if attaching_file? && visible_file_field?
|
9
|
+
|
10
|
+
raise
|
11
|
+
end
|
12
|
+
|
13
|
+
private
|
14
|
+
|
15
|
+
def visible_file_field?
|
16
|
+
(attrs(:tagName, :type).map { |val| val&.downcase } == %w[input file]) && visible?
|
17
|
+
end
|
18
|
+
|
19
|
+
def attaching_file?
|
20
|
+
caller_locations.any? { |cl| cl.base_label == 'attach_file' }
|
21
|
+
end
|
22
|
+
|
23
|
+
def emulate_click
|
24
|
+
driver.execute_script(<<~JS, self)
|
25
|
+
arguments[0].dispatchEvent(
|
26
|
+
new MouseEvent('click', {
|
27
|
+
view: window,
|
28
|
+
bubbles: true,
|
29
|
+
cancelable: true
|
30
|
+
}));
|
31
|
+
JS
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -3,67 +3,47 @@
|
|
3
3
|
module Capybara
|
4
4
|
module Selenium
|
5
5
|
module Find
|
6
|
-
def find_xpath(selector, uses_visibility: false, styles: nil, **_options)
|
7
|
-
find_by(:xpath, selector, uses_visibility: uses_visibility, texts: [], styles: styles)
|
6
|
+
def find_xpath(selector, uses_visibility: false, styles: nil, position: false, **_options)
|
7
|
+
find_by(:xpath, selector, uses_visibility: uses_visibility, texts: [], styles: styles, position: position)
|
8
8
|
end
|
9
9
|
|
10
|
-
def find_css(selector, uses_visibility: false, texts: [], styles: nil, **_options)
|
11
|
-
find_by(:css, selector, uses_visibility: uses_visibility, texts: texts, styles: styles)
|
10
|
+
def find_css(selector, uses_visibility: false, texts: [], styles: nil, position: false, **_options)
|
11
|
+
find_by(:css, selector, uses_visibility: uses_visibility, texts: texts, styles: styles, position: position)
|
12
12
|
end
|
13
13
|
|
14
14
|
private
|
15
15
|
|
16
|
-
def find_by(format, selector, uses_visibility:, texts:, styles:)
|
16
|
+
def find_by(format, selector, uses_visibility:, texts:, styles:, position:)
|
17
17
|
els = find_context.find_elements(format, selector)
|
18
18
|
hints = []
|
19
19
|
|
20
20
|
if (els.size > 2) && !ENV['DISABLE_CAPYBARA_SELENIUM_OPTIMIZATIONS']
|
21
21
|
els = filter_by_text(els, texts) unless texts.empty?
|
22
|
-
|
23
|
-
hints_js = +''
|
24
|
-
functions = []
|
25
|
-
if uses_visibility && !is_displayed_atom.empty?
|
26
|
-
hints_js << <<~VISIBILITY_JS
|
27
|
-
var vis_func = #{is_displayed_atom};
|
28
|
-
VISIBILITY_JS
|
29
|
-
functions << 'vis_func'
|
30
|
-
end
|
31
|
-
|
32
|
-
if styles.is_a? Hash
|
33
|
-
hints_js << <<~STYLE_JS
|
34
|
-
var style_func = function(el){
|
35
|
-
var el_styles = window.getComputedStyle(el);
|
36
|
-
return #{styles.keys.map(&:to_s)}.reduce(function(res, style){
|
37
|
-
res[style] = el_styles[style];
|
38
|
-
return res;
|
39
|
-
}, {});
|
40
|
-
};
|
41
|
-
STYLE_JS
|
42
|
-
functions << 'style_func'
|
43
|
-
end
|
44
|
-
|
45
|
-
unless functions.empty?
|
46
|
-
hints_js << <<~EACH_JS
|
47
|
-
return arguments[0].map(function(el){
|
48
|
-
return [#{functions.join(',')}].map(function(fn){ return fn.call(null, el) });
|
49
|
-
});
|
50
|
-
EACH_JS
|
51
|
-
|
52
|
-
hints = es_context.execute_script hints_js, els
|
53
|
-
hints.map! do |results|
|
54
|
-
result = {}
|
55
|
-
result[:style] = results.pop if styles.is_a? Hash
|
56
|
-
result[:visible] = results.pop if uses_visibility
|
57
|
-
result
|
58
|
-
end
|
59
|
-
end
|
22
|
+
hints = gather_hints(els, uses_visibility: uses_visibility, styles: styles, position: position)
|
60
23
|
end
|
61
24
|
els.map.with_index { |el, idx| build_node(el, hints[idx] || {}) }
|
62
25
|
end
|
63
26
|
|
27
|
+
def gather_hints(elements, uses_visibility:, styles:, position:)
|
28
|
+
hints_js, functions = build_hints_js(uses_visibility, styles, position)
|
29
|
+
return [] unless functions.any?
|
30
|
+
|
31
|
+
es_context.execute_script(hints_js, elements).map! do |results|
|
32
|
+
hint = {}
|
33
|
+
hint[:style] = results.pop if functions.include?(:style_func)
|
34
|
+
hint[:position] = results.pop if functions.include?(:position_func)
|
35
|
+
hint[:visible] = results.pop if functions.include?(:vis_func)
|
36
|
+
hint
|
37
|
+
end
|
38
|
+
rescue ::Selenium::WebDriver::Error::StaleElementReferenceError,
|
39
|
+
::Capybara::NotSupportedByDriverError
|
40
|
+
# warn 'Unexpected Stale Element Error - skipping optimization'
|
41
|
+
[]
|
42
|
+
end
|
43
|
+
|
64
44
|
def filter_by_text(elements, texts)
|
65
45
|
es_context.execute_script <<~JS, elements, texts
|
66
|
-
var texts = arguments[1]
|
46
|
+
var texts = arguments[1];
|
67
47
|
return arguments[0].filter(function(el){
|
68
48
|
var content = el.textContent.toLowerCase();
|
69
49
|
return texts.every(function(txt){ return content.indexOf(txt.toLowerCase()) != -1 });
|
@@ -71,12 +51,54 @@ module Capybara
|
|
71
51
|
JS
|
72
52
|
end
|
73
53
|
|
54
|
+
def build_hints_js(uses_visibility, styles, position)
|
55
|
+
functions = []
|
56
|
+
hints_js = +''
|
57
|
+
|
58
|
+
if uses_visibility && !is_displayed_atom.empty?
|
59
|
+
hints_js << <<~VISIBILITY_JS
|
60
|
+
var vis_func = #{is_displayed_atom};
|
61
|
+
VISIBILITY_JS
|
62
|
+
functions << :vis_func
|
63
|
+
end
|
64
|
+
|
65
|
+
if position
|
66
|
+
hints_js << <<~POSITION_JS
|
67
|
+
var position_func = function(el){
|
68
|
+
return el.getBoundingClientRect();
|
69
|
+
};
|
70
|
+
POSITION_JS
|
71
|
+
functions << :position_func
|
72
|
+
end
|
73
|
+
|
74
|
+
if styles.is_a? Hash
|
75
|
+
hints_js << <<~STYLE_JS
|
76
|
+
var style_func = function(el){
|
77
|
+
var el_styles = window.getComputedStyle(el);
|
78
|
+
return #{styles.keys.map(&:to_s)}.reduce(function(res, style){
|
79
|
+
res[style] = el_styles[style];
|
80
|
+
return res;
|
81
|
+
}, {});
|
82
|
+
};
|
83
|
+
STYLE_JS
|
84
|
+
functions << :style_func
|
85
|
+
end
|
86
|
+
|
87
|
+
hints_js << <<~EACH_JS
|
88
|
+
return arguments[0].map(function(el){
|
89
|
+
return [#{functions.join(',')}].map(function(fn){ return fn.call(null, el) });
|
90
|
+
});
|
91
|
+
EACH_JS
|
92
|
+
|
93
|
+
[hints_js, functions]
|
94
|
+
end
|
95
|
+
|
74
96
|
def es_context
|
75
97
|
respond_to?(:execute_script) ? self : driver
|
76
98
|
end
|
77
99
|
|
78
100
|
def is_displayed_atom # rubocop:disable Naming/PredicateName
|
79
|
-
@@is_displayed_atom ||= begin
|
101
|
+
@@is_displayed_atom ||= begin # rubocop:disable Style/ClassVars
|
80
102
|
browser.send(:bridge).send(:read_atom, 'isDisplayed')
|
81
103
|
rescue StandardError
|
82
104
|
# If the atom doesn't exist or other error
|
@@ -2,36 +2,121 @@
|
|
2
2
|
|
3
3
|
class Capybara::Selenium::Node
|
4
4
|
module Html5Drag
|
5
|
-
|
5
|
+
# Implement methods to emulate HTML5 drag and drop
|
6
6
|
|
7
|
-
|
7
|
+
def drag_to(element, html5: nil, delay: 0.05, drop_modifiers: [])
|
8
|
+
drop_modifiers = Array(drop_modifiers)
|
8
9
|
|
9
|
-
def html5_drag_to(element)
|
10
10
|
driver.execute_script MOUSEDOWN_TRACKER
|
11
11
|
scroll_if_needed { browser_action.click_and_hold(native).perform }
|
12
|
-
|
13
|
-
|
12
|
+
html5 = !driver.evaluate_script(LEGACY_DRAG_CHECK, self) if html5.nil?
|
13
|
+
if html5
|
14
|
+
perform_html5_drag(element, delay, drop_modifiers)
|
14
15
|
else
|
15
|
-
|
16
|
-
browser_action.release.perform
|
16
|
+
perform_legacy_drag(element, drop_modifiers)
|
17
17
|
end
|
18
18
|
end
|
19
19
|
|
20
|
-
|
21
|
-
|
22
|
-
|
20
|
+
private
|
21
|
+
|
22
|
+
def perform_legacy_drag(element, drop_modifiers)
|
23
|
+
element.scroll_if_needed do
|
24
|
+
# browser_action.move_to(element.native).release.perform
|
25
|
+
keys_down = modifiers_down(browser_action, drop_modifiers)
|
26
|
+
keys_up = modifiers_up(keys_down.move_to(element.native).release, drop_modifiers)
|
27
|
+
keys_up.perform
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
def perform_html5_drag(element, delay, drop_modifiers)
|
32
|
+
driver.evaluate_async_script HTML5_DRAG_DROP_SCRIPT, self, element, delay * 1000, normalize_keys(drop_modifiers)
|
33
|
+
browser_action.release.perform
|
34
|
+
end
|
35
|
+
|
36
|
+
def html5_drop(*args)
|
37
|
+
if args[0].is_a? String
|
38
|
+
input = driver.evaluate_script ATTACH_FILE
|
39
|
+
input.set_file(args)
|
40
|
+
driver.execute_script DROP_FILE, self, input
|
41
|
+
else
|
42
|
+
items = args.each_with_object([]) do |arg, arr|
|
43
|
+
arg.each_with_object(arr) do |(type, data), arr_|
|
44
|
+
arr_ << { type: type, data: data }
|
45
|
+
end
|
46
|
+
end
|
47
|
+
driver.execute_script DROP_STRING, items, self
|
48
|
+
end
|
23
49
|
end
|
24
50
|
|
51
|
+
DROP_STRING = <<~JS
|
52
|
+
var strings = arguments[0],
|
53
|
+
el = arguments[1],
|
54
|
+
dt = new DataTransfer(),
|
55
|
+
opts = { cancelable: true, bubbles: true, dataTransfer: dt };
|
56
|
+
for (var i=0; i < strings.length; i++){
|
57
|
+
if (dt.items) {
|
58
|
+
dt.items.add(strings[i]['data'], strings[i]['type']);
|
59
|
+
} else {
|
60
|
+
dt.setData(strings[i]['type'], strings[i]['data']);
|
61
|
+
}
|
62
|
+
}
|
63
|
+
var dropEvent = new DragEvent('drop', opts);
|
64
|
+
el.dispatchEvent(dropEvent);
|
65
|
+
JS
|
66
|
+
|
67
|
+
DROP_FILE = <<~JS
|
68
|
+
var el = arguments[0],
|
69
|
+
input = arguments[1],
|
70
|
+
files = input.files,
|
71
|
+
dt = new DataTransfer(),
|
72
|
+
opts = { cancelable: true, bubbles: true, dataTransfer: dt };
|
73
|
+
input.parentElement.removeChild(input);
|
74
|
+
if (dt.items){
|
75
|
+
for (var i=0; i<files.length; i++){
|
76
|
+
dt.items.add(files[i]);
|
77
|
+
}
|
78
|
+
} else {
|
79
|
+
Object.defineProperty(dt, "files", {
|
80
|
+
value: files,
|
81
|
+
writable: false
|
82
|
+
});
|
83
|
+
}
|
84
|
+
var dropEvent = new DragEvent('drop', opts);
|
85
|
+
el.dispatchEvent(dropEvent);
|
86
|
+
JS
|
87
|
+
|
88
|
+
ATTACH_FILE = <<~JS
|
89
|
+
(function(){
|
90
|
+
var input = document.createElement('INPUT');
|
91
|
+
input.type = "file";
|
92
|
+
input.id = "_capybara_drop_file";
|
93
|
+
input.multiple = true;
|
94
|
+
document.body.appendChild(input);
|
95
|
+
return input;
|
96
|
+
})()
|
97
|
+
JS
|
98
|
+
|
25
99
|
MOUSEDOWN_TRACKER = <<~JS
|
100
|
+
window.capybara_mousedown_prevented = null;
|
26
101
|
document.addEventListener('mousedown', ev => {
|
27
102
|
window.capybara_mousedown_prevented = ev.defaultPrevented;
|
28
103
|
}, { once: true, passive: true })
|
29
104
|
JS
|
30
105
|
|
31
|
-
|
32
|
-
|
33
|
-
|
106
|
+
LEGACY_DRAG_CHECK = <<~JS
|
107
|
+
(function(el){
|
108
|
+
if ([true, null].indexOf(window.capybara_mousedown_prevented) >= 0){
|
109
|
+
return true;
|
110
|
+
}
|
111
|
+
|
112
|
+
do {
|
113
|
+
if (el.draggable) return false;
|
114
|
+
} while (el = el.parentElement );
|
115
|
+
return true;
|
116
|
+
})(arguments[0])
|
117
|
+
JS
|
34
118
|
|
119
|
+
HTML5_DRAG_DROP_SCRIPT = <<~JS
|
35
120
|
function rectCenter(rect){
|
36
121
|
return new DOMPoint(
|
37
122
|
(rect.left + rect.right)/2,
|
@@ -70,9 +155,61 @@ class Capybara::Selenium::Node
|
|
70
155
|
return new DOMPoint(pt.x,pt.y);
|
71
156
|
}
|
72
157
|
|
158
|
+
function dragEnterTarget() {
|
159
|
+
target.scrollIntoView({behavior: 'instant', block: 'center', inline: 'center'});
|
160
|
+
var targetRect = target.getBoundingClientRect();
|
161
|
+
var sourceCenter = rectCenter(source.getBoundingClientRect());
|
162
|
+
|
163
|
+
for (var i = 0; i < drop_modifier_keys.length; i++) {
|
164
|
+
key = drop_modifier_keys[i];
|
165
|
+
if (key == "control"){
|
166
|
+
key = "ctrl"
|
167
|
+
}
|
168
|
+
opts[key + 'Key'] = true;
|
169
|
+
}
|
170
|
+
|
171
|
+
// fire 2 dragover events to simulate dragging with a direction
|
172
|
+
var entryPoint = pointOnRect(sourceCenter, targetRect)
|
173
|
+
var dragOverOpts = Object.assign({clientX: entryPoint.x, clientY: entryPoint.y}, opts);
|
174
|
+
var dragOverEvent = new DragEvent('dragover', dragOverOpts);
|
175
|
+
target.dispatchEvent(dragOverEvent);
|
176
|
+
window.setTimeout(dragOnTarget, step_delay);
|
177
|
+
}
|
178
|
+
|
179
|
+
function dragOnTarget() {
|
180
|
+
var targetCenter = rectCenter(target.getBoundingClientRect());
|
181
|
+
var dragOverOpts = Object.assign({clientX: targetCenter.x, clientY: targetCenter.y}, opts);
|
182
|
+
var dragOverEvent = new DragEvent('dragover', dragOverOpts);
|
183
|
+
target.dispatchEvent(dragOverEvent);
|
184
|
+
window.setTimeout(dragLeave, step_delay, dragOverEvent.defaultPrevented, dragOverOpts);
|
185
|
+
}
|
186
|
+
|
187
|
+
function dragLeave(drop, dragOverOpts) {
|
188
|
+
var dragLeaveOptions = Object.assign({}, opts, dragOverOpts);
|
189
|
+
var dragLeaveEvent = new DragEvent('dragleave', dragLeaveOptions);
|
190
|
+
target.dispatchEvent(dragLeaveEvent);
|
191
|
+
if (drop) {
|
192
|
+
var dropEvent = new DragEvent('drop', dragLeaveOptions);
|
193
|
+
target.dispatchEvent(dropEvent);
|
194
|
+
}
|
195
|
+
var dragEndEvent = new DragEvent('dragend', dragLeaveOptions);
|
196
|
+
source.dispatchEvent(dragEndEvent);
|
197
|
+
callback.call(true);
|
198
|
+
}
|
199
|
+
|
200
|
+
var source = arguments[0],
|
201
|
+
target = arguments[1],
|
202
|
+
step_delay = arguments[2],
|
203
|
+
drop_modifier_keys = arguments[3],
|
204
|
+
callback = arguments[4];
|
205
|
+
|
73
206
|
var dt = new DataTransfer();
|
74
207
|
var opts = { cancelable: true, bubbles: true, dataTransfer: dt };
|
75
208
|
|
209
|
+
while (source && !source.draggable) {
|
210
|
+
source = source.parentElement;
|
211
|
+
}
|
212
|
+
|
76
213
|
if (source.tagName == 'A'){
|
77
214
|
dt.setData('text/uri-list', source.href);
|
78
215
|
dt.setData('text', source.href);
|
@@ -84,29 +221,8 @@ class Capybara::Selenium::Node
|
|
84
221
|
|
85
222
|
var dragEvent = new DragEvent('dragstart', opts);
|
86
223
|
source.dispatchEvent(dragEvent);
|
87
|
-
|
88
|
-
|
89
|
-
var sourceCenter = rectCenter(source.getBoundingClientRect());
|
90
|
-
|
91
|
-
// fire 2 dragover events to simulate dragging with a direction
|
92
|
-
var entryPoint = pointOnRect(sourceCenter, targetRect)
|
93
|
-
var dragOverOpts = Object.assign({clientX: entryPoint.x, clientY: entryPoint.y}, opts);
|
94
|
-
var dragOverEvent = new DragEvent('dragover', dragOverOpts);
|
95
|
-
target.dispatchEvent(dragOverEvent);
|
96
|
-
|
97
|
-
var targetCenter = rectCenter(targetRect);
|
98
|
-
dragOverOpts = Object.assign({clientX: targetCenter.x, clientY: targetCenter.y}, opts);
|
99
|
-
dragOverEvent = new DragEvent('dragover', dragOverOpts);
|
100
|
-
target.dispatchEvent(dragOverEvent);
|
101
|
-
|
102
|
-
var dragLeaveEvent = new DragEvent('dragleave', opts);
|
103
|
-
target.dispatchEvent(dragLeaveEvent);
|
104
|
-
if (dragOverEvent.defaultPrevented) {
|
105
|
-
var dropEvent = new DragEvent('drop', opts);
|
106
|
-
target.dispatchEvent(dropEvent);
|
107
|
-
}
|
108
|
-
var dragEndEvent = new DragEvent('dragend', opts);
|
109
|
-
source.dispatchEvent(dragEndEvent);
|
224
|
+
|
225
|
+
window.setTimeout(dragEnterTarget, step_delay);
|
110
226
|
JS
|
111
227
|
end
|
112
228
|
end
|