capybara 2.18.0 → 3.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (172) hide show
  1. checksums.yaml +4 -4
  2. data/History.md +55 -1
  3. data/README.md +18 -17
  4. data/lib/capybara/config.rb +11 -58
  5. data/lib/capybara/cucumber.rb +2 -3
  6. data/lib/capybara/driver/base.rb +15 -16
  7. data/lib/capybara/driver/node.rb +5 -4
  8. data/lib/capybara/dsl.rb +1 -0
  9. data/lib/capybara/helpers.rb +19 -29
  10. data/lib/capybara/minitest/spec.rb +15 -14
  11. data/lib/capybara/minitest.rb +139 -138
  12. data/lib/capybara/node/actions.rb +60 -81
  13. data/lib/capybara/node/base.rb +11 -18
  14. data/lib/capybara/node/document.rb +2 -2
  15. data/lib/capybara/node/document_matchers.rb +8 -8
  16. data/lib/capybara/node/element.rb +30 -40
  17. data/lib/capybara/node/finders.rb +62 -70
  18. data/lib/capybara/node/matchers.rb +50 -71
  19. data/lib/capybara/node/simple.rb +11 -17
  20. data/lib/capybara/queries/ancestor_query.rb +11 -7
  21. data/lib/capybara/queries/base_query.rb +22 -18
  22. data/lib/capybara/queries/current_path_query.rb +8 -24
  23. data/lib/capybara/queries/match_query.rb +3 -7
  24. data/lib/capybara/queries/selector_query.rb +92 -95
  25. data/lib/capybara/queries/sibling_query.rb +4 -4
  26. data/lib/capybara/queries/text_query.rb +35 -35
  27. data/lib/capybara/queries/title_query.rb +8 -11
  28. data/lib/capybara/rack_test/browser.rb +15 -18
  29. data/lib/capybara/rack_test/css_handlers.rb +6 -4
  30. data/lib/capybara/rack_test/driver.rb +6 -10
  31. data/lib/capybara/rack_test/form.rb +50 -40
  32. data/lib/capybara/rack_test/node.rb +93 -63
  33. data/lib/capybara/rails.rb +2 -6
  34. data/lib/capybara/result.rb +22 -22
  35. data/lib/capybara/rspec/compound.rb +5 -10
  36. data/lib/capybara/rspec/features.rb +17 -48
  37. data/lib/capybara/rspec/matcher_proxies.rb +31 -15
  38. data/lib/capybara/rspec/matchers.rb +70 -61
  39. data/lib/capybara/rspec.rb +5 -10
  40. data/lib/capybara/selector/css.rb +6 -11
  41. data/lib/capybara/selector/filter.rb +1 -17
  42. data/lib/capybara/selector/filter_set.rb +18 -15
  43. data/lib/capybara/selector/filters/base.rb +7 -6
  44. data/lib/capybara/selector/filters/expression_filter.rb +6 -23
  45. data/lib/capybara/selector/filters/node_filter.rb +2 -12
  46. data/lib/capybara/selector/selector.rb +28 -34
  47. data/lib/capybara/selector.rb +129 -117
  48. data/lib/capybara/selenium/driver.rb +131 -125
  49. data/lib/capybara/selenium/node.rb +197 -115
  50. data/lib/capybara/server.rb +3 -2
  51. data/lib/capybara/session/config.rb +47 -67
  52. data/lib/capybara/session/matchers.rb +8 -7
  53. data/lib/capybara/session.rb +138 -224
  54. data/lib/capybara/spec/public/test.js +25 -4
  55. data/lib/capybara/spec/session/accept_alert_spec.rb +1 -0
  56. data/lib/capybara/spec/session/accept_confirm_spec.rb +3 -2
  57. data/lib/capybara/spec/session/accept_prompt_spec.rb +1 -0
  58. data/lib/capybara/spec/session/all_spec.rb +31 -18
  59. data/lib/capybara/spec/session/ancestor_spec.rb +6 -8
  60. data/lib/capybara/spec/session/assert_all_of_selectors_spec.rb +6 -5
  61. data/lib/capybara/spec/session/assert_current_path.rb +12 -11
  62. data/lib/capybara/spec/session/assert_selector.rb +1 -0
  63. data/lib/capybara/spec/session/assert_text.rb +23 -23
  64. data/lib/capybara/spec/session/assert_title.rb +13 -3
  65. data/lib/capybara/spec/session/attach_file_spec.rb +51 -30
  66. data/lib/capybara/spec/session/body_spec.rb +1 -0
  67. data/lib/capybara/spec/session/check_spec.rb +7 -6
  68. data/lib/capybara/spec/session/choose_spec.rb +5 -4
  69. data/lib/capybara/spec/session/click_button_spec.rb +24 -32
  70. data/lib/capybara/spec/session/click_link_or_button_spec.rb +8 -7
  71. data/lib/capybara/spec/session/click_link_spec.rb +8 -7
  72. data/lib/capybara/spec/session/current_scope_spec.rb +4 -3
  73. data/lib/capybara/spec/session/current_url_spec.rb +17 -6
  74. data/lib/capybara/spec/session/dismiss_confirm_spec.rb +1 -1
  75. data/lib/capybara/spec/session/dismiss_prompt_spec.rb +1 -0
  76. data/lib/capybara/spec/session/element/assert_match_selector.rb +1 -1
  77. data/lib/capybara/spec/session/element/match_xpath_spec.rb +1 -1
  78. data/lib/capybara/spec/session/element/matches_selector_spec.rb +5 -5
  79. data/lib/capybara/spec/session/evaluate_async_script_spec.rb +3 -2
  80. data/lib/capybara/spec/session/evaluate_script_spec.rb +4 -3
  81. data/lib/capybara/spec/session/execute_script_spec.rb +4 -3
  82. data/lib/capybara/spec/session/fill_in_spec.rb +30 -5
  83. data/lib/capybara/spec/session/find_button_spec.rb +4 -3
  84. data/lib/capybara/spec/session/find_by_id_spec.rb +2 -1
  85. data/lib/capybara/spec/session/find_field_spec.rb +9 -15
  86. data/lib/capybara/spec/session/find_link_spec.rb +6 -5
  87. data/lib/capybara/spec/session/find_spec.rb +37 -31
  88. data/lib/capybara/spec/session/first_spec.rb +60 -33
  89. data/lib/capybara/spec/session/frame/frame_title_spec.rb +23 -0
  90. data/lib/capybara/spec/session/frame/frame_url_spec.rb +23 -0
  91. data/lib/capybara/spec/session/frame/switch_to_frame_spec.rb +2 -1
  92. data/lib/capybara/spec/session/frame/within_frame_spec.rb +9 -16
  93. data/lib/capybara/spec/session/go_back_spec.rb +1 -0
  94. data/lib/capybara/spec/session/go_forward_spec.rb +1 -0
  95. data/lib/capybara/spec/session/has_all_selectors_spec.rb +15 -15
  96. data/lib/capybara/spec/session/has_button_spec.rb +2 -1
  97. data/lib/capybara/spec/session/has_css_spec.rb +3 -2
  98. data/lib/capybara/spec/session/has_current_path_spec.rb +12 -28
  99. data/lib/capybara/spec/session/has_field_spec.rb +4 -3
  100. data/lib/capybara/spec/session/has_link_spec.rb +1 -0
  101. data/lib/capybara/spec/session/has_none_selectors_spec.rb +17 -17
  102. data/lib/capybara/spec/session/has_select_spec.rb +30 -29
  103. data/lib/capybara/spec/session/has_selector_spec.rb +5 -4
  104. data/lib/capybara/spec/session/has_table_spec.rb +2 -1
  105. data/lib/capybara/spec/session/has_text_spec.rb +9 -13
  106. data/lib/capybara/spec/session/has_title_spec.rb +1 -0
  107. data/lib/capybara/spec/session/has_xpath_spec.rb +1 -0
  108. data/lib/capybara/spec/session/headers.rb +2 -1
  109. data/lib/capybara/spec/session/html_spec.rb +1 -0
  110. data/lib/capybara/spec/session/node_spec.rb +91 -56
  111. data/lib/capybara/spec/session/node_wrapper_spec.rb +36 -0
  112. data/lib/capybara/spec/session/refresh_spec.rb +6 -2
  113. data/lib/capybara/spec/session/reset_session_spec.rb +19 -0
  114. data/lib/capybara/spec/session/response_code.rb +1 -0
  115. data/lib/capybara/spec/session/save_and_open_page_spec.rb +1 -0
  116. data/lib/capybara/spec/session/save_and_open_screenshot_spec.rb +6 -11
  117. data/lib/capybara/spec/session/save_page_spec.rb +1 -17
  118. data/lib/capybara/spec/session/save_screenshot_spec.rb +3 -3
  119. data/lib/capybara/spec/session/select_spec.rb +20 -20
  120. data/lib/capybara/spec/session/selectors_spec.rb +2 -2
  121. data/lib/capybara/spec/session/sibling_spec.rb +1 -1
  122. data/lib/capybara/spec/session/text_spec.rb +17 -3
  123. data/lib/capybara/spec/session/title_spec.rb +11 -1
  124. data/lib/capybara/spec/session/uncheck_spec.rb +4 -3
  125. data/lib/capybara/spec/session/unselect_spec.rb +6 -5
  126. data/lib/capybara/spec/session/visit_spec.rb +9 -3
  127. data/lib/capybara/spec/session/window/become_closed_spec.rb +2 -1
  128. data/lib/capybara/spec/session/window/current_window_spec.rb +1 -0
  129. data/lib/capybara/spec/session/window/open_new_window_spec.rb +1 -0
  130. data/lib/capybara/spec/session/window/switch_to_window_spec.rb +2 -1
  131. data/lib/capybara/spec/session/window/window_opened_by_spec.rb +2 -1
  132. data/lib/capybara/spec/session/window/window_spec.rb +12 -12
  133. data/lib/capybara/spec/session/window/windows_spec.rb +2 -3
  134. data/lib/capybara/spec/session/window/within_window_spec.rb +15 -71
  135. data/lib/capybara/spec/session/within_spec.rb +1 -0
  136. data/lib/capybara/spec/spec_helper.rb +34 -18
  137. data/lib/capybara/spec/test_app.rb +17 -9
  138. data/lib/capybara/spec/views/form.erb +7 -0
  139. data/lib/capybara/spec/views/with_html.erb +23 -1
  140. data/lib/capybara/spec/views/within_frames.erb +4 -1
  141. data/lib/capybara/version.rb +2 -1
  142. data/lib/capybara/window.rb +6 -10
  143. data/lib/capybara.rb +28 -25
  144. data/spec/basic_node_spec.rb +1 -0
  145. data/spec/capybara_spec.rb +11 -50
  146. data/spec/dsl_spec.rb +5 -13
  147. data/spec/filter_set_spec.rb +5 -4
  148. data/spec/fixtures/selenium_driver_rspec_failure.rb +2 -1
  149. data/spec/fixtures/selenium_driver_rspec_success.rb +3 -2
  150. data/spec/minitest_spec.rb +4 -3
  151. data/spec/minitest_spec_spec.rb +3 -2
  152. data/spec/per_session_config_spec.rb +9 -8
  153. data/spec/rack_test_spec.rb +21 -20
  154. data/spec/result_spec.rb +17 -16
  155. data/spec/rspec/features_spec.rb +17 -14
  156. data/spec/rspec/scenarios_spec.rb +5 -7
  157. data/spec/rspec/shared_spec_matchers.rb +96 -99
  158. data/spec/rspec/views_spec.rb +2 -1
  159. data/spec/rspec_matchers_spec.rb +18 -2
  160. data/spec/rspec_spec.rb +11 -15
  161. data/spec/selector_spec.rb +5 -6
  162. data/spec/selenium_spec_chrome.rb +9 -4
  163. data/spec/selenium_spec_edge.rb +27 -0
  164. data/spec/selenium_spec_ie.rb +31 -0
  165. data/spec/selenium_spec_marionette.rb +28 -12
  166. data/spec/server_spec.rb +33 -33
  167. data/spec/session_spec.rb +2 -1
  168. data/spec/shared_selenium_session.rb +36 -22
  169. data/spec/spec_helper.rb +3 -6
  170. metadata +68 -85
  171. data/lib/capybara/query.rb +0 -7
  172. data/spec/selenium_spec_firefox.rb +0 -68
