watir 6.10.1 → 6.10.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.document +5 -5
- data/.github/ISSUE_TEMPLATE.md +16 -16
- data/.gitignore +21 -21
- data/.travis.yml +72 -72
- data/CHANGES.md +4 -0
- data/Gemfile +16 -16
- data/LICENSE +23 -23
- data/README.md +106 -106
- data/Rakefile +153 -153
- data/appveyor.yml +13 -13
- data/lib/watir-webdriver.rb +2 -2
- data/lib/watir.rb +158 -158
- data/lib/watir/adjacent.rb +126 -126
- data/lib/watir/after_hooks.rb +132 -132
- data/lib/watir/alert.rb +120 -120
- data/lib/watir/aliases.rb +6 -6
- data/lib/watir/attribute_helper.rb +96 -96
- data/lib/watir/browser.rb +349 -349
- data/lib/watir/capabilities.rb +123 -123
- data/lib/watir/cell_container.rb +25 -25
- data/lib/watir/container.rb +51 -51
- data/lib/watir/cookies.rb +134 -134
- data/lib/watir/element_collection.rb +138 -138
- data/lib/watir/elements/area.rb +12 -12
- data/lib/watir/elements/button.rb +37 -37
- data/lib/watir/elements/cell.rb +17 -17
- data/lib/watir/elements/checkbox.rb +56 -56
- data/lib/watir/elements/dlist.rb +12 -12
- data/lib/watir/elements/element.rb +2 -2
- data/lib/watir/elements/file_field.rb +41 -41
- data/lib/watir/elements/font.rb +25 -25
- data/lib/watir/elements/form.rb +16 -16
- data/lib/watir/elements/hidden.rb +25 -25
- data/lib/watir/elements/html_elements.rb +2035 -2035
- data/lib/watir/elements/iframe.rb +166 -166
- data/lib/watir/elements/image.rb +21 -21
- data/lib/watir/elements/input.rb +15 -15
- data/lib/watir/elements/link.rb +18 -18
- data/lib/watir/elements/list.rb +49 -49
- data/lib/watir/elements/option.rb +72 -72
- data/lib/watir/elements/radio.rb +60 -60
- data/lib/watir/elements/row.rb +17 -17
- data/lib/watir/elements/select.rb +250 -250
- data/lib/watir/elements/svg_elements.rb +613 -613
- data/lib/watir/elements/table.rb +76 -76
- data/lib/watir/elements/table_cell.rb +31 -31
- data/lib/watir/elements/table_row.rb +32 -32
- data/lib/watir/elements/table_section.rb +15 -15
- data/lib/watir/elements/text_area.rb +5 -5
- data/lib/watir/elements/text_field.rb +39 -39
- data/lib/watir/exception.rb +16 -16
- data/lib/watir/extensions/nokogiri.rb +14 -14
- data/lib/watir/generator.rb +3 -3
- data/lib/watir/generator/base.rb +11 -11
- data/lib/watir/generator/base/generator.rb +115 -115
- data/lib/watir/generator/base/idl_sorter.rb +47 -47
- data/lib/watir/generator/base/spec_extractor.rb +138 -138
- data/lib/watir/generator/base/util.rb +21 -21
- data/lib/watir/generator/base/visitor.rb +162 -162
- data/lib/watir/generator/html.rb +15 -15
- data/lib/watir/generator/html/generator.rb +36 -36
- data/lib/watir/generator/html/spec_extractor.rb +50 -50
- data/lib/watir/generator/html/visitor.rb +21 -21
- data/lib/watir/generator/svg.rb +7 -7
- data/lib/watir/generator/svg/generator.rb +38 -38
- data/lib/watir/generator/svg/spec_extractor.rb +53 -53
- data/lib/watir/generator/svg/visitor.rb +21 -21
- data/lib/watir/has_window.rb +68 -68
- data/lib/watir/js_execution.rb +143 -143
- data/lib/watir/js_snippets.rb +16 -16
- data/lib/watir/js_snippets/backgroundColor.js +7 -7
- data/lib/watir/js_snippets/fireEvent.js +31 -31
- data/lib/watir/js_snippets/focus.js +3 -3
- data/lib/watir/js_snippets/getInnerHtml.js +19 -19
- data/lib/watir/js_snippets/getInnerText.js +19 -19
- data/lib/watir/js_snippets/getOuterHtml.js +20 -20
- data/lib/watir/js_snippets/getTextContent.js +19 -19
- data/lib/watir/js_snippets/isImageLoaded.js +3 -3
- data/lib/watir/js_snippets/selectOptionsLabel.js +10 -10
- data/lib/watir/js_snippets/selectOptionsText.js +10 -10
- data/lib/watir/js_snippets/selectOptionsValue.js +10 -10
- data/lib/watir/js_snippets/selectText.js +64 -64
- data/lib/watir/js_snippets/selectedOptions.js +11 -11
- data/lib/watir/js_snippets/setValue.js +3 -3
- data/lib/watir/legacy_wait.rb +126 -126
- data/lib/watir/locators.rb +60 -60
- data/lib/watir/locators/button/locator.rb +31 -31
- data/lib/watir/locators/button/selector_builder.rb +30 -30
- data/lib/watir/locators/button/selector_builder/xpath.rb +29 -29
- data/lib/watir/locators/button/validator.rb +15 -15
- data/lib/watir/locators/cell/locator.rb +15 -15
- data/lib/watir/locators/cell/selector_builder.rb +24 -24
- data/lib/watir/locators/element/selector_builder.rb +150 -150
- data/lib/watir/locators/element/selector_builder/xpath.rb +119 -119
- data/lib/watir/locators/element/validator.rb +18 -18
- data/lib/watir/locators/row/locator.rb +15 -15
- data/lib/watir/locators/row/selector_builder.rb +29 -29
- data/lib/watir/locators/text_area/locator.rb +13 -13
- data/lib/watir/locators/text_area/selector_builder.rb +22 -22
- data/lib/watir/locators/text_field/locator.rb +30 -30
- data/lib/watir/locators/text_field/selector_builder.rb +31 -31
- data/lib/watir/locators/text_field/selector_builder/xpath.rb +17 -17
- data/lib/watir/locators/text_field/validator.rb +12 -12
- data/lib/watir/logger.rb +109 -109
- data/lib/watir/radio_set.rb +231 -231
- data/lib/watir/row_container.rb +36 -36
- data/lib/watir/screenshot.rb +50 -50
- data/lib/watir/user_editable.rb +56 -56
- data/lib/watir/wait.rb +200 -200
- data/lib/watir/wait/timer.rb +52 -52
- data/lib/watir/window.rb +257 -257
- data/lib/watir/xpath_support.rb +20 -20
- data/lib/watirspec.rb +86 -86
- data/lib/watirspec/guards.rb +60 -60
- data/lib/watirspec/implementation.rb +51 -51
- data/lib/watirspec/rake_tasks.rb +115 -115
- data/lib/watirspec/remote_server.rb +38 -38
- data/lib/watirspec/runner.rb +53 -53
- data/lib/watirspec/server.rb +99 -99
- data/lib/watirspec/server/app.rb +76 -76
- data/spec/browser_spec.rb +278 -278
- data/spec/click_spec.rb +19 -19
- data/spec/container_spec.rb +34 -34
- data/spec/element_locator_spec.rb +517 -517
- data/spec/element_spec.rb +158 -158
- data/spec/implementation_spec.rb +24 -24
- data/spec/input_spec.rb +14 -14
- data/spec/locator_spec_helper.rb +55 -55
- data/spec/logger_spec.rb +46 -46
- data/spec/spec_helper.rb +22 -22
- data/spec/special_chars_spec.rb +13 -13
- data/spec/watirspec/adjacent_spec.rb +227 -227
- data/spec/watirspec/after_hooks_spec.rb +182 -182
- data/spec/watirspec/alert_spec.rb +108 -108
- data/spec/watirspec/attributes_spec.rb +16 -16
- data/spec/watirspec/browser_spec.rb +281 -281
- data/spec/watirspec/cookies_spec.rb +154 -154
- data/spec/watirspec/drag_and_drop_spec.rb +49 -49
- data/spec/watirspec/element_hidden_spec.rb +75 -75
- data/spec/watirspec/elements/area_spec.rb +72 -72
- data/spec/watirspec/elements/areas_spec.rb +42 -42
- data/spec/watirspec/elements/button_spec.rb +277 -277
- data/spec/watirspec/elements/buttons_spec.rb +55 -55
- data/spec/watirspec/elements/checkbox_spec.rb +275 -275
- data/spec/watirspec/elements/checkboxes_spec.rb +44 -44
- data/spec/watirspec/elements/collections_spec.rb +52 -52
- data/spec/watirspec/elements/dd_spec.rb +124 -124
- data/spec/watirspec/elements/dds_spec.rb +42 -42
- data/spec/watirspec/elements/del_spec.rb +126 -126
- data/spec/watirspec/elements/dels_spec.rb +40 -40
- data/spec/watirspec/elements/div_spec.rb +245 -245
- data/spec/watirspec/elements/divs_spec.rb +42 -42
- data/spec/watirspec/elements/dl_spec.rb +142 -142
- data/spec/watirspec/elements/dls_spec.rb +43 -43
- data/spec/watirspec/elements/dt_spec.rb +122 -122
- data/spec/watirspec/elements/dts_spec.rb +42 -42
- data/spec/watirspec/elements/element_spec.rb +483 -483
- data/spec/watirspec/elements/elements_spec.rb +47 -47
- data/spec/watirspec/elements/em_spec.rb +96 -96
- data/spec/watirspec/elements/ems_spec.rb +43 -43
- data/spec/watirspec/elements/filefield_spec.rb +170 -170
- data/spec/watirspec/elements/filefields_spec.rb +43 -43
- data/spec/watirspec/elements/font_spec.rb +29 -29
- data/spec/watirspec/elements/form_spec.rb +66 -66
- data/spec/watirspec/elements/forms_spec.rb +46 -46
- data/spec/watirspec/elements/frame_spec.rb +123 -123
- data/spec/watirspec/elements/frames_spec.rb +41 -41
- data/spec/watirspec/elements/hidden_spec.rb +104 -104
- data/spec/watirspec/elements/hiddens_spec.rb +43 -43
- data/spec/watirspec/elements/hn_spec.rb +92 -92
- data/spec/watirspec/elements/hns_spec.rb +38 -38
- data/spec/watirspec/elements/iframe_spec.rb +3 -0
- data/spec/watirspec/elements/iframes_spec.rb +47 -47
- data/spec/watirspec/elements/image_spec.rb +158 -158
- data/spec/watirspec/elements/images_spec.rb +40 -40
- data/spec/watirspec/elements/ins_spec.rb +127 -127
- data/spec/watirspec/elements/inses_spec.rb +40 -40
- data/spec/watirspec/elements/label_spec.rb +75 -75
- data/spec/watirspec/elements/labels_spec.rb +40 -40
- data/spec/watirspec/elements/li_spec.rb +111 -111
- data/spec/watirspec/elements/link_spec.rb +171 -171
- data/spec/watirspec/elements/links_spec.rb +44 -44
- data/spec/watirspec/elements/lis_spec.rb +42 -42
- data/spec/watirspec/elements/list_spec.rb +32 -32
- data/spec/watirspec/elements/map_spec.rb +75 -75
- data/spec/watirspec/elements/maps_spec.rb +41 -41
- data/spec/watirspec/elements/meta_spec.rb +23 -23
- data/spec/watirspec/elements/metas_spec.rb +40 -40
- data/spec/watirspec/elements/ol_spec.rb +87 -87
- data/spec/watirspec/elements/ols_spec.rb +40 -40
- data/spec/watirspec/elements/option_spec.rb +121 -121
- data/spec/watirspec/elements/p_spec.rb +111 -111
- data/spec/watirspec/elements/pre_spec.rb +111 -111
- data/spec/watirspec/elements/pres_spec.rb +40 -40
- data/spec/watirspec/elements/ps_spec.rb +40 -40
- data/spec/watirspec/elements/radio_spec.rb +273 -273
- data/spec/watirspec/elements/radios_spec.rb +43 -43
- data/spec/watirspec/elements/select_list_spec.rb +553 -553
- data/spec/watirspec/elements/select_lists_spec.rb +46 -46
- data/spec/watirspec/elements/span_spec.rb +128 -128
- data/spec/watirspec/elements/spans_spec.rb +40 -40
- data/spec/watirspec/elements/strong_spec.rb +88 -88
- data/spec/watirspec/elements/strongs_spec.rb +43 -43
- data/spec/watirspec/elements/table_nesting_spec.rb +51 -51
- data/spec/watirspec/elements/table_spec.rb +156 -156
- data/spec/watirspec/elements/tables_spec.rb +42 -42
- data/spec/watirspec/elements/tbody_spec.rb +93 -93
- data/spec/watirspec/elements/tbodys_spec.rb +62 -62
- data/spec/watirspec/elements/td_spec.rb +93 -93
- data/spec/watirspec/elements/tds_spec.rb +53 -53
- data/spec/watirspec/elements/text_field_spec.rb +360 -360
- data/spec/watirspec/elements/text_fields_spec.rb +44 -44
- data/spec/watirspec/elements/textarea_spec.rb +26 -26
- data/spec/watirspec/elements/textareas_spec.rb +24 -24
- data/spec/watirspec/elements/tfoot_spec.rb +86 -86
- data/spec/watirspec/elements/tfoots_spec.rb +68 -68
- data/spec/watirspec/elements/thead_spec.rb +86 -86
- data/spec/watirspec/elements/theads_spec.rb +68 -68
- data/spec/watirspec/elements/tr_spec.rb +78 -78
- data/spec/watirspec/elements/trs_spec.rb +61 -61
- data/spec/watirspec/elements/ul_spec.rb +76 -76
- data/spec/watirspec/elements/uls_spec.rb +39 -39
- data/spec/watirspec/html/alerts.html +12 -12
- data/spec/watirspec/html/aria_attributes.html +9 -9
- data/spec/watirspec/html/class_locator.html +8 -8
- data/spec/watirspec/html/clicks.html +19 -19
- data/spec/watirspec/html/closeable.html +12 -12
- data/spec/watirspec/html/collections.html +14 -14
- data/spec/watirspec/html/css/jquery-ui-1.8.17.custom.css +286 -286
- data/spec/watirspec/html/data_attributes.html +9 -9
- data/spec/watirspec/html/definition_lists.html +47 -47
- data/spec/watirspec/html/drag_and_drop.html +106 -106
- data/spec/watirspec/html/font.html +10 -10
- data/spec/watirspec/html/forms_with_input_elements.html +174 -174
- data/spec/watirspec/html/frame_1.html +22 -22
- data/spec/watirspec/html/frame_2.html +15 -15
- data/spec/watirspec/html/frames.html +10 -10
- data/spec/watirspec/html/hover.html +11 -11
- data/spec/watirspec/html/iframe_1.html +22 -22
- data/spec/watirspec/html/iframes.html +12 -12
- data/spec/watirspec/html/images.html +27 -27
- data/spec/watirspec/html/inner_outer.html +4 -4
- data/spec/watirspec/html/javascript/angular.min.js +331 -331
- data/spec/watirspec/html/javascript/helpers.js +16 -16
- data/spec/watirspec/html/javascript/jquery-1.7.1.min.js +3 -3
- data/spec/watirspec/html/javascript/jquery-ui-1.8.17.custom.min.js +67 -67
- data/spec/watirspec/html/keylogger.html +14 -14
- data/spec/watirspec/html/modal_dialog.html +9 -9
- data/spec/watirspec/html/multiple_ids.html +14 -14
- data/spec/watirspec/html/nested_elements.html +33 -33
- data/spec/watirspec/html/nested_frame_2.html +9 -9
- data/spec/watirspec/html/nested_frame_3.html +14 -14
- data/spec/watirspec/html/nested_frames.html +10 -10
- data/spec/watirspec/html/nested_iframe_2.html +12 -12
- data/spec/watirspec/html/nested_iframes.html +13 -13
- data/spec/watirspec/html/nested_tables.html +202 -202
- data/spec/watirspec/html/non_control_elements.html +140 -140
- data/spec/watirspec/html/removed_element.html +24 -24
- data/spec/watirspec/html/right_click.html +10 -10
- data/spec/watirspec/html/special_chars.html +11 -11
- data/spec/watirspec/html/tables.html +120 -120
- data/spec/watirspec/html/timeout_window_location.html +19 -19
- data/spec/watirspec/html/uneven_table.html +20 -20
- data/spec/watirspec/html/wait.html +84 -84
- data/spec/watirspec/html/window_switching.html +11 -11
- data/spec/watirspec/radio_set_spec.rb +336 -336
- data/spec/watirspec/relaxed_locate_spec.rb +99 -99
- data/spec/watirspec/screenshot_spec.rb +29 -29
- data/spec/watirspec/support/raise_exception_matchers.rb +36 -36
- data/spec/watirspec/wait_spec.rb +381 -381
- data/spec/watirspec/window_switching_spec.rb +415 -415
- data/spec/watirspec_helper.rb +161 -161
- data/support/appveyor.cmd +9 -9
- data/support/doctest_helper.rb +101 -101
- data/support/travis.sh +11 -11
- data/support/version_differ.rb +59 -59
- data/watir.gemspec +1 -1
- metadata +169 -3
data/lib/watir/alert.rb
CHANGED
@@ -1,120 +1,120 @@
|
|
1
|
-
module Watir
|
2
|
-
class Alert
|
3
|
-
include EventuallyPresent
|
4
|
-
include Waitable
|
5
|
-
|
6
|
-
def initialize(browser)
|
7
|
-
@browser = browser
|
8
|
-
@alert = nil
|
9
|
-
end
|
10
|
-
|
11
|
-
#
|
12
|
-
# Returns text of alert.
|
13
|
-
#
|
14
|
-
# @example
|
15
|
-
# browser.alert.text
|
16
|
-
# #=> "ok"
|
17
|
-
#
|
18
|
-
# @return [String]
|
19
|
-
#
|
20
|
-
|
21
|
-
def text
|
22
|
-
wait_for_exists
|
23
|
-
@alert.text
|
24
|
-
end
|
25
|
-
|
26
|
-
#
|
27
|
-
# Closes alert or accepts prompts/confirms.
|
28
|
-
#
|
29
|
-
# @example
|
30
|
-
# browser.alert.ok
|
31
|
-
# browser.alert.exists?
|
32
|
-
# #=> false
|
33
|
-
#
|
34
|
-
|
35
|
-
def ok
|
36
|
-
wait_for_exists
|
37
|
-
@alert.accept
|
38
|
-
@browser.after_hooks.run
|
39
|
-
end
|
40
|
-
|
41
|
-
#
|
42
|
-
# Closes alert or cancels prompts/confirms.
|
43
|
-
#
|
44
|
-
# @example
|
45
|
-
# browser.alert.close
|
46
|
-
# browser.alert.exists?
|
47
|
-
# #=> false
|
48
|
-
#
|
49
|
-
|
50
|
-
def close
|
51
|
-
wait_for_exists
|
52
|
-
@alert.dismiss
|
53
|
-
@browser.after_hooks.run
|
54
|
-
end
|
55
|
-
|
56
|
-
#
|
57
|
-
# Enters text to prompt.
|
58
|
-
#
|
59
|
-
# @example
|
60
|
-
# browser.alert.set "Text for prompt"
|
61
|
-
# browser.alert.ok
|
62
|
-
#
|
63
|
-
# @param [String] value
|
64
|
-
#
|
65
|
-
|
66
|
-
def set(value)
|
67
|
-
wait_for_exists
|
68
|
-
@alert.send_keys(value)
|
69
|
-
end
|
70
|
-
|
71
|
-
#
|
72
|
-
# Returns true if alert, confirm or prompt is present and false otherwise.
|
73
|
-
#
|
74
|
-
# @example
|
75
|
-
# browser.alert.exists?
|
76
|
-
# #=> true
|
77
|
-
#
|
78
|
-
|
79
|
-
def exists?
|
80
|
-
assert_exists
|
81
|
-
true
|
82
|
-
rescue Exception::UnknownObjectException
|
83
|
-
false
|
84
|
-
end
|
85
|
-
alias_method :present?, :exists?
|
86
|
-
|
87
|
-
#
|
88
|
-
# @api private
|
89
|
-
#
|
90
|
-
|
91
|
-
def selector_string
|
92
|
-
'alert'
|
93
|
-
end
|
94
|
-
|
95
|
-
private
|
96
|
-
|
97
|
-
def assert_exists
|
98
|
-
@alert = @browser.driver.switch_to.alert
|
99
|
-
rescue Selenium::WebDriver::Error::NoAlertPresentError, Selenium::WebDriver::Error::NoSuchAlertError
|
100
|
-
raise Exception::UnknownObjectException, 'unable to locate alert'
|
101
|
-
end
|
102
|
-
|
103
|
-
def wait_for_exists
|
104
|
-
return assert_exists unless Watir.relaxed_locate?
|
105
|
-
|
106
|
-
begin
|
107
|
-
wait_until(message: "waiting for alert", &:exists?)
|
108
|
-
rescue Wait::TimeoutError
|
109
|
-
unless Watir.default_timeout == 0
|
110
|
-
message = "This code has slept for the duration of the default timeout "
|
111
|
-
message << "waiting for an Alert to exist. If the test is still passing, "
|
112
|
-
message << "consider using Alert#exists? instead of rescuing UnknownObjectException"
|
113
|
-
Watir.logger.warn message
|
114
|
-
end
|
115
|
-
raise Exception::UnknownObjectException, 'unable to locate alert'
|
116
|
-
end
|
117
|
-
end
|
118
|
-
|
119
|
-
end # Alert
|
120
|
-
end # Watir
|
1
|
+
module Watir
|
2
|
+
class Alert
|
3
|
+
include EventuallyPresent
|
4
|
+
include Waitable
|
5
|
+
|
6
|
+
def initialize(browser)
|
7
|
+
@browser = browser
|
8
|
+
@alert = nil
|
9
|
+
end
|
10
|
+
|
11
|
+
#
|
12
|
+
# Returns text of alert.
|
13
|
+
#
|
14
|
+
# @example
|
15
|
+
# browser.alert.text
|
16
|
+
# #=> "ok"
|
17
|
+
#
|
18
|
+
# @return [String]
|
19
|
+
#
|
20
|
+
|
21
|
+
def text
|
22
|
+
wait_for_exists
|
23
|
+
@alert.text
|
24
|
+
end
|
25
|
+
|
26
|
+
#
|
27
|
+
# Closes alert or accepts prompts/confirms.
|
28
|
+
#
|
29
|
+
# @example
|
30
|
+
# browser.alert.ok
|
31
|
+
# browser.alert.exists?
|
32
|
+
# #=> false
|
33
|
+
#
|
34
|
+
|
35
|
+
def ok
|
36
|
+
wait_for_exists
|
37
|
+
@alert.accept
|
38
|
+
@browser.after_hooks.run
|
39
|
+
end
|
40
|
+
|
41
|
+
#
|
42
|
+
# Closes alert or cancels prompts/confirms.
|
43
|
+
#
|
44
|
+
# @example
|
45
|
+
# browser.alert.close
|
46
|
+
# browser.alert.exists?
|
47
|
+
# #=> false
|
48
|
+
#
|
49
|
+
|
50
|
+
def close
|
51
|
+
wait_for_exists
|
52
|
+
@alert.dismiss
|
53
|
+
@browser.after_hooks.run
|
54
|
+
end
|
55
|
+
|
56
|
+
#
|
57
|
+
# Enters text to prompt.
|
58
|
+
#
|
59
|
+
# @example
|
60
|
+
# browser.alert.set "Text for prompt"
|
61
|
+
# browser.alert.ok
|
62
|
+
#
|
63
|
+
# @param [String] value
|
64
|
+
#
|
65
|
+
|
66
|
+
def set(value)
|
67
|
+
wait_for_exists
|
68
|
+
@alert.send_keys(value)
|
69
|
+
end
|
70
|
+
|
71
|
+
#
|
72
|
+
# Returns true if alert, confirm or prompt is present and false otherwise.
|
73
|
+
#
|
74
|
+
# @example
|
75
|
+
# browser.alert.exists?
|
76
|
+
# #=> true
|
77
|
+
#
|
78
|
+
|
79
|
+
def exists?
|
80
|
+
assert_exists
|
81
|
+
true
|
82
|
+
rescue Exception::UnknownObjectException
|
83
|
+
false
|
84
|
+
end
|
85
|
+
alias_method :present?, :exists?
|
86
|
+
|
87
|
+
#
|
88
|
+
# @api private
|
89
|
+
#
|
90
|
+
|
91
|
+
def selector_string
|
92
|
+
'alert'
|
93
|
+
end
|
94
|
+
|
95
|
+
private
|
96
|
+
|
97
|
+
def assert_exists
|
98
|
+
@alert = @browser.driver.switch_to.alert
|
99
|
+
rescue Selenium::WebDriver::Error::NoAlertPresentError, Selenium::WebDriver::Error::NoSuchAlertError
|
100
|
+
raise Exception::UnknownObjectException, 'unable to locate alert'
|
101
|
+
end
|
102
|
+
|
103
|
+
def wait_for_exists
|
104
|
+
return assert_exists unless Watir.relaxed_locate?
|
105
|
+
|
106
|
+
begin
|
107
|
+
wait_until(message: "waiting for alert", &:exists?)
|
108
|
+
rescue Wait::TimeoutError
|
109
|
+
unless Watir.default_timeout == 0
|
110
|
+
message = "This code has slept for the duration of the default timeout "
|
111
|
+
message << "waiting for an Alert to exist. If the test is still passing, "
|
112
|
+
message << "consider using Alert#exists? instead of rescuing UnknownObjectException"
|
113
|
+
Watir.logger.warn message
|
114
|
+
end
|
115
|
+
raise Exception::UnknownObjectException, 'unable to locate alert'
|
116
|
+
end
|
117
|
+
end
|
118
|
+
|
119
|
+
end # Alert
|
120
|
+
end # Watir
|
data/lib/watir/aliases.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
|
-
module Watir
|
2
|
-
module Container
|
3
|
-
alias_method :field_set, :fieldset
|
4
|
-
alias_method :field_sets, :fieldsets
|
5
|
-
end # Container
|
6
|
-
end # Watir
|
1
|
+
module Watir
|
2
|
+
module Container
|
3
|
+
alias_method :field_set, :fieldset
|
4
|
+
alias_method :field_sets, :fieldsets
|
5
|
+
end # Container
|
6
|
+
end # Watir
|
@@ -1,96 +1,96 @@
|
|
1
|
-
module Watir
|
2
|
-
|
3
|
-
#
|
4
|
-
# @private
|
5
|
-
#
|
6
|
-
# Extended by Element, provides methods for defining attributes on the element classes.
|
7
|
-
#
|
8
|
-
|
9
|
-
module AttributeHelper
|
10
|
-
|
11
|
-
def inherit_attributes_from(kls)
|
12
|
-
kls.typed_attributes.each do |type, attrs|
|
13
|
-
attrs.each { |method, attr| attribute type, method, attr }
|
14
|
-
end
|
15
|
-
end
|
16
|
-
|
17
|
-
def typed_attributes
|
18
|
-
@typed_attributes ||= Hash.new { |hash, type| hash[type] = [] }
|
19
|
-
end
|
20
|
-
|
21
|
-
def attribute_list
|
22
|
-
@attribute_list ||= (
|
23
|
-
list = typed_attributes.values.flatten
|
24
|
-
list += ancestors[1..-1].map do |e|
|
25
|
-
e.attribute_list if e.respond_to?(:attribute_list)
|
26
|
-
end.compact.flatten
|
27
|
-
).uniq
|
28
|
-
end
|
29
|
-
|
30
|
-
alias_method :attributes, :attribute_list
|
31
|
-
|
32
|
-
#
|
33
|
-
# YARD macro to generated friendly
|
34
|
-
# documentation for attributes.
|
35
|
-
#
|
36
|
-
# @macro [attach] attribute
|
37
|
-
# @method $2
|
38
|
-
# @return [$1] value of $3 property
|
39
|
-
#
|
40
|
-
def attribute(type, method, attr)
|
41
|
-
typed_attributes[type] << [method, attr]
|
42
|
-
define_attribute(type, method, attr)
|
43
|
-
end
|
44
|
-
|
45
|
-
private
|
46
|
-
|
47
|
-
def self.extended(klass)
|
48
|
-
klass.class_eval do
|
49
|
-
# undefine deprecated methods to use them for Element attributes
|
50
|
-
[:id, :type].each { |m| undef_method m if method_defined? m }
|
51
|
-
end
|
52
|
-
end
|
53
|
-
|
54
|
-
def define_attribute(type, name, attr)
|
55
|
-
case type.to_s
|
56
|
-
when 'String'
|
57
|
-
define_string_attribute(name, attr)
|
58
|
-
when 'Boolean'
|
59
|
-
define_boolean_attribute(name, attr)
|
60
|
-
when 'Integer'
|
61
|
-
define_int_attribute(name, attr)
|
62
|
-
when 'Float'
|
63
|
-
define_float_attribute(name, attr)
|
64
|
-
else
|
65
|
-
Watir.logger.debug "treating #{type.inspect} as string for now"
|
66
|
-
end
|
67
|
-
end
|
68
|
-
|
69
|
-
def define_string_attribute(mname, aname)
|
70
|
-
define_method mname do
|
71
|
-
attribute_value(aname).to_s
|
72
|
-
end
|
73
|
-
end
|
74
|
-
|
75
|
-
def define_boolean_attribute(mname, aname)
|
76
|
-
define_method mname do
|
77
|
-
attribute_value(aname) == "true"
|
78
|
-
end
|
79
|
-
end
|
80
|
-
|
81
|
-
def define_int_attribute(mname, aname)
|
82
|
-
define_method mname do
|
83
|
-
value = attribute_value(aname)
|
84
|
-
value && Integer(value)
|
85
|
-
end
|
86
|
-
end
|
87
|
-
|
88
|
-
def define_float_attribute(mname, aname)
|
89
|
-
define_method mname do
|
90
|
-
value = attribute_value(aname)
|
91
|
-
value && Float(value)
|
92
|
-
end
|
93
|
-
end
|
94
|
-
|
95
|
-
end # AttributeHelper
|
96
|
-
end # Watir
|
1
|
+
module Watir
|
2
|
+
|
3
|
+
#
|
4
|
+
# @private
|
5
|
+
#
|
6
|
+
# Extended by Element, provides methods for defining attributes on the element classes.
|
7
|
+
#
|
8
|
+
|
9
|
+
module AttributeHelper
|
10
|
+
|
11
|
+
def inherit_attributes_from(kls)
|
12
|
+
kls.typed_attributes.each do |type, attrs|
|
13
|
+
attrs.each { |method, attr| attribute type, method, attr }
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def typed_attributes
|
18
|
+
@typed_attributes ||= Hash.new { |hash, type| hash[type] = [] }
|
19
|
+
end
|
20
|
+
|
21
|
+
def attribute_list
|
22
|
+
@attribute_list ||= (
|
23
|
+
list = typed_attributes.values.flatten
|
24
|
+
list += ancestors[1..-1].map do |e|
|
25
|
+
e.attribute_list if e.respond_to?(:attribute_list)
|
26
|
+
end.compact.flatten
|
27
|
+
).uniq
|
28
|
+
end
|
29
|
+
|
30
|
+
alias_method :attributes, :attribute_list
|
31
|
+
|
32
|
+
#
|
33
|
+
# YARD macro to generated friendly
|
34
|
+
# documentation for attributes.
|
35
|
+
#
|
36
|
+
# @macro [attach] attribute
|
37
|
+
# @method $2
|
38
|
+
# @return [$1] value of $3 property
|
39
|
+
#
|
40
|
+
def attribute(type, method, attr)
|
41
|
+
typed_attributes[type] << [method, attr]
|
42
|
+
define_attribute(type, method, attr)
|
43
|
+
end
|
44
|
+
|
45
|
+
private
|
46
|
+
|
47
|
+
def self.extended(klass)
|
48
|
+
klass.class_eval do
|
49
|
+
# undefine deprecated methods to use them for Element attributes
|
50
|
+
[:id, :type].each { |m| undef_method m if method_defined? m }
|
51
|
+
end
|
52
|
+
end
|
53
|
+
|
54
|
+
def define_attribute(type, name, attr)
|
55
|
+
case type.to_s
|
56
|
+
when 'String'
|
57
|
+
define_string_attribute(name, attr)
|
58
|
+
when 'Boolean'
|
59
|
+
define_boolean_attribute(name, attr)
|
60
|
+
when 'Integer'
|
61
|
+
define_int_attribute(name, attr)
|
62
|
+
when 'Float'
|
63
|
+
define_float_attribute(name, attr)
|
64
|
+
else
|
65
|
+
Watir.logger.debug "treating #{type.inspect} as string for now"
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
def define_string_attribute(mname, aname)
|
70
|
+
define_method mname do
|
71
|
+
attribute_value(aname).to_s
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
def define_boolean_attribute(mname, aname)
|
76
|
+
define_method mname do
|
77
|
+
attribute_value(aname) == "true"
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
def define_int_attribute(mname, aname)
|
82
|
+
define_method mname do
|
83
|
+
value = attribute_value(aname)
|
84
|
+
value && Integer(value)
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
def define_float_attribute(mname, aname)
|
89
|
+
define_method mname do
|
90
|
+
value = attribute_value(aname)
|
91
|
+
value && Float(value)
|
92
|
+
end
|
93
|
+
end
|
94
|
+
|
95
|
+
end # AttributeHelper
|
96
|
+
end # Watir
|
data/lib/watir/browser.rb
CHANGED
@@ -1,349 +1,349 @@
|
|
1
|
-
module Watir
|
2
|
-
|
3
|
-
#
|
4
|
-
# The main class through which you control the browser.
|
5
|
-
#
|
6
|
-
|
7
|
-
class Browser
|
8
|
-
include Container
|
9
|
-
include HasWindow
|
10
|
-
include Waitable
|
11
|
-
|
12
|
-
attr_writer :default_context, :original_window
|
13
|
-
attr_reader :driver
|
14
|
-
attr_reader :after_hooks
|
15
|
-
alias_method :wd, :driver # ensures duck typing with Watir::Element
|
16
|
-
|
17
|
-
class << self
|
18
|
-
#
|
19
|
-
# Creates a Watir::Browser instance and goes to URL.
|
20
|
-
#
|
21
|
-
# @example
|
22
|
-
# browser = Watir::Browser.start "www.google.com", :chrome
|
23
|
-
# #=> #<Watir::Browser:0x..fa45a499cb41e1752 url="http://www.google.com" title="Google">
|
24
|
-
#
|
25
|
-
# @param [String] url
|
26
|
-
# @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance
|
27
|
-
# @return [Watir::Browser]
|
28
|
-
#
|
29
|
-
def start(url, browser = :chrome, *args)
|
30
|
-
b = new(browser, *args)
|
31
|
-
b.goto url
|
32
|
-
|
33
|
-
b
|
34
|
-
end
|
35
|
-
end
|
36
|
-
|
37
|
-
#
|
38
|
-
# Creates a Watir::Browser instance.
|
39
|
-
#
|
40
|
-
# @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance
|
41
|
-
# @param args Passed to the underlying driver
|
42
|
-
#
|
43
|
-
|
44
|
-
def initialize(browser = :chrome, *args)
|
45
|
-
case browser
|
46
|
-
when ::Symbol, String
|
47
|
-
selenium_args = Watir::Capabilities.new(browser, *args).to_args
|
48
|
-
@driver = Selenium::WebDriver.for *selenium_args
|
49
|
-
when Selenium::WebDriver::Driver
|
50
|
-
@driver = browser
|
51
|
-
else
|
52
|
-
raise ArgumentError, "expected Symbol or Selenium::WebDriver::Driver, got #{browser.class}"
|
53
|
-
end
|
54
|
-
|
55
|
-
@after_hooks = AfterHooks.new(self)
|
56
|
-
@closed = false
|
57
|
-
@default_context = true
|
58
|
-
end
|
59
|
-
|
60
|
-
def inspect
|
61
|
-
'#<%s:0x%x url=%s title=%s>' % [self.class, hash*2, url.inspect, title.inspect]
|
62
|
-
rescue Errno::ECONNREFUSED
|
63
|
-
'#<%s:0x%x closed=true>' % [self.class, hash*2]
|
64
|
-
rescue Selenium::WebDriver::Error::UnhandledAlertError, Selenium::WebDriver::Error::UnexpectedAlertOpenError
|
65
|
-
'#<%s:0x%x alert=true>' % [self.class, hash*2]
|
66
|
-
end
|
67
|
-
alias selector_string inspect
|
68
|
-
|
69
|
-
#
|
70
|
-
# Goes to the given URL.
|
71
|
-
#
|
72
|
-
# @example
|
73
|
-
# browser.goto "watir.github.io"
|
74
|
-
#
|
75
|
-
# @param [String] uri The url.
|
76
|
-
# @return [String] The url you end up at.
|
77
|
-
#
|
78
|
-
|
79
|
-
def goto(uri)
|
80
|
-
uri = "http://#{uri}" unless uri =~ URI.regexp
|
81
|
-
|
82
|
-
@driver.navigate.to uri
|
83
|
-
@after_hooks.run
|
84
|
-
|
85
|
-
uri
|
86
|
-
end
|
87
|
-
|
88
|
-
#
|
89
|
-
# Navigates back in history.
|
90
|
-
#
|
91
|
-
|
92
|
-
def back
|
93
|
-
@driver.navigate.back
|
94
|
-
@after_hooks.run
|
95
|
-
end
|
96
|
-
|
97
|
-
#
|
98
|
-
# Navigates forward in history.
|
99
|
-
#
|
100
|
-
|
101
|
-
def forward
|
102
|
-
@driver.navigate.forward
|
103
|
-
@after_hooks.run
|
104
|
-
end
|
105
|
-
|
106
|
-
#
|
107
|
-
# Returns URL of current page.
|
108
|
-
#
|
109
|
-
# @example
|
110
|
-
# browser.goto "watir.com"
|
111
|
-
# browser.url
|
112
|
-
# #=> "http://watir.com/"
|
113
|
-
#
|
114
|
-
# @return [String]
|
115
|
-
#
|
116
|
-
|
117
|
-
def url
|
118
|
-
assert_exists
|
119
|
-
@driver.current_url
|
120
|
-
end
|
121
|
-
|
122
|
-
#
|
123
|
-
# Returns title of current page.
|
124
|
-
#
|
125
|
-
# @example
|
126
|
-
# browser.goto "watir.github.io"
|
127
|
-
# browser.title
|
128
|
-
# #=> "Watir is... – Watir Project – Watir stands for Web Application Testing In Ruby. It facilitates the writing of automated tests by mimicking the behavior of a user interacting with a website."
|
129
|
-
#
|
130
|
-
# @return [String]
|
131
|
-
#
|
132
|
-
|
133
|
-
def title
|
134
|
-
@driver.title
|
135
|
-
end
|
136
|
-
|
137
|
-
#
|
138
|
-
# Closes browser.
|
139
|
-
#
|
140
|
-
|
141
|
-
def close
|
142
|
-
return if @closed
|
143
|
-
@driver.quit
|
144
|
-
@closed = true
|
145
|
-
end
|
146
|
-
alias_method :quit, :close # TODO: close vs quit
|
147
|
-
|
148
|
-
#
|
149
|
-
# Handles cookies.
|
150
|
-
#
|
151
|
-
# @return [Watir::Cookies]
|
152
|
-
#
|
153
|
-
|
154
|
-
def cookies
|
155
|
-
@cookies ||= Cookies.new driver.manage
|
156
|
-
end
|
157
|
-
|
158
|
-
#
|
159
|
-
# Returns browser name.
|
160
|
-
#
|
161
|
-
# @example
|
162
|
-
# browser = Watir::Browser.new :chrome
|
163
|
-
# browser.name
|
164
|
-
# #=> :chrome
|
165
|
-
#
|
166
|
-
# @return [Symbol]
|
167
|
-
#
|
168
|
-
|
169
|
-
def name
|
170
|
-
@driver.browser
|
171
|
-
end
|
172
|
-
|
173
|
-
#
|
174
|
-
# Returns text of page body.
|
175
|
-
#
|
176
|
-
# @return [String]
|
177
|
-
#
|
178
|
-
|
179
|
-
def text
|
180
|
-
body.text
|
181
|
-
end
|
182
|
-
|
183
|
-
#
|
184
|
-
# Returns HTML code of current page.
|
185
|
-
#
|
186
|
-
# @return [String]
|
187
|
-
#
|
188
|
-
|
189
|
-
def html
|
190
|
-
# use body.html instead?
|
191
|
-
@driver.page_source
|
192
|
-
end
|
193
|
-
|
194
|
-
#
|
195
|
-
# Handles JavaScript alerts, confirms and prompts.
|
196
|
-
#
|
197
|
-
# @return [Watir::Alert]
|
198
|
-
#
|
199
|
-
|
200
|
-
def alert
|
201
|
-
Alert.new(self)
|
202
|
-
end
|
203
|
-
|
204
|
-
#
|
205
|
-
# Refreshes current page.
|
206
|
-
#
|
207
|
-
|
208
|
-
def refresh
|
209
|
-
@driver.navigate.refresh
|
210
|
-
@after_hooks.run
|
211
|
-
end
|
212
|
-
|
213
|
-
#
|
214
|
-
# Waits until readyState of document is complete.
|
215
|
-
#
|
216
|
-
# @example
|
217
|
-
# browser.wait
|
218
|
-
#
|
219
|
-
# @param [Integer] timeout
|
220
|
-
# @raise [Watir::Wait::TimeoutError] if timeout is exceeded
|
221
|
-
#
|
222
|
-
|
223
|
-
def wait(timeout = 5)
|
224
|
-
wait_until(timeout: timeout, message: "waiting for document.readyState == 'complete'") do
|
225
|
-
ready_state == "complete"
|
226
|
-
end
|
227
|
-
end
|
228
|
-
|
229
|
-
#
|
230
|
-
# Returns readyState of document.
|
231
|
-
#
|
232
|
-
# @return [String]
|
233
|
-
#
|
234
|
-
|
235
|
-
def ready_state
|
236
|
-
execute_script 'return document.readyState'
|
237
|
-
end
|
238
|
-
|
239
|
-
#
|
240
|
-
# Returns the text of status bar.
|
241
|
-
#
|
242
|
-
# @return [String]
|
243
|
-
#
|
244
|
-
|
245
|
-
def status
|
246
|
-
execute_script "return window.status;"
|
247
|
-
end
|
248
|
-
|
249
|
-
#
|
250
|
-
# Executes JavaScript snippet.
|
251
|
-
#
|
252
|
-
# If you are going to use the value snippet returns, make sure to use
|
253
|
-
# `return` explicitly.
|
254
|
-
#
|
255
|
-
# @example Check that Ajax requests are completed with jQuery
|
256
|
-
# browser.execute_script("return jQuery.active") == 0
|
257
|
-
# #=> true
|
258
|
-
#
|
259
|
-
# @param [String] script JavaScript snippet to execute
|
260
|
-
# @param args Arguments will be available in the given script in the 'arguments' pseudo-array
|
261
|
-
#
|
262
|
-
|
263
|
-
def execute_script(script, *args)
|
264
|
-
args.map! { |e| e.kind_of?(Watir::Element) ? e.wd : e }
|
265
|
-
returned = @driver.execute_script(script, *args)
|
266
|
-
|
267
|
-
wrap_elements_in(self, returned)
|
268
|
-
end
|
269
|
-
|
270
|
-
#
|
271
|
-
# Sends sequence of keystrokes to currently active element.
|
272
|
-
#
|
273
|
-
# @example
|
274
|
-
# browser.goto "www.google.com"
|
275
|
-
# browser.send_keys "Watir", :return
|
276
|
-
#
|
277
|
-
# @param [String, Symbol] args
|
278
|
-
#
|
279
|
-
|
280
|
-
def send_keys(*args)
|
281
|
-
@driver.switch_to.active_element.send_keys(*args)
|
282
|
-
end
|
283
|
-
|
284
|
-
#
|
285
|
-
# Handles screenshots of current pages.
|
286
|
-
#
|
287
|
-
# @return [Watir::Screenshot]
|
288
|
-
#
|
289
|
-
|
290
|
-
def screenshot
|
291
|
-
Screenshot.new driver
|
292
|
-
end
|
293
|
-
|
294
|
-
#
|
295
|
-
# Returns true if browser is not closed and false otherwise.
|
296
|
-
#
|
297
|
-
# @return [Boolean]
|
298
|
-
#
|
299
|
-
|
300
|
-
def exist?
|
301
|
-
!@closed && window.present?
|
302
|
-
end
|
303
|
-
alias_method :exists?, :exist?
|
304
|
-
|
305
|
-
#
|
306
|
-
# Protocol shared with Watir::Element
|
307
|
-
#
|
308
|
-
# @api private
|
309
|
-
#
|
310
|
-
|
311
|
-
def assert_exists
|
312
|
-
ensure_context
|
313
|
-
return if window.present?
|
314
|
-
raise Exception::NoMatchingWindowFoundException, "browser window was closed"
|
315
|
-
end
|
316
|
-
|
317
|
-
def ensure_context
|
318
|
-
raise Exception::Error, "browser was closed" if @closed
|
319
|
-
driver.switch_to.default_content unless @default_context
|
320
|
-
@default_context = true
|
321
|
-
end
|
322
|
-
|
323
|
-
def browser
|
324
|
-
self
|
325
|
-
end
|
326
|
-
|
327
|
-
private
|
328
|
-
|
329
|
-
def wrap_elements_in(scope, obj)
|
330
|
-
case obj
|
331
|
-
when Selenium::WebDriver::Element
|
332
|
-
wrap_element(scope, obj)
|
333
|
-
when Array
|
334
|
-
obj.map { |e| wrap_elements_in(scope, e) }
|
335
|
-
when Hash
|
336
|
-
obj.each { |k,v| obj[k] = wrap_elements_in(scope, v) }
|
337
|
-
|
338
|
-
obj
|
339
|
-
else
|
340
|
-
obj
|
341
|
-
end
|
342
|
-
end
|
343
|
-
|
344
|
-
def wrap_element(scope, element)
|
345
|
-
Watir.element_class_for(element.tag_name.downcase).new(scope, element: element)
|
346
|
-
end
|
347
|
-
|
348
|
-
end # Browser
|
349
|
-
end # Watir
|
1
|
+
module Watir
|
2
|
+
|
3
|
+
#
|
4
|
+
# The main class through which you control the browser.
|
5
|
+
#
|
6
|
+
|
7
|
+
class Browser
|
8
|
+
include Container
|
9
|
+
include HasWindow
|
10
|
+
include Waitable
|
11
|
+
|
12
|
+
attr_writer :default_context, :original_window
|
13
|
+
attr_reader :driver
|
14
|
+
attr_reader :after_hooks
|
15
|
+
alias_method :wd, :driver # ensures duck typing with Watir::Element
|
16
|
+
|
17
|
+
class << self
|
18
|
+
#
|
19
|
+
# Creates a Watir::Browser instance and goes to URL.
|
20
|
+
#
|
21
|
+
# @example
|
22
|
+
# browser = Watir::Browser.start "www.google.com", :chrome
|
23
|
+
# #=> #<Watir::Browser:0x..fa45a499cb41e1752 url="http://www.google.com" title="Google">
|
24
|
+
#
|
25
|
+
# @param [String] url
|
26
|
+
# @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance
|
27
|
+
# @return [Watir::Browser]
|
28
|
+
#
|
29
|
+
def start(url, browser = :chrome, *args)
|
30
|
+
b = new(browser, *args)
|
31
|
+
b.goto url
|
32
|
+
|
33
|
+
b
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
#
|
38
|
+
# Creates a Watir::Browser instance.
|
39
|
+
#
|
40
|
+
# @param [Symbol, Selenium::WebDriver] browser :firefox, :ie, :chrome, :remote or Selenium::WebDriver instance
|
41
|
+
# @param args Passed to the underlying driver
|
42
|
+
#
|
43
|
+
|
44
|
+
def initialize(browser = :chrome, *args)
|
45
|
+
case browser
|
46
|
+
when ::Symbol, String
|
47
|
+
selenium_args = Watir::Capabilities.new(browser, *args).to_args
|
48
|
+
@driver = Selenium::WebDriver.for *selenium_args
|
49
|
+
when Selenium::WebDriver::Driver
|
50
|
+
@driver = browser
|
51
|
+
else
|
52
|
+
raise ArgumentError, "expected Symbol or Selenium::WebDriver::Driver, got #{browser.class}"
|
53
|
+
end
|
54
|
+
|
55
|
+
@after_hooks = AfterHooks.new(self)
|
56
|
+
@closed = false
|
57
|
+
@default_context = true
|
58
|
+
end
|
59
|
+
|
60
|
+
def inspect
|
61
|
+
'#<%s:0x%x url=%s title=%s>' % [self.class, hash*2, url.inspect, title.inspect]
|
62
|
+
rescue Errno::ECONNREFUSED
|
63
|
+
'#<%s:0x%x closed=true>' % [self.class, hash*2]
|
64
|
+
rescue Selenium::WebDriver::Error::UnhandledAlertError, Selenium::WebDriver::Error::UnexpectedAlertOpenError
|
65
|
+
'#<%s:0x%x alert=true>' % [self.class, hash*2]
|
66
|
+
end
|
67
|
+
alias selector_string inspect
|
68
|
+
|
69
|
+
#
|
70
|
+
# Goes to the given URL.
|
71
|
+
#
|
72
|
+
# @example
|
73
|
+
# browser.goto "watir.github.io"
|
74
|
+
#
|
75
|
+
# @param [String] uri The url.
|
76
|
+
# @return [String] The url you end up at.
|
77
|
+
#
|
78
|
+
|
79
|
+
def goto(uri)
|
80
|
+
uri = "http://#{uri}" unless uri =~ URI.regexp
|
81
|
+
|
82
|
+
@driver.navigate.to uri
|
83
|
+
@after_hooks.run
|
84
|
+
|
85
|
+
uri
|
86
|
+
end
|
87
|
+
|
88
|
+
#
|
89
|
+
# Navigates back in history.
|
90
|
+
#
|
91
|
+
|
92
|
+
def back
|
93
|
+
@driver.navigate.back
|
94
|
+
@after_hooks.run
|
95
|
+
end
|
96
|
+
|
97
|
+
#
|
98
|
+
# Navigates forward in history.
|
99
|
+
#
|
100
|
+
|
101
|
+
def forward
|
102
|
+
@driver.navigate.forward
|
103
|
+
@after_hooks.run
|
104
|
+
end
|
105
|
+
|
106
|
+
#
|
107
|
+
# Returns URL of current page.
|
108
|
+
#
|
109
|
+
# @example
|
110
|
+
# browser.goto "watir.com"
|
111
|
+
# browser.url
|
112
|
+
# #=> "http://watir.com/"
|
113
|
+
#
|
114
|
+
# @return [String]
|
115
|
+
#
|
116
|
+
|
117
|
+
def url
|
118
|
+
assert_exists
|
119
|
+
@driver.current_url
|
120
|
+
end
|
121
|
+
|
122
|
+
#
|
123
|
+
# Returns title of current page.
|
124
|
+
#
|
125
|
+
# @example
|
126
|
+
# browser.goto "watir.github.io"
|
127
|
+
# browser.title
|
128
|
+
# #=> "Watir is... – Watir Project – Watir stands for Web Application Testing In Ruby. It facilitates the writing of automated tests by mimicking the behavior of a user interacting with a website."
|
129
|
+
#
|
130
|
+
# @return [String]
|
131
|
+
#
|
132
|
+
|
133
|
+
def title
|
134
|
+
@driver.title
|
135
|
+
end
|
136
|
+
|
137
|
+
#
|
138
|
+
# Closes browser.
|
139
|
+
#
|
140
|
+
|
141
|
+
def close
|
142
|
+
return if @closed
|
143
|
+
@driver.quit
|
144
|
+
@closed = true
|
145
|
+
end
|
146
|
+
alias_method :quit, :close # TODO: close vs quit
|
147
|
+
|
148
|
+
#
|
149
|
+
# Handles cookies.
|
150
|
+
#
|
151
|
+
# @return [Watir::Cookies]
|
152
|
+
#
|
153
|
+
|
154
|
+
def cookies
|
155
|
+
@cookies ||= Cookies.new driver.manage
|
156
|
+
end
|
157
|
+
|
158
|
+
#
|
159
|
+
# Returns browser name.
|
160
|
+
#
|
161
|
+
# @example
|
162
|
+
# browser = Watir::Browser.new :chrome
|
163
|
+
# browser.name
|
164
|
+
# #=> :chrome
|
165
|
+
#
|
166
|
+
# @return [Symbol]
|
167
|
+
#
|
168
|
+
|
169
|
+
def name
|
170
|
+
@driver.browser
|
171
|
+
end
|
172
|
+
|
173
|
+
#
|
174
|
+
# Returns text of page body.
|
175
|
+
#
|
176
|
+
# @return [String]
|
177
|
+
#
|
178
|
+
|
179
|
+
def text
|
180
|
+
body.text
|
181
|
+
end
|
182
|
+
|
183
|
+
#
|
184
|
+
# Returns HTML code of current page.
|
185
|
+
#
|
186
|
+
# @return [String]
|
187
|
+
#
|
188
|
+
|
189
|
+
def html
|
190
|
+
# use body.html instead?
|
191
|
+
@driver.page_source
|
192
|
+
end
|
193
|
+
|
194
|
+
#
|
195
|
+
# Handles JavaScript alerts, confirms and prompts.
|
196
|
+
#
|
197
|
+
# @return [Watir::Alert]
|
198
|
+
#
|
199
|
+
|
200
|
+
def alert
|
201
|
+
Alert.new(self)
|
202
|
+
end
|
203
|
+
|
204
|
+
#
|
205
|
+
# Refreshes current page.
|
206
|
+
#
|
207
|
+
|
208
|
+
def refresh
|
209
|
+
@driver.navigate.refresh
|
210
|
+
@after_hooks.run
|
211
|
+
end
|
212
|
+
|
213
|
+
#
|
214
|
+
# Waits until readyState of document is complete.
|
215
|
+
#
|
216
|
+
# @example
|
217
|
+
# browser.wait
|
218
|
+
#
|
219
|
+
# @param [Integer] timeout
|
220
|
+
# @raise [Watir::Wait::TimeoutError] if timeout is exceeded
|
221
|
+
#
|
222
|
+
|
223
|
+
def wait(timeout = 5)
|
224
|
+
wait_until(timeout: timeout, message: "waiting for document.readyState == 'complete'") do
|
225
|
+
ready_state == "complete"
|
226
|
+
end
|
227
|
+
end
|
228
|
+
|
229
|
+
#
|
230
|
+
# Returns readyState of document.
|
231
|
+
#
|
232
|
+
# @return [String]
|
233
|
+
#
|
234
|
+
|
235
|
+
def ready_state
|
236
|
+
execute_script 'return document.readyState'
|
237
|
+
end
|
238
|
+
|
239
|
+
#
|
240
|
+
# Returns the text of status bar.
|
241
|
+
#
|
242
|
+
# @return [String]
|
243
|
+
#
|
244
|
+
|
245
|
+
def status
|
246
|
+
execute_script "return window.status;"
|
247
|
+
end
|
248
|
+
|
249
|
+
#
|
250
|
+
# Executes JavaScript snippet.
|
251
|
+
#
|
252
|
+
# If you are going to use the value snippet returns, make sure to use
|
253
|
+
# `return` explicitly.
|
254
|
+
#
|
255
|
+
# @example Check that Ajax requests are completed with jQuery
|
256
|
+
# browser.execute_script("return jQuery.active") == 0
|
257
|
+
# #=> true
|
258
|
+
#
|
259
|
+
# @param [String] script JavaScript snippet to execute
|
260
|
+
# @param args Arguments will be available in the given script in the 'arguments' pseudo-array
|
261
|
+
#
|
262
|
+
|
263
|
+
def execute_script(script, *args)
|
264
|
+
args.map! { |e| e.kind_of?(Watir::Element) ? e.wd : e }
|
265
|
+
returned = @driver.execute_script(script, *args)
|
266
|
+
|
267
|
+
wrap_elements_in(self, returned)
|
268
|
+
end
|
269
|
+
|
270
|
+
#
|
271
|
+
# Sends sequence of keystrokes to currently active element.
|
272
|
+
#
|
273
|
+
# @example
|
274
|
+
# browser.goto "www.google.com"
|
275
|
+
# browser.send_keys "Watir", :return
|
276
|
+
#
|
277
|
+
# @param [String, Symbol] args
|
278
|
+
#
|
279
|
+
|
280
|
+
def send_keys(*args)
|
281
|
+
@driver.switch_to.active_element.send_keys(*args)
|
282
|
+
end
|
283
|
+
|
284
|
+
#
|
285
|
+
# Handles screenshots of current pages.
|
286
|
+
#
|
287
|
+
# @return [Watir::Screenshot]
|
288
|
+
#
|
289
|
+
|
290
|
+
def screenshot
|
291
|
+
Screenshot.new driver
|
292
|
+
end
|
293
|
+
|
294
|
+
#
|
295
|
+
# Returns true if browser is not closed and false otherwise.
|
296
|
+
#
|
297
|
+
# @return [Boolean]
|
298
|
+
#
|
299
|
+
|
300
|
+
def exist?
|
301
|
+
!@closed && window.present?
|
302
|
+
end
|
303
|
+
alias_method :exists?, :exist?
|
304
|
+
|
305
|
+
#
|
306
|
+
# Protocol shared with Watir::Element
|
307
|
+
#
|
308
|
+
# @api private
|
309
|
+
#
|
310
|
+
|
311
|
+
def assert_exists
|
312
|
+
ensure_context
|
313
|
+
return if window.present?
|
314
|
+
raise Exception::NoMatchingWindowFoundException, "browser window was closed"
|
315
|
+
end
|
316
|
+
|
317
|
+
def ensure_context
|
318
|
+
raise Exception::Error, "browser was closed" if @closed
|
319
|
+
driver.switch_to.default_content unless @default_context
|
320
|
+
@default_context = true
|
321
|
+
end
|
322
|
+
|
323
|
+
def browser
|
324
|
+
self
|
325
|
+
end
|
326
|
+
|
327
|
+
private
|
328
|
+
|
329
|
+
def wrap_elements_in(scope, obj)
|
330
|
+
case obj
|
331
|
+
when Selenium::WebDriver::Element
|
332
|
+
wrap_element(scope, obj)
|
333
|
+
when Array
|
334
|
+
obj.map { |e| wrap_elements_in(scope, e) }
|
335
|
+
when Hash
|
336
|
+
obj.each { |k,v| obj[k] = wrap_elements_in(scope, v) }
|
337
|
+
|
338
|
+
obj
|
339
|
+
else
|
340
|
+
obj
|
341
|
+
end
|
342
|
+
end
|
343
|
+
|
344
|
+
def wrap_element(scope, element)
|
345
|
+
Watir.element_class_for(element.tag_name.downcase).new(scope, element: element)
|
346
|
+
end
|
347
|
+
|
348
|
+
end # Browser
|
349
|
+
end # Watir
|