capybara 3.8.1 → 3.33.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (242) hide show
  1. checksums.yaml +4 -4
  2. data/.yardopts +1 -0
  3. data/History.md +465 -0
  4. data/License.txt +1 -1
  5. data/README.md +58 -57
  6. data/lib/capybara/config.rb +10 -4
  7. data/lib/capybara/cucumber.rb +1 -1
  8. data/lib/capybara/driver/base.rb +2 -2
  9. data/lib/capybara/driver/node.rb +26 -5
  10. data/lib/capybara/dsl.rb +12 -4
  11. data/lib/capybara/helpers.rb +8 -4
  12. data/lib/capybara/minitest/spec.rb +162 -85
  13. data/lib/capybara/minitest.rb +248 -148
  14. data/lib/capybara/node/actions.rb +149 -96
  15. data/lib/capybara/node/base.rb +27 -10
  16. data/lib/capybara/node/document.rb +12 -0
  17. data/lib/capybara/node/document_matchers.rb +9 -5
  18. data/lib/capybara/node/element.rb +254 -109
  19. data/lib/capybara/node/finders.rb +83 -76
  20. data/lib/capybara/node/matchers.rb +279 -141
  21. data/lib/capybara/node/simple.rb +25 -6
  22. data/lib/capybara/queries/ancestor_query.rb +5 -7
  23. data/lib/capybara/queries/base_query.rb +11 -5
  24. data/lib/capybara/queries/current_path_query.rb +3 -3
  25. data/lib/capybara/queries/match_query.rb +1 -0
  26. data/lib/capybara/queries/selector_query.rb +467 -103
  27. data/lib/capybara/queries/sibling_query.rb +5 -4
  28. data/lib/capybara/queries/style_query.rb +6 -2
  29. data/lib/capybara/queries/text_query.rb +17 -3
  30. data/lib/capybara/queries/title_query.rb +2 -2
  31. data/lib/capybara/rack_test/browser.rb +22 -15
  32. data/lib/capybara/rack_test/driver.rb +10 -1
  33. data/lib/capybara/rack_test/errors.rb +6 -0
  34. data/lib/capybara/rack_test/form.rb +33 -28
  35. data/lib/capybara/rack_test/node.rb +74 -6
  36. data/lib/capybara/registration_container.rb +44 -0
  37. data/lib/capybara/registrations/drivers.rb +36 -0
  38. data/lib/capybara/registrations/patches/puma_ssl.rb +27 -0
  39. data/lib/capybara/registrations/servers.rb +44 -0
  40. data/lib/capybara/result.rb +55 -23
  41. data/lib/capybara/rspec/features.rb +4 -4
  42. data/lib/capybara/rspec/matcher_proxies.rb +36 -15
  43. data/lib/capybara/rspec/matchers/base.rb +111 -0
  44. data/lib/capybara/rspec/matchers/become_closed.rb +33 -0
  45. data/lib/capybara/rspec/matchers/compound.rb +88 -0
  46. data/lib/capybara/rspec/matchers/count_sugar.rb +37 -0
  47. data/lib/capybara/rspec/matchers/have_ancestor.rb +28 -0
  48. data/lib/capybara/rspec/matchers/have_current_path.rb +29 -0
  49. data/lib/capybara/rspec/matchers/have_selector.rb +77 -0
  50. data/lib/capybara/rspec/matchers/have_sibling.rb +27 -0
  51. data/lib/capybara/rspec/matchers/have_text.rb +33 -0
  52. data/lib/capybara/rspec/matchers/have_title.rb +29 -0
  53. data/lib/capybara/rspec/matchers/match_selector.rb +27 -0
  54. data/lib/capybara/rspec/matchers/match_style.rb +38 -0
  55. data/lib/capybara/rspec/matchers/spatial_sugar.rb +39 -0
  56. data/lib/capybara/rspec/matchers.rb +117 -311
  57. data/lib/capybara/selector/builders/css_builder.rb +84 -0
  58. data/lib/capybara/selector/builders/xpath_builder.rb +69 -0
  59. data/lib/capybara/selector/css.rb +17 -15
  60. data/lib/capybara/selector/definition/button.rb +52 -0
  61. data/lib/capybara/selector/definition/checkbox.rb +26 -0
  62. data/lib/capybara/selector/definition/css.rb +10 -0
  63. data/lib/capybara/selector/definition/datalist_input.rb +35 -0
  64. data/lib/capybara/selector/definition/datalist_option.rb +25 -0
  65. data/lib/capybara/selector/definition/element.rb +27 -0
  66. data/lib/capybara/selector/definition/field.rb +40 -0
  67. data/lib/capybara/selector/definition/fieldset.rb +14 -0
  68. data/lib/capybara/selector/definition/file_field.rb +13 -0
  69. data/lib/capybara/selector/definition/fillable_field.rb +33 -0
  70. data/lib/capybara/selector/definition/frame.rb +17 -0
  71. data/lib/capybara/selector/definition/id.rb +6 -0
  72. data/lib/capybara/selector/definition/label.rb +62 -0
  73. data/lib/capybara/selector/definition/link.rb +54 -0
  74. data/lib/capybara/selector/definition/link_or_button.rb +16 -0
  75. data/lib/capybara/selector/definition/option.rb +27 -0
  76. data/lib/capybara/selector/definition/radio_button.rb +27 -0
  77. data/lib/capybara/selector/definition/select.rb +81 -0
  78. data/lib/capybara/selector/definition/table.rb +109 -0
  79. data/lib/capybara/selector/definition/table_row.rb +21 -0
  80. data/lib/capybara/selector/definition/xpath.rb +5 -0
  81. data/lib/capybara/selector/definition.rb +277 -0
  82. data/lib/capybara/selector/filter.rb +1 -0
  83. data/lib/capybara/selector/filter_set.rb +26 -19
  84. data/lib/capybara/selector/filters/base.rb +24 -5
  85. data/lib/capybara/selector/filters/expression_filter.rb +3 -3
  86. data/lib/capybara/selector/filters/locator_filter.rb +29 -0
  87. data/lib/capybara/selector/filters/node_filter.rb +16 -2
  88. data/lib/capybara/selector/regexp_disassembler.rb +214 -0
  89. data/lib/capybara/selector/selector.rb +73 -367
  90. data/lib/capybara/selector/xpath_extensions.rb +17 -0
  91. data/lib/capybara/selector.rb +221 -480
  92. data/lib/capybara/selenium/atoms/getAttribute.min.js +1 -0
  93. data/lib/capybara/selenium/atoms/isDisplayed.min.js +1 -0
  94. data/lib/capybara/selenium/atoms/src/getAttribute.js +161 -0
  95. data/lib/capybara/selenium/atoms/src/isDisplayed.js +454 -0
  96. data/lib/capybara/selenium/driver.rb +203 -86
  97. data/lib/capybara/selenium/driver_specializations/chrome_driver.rb +88 -14
  98. data/lib/capybara/selenium/driver_specializations/edge_driver.rb +124 -0
  99. data/lib/capybara/selenium/driver_specializations/firefox_driver.rb +89 -0
  100. data/lib/capybara/selenium/driver_specializations/internet_explorer_driver.rb +26 -0
  101. data/lib/capybara/selenium/driver_specializations/safari_driver.rb +24 -0
  102. data/lib/capybara/selenium/extensions/file_input_click_emulation.rb +34 -0
  103. data/lib/capybara/selenium/extensions/find.rb +110 -0
  104. data/lib/capybara/selenium/extensions/html5_drag.rb +191 -22
  105. data/lib/capybara/selenium/extensions/modifier_keys_stack.rb +28 -0
  106. data/lib/capybara/selenium/extensions/scroll.rb +78 -0
  107. data/lib/capybara/selenium/logger_suppressor.rb +34 -0
  108. data/lib/capybara/selenium/node.rb +298 -93
  109. data/lib/capybara/selenium/nodes/chrome_node.rb +100 -8
  110. data/lib/capybara/selenium/nodes/edge_node.rb +104 -0
  111. data/lib/capybara/selenium/nodes/firefox_node.rb +131 -0
  112. data/lib/capybara/selenium/nodes/ie_node.rb +22 -0
  113. data/lib/capybara/selenium/nodes/safari_node.rb +118 -0
  114. data/lib/capybara/selenium/patches/action_pauser.rb +26 -0
  115. data/lib/capybara/selenium/patches/atoms.rb +18 -0
  116. data/lib/capybara/selenium/patches/is_displayed.rb +16 -0
  117. data/lib/capybara/selenium/patches/logs.rb +45 -0
  118. data/lib/capybara/selenium/patches/pause_duration_fix.rb +1 -3
  119. data/lib/capybara/selenium/patches/persistent_client.rb +20 -0
  120. data/lib/capybara/server/animation_disabler.rb +4 -3
  121. data/lib/capybara/server/checker.rb +6 -2
  122. data/lib/capybara/server/middleware.rb +23 -13
  123. data/lib/capybara/server.rb +30 -7
  124. data/lib/capybara/session/config.rb +14 -10
  125. data/lib/capybara/session/matchers.rb +11 -7
  126. data/lib/capybara/session.rb +152 -111
  127. data/lib/capybara/spec/public/offset.js +6 -0
  128. data/lib/capybara/spec/public/test.js +101 -10
  129. data/lib/capybara/spec/session/all_spec.rb +96 -6
  130. data/lib/capybara/spec/session/ancestor_spec.rb +5 -0
  131. data/lib/capybara/spec/session/assert_all_of_selectors_spec.rb +29 -0
  132. data/lib/capybara/spec/session/assert_current_path_spec.rb +5 -2
  133. data/lib/capybara/spec/session/assert_selector_spec.rb +0 -10
  134. data/lib/capybara/spec/session/assert_style_spec.rb +4 -4
  135. data/lib/capybara/spec/session/assert_text_spec.rb +9 -5
  136. data/lib/capybara/spec/session/attach_file_spec.rb +63 -36
  137. data/lib/capybara/spec/session/check_spec.rb +10 -4
  138. data/lib/capybara/spec/session/choose_spec.rb +8 -2
  139. data/lib/capybara/spec/session/click_button_spec.rb +117 -61
  140. data/lib/capybara/spec/session/click_link_or_button_spec.rb +16 -0
  141. data/lib/capybara/spec/session/click_link_spec.rb +17 -6
  142. data/lib/capybara/spec/session/element/matches_selector_spec.rb +40 -39
  143. data/lib/capybara/spec/session/evaluate_script_spec.rb +13 -0
  144. data/lib/capybara/spec/session/execute_script_spec.rb +1 -0
  145. data/lib/capybara/spec/session/fill_in_spec.rb +47 -6
  146. data/lib/capybara/spec/session/find_field_spec.rb +1 -1
  147. data/lib/capybara/spec/session/find_spec.rb +74 -4
  148. data/lib/capybara/spec/session/first_spec.rb +1 -1
  149. data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +13 -1
  150. data/lib/capybara/spec/session/frame/within_frame_spec.rb +12 -1
  151. data/lib/capybara/spec/session/has_all_selectors_spec.rb +1 -1
  152. data/lib/capybara/spec/session/has_ancestor_spec.rb +46 -0
  153. data/lib/capybara/spec/session/has_any_selectors_spec.rb +25 -0
  154. data/lib/capybara/spec/session/has_button_spec.rb +16 -0
  155. data/lib/capybara/spec/session/has_css_spec.rb +122 -12
  156. data/lib/capybara/spec/session/has_current_path_spec.rb +6 -4
  157. data/lib/capybara/spec/session/has_field_spec.rb +55 -0
  158. data/lib/capybara/spec/session/has_select_spec.rb +34 -6
  159. data/lib/capybara/spec/session/has_selector_spec.rb +11 -4
  160. data/lib/capybara/spec/session/has_sibling_spec.rb +50 -0
  161. data/lib/capybara/spec/session/has_table_spec.rb +166 -0
  162. data/lib/capybara/spec/session/has_text_spec.rb +48 -1
  163. data/lib/capybara/spec/session/has_xpath_spec.rb +17 -0
  164. data/lib/capybara/spec/session/html_spec.rb +7 -0
  165. data/lib/capybara/spec/session/matches_style_spec.rb +35 -0
  166. data/lib/capybara/spec/session/node_spec.rb +643 -18
  167. data/lib/capybara/spec/session/node_wrapper_spec.rb +1 -1
  168. data/lib/capybara/spec/session/refresh_spec.rb +4 -0
  169. data/lib/capybara/spec/session/reset_session_spec.rb +23 -8
  170. data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +2 -2
  171. data/lib/capybara/spec/session/save_screenshot_spec.rb +4 -4
  172. data/lib/capybara/spec/session/scroll_spec.rb +117 -0
  173. data/lib/capybara/spec/session/select_spec.rb +10 -10
  174. data/lib/capybara/spec/session/selectors_spec.rb +36 -5
  175. data/lib/capybara/spec/session/uncheck_spec.rb +2 -2
  176. data/lib/capybara/spec/session/unselect_spec.rb +1 -1
  177. data/lib/capybara/spec/session/window/become_closed_spec.rb +20 -17
  178. data/lib/capybara/spec/session/window/switch_to_window_spec.rb +4 -0
  179. data/lib/capybara/spec/session/window/window_opened_by_spec.rb +4 -0
  180. data/lib/capybara/spec/session/window/window_spec.rb +59 -58
  181. data/lib/capybara/spec/session/window/windows_spec.rb +4 -0
  182. data/lib/capybara/spec/session/within_spec.rb +23 -0
  183. data/lib/capybara/spec/spec_helper.rb +16 -6
  184. data/lib/capybara/spec/test_app.rb +28 -23
  185. data/lib/capybara/spec/views/animated.erb +49 -0
  186. data/lib/capybara/spec/views/form.erb +48 -7
  187. data/lib/capybara/spec/views/frame_child.erb +3 -2
  188. data/lib/capybara/spec/views/frame_one.erb +1 -0
  189. data/lib/capybara/spec/views/obscured.erb +47 -0
  190. data/lib/capybara/spec/views/offset.erb +32 -0
  191. data/lib/capybara/spec/views/react.erb +45 -0
  192. data/lib/capybara/spec/views/scroll.erb +20 -0
  193. data/lib/capybara/spec/views/spatial.erb +31 -0
  194. data/lib/capybara/spec/views/tables.erb +67 -0
  195. data/lib/capybara/spec/views/with_animation.erb +29 -1
  196. data/lib/capybara/spec/views/with_dragula.erb +24 -0
  197. data/lib/capybara/spec/views/with_hover.erb +1 -0
  198. data/lib/capybara/spec/views/with_hover1.erb +10 -0
  199. data/lib/capybara/spec/views/with_html.erb +32 -6
  200. data/lib/capybara/spec/views/with_js.erb +3 -1
  201. data/lib/capybara/spec/views/with_jstree.erb +26 -0
  202. data/lib/capybara/spec/views/with_scope_other.erb +6 -0
  203. data/lib/capybara/spec/views/with_sortable_js.erb +21 -0
  204. data/lib/capybara/version.rb +1 -1
  205. data/lib/capybara/window.rb +11 -11
  206. data/lib/capybara.rb +118 -111
  207. data/spec/basic_node_spec.rb +14 -3
  208. data/spec/capybara_spec.rb +29 -29
  209. data/spec/css_builder_spec.rb +101 -0
  210. data/spec/dsl_spec.rb +46 -21
  211. data/spec/filter_set_spec.rb +5 -5
  212. data/spec/fixtures/selenium_driver_rspec_failure.rb +1 -1
  213. data/spec/fixtures/selenium_driver_rspec_success.rb +1 -1
  214. data/spec/minitest_spec.rb +18 -4
  215. data/spec/minitest_spec_spec.rb +59 -44
  216. data/spec/rack_test_spec.rb +117 -89
  217. data/spec/regexp_dissassembler_spec.rb +250 -0
  218. data/spec/result_spec.rb +51 -49
  219. data/spec/rspec/features_spec.rb +3 -0
  220. data/spec/rspec/shared_spec_matchers.rb +112 -97
  221. data/spec/rspec_spec.rb +35 -17
  222. data/spec/sauce_spec_chrome.rb +43 -0
  223. data/spec/selector_spec.rb +244 -28
  224. data/spec/selenium_spec_chrome.rb +125 -54
  225. data/spec/selenium_spec_chrome_remote.rb +26 -12
  226. data/spec/selenium_spec_edge.rb +23 -8
  227. data/spec/selenium_spec_firefox.rb +208 -0
  228. data/spec/selenium_spec_firefox_remote.rb +15 -18
  229. data/spec/selenium_spec_ie.rb +82 -13
  230. data/spec/selenium_spec_safari.rb +148 -0
  231. data/spec/server_spec.rb +118 -77
  232. data/spec/session_spec.rb +19 -3
  233. data/spec/shared_selenium_node.rb +83 -0
  234. data/spec/shared_selenium_session.rb +110 -65
  235. data/spec/spec_helper.rb +57 -9
  236. data/spec/xpath_builder_spec.rb +93 -0
  237. metadata +257 -17
  238. data/lib/capybara/rspec/compound.rb +0 -94
  239. data/lib/capybara/selenium/driver_specializations/marionette_driver.rb +0 -49
  240. data/lib/capybara/selenium/nodes/marionette_node.rb +0 -121
  241. data/lib/capybara/spec/session/has_style_spec.rb +0 -25
  242. data/spec/selenium_spec_marionette.rb +0 -172