@@ -1,4 +1,5 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'minitest'
3
4
  require 'capybara/dsl'
4
5
 
@@ -41,17 +42,16 @@ module Capybara
41
42
  # @!method assert_no_current_path
42
43
  # see {Capybara::SessionMatchers#assert_no_current_path}
43
44
 
44
-
45
- %w(assert_text assert_no_text assert_title assert_no_title assert_current_path assert_no_current_path).each do |assertion_name|
46
- self.class_eval <<-EOM, __FILE__, __LINE__ + 1
45
+ %w[assert_text assert_no_text assert_title assert_no_title assert_current_path assert_no_current_path].each do |assertion_name|
46
+ class_eval <<-ASSERTION, __FILE__, __LINE__ + 1
47
47
  def #{assertion_name} *args
48
48
  self.assertions +=1
49
- subject, *args = determine_subject(args)
49
+ subject, args = determine_subject(args)
50
50
  subject.#{assertion_name}(*args)
51
51
  rescue Capybara::ExpectationNotMet => e
52
52
  raise ::Minitest::Assertion, e.message
53
53
  end
54
- EOM
54
+ ASSERTION
55
55
  end
56
56
 
57
57
  alias_method :refute_title, :assert_no_title
@@ -81,190 +81,191 @@ module Capybara
81
81
  # @!method assert_xpath
