selenium-webdriver 3.0.8 → 3.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGES +9 -3
- data/lib/selenium/webdriver/chrome/bridge.rb +0 -1
- data/lib/selenium/webdriver/common.rb +9 -1
- data/lib/selenium/webdriver/common/action_builder.rb +1 -3
- data/lib/selenium/webdriver/common/driver.rb +17 -0
- data/lib/selenium/webdriver/common/interactions/input_device.rb +53 -0
- data/lib/selenium/webdriver/common/interactions/interaction.rb +52 -0
- data/lib/selenium/webdriver/common/interactions/interactions.rb +43 -0
- data/lib/selenium/webdriver/common/interactions/key_actions.rb +112 -0
- data/lib/selenium/webdriver/common/interactions/key_input.rb +64 -0
- data/lib/selenium/webdriver/common/{driver_extensions/has_input_devices.rb → interactions/none_input.rb} +9 -29
- data/lib/selenium/webdriver/common/interactions/pointer_actions.rb +355 -0
- data/lib/selenium/webdriver/common/interactions/pointer_input.rb +134 -0
- data/lib/selenium/webdriver/common/keys.rb +33 -12
- data/lib/selenium/webdriver/common/w3c_action_builder.rb +211 -0
- data/lib/selenium/webdriver/edge/bridge.rb +1 -4
- data/lib/selenium/webdriver/firefox/bridge.rb +1 -2
- data/lib/selenium/webdriver/firefox/extension/webdriver.xpi +0 -0
- data/lib/selenium/webdriver/firefox/w3c_bridge.rb +0 -1
- data/lib/selenium/webdriver/ie/bridge.rb +1 -1
- data/lib/selenium/webdriver/phantomjs/bridge.rb +1 -4
- data/lib/selenium/webdriver/remote/bridge.rb +25 -1
- data/lib/selenium/webdriver/remote/w3c_bridge.rb +16 -29
- data/lib/selenium/webdriver/remote/w3c_commands.rb +1 -0
- data/selenium-webdriver.gemspec +1 -1
- metadata +11 -3
@@ -88,7 +88,22 @@ module Selenium
|
|
88
88
|
f11: "\ue03B",
|
89
89
|
f12: "\ue03C",
|
90
90
|
meta: "\ue03D",
|
91
|
-
command: "\ue03D" # alias
|
91
|
+
command: "\ue03D", # alias
|
92
|
+
left_meta: "\ue03D", # alias
|
93
|
+
right_shift: "\ue050",
|
94
|
+
right_control: "\ue051",
|
95
|
+
right_alt: "\ue052",
|
96
|
+
right_meta: "\ue053",
|
97
|
+
numpad_page_up: "\ue054",
|
98
|
+
numpad_page_down: "\ue055",
|
99
|
+
numpad_end: "\ue056",
|
100
|
+
numpad_home: "\ue057",
|
101
|
+
numpad_left: "\ue058",
|
102
|
+
numpad_up: "\ue059",
|
103
|
+
numpad_right: "\ue05A",
|
104
|
+
numpad_down: "\ue05B",
|
105
|
+
numpad_insert: "\ue05C",
|
106
|
+
numpad_delete: "\ue05D"
|
92
107
|
}.freeze
|
93
108
|
|
94
109
|
#
|
@@ -105,18 +120,24 @@ module Selenium
|
|
105
120
|
#
|
106
121
|
|
107
122
|
def self.encode(keys)
|
108
|
-
keys.map
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
123
|
+
keys.map { |key| encode_key(key) }
|
124
|
+
end
|
125
|
+
|
126
|
+
#
|
127
|
+
# @api private
|
128
|
+
#
|
129
|
+
|
130
|
+
def self.encode_key(key)
|
131
|
+
case key
|
132
|
+
when Symbol
|
133
|
+
Keys[key]
|
134
|
+
when Array
|
135
|
+
key = key.map { |e| e.is_a?(Symbol) ? Keys[e] : e }.join
|
136
|
+
key << Keys[:null]
|
115
137
|
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
end
|
138
|
+
key
|
139
|
+
else
|
140
|
+
key.to_s
|
120
141
|
end
|
121
142
|
end
|
122
143
|
end # Keys
|
@@ -0,0 +1,211 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
#
|
3
|
+
# Licensed to the Software Freedom Conservancy (SFC) under one
|
4
|
+
# or more contributor license agreements. See the NOTICE file
|
5
|
+
# distributed with this work for additional information
|
6
|
+
# regarding copyright ownership. The SFC licenses this file
|
7
|
+
# to you under the Apache License, Version 2.0 (the
|
8
|
+
# "License"); you may not use this file except in compliance
|
9
|
+
# with the License. You may obtain a copy of the License at
|
10
|
+
#
|
11
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
12
|
+
#
|
13
|
+
# Unless required by applicable law or agreed to in writing,
|
14
|
+
# software distributed under the License is distributed on an
|
15
|
+
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
16
|
+
# KIND, either express or implied. See the License for the
|
17
|
+
# specific language governing permissions and limitations
|
18
|
+
# under the License.
|
19
|
+
|
20
|
+
module Selenium
|
21
|
+
module WebDriver
|
22
|
+
class W3CActionBuilder
|
23
|
+
include KeyActions # Actions specific to key inputs
|
24
|
+
include PointerActions # Actions specific to pointer inputs
|
25
|
+
attr_reader :devices
|
26
|
+
|
27
|
+
#
|
28
|
+
# Initialize a W3C Action Builder. Differs from previous by requiring a bridge and allowing asynchronous actions.
|
29
|
+
# The W3C implementation allows asynchronous actions per device. e.g. A key can be pressed at the same time that
|
30
|
+
# the mouse is moving. Keep in mind that pauses must be added for other devices in order to line up the actions
|
31
|
+
# correctly when using asynchronous.
|
32
|
+
#
|
33
|
+
# @param [Selenium::WebDriver::Remote::W3CBridge] bridge the bridge for the current driver instance
|
34
|
+
# @param [Selenium::WebDriver::Interactions::PointerInput] mouse PointerInput for the mouse.
|
35
|
+
# @param [Selenium::WebDriver::Interactions::KeyInput] keyboard KeyInput for the keyboard.
|
36
|
+
# @param [Boolean] async Whether to perform the actions asynchronously per device. Defaults to false for
|
37
|
+
# backwards compatibility.
|
38
|
+
# @return [W3CActionBuilder] A self reference.
|
39
|
+
#
|
40
|
+
|
41
|
+
def initialize(bridge, mouse, keyboard, async = false)
|
42
|
+
# For backwards compatibility, automatically include mouse & keyboard
|
43
|
+
@bridge = bridge
|
44
|
+
@devices = [mouse, keyboard]
|
45
|
+
@async = async
|
46
|
+
end
|
47
|
+
|
48
|
+
#
|
49
|
+
# Adds a PointerInput device of the given kind
|
50
|
+
#
|
51
|
+
# @example Add a touch pointer input device
|
52
|
+
#
|
53
|
+
# builder = device.action
|
54
|
+
# builder.add_pointer_input('touch', :touch)
|
55
|
+
#
|
56
|
+
# @param [String] name name for the device
|
57
|
+
# @param [Symbol] kind kind of pointer device to create
|
58
|
+
# @return [Interactions::PointerInput] The pointer input added
|
59
|
+
#
|
60
|
+
#
|
61
|
+
|
62
|
+
def add_pointer_input(kind, name)
|
63
|
+
new_input = Interactions.pointer(kind, name: name)
|
64
|
+
add_input(new_input)
|
65
|
+
new_input
|
66
|
+
end
|
67
|
+
|
68
|
+
#
|
69
|
+
# Adds a KeyInput device
|
70
|
+
#
|
71
|
+
# @example Add a key input device
|
72
|
+
#
|
73
|
+
# builder = device.action
|
74
|
+
# builder.add_key_input('keyboard2')
|
75
|
+
#
|
76
|
+
# @param [String] name name for the device
|
77
|
+
# @return [Interactions::KeyInput] The key input added
|
78
|
+
#
|
79
|
+
|
80
|
+
def add_key_input(name)
|
81
|
+
new_input = Interactions.key(name)
|
82
|
+
add_input(new_input)
|
83
|
+
new_input
|
84
|
+
end
|
85
|
+
|
86
|
+
#
|
87
|
+
# Retrieves the input device for the given name
|
88
|
+
#
|
89
|
+
# @param [String] name name of the input device
|
90
|
+
# @return [Selenium::WebDriver::Interactions::InputDevice] input device with given name
|
91
|
+
#
|
92
|
+
|
93
|
+
def get_device(name)
|
94
|
+
@devices.find { |device| device.name == name.to_s }
|
95
|
+
end
|
96
|
+
|
97
|
+
#
|
98
|
+
# Retrieves the current PointerInput devices
|
99
|
+
#
|
100
|
+
# @return [Array] array of current PointerInput devices
|
101
|
+
#
|
102
|
+
|
103
|
+
def pointer_inputs
|
104
|
+
@devices.select { |device| device.type == Interactions::POINTER }
|
105
|
+
end
|
106
|
+
|
107
|
+
#
|
108
|
+
# Retrieves the current KeyInput device
|
109
|
+
#
|
110
|
+
# @return [Selenium::WebDriver::Interactions::InputDevice] current KeyInput device
|
111
|
+
#
|
112
|
+
|
113
|
+
def key_inputs
|
114
|
+
@devices.select { |device| device.type == Interactions::KEY }
|
115
|
+
end
|
116
|
+
|
117
|
+
#
|
118
|
+
# Creates a pause for the given device of the given duration. If no duration is given, the pause will only wait
|
119
|
+
# for all actions to complete in that tick.
|
120
|
+
#
|
121
|
+
# @example Send keys to an element
|
122
|
+
#
|
123
|
+
# action_builder = driver.action
|
124
|
+
# keyboard = action_builder.key_input
|
125
|
+
# el = driver.find_element(id: "some_id")
|
126
|
+
# driver.action.click(el).pause(keyboard).pause(keyboard).pause(keyboard).send_keys('keys').perform
|
127
|
+
#
|
128
|
+
# @param [InputDevice] device Input device to pause
|
129
|
+
# @param [Float] duration Duration to pause
|
130
|
+
# @return [W3CActionBuilder] A self reference.
|
131
|
+
#
|
132
|
+
|
133
|
+
def pause(device, duration = nil)
|
134
|
+
device.create_pause(duration)
|
135
|
+
self
|
136
|
+
end
|
137
|
+
|
138
|
+
#
|
139
|
+
# Creates multiple pauses for the given device of the given duration.
|
140
|
+
#
|
141
|
+
# @example Send keys to an element
|
142
|
+
#
|
143
|
+
# action_builder = driver.action
|
144
|
+
# keyboard = action_builder.key_input
|
145
|
+
# el = driver.find_element(id: "some_id")
|
146
|
+
# driver.action.click(el).pauses(keyboard, 3).send_keys('keys').perform
|
147
|
+
#
|
148
|
+
# @param [InputDevice] device Input device to pause
|
149
|
+
# @param [Integer] number of pauses to add for the device
|
150
|
+
# @param [Float] duration Duration to pause
|
151
|
+
# @return [W3CActionBuilder] A self reference.
|
152
|
+
#
|
153
|
+
|
154
|
+
def pauses(device, number, duration = nil)
|
155
|
+
number.times { device.create_pause(duration) }
|
156
|
+
self
|
157
|
+
end
|
158
|
+
|
159
|
+
#
|
160
|
+
# Executes the actions added to the builder.
|
161
|
+
#
|
162
|
+
|
163
|
+
def perform
|
164
|
+
@bridge.send_actions @devices.map(&:encode).compact
|
165
|
+
clear_all_actions
|
166
|
+
nil
|
167
|
+
end
|
168
|
+
|
169
|
+
#
|
170
|
+
# Clears all actions from the builder.
|
171
|
+
#
|
172
|
+
|
173
|
+
def clear_all_actions
|
174
|
+
@devices.each(&:clear_actions)
|
175
|
+
end
|
176
|
+
|
177
|
+
#
|
178
|
+
# Releases all action states from the browser.
|
179
|
+
#
|
180
|
+
|
181
|
+
def release_actions
|
182
|
+
@bridge.release_actions
|
183
|
+
end
|
184
|
+
|
185
|
+
private
|
186
|
+
|
187
|
+
#
|
188
|
+
# Adds pauses for all devices but the given devices
|
189
|
+
#
|
190
|
+
# @param [Array[InputDevice]] action_devices Array of Input Devices performing an action in this tick.
|
191
|
+
#
|
192
|
+
|
193
|
+
def tick(*action_devices)
|
194
|
+
return if @async
|
195
|
+
@devices.each { |device| device.create_pause unless action_devices.include? device }
|
196
|
+
end
|
197
|
+
|
198
|
+
#
|
199
|
+
# Adds an InputDevice
|
200
|
+
#
|
201
|
+
|
202
|
+
def add_input(device)
|
203
|
+
unless @async
|
204
|
+
max_device = @devices.max { |a, b| a.actions.length <=> b.actions.length }
|
205
|
+
pauses(device, max_device.actions.length)
|
206
|
+
end
|
207
|
+
@devices << device
|
208
|
+
end
|
209
|
+
end # W3CActionBuilder
|
210
|
+
end # WebDriver
|
211
|
+
end # Selenium
|
Binary file
|
@@ -80,7 +80,6 @@ module Selenium
|
|
80
80
|
|
81
81
|
def driver_extensions
|
82
82
|
[
|
83
|
-
DriverExtensions::HasInputDevices,
|
84
83
|
DriverExtensions::UploadsFiles,
|
85
84
|
DriverExtensions::TakesScreenshot,
|
86
85
|
DriverExtensions::HasSessionId,
|
@@ -374,6 +373,31 @@ module Selenium
|
|
374
373
|
# actions
|
375
374
|
#
|
376
375
|
|
376
|
+
#
|
377
|
+
# @return [ActionBuilder]
|
378
|
+
# @api public
|
379
|
+
#
|
380
|
+
|
381
|
+
def action
|
382
|
+
ActionBuilder.new Mouse.new(self), Keyboard.new(self)
|
383
|
+
end
|
384
|
+
|
385
|
+
def mouse
|
386
|
+
warn <<-DEPRECATE.gsub(/\n +| {2,}/, ' ').freeze
|
387
|
+
[DEPRECATION] `Driver#mouse` is deprecated with w3c implementation. Instead use
|
388
|
+
driver.action.<command>.perform
|
389
|
+
DEPRECATE
|
390
|
+
Mouse.new self
|
391
|
+
end
|
392
|
+
|
393
|
+
def keyboard
|
394
|
+
warn <<-DEPRECATE.gsub(/\n +| {2,}/, ' ').freeze
|
395
|
+
[DEPRECATION] `Driver#keyboard` is deprecated with w3c implementation. Instead use
|
396
|
+
driver.action.<command>.perform
|
397
|
+
DEPRECATE
|
398
|
+
Keyboard.new self
|
399
|
+
end
|
400
|
+
|
377
401
|
def click_element(element)
|
378
402
|
execute :click_element, id: element
|
379
403
|
end
|
@@ -79,12 +79,10 @@ module Selenium
|
|
79
79
|
|
80
80
|
def driver_extensions
|
81
81
|
[
|
82
|
-
DriverExtensions::HasInputDevices,
|
83
82
|
DriverExtensions::UploadsFiles,
|
84
83
|
DriverExtensions::TakesScreenshot,
|
85
84
|
DriverExtensions::HasSessionId,
|
86
85
|
DriverExtensions::Rotatable,
|
87
|
-
DriverExtensions::HasTouchScreen,
|
88
86
|
DriverExtensions::HasRemoteStatus,
|
89
87
|
DriverExtensions::HasWebStorage
|
90
88
|
]
|
@@ -381,43 +379,32 @@ module Selenium
|
|
381
379
|
# actions
|
382
380
|
#
|
383
381
|
|
384
|
-
def
|
385
|
-
|
386
|
-
|
387
|
-
|
388
|
-
|
389
|
-
execute :click, {}, {button: 0}
|
382
|
+
def action(async = false)
|
383
|
+
W3CActionBuilder.new self,
|
384
|
+
Interactions.pointer(:mouse, name: 'mouse'),
|
385
|
+
Interactions.key('keyboard'),
|
386
|
+
async
|
390
387
|
end
|
388
|
+
alias_method :actions, :action
|
391
389
|
|
392
|
-
def
|
393
|
-
|
390
|
+
def mouse
|
391
|
+
raise Error::UnsupportedOperationError, '#mouse is no longer supported, use #action instead'
|
394
392
|
end
|
395
393
|
|
396
|
-
def
|
397
|
-
|
394
|
+
def keyboard
|
395
|
+
raise Error::UnsupportedOperationError, '#keyboard is no longer supported, use #action instead'
|
398
396
|
end
|
399
397
|
|
400
|
-
def
|
401
|
-
execute :
|
398
|
+
def send_actions(data)
|
399
|
+
execute :actions, {}, {actions: data}
|
402
400
|
end
|
403
401
|
|
404
|
-
def
|
405
|
-
execute :
|
402
|
+
def release_actions
|
403
|
+
execute :release_actions
|
406
404
|
end
|
407
405
|
|
408
|
-
def
|
409
|
-
|
410
|
-
|
411
|
-
if x && y
|
412
|
-
params[:xoffset] = x
|
413
|
-
params[:yoffset] = y
|
414
|
-
end
|
415
|
-
|
416
|
-
execute :mouse_move_to, {}, params
|
417
|
-
end
|
418
|
-
|
419
|
-
def send_keys_to_active_element(keys)
|
420
|
-
send_keys_to_element(active_element, keys)
|
406
|
+
def click_element(element)
|
407
|
+
execute :element_click, id: element
|
421
408
|
end
|
422
409
|
|
423
410
|
# TODO: - Implement file verification
|