@@ -0,0 +1,124 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'capybara/selenium/nodes/edge_node'
4
+
5
+ module Capybara::Selenium::Driver::EdgeDriver
6
+ def self.extended(base)
7
+ bridge = base.send(:bridge)
8
+ bridge.extend Capybara::Selenium::IsDisplayed unless bridge.commands(:is_element_displayed)
9
+ base.options[:native_displayed] = false if base.options[:native_displayed].nil?
10
+ end
11
+
12
+ def fullscreen_window(handle)
13
+ return super if edgedriver_version < 75
14
+
15
+ within_given_window(handle) do
16
+ super
17
+ rescue NoMethodError => e
18
+ raise unless e.message.match?(/full_screen_window/)
19
+
20
+ result = bridge.http.call(:post, "session/#{bridge.session_id}/window/fullscreen", {})
21
+ result['value']
22
+ end
23
+ end
24
+
25
+ def resize_window_to(handle, width, height)
26
+ super
27
+ rescue Selenium::WebDriver::Error::UnknownError => e
28
+ raise unless e.message.match?(/failed to change window state/)
29
+
30
+ # Chromedriver doesn't wait long enough for state to change when coming out of fullscreen
31
+ # and raises unnecessary error. Wait a bit and try again.
32
+ sleep 0.25
33
+ super
34
+ end
35
+
36
+ def reset!
37
+ return super if edgedriver_version < 75
38
+ # Use instance variable directly so we avoid starting the browser just to reset the session
39
+ return unless @browser
40
+
41
+ switch_to_window(window_handles.first)
42
+ window_handles.slice(1..-1).each { |win| close_window(win) }
43
+
44
+ timer = Capybara::Helpers.timer(expire_in: 10)
45
+ begin
46
+ clear_storage unless uniform_storage_clear?
47
+ @browser.navigate.to('about:blank')
48
+ wait_for_empty_page(timer)
49
+ rescue *unhandled_alert_errors
50
+ accept_unhandled_reset_alert
51
+ retry
52
+ end
53
+
54
+ execute_cdp('Storage.clearDataForOrigin', origin: '*', storageTypes: storage_types_to_clear)
55
+ end
56
+
57
+ def download_path=(path)
58
+ if @browser.respond_to?(:download_path=)
59
+ @browser.download_path = path
60
+ else
61
+ # Not yet implemented in seleniun-webdriver for edge so do it ourselves
62
+ execute_cdp('Page.setDownloadBehavior', behavior: 'allow', downloadPath: path)
63
+ end
64
+ end
65
+
66
+ private
67
+
68
+ def storage_types_to_clear
69
+ types = ['cookies']
70
+ types << 'local_storage' if clear_all_storage?
71
+ types.join(',')
72
+ end
73
+
74
+ def clear_all_storage?
75
+ storage_clears.none? false
76
+ end
77
+
78
+ def uniform_storage_clear?
79
+ storage_clears.uniq { |s| s == false }.length <= 1
80
+ end
81
+
82
+ def storage_clears
83
+ options.values_at(:clear_session_storage, :clear_local_storage)
84
+ end
85
+
86
+ def clear_storage
87
+ # Edgedriver crashes if attempt to clear storage on about:blank
88
+ url = current_url
89
+ super unless url.nil? || url.start_with?('about:')
90
+ end
91
+
92
+ def delete_all_cookies
93
+ return super if edgedriver_version < 75
94
+
95
+ execute_cdp('Network.clearBrowserCookies')
96
+ rescue *cdp_unsupported_errors
97
+ # If the CDP clear isn't supported do original limited clear
98
+ super
99
+ end
100
+
101
+ def cdp_unsupported_errors
102
+ @cdp_unsupported_errors ||= [Selenium::WebDriver::Error::WebDriverError]
103
+ end
104
+
105
+ def execute_cdp(cmd, params = {})
106
+ args = { cmd: cmd, params: params }
107
+ result = bridge.http.call(:post, "session/#{bridge.session_id}/goog/cdp/execute", args)
108
+ result['value']
109
+ end
110
+
111
+ def build_node(native_node, initial_cache = {})
112
+ ::Capybara::Selenium::EdgeNode.new(self, native_node, initial_cache)
113
+ end
114
+
115
+ def edgedriver_version
116
+ @edgedriver_version ||= begin
117
+ caps = browser.capabilities
118
+ caps['chrome']&.fetch('chromedriverVersion', nil).to_f
119
+ end
120
+ end
121
+ end
122
+
123
+ Capybara::Selenium::Driver.register_specialization :edge, Capybara::Selenium::Driver::EdgeDriver
124
+ Capybara::Selenium::Driver.register_specialization :edge_chrome, Capybara::Selenium::Driver::EdgeDriver
@@ -0,0 +1,89 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'capybara/selenium/nodes/firefox_node'
4
+
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
+
30
+ def resize_window_to(handle, width, height)
31
+ within_given_window(handle) do
32
+ # Don't set the size if already set - See https://github.com/mozilla/geckodriver/issues/643
33
+ if window_size(handle) == [width, height]
34
+ {}
35
+ else
36
+ super
37
+ end
38
+ end
39
+ end
40
+
41
+ def reset!
42
+ # Use instance variable directly so we avoid starting the browser just to reset the session
43
+ return unless @browser
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
+
54
+ switch_to_window(window_handles.first)
55
+ window_handles.slice(1..-1).each { |win| close_window(win) }
56
+ super
57
+ end
58
+
59
+ def refresh
60
+ # Accept any "will repost content" confirmation that occurs
61
+ accept_modal :confirm, wait: 0.1 do
62
+ super
63
+ end
64
+ rescue Capybara::ModalNotFound
65
+ # No modal was opened - page has refreshed - ignore
66
+ end
67
+
68
+ def switch_to_frame(frame)
69
+ return super unless frame == :parent
70
+
71
+ # geckodriver/firefox has an issue if the current frame is removed from within it
72
+ # so we have to move to the default_content and iterate back through the frames
73
+ handles = @frame_handles[current_window_handle]
74
+ browser.switch_to.default_content
75
+ handles.tap(&:pop).each { |fh| browser.switch_to.frame(fh.native) }
76
+ end
77
+
78
+ private
79
+
80
+ def build_node(native_node, initial_cache = {})
81
+ ::Capybara::Selenium::FirefoxNode.new(self, native_node, initial_cache)
82
+ end
83
+
84
+ def browser_version
85
+ browser.capabilities[:browser_version].to_f
86
+ end
87
+ end
88
+
89
+ Capybara::Selenium::Driver.register_specialization :firefox, Capybara::Selenium::Driver::FirefoxDriver
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'capybara/selenium/nodes/ie_node'
4
+
5
+ module Capybara::Selenium::Driver::InternetExplorerDriver
6
+ def switch_to_frame(frame)
7
+ return super unless frame == :parent
8
+
9
+ # iedriverserver has an issue if the current frame is removed from within it
10
+ # so we have to move to the default_content and iterate back through the frames
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) }
14
+ end
15
+
16
+ private
17
+
18
+ def build_node(native_node, initial_cache = {})
19
+ ::Capybara::Selenium::IENode.new(self, native_node, initial_cache)
20
+ end
21
+ end
22
+
23
+ module Capybara::Selenium
24
+ Driver.register_specialization :ie, Driver::InternetExplorerDriver
25
+ Driver.register_specialization :internet_explorer, Driver::InternetExplorerDriver
26
+ end
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'capybara/selenium/nodes/safari_node'
4
+
5
+ module Capybara::Selenium::Driver::SafariDriver
6
+ def switch_to_frame(frame)
7
+ return super unless frame == :parent
8
+
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) }
14
+ end
15
+
16
+ private
17
+
18
+ def build_node(native_node, initial_cache = {})
19
+ ::Capybara::Selenium::SafariNode.new(self, native_node, initial_cache)
20
+ end
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
@@ -0,0 +1,110 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Capybara
4
+ module Selenium
5
+ module Find
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
+ end
9
+
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
+ end
13
+
14
+ private
15
+
16
+ def find_by(format, selector, uses_visibility:, texts:, styles:, position:)
17
+ els = find_context.find_elements(format, selector)
18
+ hints = []
19
+
20
+ if (els.size > 2) && !ENV['DISABLE_CAPYBARA_SELENIUM_OPTIMIZATIONS']
21
+ els = filter_by_text(els, texts) unless texts.empty?
22
+ hints = gather_hints(els, uses_visibility: uses_visibility, styles: styles, position: position)
23
+ end
24
+ els.map.with_index { |el, idx| build_node(el, hints[idx] || {}) }
25
+ end
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
+
44
+ def filter_by_text(elements, texts)
45
+ es_context.execute_script <<~JS, elements, texts
46
+ var texts = arguments[1];
47
+ return arguments[0].filter(function(el){
48
+ var content = el.textContent.toLowerCase();
49
+ return texts.every(function(txt){ return content.indexOf(txt.toLowerCase()) != -1 });
50
+ })
51
+ JS
52
+ end
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
+
96
+ def es_context
97
+ respond_to?(:execute_script) ? self : driver
98
+ end
99
+
100
+ def is_displayed_atom # rubocop:disable Naming/PredicateName
101
+ @@is_displayed_atom ||= begin # rubocop:disable Style/ClassVars
102
+ browser.send(:bridge).send(:read_atom, 'isDisplayed')
103
+ rescue StandardError
104
+ # If the atom doesn't exist or other error
105
+ ''
106
+ end
107
+ end
108
+ end
109
+ end
110
+ end
@@ -2,37 +2,214 @@
2
2
 