82
82
  # see {Capybara::Node::Matchers#assert_not_matches_selector}
83
83
 
84
- %w(assert_selector assert_no_selector
84
+ %w[assert_selector assert_no_selector
85
85
  assert_all_of_selectors assert_none_of_selectors
86
- assert_matches_selector assert_not_matches_selector).each do |assertion_name|
87
- self.class_eval <<-EOM, __FILE__, __LINE__ + 1
86
+ assert_matches_selector assert_not_matches_selector].each do |assertion_name|
87
+ class_eval <<-ASSERTION, __FILE__, __LINE__ + 1
88
88
  def #{assertion_name} *args, &optional_filter_block
89
89
  self.assertions +=1
90
- subject, *args = determine_subject(args)
90
+ subject, args = determine_subject(args)
91
91
  subject.#{assertion_name}(*args, &optional_filter_block)
92
92
  rescue Capybara::ExpectationNotMet => e
93
93
  raise ::Minitest::Assertion, e.message
94
94
  end
95
- EOM
95
+ ASSERTION
96
96
  end
97
97
 
98
98
  alias_method :refute_selector, :assert_no_selector
99
99
  alias_method :refute_matches_selector, :assert_not_matches_selector
100
100
 
101
- %w(xpath css link button field select table).each do |selector_type|
101
+ %w[xpath css link button field select table].each do |selector_type|
102
102
  define_method "assert_#{selector_type}" do |*args, &optional_filter_block|
