page_object 1.1.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +7 -0
- data/.rspec +2 -0
- data/.ruby-gemset +1 -0
- data/.ruby-version +1 -0
- data/.travis.yml +7 -0
- data/ChangeLog +809 -0
- data/Gemfile +14 -0
- data/Guardfile +20 -0
- data/LICENSE +20 -0
- data/README.md +111 -0
- data/Rakefile +35 -0
- data/cucumber.yml +10 -0
- data/features/area.feature +35 -0
- data/features/async.feature +30 -0
- data/features/audio.feature +68 -0
- data/features/bold.feature +21 -0
- data/features/button.feature +98 -0
- data/features/canvas.feature +37 -0
- data/features/check_box.feature +55 -0
- data/features/div.feature +45 -0
- data/features/element.feature +322 -0
- data/features/file_field.feature +40 -0
- data/features/form.feature +43 -0
- data/features/frames.feature +75 -0
- data/features/generic_elements.feature +29 -0
- data/features/gxt_table_extension.feature +24 -0
- data/features/headings.feature +97 -0
- data/features/hidden_field.feature +45 -0
- data/features/html/04-Death_Becomes_Fur.mp4 +0 -0
- data/features/html/04-Death_Becomes_Fur.oga +0 -0
- data/features/html/async.html +36 -0
- data/features/html/double_click.html +13 -0
- data/features/html/failure.html +8 -0
- data/features/html/frame_1.html +18 -0
- data/features/html/frame_2.html +16 -0
- data/features/html/frame_3.html +14 -0
- data/features/html/frames.html +12 -0
- data/features/html/hover.html +12 -0
- data/features/html/iframes.html +12 -0
- data/features/html/images/circle.png +0 -0
- data/features/html/images/img_pulpit.jpg +0 -0
- data/features/html/images/submit.gif +0 -0
- data/features/html/indexed_property.html +55 -0
- data/features/html/modal.html +17 -0
- data/features/html/modal_1.html +38 -0
- data/features/html/modal_2.html +27 -0
- data/features/html/movie.mp4 +0 -0
- data/features/html/movie.ogg +0 -0
- data/features/html/multi_elements.html +144 -0
- data/features/html/nested_elements.html +77 -0
- data/features/html/nested_frame_1.html +1 -0
- data/features/html/nested_frame_2.html +11 -0
- data/features/html/nested_frame_3.html +14 -0
- data/features/html/nested_frames.html +10 -0
- data/features/html/planets.gif +0 -0
- data/features/html/static_elements.html +207 -0
- data/features/html/success.html +8 -0
- data/features/html/sun.gif +0 -0
- data/features/html/sun.html +7 -0
- data/features/image.feature +50 -0
- data/features/indexed_property.feature +117 -0
- data/features/javascript.feature +28 -0
- data/features/label.feature +46 -0
- data/features/link.feature +52 -0
- data/features/list_item.feature +36 -0
- data/features/modal_dialog.feature +15 -0
- data/features/multi_elements.feature +492 -0
- data/features/nested_elements.feature +117 -0
- data/features/ordered_list.feature +56 -0
- data/features/page_level_actions.feature +90 -0
- data/features/paragraph.feature +35 -0
- data/features/radio_button.feature +58 -0
- data/features/radio_button_group.feature +29 -0
- data/features/sample-app/public/jquery-1.3.2.js +4376 -0
- data/features/sample-app/public/jquery.html +30 -0
- data/features/sample-app/public/prototype-1.6.0.3.js +4320 -0
- data/features/sample-app/public/prototype.html +35 -0
- data/features/sample-app/sample_app.rb +35 -0
- data/features/section.feature +132 -0
- data/features/select_list.feature +86 -0
- data/features/span.feature +37 -0
- data/features/step_definitions/accessor_steps.rb +64 -0
- data/features/step_definitions/area_steps.rb +19 -0
- data/features/step_definitions/async_steps.rb +83 -0
- data/features/step_definitions/audio_steps.rb +27 -0
- data/features/step_definitions/bold_steps.rb +12 -0
- data/features/step_definitions/button_steps.rb +48 -0
- data/features/step_definitions/canvas_steps.rb +15 -0
- data/features/step_definitions/check_box_steps.rb +35 -0
- data/features/step_definitions/div_steps.rb +19 -0
- data/features/step_definitions/element_steps.rb +266 -0
- data/features/step_definitions/file_field_steps.rb +19 -0
- data/features/step_definitions/form_steps.rb +19 -0
- data/features/step_definitions/frames_steps.rb +159 -0
- data/features/step_definitions/generic_element_steps.rb +31 -0
- data/features/step_definitions/gxt_table_steps.rb +58 -0
- data/features/step_definitions/headings_steps.rb +12 -0
- data/features/step_definitions/hidden_field_steps.rb +27 -0
- data/features/step_definitions/image_steps.rb +27 -0
- data/features/step_definitions/indexed_property_steps.rb +163 -0
- data/features/step_definitions/javascript_steps.rb +53 -0
- data/features/step_definitions/label_steps.rb +19 -0
- data/features/step_definitions/link_steps.rb +40 -0
- data/features/step_definitions/list_item_steps.rb +19 -0
- data/features/step_definitions/modal_dialog_steps.rb +62 -0
- data/features/step_definitions/multi_elements_steps.rb +541 -0
- data/features/step_definitions/nested_elements_steps.rb +212 -0
- data/features/step_definitions/ordered_list_steps.rb +23 -0
- data/features/step_definitions/page_level_actions_steps.rb +135 -0
- data/features/step_definitions/page_traversal_steps.rb +4 -0
- data/features/step_definitions/paragraph_steps.rb +28 -0
- data/features/step_definitions/radio_button_group_steps.rb +36 -0
- data/features/step_definitions/radio_button_steps.rb +27 -0
- data/features/step_definitions/section_steps.rb +268 -0
- data/features/step_definitions/select_list_steps.rb +65 -0
- data/features/step_definitions/span_steps.rb +19 -0
- data/features/step_definitions/table_cell_steps.rb +15 -0
- data/features/step_definitions/table_row_steps.rb +23 -0
- data/features/step_definitions/table_steps.rb +70 -0
- data/features/step_definitions/text_area_steps.rb +35 -0
- data/features/step_definitions/text_field_steps.rb +35 -0
- data/features/step_definitions/unordered_list_steps.rb +23 -0
- data/features/step_definitions/video_steps.rb +45 -0
- data/features/support/ajax_text_environment.rb +26 -0
- data/features/support/env.rb +8 -0
- data/features/support/hooks.rb +8 -0
- data/features/support/page.rb +382 -0
- data/features/support/persistent_browser.rb +70 -0
- data/features/support/targets/firefox14_osx.rb +6 -0
- data/features/support/targets/firefox14_windows7.rb +6 -0
- data/features/support/url_helper.rb +57 -0
- data/features/table.feature +122 -0
- data/features/table_cell.feature +45 -0
- data/features/table_row.feature +43 -0
- data/features/text_area.feature +51 -0
- data/features/text_field.feature +70 -0
- data/features/unordered_list.feature +56 -0
- data/features/video.feature +73 -0
- data/lib/page-object.rb +421 -0
- data/lib/page-object/accessors.rb +1412 -0
- data/lib/page-object/core_ext/string.rb +5 -0
- data/lib/page-object/element_locators.rb +21 -0
- data/lib/page-object/elements.rb +60 -0
- data/lib/page-object/elements/area.rb +31 -0
- data/lib/page-object/elements/audio.rb +9 -0
- data/lib/page-object/elements/bold.rb +11 -0
- data/lib/page-object/elements/button.rb +35 -0
- data/lib/page-object/elements/canvas.rb +23 -0
- data/lib/page-object/elements/check_box.rb +37 -0
- data/lib/page-object/elements/div.rb +19 -0
- data/lib/page-object/elements/element.rb +226 -0
- data/lib/page-object/elements/file_field.rb +38 -0
- data/lib/page-object/elements/form.rb +36 -0
- data/lib/page-object/elements/heading.rb +15 -0
- data/lib/page-object/elements/hidden_field.rb +22 -0
- data/lib/page-object/elements/image.rb +36 -0
- data/lib/page-object/elements/label.rb +19 -0
- data/lib/page-object/elements/link.rb +46 -0
- data/lib/page-object/elements/list_item.rb +19 -0
- data/lib/page-object/elements/media.rb +45 -0
- data/lib/page-object/elements/option.rb +10 -0
- data/lib/page-object/elements/ordered_list.rb +50 -0
- data/lib/page-object/elements/paragraph.rb +9 -0
- data/lib/page-object/elements/radio_button.rb +37 -0
- data/lib/page-object/elements/select_list.rb +42 -0
- data/lib/page-object/elements/span.rb +19 -0
- data/lib/page-object/elements/table.rb +79 -0
- data/lib/page-object/elements/table_cell.rb +28 -0
- data/lib/page-object/elements/table_row.rb +50 -0
- data/lib/page-object/elements/text_area.rb +38 -0
- data/lib/page-object/elements/text_field.rb +42 -0
- data/lib/page-object/elements/unordered_list.rb +51 -0
- data/lib/page-object/elements/video.rb +18 -0
- data/lib/page-object/indexed_properties.rb +40 -0
- data/lib/page-object/javascript/angularjs.rb +14 -0
- data/lib/page-object/javascript/jquery.rb +14 -0
- data/lib/page-object/javascript/prototype.rb +14 -0
- data/lib/page-object/javascript/yui.rb +19 -0
- data/lib/page-object/javascript_framework_facade.rb +80 -0
- data/lib/page-object/loads_platform.rb +45 -0
- data/lib/page-object/locator_generator.rb +131 -0
- data/lib/page-object/nested_elements.rb +17 -0
- data/lib/page-object/page_factory.rb +108 -0
- data/lib/page-object/page_populator.rb +83 -0
- data/lib/page-object/platforms.rb +18 -0
- data/lib/page-object/platforms/selenium_webdriver.rb +30 -0
- data/lib/page-object/platforms/selenium_webdriver/button.rb +15 -0
- data/lib/page-object/platforms/selenium_webdriver/check_box.rb +29 -0
- data/lib/page-object/platforms/selenium_webdriver/element.rb +291 -0
- data/lib/page-object/platforms/selenium_webdriver/file_field.rb +16 -0
- data/lib/page-object/platforms/selenium_webdriver/form.rb +16 -0
- data/lib/page-object/platforms/selenium_webdriver/image.rb +28 -0
- data/lib/page-object/platforms/selenium_webdriver/link.rb +23 -0
- data/lib/page-object/platforms/selenium_webdriver/ordered_list.rb +39 -0
- data/lib/page-object/platforms/selenium_webdriver/page_object.rb +1237 -0
- data/lib/page-object/platforms/selenium_webdriver/radio_button.rb +22 -0
- data/lib/page-object/platforms/selenium_webdriver/select_list.rb +93 -0
- data/lib/page-object/platforms/selenium_webdriver/surrogate_selenium_element.rb +42 -0
- data/lib/page-object/platforms/selenium_webdriver/table.rb +42 -0
- data/lib/page-object/platforms/selenium_webdriver/table_row.rb +43 -0
- data/lib/page-object/platforms/selenium_webdriver/text_area.rb +17 -0
- data/lib/page-object/platforms/selenium_webdriver/text_field.rb +17 -0
- data/lib/page-object/platforms/selenium_webdriver/unordered_list.rb +39 -0
- data/lib/page-object/platforms/watir_webdriver.rb +30 -0
- data/lib/page-object/platforms/watir_webdriver/check_box.rb +29 -0
- data/lib/page-object/platforms/watir_webdriver/element.rb +249 -0
- data/lib/page-object/platforms/watir_webdriver/file_field.rb +16 -0
- data/lib/page-object/platforms/watir_webdriver/form.rb +16 -0
- data/lib/page-object/platforms/watir_webdriver/image.rb +22 -0
- data/lib/page-object/platforms/watir_webdriver/link.rb +15 -0
- data/lib/page-object/platforms/watir_webdriver/ordered_list.rb +35 -0
- data/lib/page-object/platforms/watir_webdriver/page_object.rb +1082 -0
- data/lib/page-object/platforms/watir_webdriver/radio_button.rb +22 -0
- data/lib/page-object/platforms/watir_webdriver/select_list.rb +74 -0
- data/lib/page-object/platforms/watir_webdriver/table.rb +38 -0
- data/lib/page-object/platforms/watir_webdriver/table_row.rb +37 -0
- data/lib/page-object/platforms/watir_webdriver/text_area.rb +23 -0
- data/lib/page-object/platforms/watir_webdriver/text_field.rb +16 -0
- data/lib/page-object/platforms/watir_webdriver/unordered_list.rb +36 -0
- data/lib/page-object/sections.rb +29 -0
- data/lib/page-object/version.rb +4 -0
- data/lib/page-object/widgets.rb +133 -0
- data/page_object.gemspec +31 -0
- data/pageobject.gems +1 -0
- data/spec/page-object/accessors_spec.rb +40 -0
- data/spec/page-object/element_locators_spec.rb +1122 -0
- data/spec/page-object/elements/area_spec.rb +45 -0
- data/spec/page-object/elements/bold_spec.rb +29 -0
- data/spec/page-object/elements/button_spec.rb +64 -0
- data/spec/page-object/elements/canvas_spec.rb +40 -0
- data/spec/page-object/elements/check_box_spec.rb +49 -0
- data/spec/page-object/elements/div_spec.rb +28 -0
- data/spec/page-object/elements/element_spec.rb +114 -0
- data/spec/page-object/elements/file_field_spec.rb +39 -0
- data/spec/page-object/elements/form_spec.rb +28 -0
- data/spec/page-object/elements/heading_spec.rb +48 -0
- data/spec/page-object/elements/hidden_field_spec.rb +28 -0
- data/spec/page-object/elements/image_spec.rb +66 -0
- data/spec/page-object/elements/label_spec.rb +29 -0
- data/spec/page-object/elements/link_spec.rb +49 -0
- data/spec/page-object/elements/list_item_spec.rb +29 -0
- data/spec/page-object/elements/nested_element_spec.rb +254 -0
- data/spec/page-object/elements/option_spec.rb +11 -0
- data/spec/page-object/elements/ordered_list_spec.rb +94 -0
- data/spec/page-object/elements/paragraph_spec.rb +27 -0
- data/spec/page-object/elements/select_list_spec.rb +142 -0
- data/spec/page-object/elements/selenium/radio_button_spec.rb +44 -0
- data/spec/page-object/elements/selenium/text_field_spec.rb +49 -0
- data/spec/page-object/elements/selenium_element_spec.rb +177 -0
- data/spec/page-object/elements/span_spec.rb +26 -0
- data/spec/page-object/elements/table_cell_spec.rb +21 -0
- data/spec/page-object/elements/table_row_spec.rb +70 -0
- data/spec/page-object/elements/table_spec.rb +98 -0
- data/spec/page-object/elements/text_area_spec.rb +39 -0
- data/spec/page-object/elements/unordered_list_spec.rb +94 -0
- data/spec/page-object/elements/watir_element_spec.rb +145 -0
- data/spec/page-object/javascript_framework_facade_spec.rb +61 -0
- data/spec/page-object/loads_platform_spec.rb +53 -0
- data/spec/page-object/page-object_spec.rb +407 -0
- data/spec/page-object/page_factory_spec.rb +238 -0
- data/spec/page-object/page_populator_spec.rb +122 -0
- data/spec/page-object/page_section_spec.rb +73 -0
- data/spec/page-object/platforms/selenium_webdriver/selenium_page_object_spec.rb +68 -0
- data/spec/page-object/platforms/selenium_webdriver_spec.rb +29 -0
- data/spec/page-object/platforms/watir_webdriver/watir_page_object_spec.rb +29 -0
- data/spec/page-object/platforms/watir_webdriver_spec.rb +9 -0
- data/spec/page-object/selenium_accessors_spec.rb +609 -0
- data/spec/page-object/watir_accessors_spec.rb +1134 -0
- data/spec/page-object/widget_spec.rb +226 -0
- data/spec/spec_helper.rb +47 -0
- metadata +601 -0
@@ -0,0 +1,56 @@
|
|
1
|
+
Feature: Unordered list
|
2
|
+
|
3
|
+
Background:
|
4
|
+
Given I am on the static elements page
|
5
|
+
|
6
|
+
Scenario: Getting the first element from the unordered list
|
7
|
+
When I get the first item from the unordered list
|
8
|
+
Then the list items text should be "Item One"
|
9
|
+
|
10
|
+
Scenario Outline: Locating unordered lists on the page
|
11
|
+
When I search for the unordered list by "<search_by>"
|
12
|
+
And I get the first item from the list
|
13
|
+
Then the list items text should be "Item One"
|
14
|
+
And the list should contain 3 items
|
15
|
+
And each item should contain "Item"
|
16
|
+
|
17
|
+
Scenarios:
|
18
|
+
| search_by |
|
19
|
+
| id |
|
20
|
+
| class |
|
21
|
+
| xpath |
|
22
|
+
| index |
|
23
|
+
| name |
|
24
|
+
|
25
|
+
@selenium_only
|
26
|
+
Scenario Outline: Locating unordered lists on the page
|
27
|
+
When I search for the unordered list by "<search_by>"
|
28
|
+
And I get the first item from the list
|
29
|
+
Then the list items text should be "Item One"
|
30
|
+
And the list should contain 3 items
|
31
|
+
And each item should contain "Item"
|
32
|
+
|
33
|
+
Scenarios:
|
34
|
+
| search_by |
|
35
|
+
| css |
|
36
|
+
|
37
|
+
Scenario Outline: Locating unordered lists using multiple parameters
|
38
|
+
When I search for the unordered list by "<param1>" and "<param2>"
|
39
|
+
And I get the first item from the list
|
40
|
+
Then the list items text should be "Item One"
|
41
|
+
|
42
|
+
Scenarios:
|
43
|
+
| param1 | param2 |
|
44
|
+
| class | index |
|
45
|
+
| name | index |
|
46
|
+
|
47
|
+
Scenario: Finding a unordered list dynamically
|
48
|
+
When I search for the unordered list while the script is executing
|
49
|
+
Then I should see that the unordered list exists
|
50
|
+
When I get the first item from the list
|
51
|
+
Then the list items text should be "Item One"
|
52
|
+
|
53
|
+
Scenario: Getting the text from an unordered list
|
54
|
+
Then the text for the unordered list should contain "Item One"
|
55
|
+
And the text for the unordered list should contain "Item Two"
|
56
|
+
And the text for the unordered list should contain "Item Three"
|
@@ -0,0 +1,73 @@
|
|
1
|
+
Feature: Support for video element
|
2
|
+
|
3
|
+
Background:
|
4
|
+
Given I am on the static elements page
|
5
|
+
|
6
|
+
Scenario: finding an video element
|
7
|
+
When I retrieve the video element
|
8
|
+
Then I should know it exists
|
9
|
+
And I should know it is visible
|
10
|
+
|
11
|
+
Scenario Outline: Locating a video element on the page
|
12
|
+
When I search for the video element by "<search_by>"
|
13
|
+
Then I should know it is visible
|
14
|
+
|
15
|
+
Examples:
|
16
|
+
| search_by |
|
17
|
+
| id |
|
18
|
+
| class |
|
19
|
+
| name |
|
20
|
+
| xpath |
|
21
|
+
| index |
|
22
|
+
| css |
|
23
|
+
|
24
|
+
Scenario Outline: Locating videos using multiple parameters
|
25
|
+
When I search for the video element by "<param1>" and "<param2>"
|
26
|
+
Then I should know it is visible
|
27
|
+
|
28
|
+
Examples:
|
29
|
+
| param1 | param2 |
|
30
|
+
| class | index |
|
31
|
+
| name | index |
|
32
|
+
|
33
|
+
Scenario: Should know if it is autoplay
|
34
|
+
When I retrieve the video element
|
35
|
+
Then I should know the video is not autoplay
|
36
|
+
|
37
|
+
Scenario: Should know if the controls are displayed
|
38
|
+
When I retrieve the video element
|
39
|
+
Then I should know that the controls are displayed
|
40
|
+
|
41
|
+
Scenario: Should know if it is paused
|
42
|
+
When I retrieve the video element
|
43
|
+
Then I should know that the video is paused
|
44
|
+
|
45
|
+
Scenario: Should know its duration
|
46
|
+
When I retrieve the video element
|
47
|
+
Then I should know that the duration is greater than 12 seconds
|
48
|
+
|
49
|
+
Scenario: Should know its volume
|
50
|
+
When I retrieve the video element
|
51
|
+
Then I should know that its volume is 1
|
52
|
+
|
53
|
+
Scenario: Should know its height and width
|
54
|
+
When I retrieve the video element
|
55
|
+
Then I should know that its height is 240 pixels
|
56
|
+
And I should knot what its width is 320 pixels
|
57
|
+
|
58
|
+
Scenario: Should know if it has ended
|
59
|
+
When I retrieve the video element
|
60
|
+
Then I should know that it has not ended
|
61
|
+
|
62
|
+
Scenario: Should know if it is seeking
|
63
|
+
When I retrieve the video element
|
64
|
+
Then I should know that it is not seeking
|
65
|
+
|
66
|
+
Scenario: Should know if it is in a loop
|
67
|
+
When I retrieve the video element
|
68
|
+
Then I should know that it is not in a loop
|
69
|
+
|
70
|
+
Scenario: Should know if it is muted
|
71
|
+
When I retrieve the video element
|
72
|
+
Then I should know that it is muted
|
73
|
+
|
data/lib/page-object.rb
ADDED
@@ -0,0 +1,421 @@
|
|
1
|
+
require 'page-object/version'
|
2
|
+
require 'page-object/accessors'
|
3
|
+
require 'page-object/platforms'
|
4
|
+
require 'page-object/element_locators'
|
5
|
+
require 'page-object/nested_elements'
|
6
|
+
require 'page-object/page_factory'
|
7
|
+
require 'page-object/page_populator'
|
8
|
+
require 'page-object/javascript_framework_facade'
|
9
|
+
require 'page-object/indexed_properties'
|
10
|
+
require 'page-object/sections'
|
11
|
+
require 'page-object/widgets'
|
12
|
+
|
13
|
+
require 'watir-webdriver'
|
14
|
+
require 'page-object/platforms/watir_webdriver/element'
|
15
|
+
require 'page-object/platforms/watir_webdriver/page_object'
|
16
|
+
require 'page-object/platforms/selenium_webdriver/element'
|
17
|
+
require 'page-object/platforms/selenium_webdriver/page_object'
|
18
|
+
|
19
|
+
require 'selenium/webdriver/common/error'
|
20
|
+
#
|
21
|
+
# Module that when included adds functionality to a page object. This module
|
22
|
+
# will add numerous class and instance methods that you use to define and
|
23
|
+
# interact with web pages.
|
24
|
+
#
|
25
|
+
# If we have a login page with a username and password textfield and a login
|
26
|
+
# button we might define our page like the one below. We can then interact with
|
27
|
+
# the object using the generated methods.
|
28
|
+
#
|
29
|
+
# @example Login page example
|
30
|
+
# class LoginPage
|
31
|
+
# include PageObject
|
32
|
+
#
|
33
|
+
# text_field(:username, :id => 'user')
|
34
|
+
# text_field(:password, :id => 'pass')
|
35
|
+
# button(:login, :value => 'Login')
|
36
|
+
# end
|
37
|
+
#
|
38
|
+
# ...
|
39
|
+
#
|
40
|
+
# browser = Watir::Browser.new :firefox
|
41
|
+
# login_page = LoginPage.new(browser)
|
42
|
+
# login_page.username = 'cheezy'
|
43
|
+
# login_page.password = 'secret'
|
44
|
+
# login_page.login
|
45
|
+
#
|
46
|
+
# @see PageObject::Accessors to see what class level methods are added to this module at runtime.
|
47
|
+
#
|
48
|
+
module PageObject
|
49
|
+
include LoadsPlatform
|
50
|
+
include ElementLocators
|
51
|
+
include PagePopulator
|
52
|
+
|
53
|
+
# @return [Watir::Browser or Selenium::WebDriver::Driver] the platform browser passed to the constructor
|
54
|
+
attr_reader :browser
|
55
|
+
# @return [PageObject::WatirPageObject or PageObject::SeleniumPageObject] the platform page object
|
56
|
+
attr_reader :platform
|
57
|
+
|
58
|
+
#
|
59
|
+
# Construct a new page object. Prior to browser initialization it will call
|
60
|
+
# a method named initialize_accessors if it exists. Upon initialization of
|
61
|
+
# the page it will call a method named initialize_page if it exists.
|
62
|
+
#
|
63
|
+
# @param [Watir::Browser, Watir::HTMLElement or Selenium::WebDriver::Driver, Selenium::WebDriver::Element] the platform browser/element to use
|
64
|
+
# @param [bool] open the page if page_url is set
|
65
|
+
#
|
66
|
+
def initialize(root, visit=false)
|
67
|
+
initialize_accessors if respond_to?(:initialize_accessors)
|
68
|
+
initialize_browser(root)
|
69
|
+
goto if visit && respond_to?(:goto)
|
70
|
+
initialize_page if respond_to?(:initialize_page)
|
71
|
+
end
|
72
|
+
|
73
|
+
def initialize_browser(root)
|
74
|
+
@root_element = root_element_for root, PageObject::Platforms.get
|
75
|
+
@browser = browser_for root, PageObject::Platforms.get
|
76
|
+
include_platform_driver(root)
|
77
|
+
end
|
78
|
+
|
79
|
+
# @private
|
80
|
+
def self.included(cls)
|
81
|
+
cls.extend PageObject::Accessors
|
82
|
+
end
|
83
|
+
|
84
|
+
#
|
85
|
+
# Set the default timeout for page level waits
|
86
|
+
#
|
87
|
+
def self.default_page_wait=(timeout)
|
88
|
+
@page_wait = timeout
|
89
|
+
end
|
90
|
+
|
91
|
+
#
|
92
|
+
# Returns the default timeout for page lavel waits
|
93
|
+
#
|
94
|
+
def self.default_page_wait
|
95
|
+
@page_wait ||= 30
|
96
|
+
end
|
97
|
+
|
98
|
+
#
|
99
|
+
# Sets the default timeout for element level waits
|
100
|
+
#
|
101
|
+
def self.default_element_wait=(timeout)
|
102
|
+
@element_wait = timeout
|
103
|
+
end
|
104
|
+
|
105
|
+
#
|
106
|
+
# Returns the default timeout for element level waits
|
107
|
+
#
|
108
|
+
def self.default_element_wait
|
109
|
+
@element_wait ||= 5
|
110
|
+
end
|
111
|
+
|
112
|
+
#
|
113
|
+
# Set the javascript framework to use when determining number of
|
114
|
+
# ajax requests. Valid frameworks are :jquery, :prototype, :yui,
|
115
|
+
# and :angularjs
|
116
|
+
#
|
117
|
+
def self.javascript_framework=(framework)
|
118
|
+
PageObject::JavascriptFrameworkFacade.framework = framework
|
119
|
+
end
|
120
|
+
|
121
|
+
#
|
122
|
+
# Add a new javascript framework to page-object. The module passed
|
123
|
+
# in must adhere to the same prototype as the JQuery and Prototype
|
124
|
+
# modules.
|
125
|
+
#
|
126
|
+
# @param [Symbol] the name used to reference the framework in
|
127
|
+
# subsequent calls
|
128
|
+
# @param [Module] a module that has the necessary methods to perform
|
129
|
+
# the required actions.
|
130
|
+
#
|
131
|
+
def self.add_framework(key, framework)
|
132
|
+
PageObject::JavascriptFrameworkFacade.add_framework(key, framework)
|
133
|
+
end
|
134
|
+
|
135
|
+
#
|
136
|
+
# get the current page url
|
137
|
+
#
|
138
|
+
def current_url
|
139
|
+
platform.current_url
|
140
|
+
end
|
141
|
+
|
142
|
+
#
|
143
|
+
# navigate to the provided url
|
144
|
+
#
|
145
|
+
# @param [String] the full url to navigate to
|
146
|
+
#
|
147
|
+
def navigate_to(url)
|
148
|
+
platform.navigate_to(url)
|
149
|
+
end
|
150
|
+
|
151
|
+
#
|
152
|
+
# Returns the text of the current page
|
153
|
+
#
|
154
|
+
def text
|
155
|
+
root.text
|
156
|
+
end
|
157
|
+
|
158
|
+
#
|
159
|
+
# Returns the html of the current page
|
160
|
+
#
|
161
|
+
def html
|
162
|
+
platform.html
|
163
|
+
end
|
164
|
+
|
165
|
+
#
|
166
|
+
# Returns the title of the current page
|
167
|
+
#
|
168
|
+
def title
|
169
|
+
platform.title
|
170
|
+
end
|
171
|
+
|
172
|
+
#
|
173
|
+
# Wait until the block returns true or times out
|
174
|
+
#
|
175
|
+
# @example
|
176
|
+
# @page.wait_until(5, 'Success not found on page') do
|
177
|
+
# @page.text.include? 'Success'
|
178
|
+
# end
|
179
|
+
#
|
180
|
+
# @param [Numeric] the amount of time to wait for the block to return true.
|
181
|
+
# @param [String] the message to include with the error if we exceed the timeout duration.
|
182
|
+
# @param block the block to execute. It should return true when successful.
|
183
|
+
#
|
184
|
+
def wait_until(timeout = PageObject.default_page_wait, message = nil, &block)
|
185
|
+
platform.wait_until(timeout, message, &block)
|
186
|
+
end
|
187
|
+
|
188
|
+
|
189
|
+
#
|
190
|
+
# Wait until there are no pending ajax requests. This requires you
|
191
|
+
# to set the javascript framework in advance.
|
192
|
+
#
|
193
|
+
# @param [Numeric] the amount of time to wait for the block to return true.
|
194
|
+
# @param [String] the message to include with the error if we exceed
|
195
|
+
# the timeout duration.
|
196
|
+
#
|
197
|
+
def wait_for_ajax(timeout = 30, message = nil)
|
198
|
+
end_time = ::Time.now + timeout
|
199
|
+
until ::Time.now > end_time
|
200
|
+
return if browser.execute_script(::PageObject::JavascriptFrameworkFacade.pending_requests) == 0
|
201
|
+
sleep 0.5
|
202
|
+
end
|
203
|
+
message = "Timed out waiting for ajax requests to complete" unless message
|
204
|
+
raise message
|
205
|
+
end
|
206
|
+
|
207
|
+
#
|
208
|
+
# Override the normal alert popup so it does not occur.
|
209
|
+
#
|
210
|
+
# @example
|
211
|
+
# message = @page.alert do
|
212
|
+
# @page.button_that_causes_alert
|
213
|
+
# end
|
214
|
+
#
|
215
|
+
# @param frame optional parameter used when alert is nested within a frame
|
216
|
+
# @param block a block that has the call that will cause the alert to display
|
217
|
+
# @return [String] the message that was contained in the alert
|
218
|
+
#
|
219
|
+
def alert(frame=nil, &block)
|
220
|
+
platform.alert(frame, &block)
|
221
|
+
end
|
222
|
+
|
223
|
+
#
|
224
|
+
# Override the normal confirm popup so it does not occur.
|
225
|
+
#
|
226
|
+
# @example
|
227
|
+
# message = @popup.confirm(true) do
|
228
|
+
# @page.button_that_causes_confirm
|
229
|
+
# end
|
230
|
+
#
|
231
|
+
# @param [bool] what response you want to return back from the confirm popup
|
232
|
+
# @param frame optional parameter used when the confirm is nested within a frame
|
233
|
+
# @param block a block that has the call that will cause the confirm to display
|
234
|
+
# @return [String] the message that was prompted in the confirm
|
235
|
+
#
|
236
|
+
def confirm(response, frame=nil, &block)
|
237
|
+
platform.confirm(response, frame, &block)
|
238
|
+
end
|
239
|
+
|
240
|
+
#
|
241
|
+
# Override the normal prompt popup so it does not occur.
|
242
|
+
#
|
243
|
+
# @example
|
244
|
+
# message = @popup.prompt("Some Value") do
|
245
|
+
# @page.button_that_causes_prompt
|
246
|
+
# end
|
247
|
+
#
|
248
|
+
# @param [string] the value returned to the caller of the prompt
|
249
|
+
# @param frame optional parameter used with the prompt is nested within a frame
|
250
|
+
# @param block a block that has the call that will cause the prompt to display
|
251
|
+
# @return [Hash] A has containing two keys - :message contains the prompt message and
|
252
|
+
# :default_value contains the default value for the prompt if provided
|
253
|
+
#
|
254
|
+
def prompt(answer, frame=nil, &block)
|
255
|
+
platform.prompt(answer, frame, &block)
|
256
|
+
end
|
257
|
+
|
258
|
+
#
|
259
|
+
# Execute javascript on the browser
|
260
|
+
#
|
261
|
+
# @example Get inner HTML of element
|
262
|
+
# span = @page.span_element
|
263
|
+
# @page.execute_script "return arguments[0].innerHTML", span
|
264
|
+
# #=> "Span innerHTML"
|
265
|
+
#
|
266
|
+
def execute_script(script, *args)
|
267
|
+
args.map! { |e| e.kind_of?(PageObject::Elements::Element) ? e.element : e }
|
268
|
+
platform.execute_script(script, *args)
|
269
|
+
end
|
270
|
+
|
271
|
+
#
|
272
|
+
# Identify an element as existing within a frame. A frame parameter
|
273
|
+
# is passed to the block and must be passed to the other calls to PageObject.
|
274
|
+
# You can nest calls to in_frame by passing the frame to the next level.
|
275
|
+
#
|
276
|
+
# @example
|
277
|
+
# in_frame(:id => 'frame_id') do |frame|
|
278
|
+
# text_field_element(:id => 'fname', :frame => frame)
|
279
|
+
# end
|
280
|
+
#
|
281
|
+
# @param [Hash] identifier how we find the frame. The valid keys are:
|
282
|
+
# * :id => Watir and Selenium
|
283
|
+
# * :index => Watir and Selenium
|
284
|
+
# * :name => Watir and Selenium
|
285
|
+
# * :class => Watir only
|
286
|
+
# @param frame passed from a previous call to in_frame. Used to nest calls
|
287
|
+
# @param block that contains the calls to elements that exist inside the frame.
|
288
|
+
#
|
289
|
+
def in_frame(identifier, frame=nil, &block)
|
290
|
+
platform.in_frame(identifier, frame, &block)
|
291
|
+
end
|
292
|
+
|
293
|
+
# Identify an element as existing within an iframe. A frame parameter
|
294
|
+
# is passed to the block and must be passed to the other calls to PageObject.
|
295
|
+
# You can nest calls to in_iframe by passing the frame to the next level.
|
296
|
+
#
|
297
|
+
# @example
|
298
|
+
# in_iframe(:id => 'iframe_id') do |iframe|
|
299
|
+
# text_field_element(:id => 'ifname', :frame => iframe)
|
300
|
+
# end
|
301
|
+
#
|
302
|
+
# @param [Hash] identifier how we find the iframe. The valid keys are:
|
303
|
+
# * :id => Watir and Selenium
|
304
|
+
# * :index => Watir and Selenium
|
305
|
+
# * :name => Watir and Selenium
|
306
|
+
# * :class => Watir only
|
307
|
+
# @param frame passed from a previous call to in_iframe. Used to nest calls
|
308
|
+
# @param block that contains the calls to elements that exist inside the iframe.
|
309
|
+
#
|
310
|
+
def in_iframe(identifier, frame=nil, &block)
|
311
|
+
platform.in_iframe(identifier, frame, &block)
|
312
|
+
end
|
313
|
+
|
314
|
+
#
|
315
|
+
# Override the normal showModalDialog call is it opens a window instead
|
316
|
+
# of a dialog. You will need to attach to the new window in order to
|
317
|
+
# continue.
|
318
|
+
#
|
319
|
+
# @example
|
320
|
+
# @page.modal_dialog do
|
321
|
+
# @page.action_that_spawns_the_modal
|
322
|
+
# end
|
323
|
+
#
|
324
|
+
# @param block a block that contains the call that will cause the modal dialog.
|
325
|
+
#
|
326
|
+
def modal_dialog(&block)
|
327
|
+
script =
|
328
|
+
%Q{
|
329
|
+
window.showModalDialog = function(sURL, vArguments, sFeatures) {
|
330
|
+
window.dialogArguments = vArguments;
|
331
|
+
modalWin = window.open(sURL, 'modal', sFeatures);
|
332
|
+
return modalWin;
|
333
|
+
}
|
334
|
+
}
|
335
|
+
browser.execute_script script
|
336
|
+
yield if block_given?
|
337
|
+
end
|
338
|
+
|
339
|
+
#
|
340
|
+
# Attach to a running window. You can locate the window using either
|
341
|
+
# the window's title or url. If it fails to connect to a window it will
|
342
|
+
# pause for 1 second and try again.
|
343
|
+
#
|
344
|
+
# @example
|
345
|
+
# page.attach_to_window(:title => "other window's title")
|
346
|
+
#
|
347
|
+
# @param [Hash] either :title or :url of the other window. The url does not need to
|
348
|
+
# be the entire url - it can just be the page name like index.html
|
349
|
+
# @param block if present the block is executed and then execution is returned to the
|
350
|
+
# calling window
|
351
|
+
#
|
352
|
+
def attach_to_window(identifier, &block)
|
353
|
+
begin
|
354
|
+
platform.attach_to_window(identifier, &block)
|
355
|
+
rescue
|
356
|
+
sleep 1
|
357
|
+
platform.attach_to_window(identifier, &block)
|
358
|
+
end
|
359
|
+
end
|
360
|
+
|
361
|
+
#
|
362
|
+
# Find the element that has focus on the page
|
363
|
+
#
|
364
|
+
def element_with_focus
|
365
|
+
platform.element_with_focus
|
366
|
+
end
|
367
|
+
|
368
|
+
#
|
369
|
+
# Refresh to current page
|
370
|
+
#
|
371
|
+
def refresh
|
372
|
+
platform.refresh
|
373
|
+
end
|
374
|
+
|
375
|
+
#
|
376
|
+
# Go back to the previous page
|
377
|
+
#
|
378
|
+
def back
|
379
|
+
platform.back
|
380
|
+
end
|
381
|
+
|
382
|
+
#
|
383
|
+
# Go forward to the next page
|
384
|
+
#
|
385
|
+
def forward
|
386
|
+
platform.forward
|
387
|
+
end
|
388
|
+
|
389
|
+
#
|
390
|
+
# Clear the cookies from the browser
|
391
|
+
#
|
392
|
+
def clear_cookies
|
393
|
+
platform.clear_cookies
|
394
|
+
end
|
395
|
+
|
396
|
+
#
|
397
|
+
# Save the current screenshot to the provided url. File
|
398
|
+
# is saved as a png file.
|
399
|
+
#
|
400
|
+
def save_screenshot(file_name)
|
401
|
+
platform.save_screenshot file_name
|
402
|
+
end
|
403
|
+
|
404
|
+
def self.register_widget(widget_tag, widget_class, base_element_tag)
|
405
|
+
Widgets.register_widget(widget_tag, widget_class, base_element_tag)
|
406
|
+
end
|
407
|
+
|
408
|
+
private
|
409
|
+
|
410
|
+
def root
|
411
|
+
@root_element || browser_root_for(browser, PageObject::Platforms.get)
|
412
|
+
end
|
413
|
+
|
414
|
+
def include_platform_driver(browser)
|
415
|
+
@platform = load_platform(browser, PageObject::Platforms.get)
|
416
|
+
end
|
417
|
+
|
418
|
+
def call_block(&block)
|
419
|
+
block.arity == 1 ? block.call(self) : self.instance_eval(&block)
|
420
|
+
end
|
421
|
+
end
|