3
3
  class Capybara::Selenium::Node
4
4
  module Html5Drag
5
- private
5
+ # Implement methods to emulate HTML5 drag and drop
6
+
7
+ def drag_to(element, html5: nil, delay: 0.05, drop_modifiers: [])
8
+ drop_modifiers = Array(drop_modifiers)
6
9
 
7
- def html5_drag_to(element)
8
10
  driver.execute_script MOUSEDOWN_TRACKER
9
11
  scroll_if_needed { browser_action.click_and_hold(native).perform }
10
- if driver.evaluate_script('window.capybara_mousedown_prevented')
11
- 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)
12
15
  else
13
- driver.execute_script HTML5_DRAG_DROP_SCRIPT, self, element
14
- browser_action.release.perform
16
+ perform_legacy_drag(element, drop_modifiers)
15
17
  end
16
18
  end
17
19
 
18
- def html5_draggable?
19
- # Workaround https://github.com/SeleniumHQ/selenium/issues/6396
20
- 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
21
49
  end
22
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
+
23
99
  MOUSEDOWN_TRACKER = <<~JS
100
+ window.capybara_mousedown_prevented = null;
24
101
  document.addEventListener('mousedown', ev => {
25
102
  window.capybara_mousedown_prevented = ev.defaultPrevented;
26
103
  }, { once: true, passive: true })