103
- subject, *args = determine_subject(args)
103
+ subject, args = determine_subject(args)
104
104
  locator, options = extract_locator(args)
105
105
  assert_selector(subject, selector_type.to_sym, locator, options, &optional_filter_block)
106
106
  end
107
107
 
108
108
  define_method "assert_no_#{selector_type}" do |*args, &optional_filter_block|
109
- subject, *args = determine_subject(args)
109
+ subject, args = determine_subject(args)
110
110
  locator, options = extract_locator(args)
111
111
  assert_no_selector(subject, selector_type.to_sym, locator, options, &optional_filter_block)
112
112
  end
113
113
  alias_method "refute_#{selector_type}", "assert_no_#{selector_type}"
114
114
  end
115
115
 
116
- %w(checked unchecked).each do |field_type|
116
+ %w[checked unchecked].each do |field_type|
117
117
  define_method "assert_#{field_type}_field" do |*args, &optional_filter_block|
118
- subject, *args = determine_subject(args)
118
+ subject, args = determine_subject(args)
119
119
  locator, options = extract_locator(args)
120
120
  assert_selector(subject, :field, locator, options.merge(field_type.to_sym => true), &optional_filter_block)
121
121
  end
122
122
 
123
123
  define_method "assert_no_#{field_type}_field" do |*args, &optional_filter_block|
124
- subject, *args = determine_subject(args)
124
+ subject, args = determine_subject(args)
125
125
  locator, options = extract_locator(args)
126
126
  assert_no_selector(subject, :field, locator, options.merge(field_type.to_sym => true), &optional_filter_block)
127
127
  end
128
128
  alias_method "refute_#{field_type}_field", "assert_no_#{field_type}_field"
129
129
  end
130
130
 
131
- %w(xpath css).each do |selector_type|
131
+ %w[xpath css].each do |selector_type|
132
132
  define_method "assert_matches_#{selector_type}" do |*args, &optional_filter_block|
133
- subject, *args = determine_subject(args)
133
+ subject, args = determine_subject(args)
134
134
  assert_matches_selector(subject, selector_type.to_sym, *args, &optional_filter_block)
135
135
  end
136
136
 
137
137
  define_method "assert_not_matches_#{selector_type}" do |*args, &optional_filter_block|
138
- subject, *args = determine_subject(args)
138
+ subject, args = determine_subject(args)
139
139
  assert_not_matches_selector(subject, selector_type.to_sym, *args, &optional_filter_block)
140
140
  end
141
141
  alias_method "refute_matches_#{selector_type}", "assert_not_matches_#{selector_type}"
142
142
  end
143
143
 
