capybara 3.16.1 → 3.33.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (197) hide show
  1. checksums.yaml +4 -4
  2. data/.yardopts +1 -0
  3. data/History.md +321 -0
  4. data/README.md +51 -60
  5. data/lib/capybara.rb +71 -114
  6. data/lib/capybara/config.rb +8 -5
  7. data/lib/capybara/cucumber.rb +1 -1
  8. data/lib/capybara/driver/node.rb +15 -3
  9. data/lib/capybara/dsl.rb +10 -2
  10. data/lib/capybara/helpers.rb +5 -3
  11. data/lib/capybara/minitest.rb +242 -141
  12. data/lib/capybara/minitest/spec.rb +159 -90
  13. data/lib/capybara/node/actions.rb +85 -74
  14. data/lib/capybara/node/base.rb +4 -4
  15. data/lib/capybara/node/document.rb +2 -2
  16. data/lib/capybara/node/document_matchers.rb +3 -3
  17. data/lib/capybara/node/element.rb +216 -117
  18. data/lib/capybara/node/finders.rb +65 -65
  19. data/lib/capybara/node/matchers.rb +228 -126
  20. data/lib/capybara/node/simple.rb +9 -4
  21. data/lib/capybara/queries/ancestor_query.rb +5 -7
  22. data/lib/capybara/queries/base_query.rb +2 -1
  23. data/lib/capybara/queries/current_path_query.rb +1 -1
  24. data/lib/capybara/queries/selector_query.rb +296 -30
  25. data/lib/capybara/queries/sibling_query.rb +5 -4
  26. data/lib/capybara/queries/style_query.rb +2 -2
  27. data/lib/capybara/queries/text_query.rb +13 -1
  28. data/lib/capybara/queries/title_query.rb +1 -1
  29. data/lib/capybara/rack_test/browser.rb +7 -2
  30. data/lib/capybara/rack_test/driver.rb +1 -1
  31. data/lib/capybara/rack_test/form.rb +1 -1
  32. data/lib/capybara/rack_test/node.rb +43 -7
  33. data/lib/capybara/registration_container.rb +44 -0
  34. data/lib/capybara/registrations/drivers.rb +36 -0
  35. data/lib/capybara/registrations/patches/puma_ssl.rb +27 -0
  36. data/lib/capybara/registrations/servers.rb +44 -0
  37. data/lib/capybara/result.rb +36 -8
  38. data/lib/capybara/rspec/matcher_proxies.rb +6 -4
  39. data/lib/capybara/rspec/matchers.rb +100 -63
  40. data/lib/capybara/rspec/matchers/base.rb +23 -10
  41. data/lib/capybara/rspec/matchers/count_sugar.rb +37 -0
  42. data/lib/capybara/rspec/matchers/have_ancestor.rb +28 -0
  43. data/lib/capybara/rspec/matchers/have_current_path.rb +2 -2
  44. data/lib/capybara/rspec/matchers/have_selector.rb +16 -8
  45. data/lib/capybara/rspec/matchers/have_sibling.rb +27 -0
  46. data/lib/capybara/rspec/matchers/have_text.rb +4 -4
  47. data/lib/capybara/rspec/matchers/have_title.rb +2 -2
  48. data/lib/capybara/rspec/matchers/match_selector.rb +3 -3
  49. data/lib/capybara/rspec/matchers/match_style.rb +2 -2
  50. data/lib/capybara/rspec/matchers/spatial_sugar.rb +39 -0
  51. data/lib/capybara/selector.rb +219 -588
  52. data/lib/capybara/selector/builders/css_builder.rb +10 -6
  53. data/lib/capybara/selector/builders/xpath_builder.rb +1 -1
  54. data/lib/capybara/selector/css.rb +4 -2
  55. data/lib/capybara/selector/definition.rb +277 -0
  56. data/lib/capybara/selector/definition/button.rb +52 -0
  57. data/lib/capybara/selector/definition/checkbox.rb +26 -0
  58. data/lib/capybara/selector/definition/css.rb +10 -0
  59. data/lib/capybara/selector/definition/datalist_input.rb +35 -0
  60. data/lib/capybara/selector/definition/datalist_option.rb +25 -0
  61. data/lib/capybara/selector/definition/element.rb +27 -0
  62. data/lib/capybara/selector/definition/field.rb +40 -0
  63. data/lib/capybara/selector/definition/fieldset.rb +14 -0
  64. data/lib/capybara/selector/definition/file_field.rb +13 -0
  65. data/lib/capybara/selector/definition/fillable_field.rb +33 -0
  66. data/lib/capybara/selector/definition/frame.rb +17 -0
  67. data/lib/capybara/selector/definition/id.rb +6 -0
  68. data/lib/capybara/selector/definition/label.rb +62 -0
  69. data/lib/capybara/selector/definition/link.rb +54 -0
  70. data/lib/capybara/selector/definition/link_or_button.rb +16 -0
  71. data/lib/capybara/selector/definition/option.rb +27 -0
  72. data/lib/capybara/selector/definition/radio_button.rb +27 -0
  73. data/lib/capybara/selector/definition/select.rb +81 -0
  74. data/lib/capybara/selector/definition/table.rb +109 -0
  75. data/lib/capybara/selector/definition/table_row.rb +21 -0
  76. data/lib/capybara/selector/definition/xpath.rb +5 -0
  77. data/lib/capybara/selector/filter_set.rb +13 -9
  78. data/lib/capybara/selector/filters/base.rb +11 -2
  79. data/lib/capybara/selector/filters/locator_filter.rb +13 -3
  80. data/lib/capybara/selector/regexp_disassembler.rb +9 -2
  81. data/lib/capybara/selector/selector.rb +43 -448
  82. data/lib/capybara/selenium/atoms/getAttribute.min.js +1 -0
  83. data/lib/capybara/selenium/atoms/isDisplayed.min.js +1 -0
  84. data/lib/capybara/selenium/atoms/src/getAttribute.js +161 -0
  85. data/lib/capybara/selenium/atoms/src/isDisplayed.js +454 -0
  86. data/lib/capybara/selenium/driver.rb +125 -56
  87. data/lib/capybara/selenium/driver_specializations/chrome_driver.rb +73 -17
  88. data/lib/capybara/selenium/driver_specializations/edge_driver.rb +124 -0
  89. data/lib/capybara/selenium/driver_specializations/firefox_driver.rb +41 -2
  90. data/lib/capybara/selenium/driver_specializations/internet_explorer_driver.rb +14 -1
  91. data/lib/capybara/selenium/driver_specializations/safari_driver.rb +14 -5
  92. data/lib/capybara/selenium/extensions/file_input_click_emulation.rb +34 -0
  93. data/lib/capybara/selenium/extensions/find.rb +67 -45
  94. data/lib/capybara/selenium/extensions/html5_drag.rb +152 -36
  95. data/lib/capybara/selenium/extensions/modifier_keys_stack.rb +28 -0
  96. data/lib/capybara/selenium/logger_suppressor.rb +34 -0
  97. data/lib/capybara/selenium/node.rb +227 -56
  98. data/lib/capybara/selenium/nodes/chrome_node.rb +93 -8
  99. data/lib/capybara/selenium/nodes/edge_node.rb +104 -0
  100. data/lib/capybara/selenium/nodes/firefox_node.rb +37 -59
  101. data/lib/capybara/selenium/nodes/ie_node.rb +22 -0
  102. data/lib/capybara/selenium/nodes/safari_node.rb +27 -54
  103. data/lib/capybara/selenium/patches/action_pauser.rb +26 -0
  104. data/lib/capybara/selenium/patches/atoms.rb +18 -0
  105. data/lib/capybara/selenium/patches/is_displayed.rb +16 -0
  106. data/lib/capybara/selenium/patches/logs.rb +45 -0
  107. data/lib/capybara/server.rb +19 -3
  108. data/lib/capybara/server/animation_disabler.rb +2 -2
  109. data/lib/capybara/server/checker.rb +6 -2
  110. data/lib/capybara/server/middleware.rb +23 -13
  111. data/lib/capybara/session.rb +124 -106
  112. data/lib/capybara/session/config.rb +12 -10
  113. data/lib/capybara/session/matchers.rb +6 -6
  114. data/lib/capybara/spec/public/offset.js +6 -0
  115. data/lib/capybara/spec/public/test.js +94 -5
  116. data/lib/capybara/spec/session/all_spec.rb +84 -6
  117. data/lib/capybara/spec/session/ancestor_spec.rb +5 -0
  118. data/lib/capybara/spec/session/assert_current_path_spec.rb +5 -2
  119. data/lib/capybara/spec/session/assert_text_spec.rb +9 -5
  120. data/lib/capybara/spec/session/attach_file_spec.rb +14 -6
  121. data/lib/capybara/spec/session/check_spec.rb +10 -4
  122. data/lib/capybara/spec/session/choose_spec.rb +8 -2
  123. data/lib/capybara/spec/session/click_button_spec.rb +44 -1
  124. data/lib/capybara/spec/session/click_link_spec.rb +11 -0
  125. data/lib/capybara/spec/session/evaluate_script_spec.rb +12 -0
  126. data/lib/capybara/spec/session/fill_in_spec.rb +37 -2
  127. data/lib/capybara/spec/session/find_spec.rb +60 -6
  128. data/lib/capybara/spec/session/first_spec.rb +1 -1
  129. data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +14 -1
  130. data/lib/capybara/spec/session/frame/within_frame_spec.rb +12 -1
  131. data/lib/capybara/spec/session/has_ancestor_spec.rb +46 -0
  132. data/lib/capybara/spec/session/has_button_spec.rb +16 -0
  133. data/lib/capybara/spec/session/has_css_spec.rb +35 -6
  134. data/lib/capybara/spec/session/has_current_path_spec.rb +6 -4
  135. data/lib/capybara/spec/session/has_field_spec.rb +34 -0
  136. data/lib/capybara/spec/session/has_select_spec.rb +32 -4
  137. data/lib/capybara/spec/session/has_selector_spec.rb +4 -4
  138. data/lib/capybara/spec/session/has_sibling_spec.rb +50 -0
  139. data/lib/capybara/spec/session/has_table_spec.rb +51 -5
  140. data/lib/capybara/spec/session/has_text_spec.rb +47 -0
  141. data/lib/capybara/spec/session/matches_style_spec.rb +2 -2
  142. data/lib/capybara/spec/session/node_spec.rb +574 -16
  143. data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +2 -2
  144. data/lib/capybara/spec/session/save_screenshot_spec.rb +4 -4
  145. data/lib/capybara/spec/session/scroll_spec.rb +1 -1
  146. data/lib/capybara/spec/session/select_spec.rb +5 -10
  147. data/lib/capybara/spec/session/selectors_spec.rb +24 -3
  148. data/lib/capybara/spec/session/uncheck_spec.rb +2 -2
  149. data/lib/capybara/spec/session/unselect_spec.rb +1 -1
  150. data/lib/capybara/spec/session/window/window_spec.rb +10 -9
  151. data/lib/capybara/spec/spec_helper.rb +7 -2
  152. data/lib/capybara/spec/test_app.rb +26 -21
  153. data/lib/capybara/spec/views/animated.erb +49 -0
  154. data/lib/capybara/spec/views/form.erb +25 -4
  155. data/lib/capybara/spec/views/frame_child.erb +2 -1
  156. data/lib/capybara/spec/views/frame_one.erb +1 -0
  157. data/lib/capybara/spec/views/obscured.erb +9 -9
  158. data/lib/capybara/spec/views/offset.erb +32 -0
  159. data/lib/capybara/spec/views/react.erb +45 -0
  160. data/lib/capybara/spec/views/spatial.erb +31 -0
  161. data/lib/capybara/spec/views/with_animation.erb +29 -1
  162. data/lib/capybara/spec/views/with_dragula.erb +24 -0
  163. data/lib/capybara/spec/views/with_html.erb +28 -2
  164. data/lib/capybara/spec/views/with_js.erb +2 -1
  165. data/lib/capybara/spec/views/with_jstree.erb +26 -0
  166. data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
  167. data/lib/capybara/version.rb +1 -1
  168. data/lib/capybara/window.rb +10 -10
  169. data/spec/basic_node_spec.rb +6 -6
  170. data/spec/capybara_spec.rb +28 -28
  171. data/spec/dsl_spec.rb +16 -3
  172. data/spec/filter_set_spec.rb +5 -5
  173. data/spec/fixtures/selenium_driver_rspec_failure.rb +1 -1
  174. data/spec/fixtures/selenium_driver_rspec_success.rb +1 -1
  175. data/spec/minitest_spec.rb +12 -2
  176. data/spec/minitest_spec_spec.rb +56 -45
  177. data/spec/rack_test_spec.rb +25 -12
  178. data/spec/regexp_dissassembler_spec.rb +53 -39
  179. data/spec/result_spec.rb +50 -54
  180. data/spec/rspec/features_spec.rb +1 -0
  181. data/spec/rspec/shared_spec_matchers.rb +78 -62
  182. data/spec/rspec_spec.rb +5 -5
  183. data/spec/sauce_spec_chrome.rb +1 -0
  184. data/spec/selector_spec.rb +26 -16
  185. data/spec/selenium_spec_chrome.rb +84 -5
  186. data/spec/selenium_spec_chrome_remote.rb +23 -8
  187. data/spec/selenium_spec_edge.rb +23 -8
  188. data/spec/selenium_spec_firefox.rb +16 -21
  189. data/spec/selenium_spec_firefox_remote.rb +4 -13
  190. data/spec/selenium_spec_ie.rb +23 -15
  191. data/spec/selenium_spec_safari.rb +17 -17
  192. data/spec/server_spec.rb +87 -42
  193. data/spec/session_spec.rb +11 -4
  194. data/spec/shared_selenium_node.rb +83 -0
  195. data/spec/shared_selenium_session.rb +62 -72
  196. data/spec/spec_helper.rb +43 -5
  197. 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 # rubocop:disable Lint/HandleExceptions
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
- private # rubocop:disable Layout/IndentationWidth
6
+ def switch_to_frame(frame)
7
+ return super unless frame == :parent
7
8
 
8
- def build_node(native_node, initial_cache = {})
9
- ::Capybara::Selenium::SafariNode.new(self, native_node, initial_cache)
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
- def bridge
13
- browser.send(:bridge)
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
- # Implement methods to emulate HTML5 drag and drop
5
+ # Implement methods to emulate HTML5 drag and drop
6
6
 
7
- private # rubocop:disable Layout/IndentationWidth
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
- if driver.evaluate_script('window.capybara_mousedown_prevented')
13
- element.scroll_if_needed { browser_action.move_to(element.native).release.perform }
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
- driver.execute_script HTML5_DRAG_DROP_SCRIPT, self, element
16
- browser_action.release.perform
16
+ perform_legacy_drag(element, drop_modifiers)
17
17
  end
18
18
  end
19
19
 
20
- def html5_draggable?
21
- # Workaround https://github.com/SeleniumHQ/selenium/issues/6396
22
- native.property('draggable')
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
- HTML5_DRAG_DROP_SCRIPT = <<~JS
32
- var source = arguments[0];
33
- var target = arguments[1];
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
- target.scrollIntoView({behavior: 'instant', block: 'center', inline: 'center'});
88
- var targetRect = target.getBoundingClientRect();
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