27
104
  JS
28
105
 
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
118
+
29
119
  HTML5_DRAG_DROP_SCRIPT = <<~JS
30
- var source = arguments[0];
31
- var target = arguments[1];
120
+ function rectCenter(rect){
121
+ return new DOMPoint(
122
+ (rect.left + rect.right)/2,
123
+ (rect.top + rect.bottom)/2
124
+ );
125
+ }
126
+
127
+ function pointOnRect(pt, rect) {
128
+ var rectPt = rectCenter(rect);
129
+ var slope = (rectPt.y - pt.y) / (rectPt.x - pt.x);
130
+
131
+ if (pt.x <= rectPt.x) { // left side
132
+ var minXy = slope * (rect.left - pt.x) + pt.y;
133
+ if (rect.top <= minXy && minXy <= rect.bottom)
134
+ return new DOMPoint(rect.left, minXy);
135
+ }
136
+
137
+ if (pt.x >= rectPt.x) { // right side
138
+ var maxXy = slope * (rect.right - pt.x) + pt.y;
139
+ if (rect.top <= maxXy && maxXy <= rect.bottom)
140
+ return new DOMPoint(rect.right, maxXy);
141
+ }
142
+
143
+ if (pt.y <= rectPt.y) { // top side
144
+ var minYx = (rectPt.top - pt.y) / slope + pt.x;
145
+ if (rect.left <= minYx && minYx <= rect.right)
146
+ return new DOMPoint(minYx, rect.top);
147
+ }
148
+
149
+ if (pt.y >= rectPt.y) { // bottom side
150
+ var maxYx = (rect.bottom - pt.y) / slope + pt.x;
151
+ if (rect.left <= maxYx && maxYx <= rect.right)
152
+ return new DOMPoint(maxYx, rect.bottom);
153
+ }
154
+
155
+ return new DOMPoint(pt.x,pt.y);
156
+ }
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];
32
205
 