144
-
145
- ##
146
- # Assertion that there is xpath
147
- #
148
- # @!method assert_xpath
149
- # see Capybara::Node::Matchers#has_xpath?
150
-
151
- ##
152
- # Assertion that there is no xpath
153
- #
154
- # @!method refute_xpath
155
- # @!method assert_no_xpath
156
- # see Capybara::Node::Matchers#has_no_xpath?
157
-
158
- ##
159
- # Assertion that there is css
160
- #
161
- # @!method assert_css
162
- # see Capybara::Node::Matchers#has_css?
163
-
164
- ##
165
- # Assertion that there is no css
166
- #
167
- # @!method refute_css
168
- # @!method assert_no_css
169
- # see Capybara::Node::Matchers#has_no_css?
170
-
171
- ##
172
- # Assertion that there is link
173
- #
174
- # @!method assert_link
175
- # see {Capybara::Node::Matchers#has_link?}
176
-
177
- ##
178
- # Assertion that there is no link
179
- #
180
- # @!method assert_no_link
181
- # @!method refute_link
182
- # see {Capybara::Node::Matchers#has_no_link?}
183
-
184
- ##
185
- # Assertion that there is button
186
- #
187
- # @!method assert_button
188
- # see {Capybara::Node::Matchers#has_button?}
189
-
190
- ##
191
- # Assertion that there is no button
192
- #
193
- # @!method refute_button
194
- # @!method assert_no_button
195
- # see {Capybara::Node::Matchers#has_no_button?}
196
-
197
- ##
198
- # Assertion that there is field
199
- #
200
- # @!method assert_field
201
- # see {Capybara::Node::Matchers#has_field?}
202
-
203
- ##
204
- # Assertion that there is no field
205
- #
206
- # @!method refute_field
207
- # @!method assert_no_field
208
- # see {Capybara::Node::Matchers#has_no_field?}
209
-
210
- ##
211
- # Assertion that there is checked_field
212
- #
213
- # @!method assert_checked_field
214
- # see {Capybara::Node::Matchers#has_checked_field?}
215
-
216
- ##
217
- # Assertion that there is no checked_field
218
- #
219
- # @!method assert_no_checked_field
220
- # @!method refute_checked_field
221
-
222
- ##
223
- # Assertion that there is unchecked_field
224
- #
225
- # @!method assert_unchecked_field
226
- # see {Capybara::Node::Matchers#has_unchecked_field?}
227
-
228
- ##
229
- # Assertion that there is no unchecked_field
230
- #
231
- # @!method assert_no_unchecked_field
232
- # @!method refute_unchecked_field
233
-
234
- ##
235
- # Assertion that there is select
236
- #
237
- # @!method assert_select
238
- # see {Capybara::Node::Matchers#has_select?}
239
-
240
- ##
241
- # Assertion that there is no select
242
- #
243
- # @!method refute_select
244
- # @!method assert_no_select
245
- # see {Capybara::Node::Matchers#has_no_select?}
246
-
247
- ##
248
- # Assertion that there is table
249
- #
250
- # @!method assert_table
251
- # see {Capybara::Node::Matchers#has_table?}
252
-
253
- ##
254
- # Assertion that there is no table
255
- #
256
- # @!method refute_table
257
- # @!method assert_no_table
258
- # see {Capybara::Node::Matchers#has_no_table?}
259
-
260
- private
144
+ ##
145
+ # Assertion that there is xpath
146
+ #
147
+ # @!method assert_xpath
148
+ # see Capybara::Node::Matchers#has_xpath?
149
+
150
+ ##
151
+ # Assertion that there is no xpath
152
+ #
153
+ # @!method refute_xpath
154
+ # @!method assert_no_xpath
155
+ # see Capybara::Node::Matchers#has_no_xpath?
156
+
157
+ ##
158
+ # Assertion that there is css
159
+ #
160
+ # @!method assert_css
161
+ # see Capybara::Node::Matchers#has_css?
162
+
163
+ ##
164
+ # Assertion that there is no css
165
+ #
166
+ # @!method refute_css
167
+ # @!method assert_no_css
168
+ # see Capybara::Node::Matchers#has_no_css?
169
+
170
+ ##
171
+ # Assertion that there is link
172
+ #
173
+ # @!method assert_link
174
+ # see {Capybara::Node::Matchers#has_link?}
175
+
176
+ ##
177
+ # Assertion that there is no link
178
+ #
179
+ # @!method assert_no_link
180
+ # @!method refute_link
181
+ # see {Capybara::Node::Matchers#has_no_link?}
182
+
183
+ ##
184
+ # Assertion that there is button
185
+ #
186
+ # @!method assert_button
187
+ # see {Capybara::Node::Matchers#has_button?}
188
+
189
+ ##
190
+ # Assertion that there is no button
191
+ #
192
+ # @!method refute_button
193
+ # @!method assert_no_button
194
+ # see {Capybara::Node::Matchers#has_no_button?}
195
+
196
+ ##
197
+ # Assertion that there is field
198
+ #
199
+ # @!method assert_field
200
+ # see {Capybara::Node::Matchers#has_field?}
201
+
202
+ ##
203
+ # Assertion that there is no field
204
+ #
205
+ # @!method refute_field
206
+ # @!method assert_no_field
207
+ # see {Capybara::Node::Matchers#has_no_field?}
208
+
209
+ ##
210
+ # Assertion that there is checked_field
211
+ #
212
+ # @!method assert_checked_field
213
+ # see {Capybara::Node::Matchers#has_checked_field?}
214
+
215
+ ##
216
+ # Assertion that there is no checked_field
217
+ #
218
+ # @!method assert_no_checked_field
219
+ # @!method refute_checked_field
220
+
221
+ ##
222
+ # Assertion that there is unchecked_field
223
+ #
224
+ # @!method assert_unchecked_field
225
+ # see {Capybara::Node::Matchers#has_unchecked_field?}
226
+
227
+ ##
228
+ # Assertion that there is no unchecked_field
229
+ #
230
+ # @!method assert_no_unchecked_field
231
+ # @!method refute_unchecked_field
232
+
233
+ ##
234
+ # Assertion that there is select
235
+ #
236
+ # @!method assert_select
237
+ # see {Capybara::Node::Matchers#has_select?}
238
+
239
+ ##
240
+ # Assertion that there is no select
241
+ #
242
+ # @!method refute_select
243
+ # @!method assert_no_select
244
+ # see {Capybara::Node::Matchers#has_no_select?}
245
+
246
+ ##
247
+ # Assertion that there is table
248
+ #
249
+ # @!method assert_table
250
+ # see {Capybara::Node::Matchers#has_table?}
251
+
252
+ ##
253
+ # Assertion that there is no table
254
+ #
255
+ # @!method refute_table
256
+ # @!method assert_no_table
257
+ # see {Capybara::Node::Matchers#has_no_table?}
258
+
259
+ private
261
260
 
262
261
  def determine_subject(args)
263
262
  case args.first
264
263
  when Capybara::Session, Capybara::Node::Base, Capybara::Node::Simple
265
- args
264
+ [args.shift, args]
265
+ when ->(arg) { arg.respond_to?(:to_capybara_node) }
266
+ [args.shift.to_capybara_node, args]
266
267
  else
267
- [page, *args]
268
+ [page, args]
268
269
  end
269
270
  end
270
271
 
@@ -1,8 +1,8 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  module Capybara
3
4
  module Node
4
5
  module Actions
5
-
6
6
  ##
7
7
  #
8
8
  # Finds a button or link and clicks it. See {Capybara::Node::Actions#click_button} and
@@ -20,8 +20,7 @@ module Capybara
20
20
  #
21
21
  # @return [Capybara::Node::Element] The element clicked
22
22
  #
23
- def click_link_or_button(locator=nil, options={})
24
- locator, options = nil, locator if locator.is_a? Hash
23
+ def click_link_or_button(locator = nil, **options)
25
24
  find(:link_or_button, locator, options).click
26
25
  end
27
26
  alias_method :click_on, :click_link_or_button
@@ -38,8 +37,7 @@ module Capybara
38
37
  # @param options See {Capybara::Node::Finders#find_link}
39
38
  #
40
39
  # @return [Capybara::Node::Element] The element clicked
41
- def click_link(locator=nil, options={})
42
- locator, options = nil, locator if locator.is_a? Hash
40
+ def click_link(locator = nil, **options)
43
41
  find(:link, locator, options).click
44
42
  end
45
43
 
@@ -56,8 +54,7 @@ module Capybara
56
54
  # @param [String] locator Which button to find
57
55
  # @param options See {Capybara::Node::Finders#find_button}
58
56
  # @return [Capybara::Node::Element] The element clicked
59
- def click_button(locator=nil, options={})
60
- locator, options = nil, locator if locator.is_a? Hash
57
+ def click_button(locator = nil, **options)
61
58
  find(:button, locator, options).click
62
59
  end
63
60
 
@@ -83,12 +80,8 @@ module Capybara
83
80
  # @option options [String, Array<String>] :class Match fields that match the class(es) provided
84
81
  #
85
82
  # @return [Capybara::Node::Element] The element filled_in
86
- def fill_in(locator, options={})
87
- locator, options = nil, locator if locator.is_a? Hash
88
- raise "Must pass a hash containing 'with'" if not options.is_a?(Hash) or not options.has_key?(:with)
89
- with = options.delete(:with)
90
- fill_options = options.delete(:fill_options)
91
- options[:with] = options.delete(:currently_with) if options.has_key?(:currently_with)
83
+ def fill_in(locator = nil, with:, fill_options: {}, **options)
84
+ options[:with] = options.delete(:currently_with) if options.key?(:currently_with)
92
85
  find(:fillable_field, locator, options).set(with, fill_options)
93
86
  end
94
87
 
@@ -113,8 +106,8 @@ module Capybara
113
106
  # @macro label_click
114
107
  #
115
108
  # @return [Capybara::Node::Element] The element chosen or the label clicked
116
- def choose(locator, options={})
117
- _check_with_label(:radio_button, true, locator, options)
109
+ def choose(locator = nil, **options)
110
+ _check_with_label(:radio_button, true, locator, **options)
118
111
  end
119
112
 
120
113
  ##
@@ -136,8 +129,8 @@ module Capybara
136
129
  # @macro waiting_behavior
137
130
  #
138
131
  # @return [Capybara::Node::Element] The element checked or the label clicked
139
- def check(locator, options={})
140
- _check_with_label(:checkbox, true, locator, options)
132
+ def check(locator, **options)
133
+ _check_with_label(:checkbox, true, locator, **options)
141
134
  end
142
135
 
143
136
  ##
@@ -159,8 +152,8 @@ module Capybara
159
152
  # @macro waiting_behavior
160
153
  #
161
154
  # @return [Capybara::Node::Element] The element unchecked or the label clicked
162
- def uncheck(locator, options={})
163
- _check_with_label(:checkbox, false, locator, options)
155
+ def uncheck(locator = nil, **options)
156
+ _check_with_label(:checkbox, false, locator, **options)
164
157
  end
165
158
 
166
159
  ##