33
206
  var dt = new DataTransfer();
34
207
  var opts = { cancelable: true, bubbles: true, dataTransfer: dt };
35
208
 
209
+ while (source && !source.draggable) {
210
+ source = source.parentElement;
211
+ }
212
+
36
213
  if (source.tagName == 'A'){
37
214
  dt.setData('text/uri-list', source.href);
38
215
  dt.setData('text', source.href);
@@ -41,19 +218,11 @@ class Capybara::Selenium::Node
41
218
  dt.setData('text/uri-list', source.src);
42
219
  dt.setData('text', source.src);
43
220
  }
221
+
44
222
  var dragEvent = new DragEvent('dragstart', opts);
45
223
  source.dispatchEvent(dragEvent);
46
- target.scrollIntoView({behavior: 'instant', block: 'center', inline: 'center'});
47
- var dragOverEvent = new DragEvent('dragover', opts);
48
- target.dispatchEvent(dragOverEvent);
49
- var dragLeaveEvent = new DragEvent('dragleave', opts);
50
- target.dispatchEvent(dragLeaveEvent);
51
- if (dragOverEvent.defaultPrevented) {
52
- var dropEvent = new DragEvent('drop', opts);
53
- target.dispatchEvent(dropEvent);
54
- }
55
- var dragEndEvent = new DragEvent('dragend', opts);
56
- source.dispatchEvent(dragEndEvent);
224
+
225
+ window.setTimeout(dragEnterTarget, step_delay);
57
226
  JS
58
227
  end
59
228
  end
@@ -0,0 +1,28 @@
1
+ # frozen_string_literal: true
2
+
3
+ class Capybara::Selenium::Node
4
+ #
5
+ # @api private
6
+ #
7
+ class ModifierKeysStack
8
+ def initialize
9
+ @stack = []
10
+ end
11
+
12
+ def include?(key)
13
+ @stack.flatten.include?(key)
14
+ end
15
+
16
+ def press(key)
17
+ @stack.last.push(key)
18
+ end
19
+
20
+ def push
21
+ @stack.push []
22
+ end
23
+
24
+ def pop
25
+ @stack.pop
26
+ end
27
+ end
28
+ end