@@ -180,13 +173,9 @@ module Capybara
180
173
  # @option options [String] :from The id, name or label of the select box
181
174
  #
182
175
  # @return [Capybara::Node::Element] The option element selected
183
- def select(value, options={})
184
- if options.has_key?(:from)
185
- from = options.delete(:from)
186
- find(:select, from, options).find(:option, value, options).select_option
187
- else
188
- find(:option, value, options).select_option
189
- end
176
+ def select(value = nil, from: nil, **options)
177
+ scope = from ? find(:select, from, options) : self
178
+ scope.find(:option, value, options).select_option
190
179
  end
191
180
 
192
181
  ##
@@ -203,13 +192,9 @@ module Capybara
203
192
  # @param [Hash{:from => String}] options The id, name or label of the select box
204
193
  #
205
194
  # @return [Capybara::Node::Element] The option element unselected
206
- def unselect(value, options={})
207
- if options.has_key?(:from)
208
- from = options.delete(:from)
209
- find(:select, from, options).find(:option, value, options).unselect_option
210
- else
211
- find(:option, value, options).unselect_option
212
- end
195
+ def unselect(value = nil, from: nil, **options)
196
+ scope = from ? find(:select, from, options) : self
197
+ scope.find(:option, value, options).unselect_option
213
198
  end
214
199
 
215
200
  ##
@@ -233,68 +218,44 @@ module Capybara
233
218
  # @option options [true, Hash] make_visible A Hash of CSS styles to change before attempting to attach the file, if `true` { opacity: 1, display: 'block', visibility: 'visible' } is used (may not be supported by all drivers)
234
219
  #
235
220
  # @return [Capybara::Node::Element] The file field element
236
- def attach_file(locator, path, options={})
237
- locator, path, options = nil, locator, path if path.is_a? Hash
221
+ def attach_file(locator = nil, path, make_visible: nil, **options) # rubocop:disable Style/OptionalArguments
238
222
  Array(path).each do |p|
239
223
  raise Capybara::FileNotFound, "cannot attach file, #{p} does not exist" unless File.exist?(p.to_s)
240
224
  end
241
225
  # Allow user to update the CSS style of the file input since they are so often hidden on a page
242
- if style = options.delete(:make_visible)
243
- style = { opacity: 1, display: 'block', visibility: 'visible' } if style == true
244
- ff = find(:file_field, locator, options.merge({visible: :all}))
245
- _update_style(ff, style)
246
- if ff.visible?
247
- begin
248
- ff.set(path)
249
- ensure
250
- _reset_style(ff)
251
- end
252
- else
253
- raise ExpectationNotMet, "The style changes in :make_visible did not make the file input visible"
254
- end
226
+ if make_visible
227
+ ff = find(:file_field, locator, options.merge(visible: :all))
228
+ while_visible(ff, make_visible) { |el| el.set(path) }
255
229
  else
256
230
  find(:file_field, locator, options).set(path)
257
231
  end
258
232
  end
259
233
 
260
234
  private
261
- def _update_style(element, style)
262
- script = <<-JS
263
- var el = arguments[0];
264
- el.capybara_style_cache = el.style.cssText;
265
- var css = arguments[1];
266
- for (var prop in css){
267
- if (css.hasOwnProperty(prop)) {
268
- el.style[prop] = css[prop]
269
- }
270
- }
271
- JS
272
- begin
273
- session.execute_script(script, element, style)
274
- rescue Capybara::NotSupportedByDriverError
275
- warn "The :make_visible option is not supported by the current driver - ignoring"
276
- end
277
- end
278
235
 
279
- def _reset_style(element)
280
- script = <<-JS
281
- var el = arguments[0];
282
- if (el.hasOwnProperty('capybara_style_cache')) {
283
- el.style.cssText = el.capybara_style_cache;
284
- delete el.capybara_style_cache;
285
- }
286
- JS
236
+ def while_visible(element, visible_css)
237
+ visible_css = { opacity: 1, display: 'block', visibility: 'visible' } if visible_css == true
238
+ _update_style(element, visible_css)
239
+ raise ExpectationNotMet, "The style changes in :make_visible did not make the file input visible" unless element.visible?
287
240
  begin
288
- session.execute_script(script, element)
289
- rescue
241
+ yield element
242
+ ensure
243
+ _reset_style(element)
290
244
  end
291
245
  end
292
246
 
247
+ def _update_style(element, style)
248
+ session.execute_script(UPDATE_STYLE_SCRIPT, element, style)
249
+ rescue Capybara::NotSupportedByDriverError
250
+ warn "The :make_visible option is not supported by the current driver - ignoring"
251
+ end
293
252
 
294
- def _check_with_label(selector, checked, locator, options)
295
- locator, options = nil, locator if locator.is_a? Hash
296
- allow_label_click = options.delete(:allow_label_click) { session_options.automatic_label_click }
253
+ def _reset_style(element)
254
+ session.execute_script(RESET_STYLE_SCRIPT, element)
255
+ rescue # swallow extra errors
256
+ end
297
257
 
258
+ def _check_with_label(selector, checked, locator, allow_label_click: session_options.automatic_label_click, **options)
298
259
  synchronize(Capybara::Queries::BaseQuery.wait(options, session_options.default_max_wait_time)) do
299
260
  begin
300
261
  el = find(selector, locator, options)
@@ -303,15 +264,33 @@ module Capybara
303
264
  raise unless allow_label_click && catch_error?(e)
304
265
  begin
305
266
  el ||= find(selector, locator, options.merge(visible: :all))
306
- label = find(:label, for: el, visible: true)
307
- label.click unless (el.checked? == checked)
308
- rescue
267
+ res = find(:label, for: el, visible: true).click unless el.checked? == checked
268
+ res
269
+ rescue # swallow extra errors - raise original
309
270
  raise e
310
271
  end
311
272
  end
312
273
  end
313
274
  end
314
275
 
276
+ UPDATE_STYLE_SCRIPT = <<-'JS'.freeze
277
+ var el = arguments[0];
278
+ el.capybara_style_cache = el.style.cssText;
279
+ var css = arguments[1];
280
+ for (var prop in css){
281
+ if (css.hasOwnProperty(prop)) {
282
+ el.style[prop] = css[prop]
283
+ }
284
+ }
285
+ JS
286
+
287
+ RESET_STYLE_SCRIPT = <<-'JS'.freeze
288
+ var el = arguments[0];
289
+ if (el.hasOwnProperty('capybara_style_cache')) {
290
+ el.style.cssText = el.capybara_style_cache;
291
+ delete el.capybara_style_cache;
292
+ }
293
+ JS
315
294
  end
316
295
  end
317
296
  end
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  module Capybara
3
4
  module Node
4
-
5
5
  ##
6
6
  #
7
7
  # A {Capybara::Node::Base} represents either an element on a page through the subclass
@@ -67,26 +67,23 @@ module Capybara
67
67
  # time has passed. On rubies/platforms which don't support access to a monotonic process clock
68
68
  # if the return value of `Time.now` is stubbed out, Capybara will raise `Capybara::FrozenInTime`.
69
69
  #
70
- # @param [Integer] seconds Number of seconds to retry this block
71
- # @param options [Hash]
72
- # @option options [Array<Exception>] :errors (driver.invalid_element_errors +
70
+ # @param [Integer] seconds (current sessions default_max_wait_time) Maximum number of seconds to retry this block
71
+ # @param [Array<Exception>] errors (driver.invalid_element_errors +
73
72
  # [Capybara::ElementNotFound]) exception types that cause the block to be rerun
74
73
  # @return [Object] The result of the given block
75
74
  # @raise [Capybara::FrozenInTime] If the return value of `Time.now` appears stuck
76
75
  #
77
- def synchronize(seconds=session_options.default_max_wait_time, options = {})
78
- start_time = Capybara::Helpers.monotonic_time
79
-
76
+ def synchronize(seconds = session_options.default_max_wait_time, errors: nil)
80
77
  if session.synchronized
81
78
  yield
82
79
  else
83
80
  session.synchronized = true
81
+ start_time = Capybara::Helpers.monotonic_time
84
82
  begin
85
83
  yield
86
84
  rescue => e
87
85
  session.raise_server_error!
88
- raise e unless driver.wait?
89
- raise e unless catch_error?(e, options[:errors])
86
+ raise e unless driver.wait? && catch_error?(e, errors)
90
87
  raise e if (Capybara::Helpers.monotonic_time - start_time) >= seconds
91
88
  sleep(0.05)
92
89
  raise Capybara::FrozenInTime, "time appears to be frozen, Capybara does not work with libraries which freeze time, consider using time travelling instead" if Capybara::Helpers.monotonic_time == start_time
@@ -108,24 +105,20 @@ module Capybara
108
105
  base.find_xpath(xpath)
109
106
  end
110
107
 
111
- # @deprecated Use query_scope instead
112
- def parent
113
- warn "DEPRECATED: #parent is deprecated in favor of #query_scope - Note: #parent was not the elements parent in the document so it's most likely not what you wanted anyway"
114
- query_scope
115
- end
116
-
117
108
  # @api private
118
109
  def session_options
119
110
  session.config
120
111
  end
121
112
 
113
+ def to_capybara_node
114
+ self
115
+ end
116
+
122
117
  protected
123
118
 
124
119
  def catch_error?(error, errors = nil)
125
120
  errors ||= (driver.invalid_element_errors + [Capybara::ElementNotFound])
126
- errors.any? do |type|
127
- error.is_a?(type)
128
- end
121
+ errors.any? { |type| error.is_a?(type) }
129
122
  end
130
123
 
131
124
  def driver