appium_lib_core 4.1.0 → 9.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +325 -270
- data/README.md +65 -15
- data/Rakefile +5 -22
- data/appium_lib_core.gemspec +6 -9
- data/bin/console +0 -4
- data/lib/appium_lib_core/android/device/auth_finger_print.rb +4 -1
- data/lib/appium_lib_core/android/device/network.rb +10 -0
- data/lib/appium_lib_core/android/device/performance.rb +3 -0
- data/lib/appium_lib_core/android/device/screen.rb +2 -0
- data/lib/appium_lib_core/android/device.rb +80 -17
- data/lib/appium_lib_core/common/base/bridge.rb +309 -93
- data/lib/appium_lib_core/common/base/capabilities.rb +21 -8
- data/lib/appium_lib_core/common/{command/mjsonwp.rb → base/device_ime.rb} +33 -12
- data/lib/appium_lib_core/common/base/driver.rb +258 -331
- data/lib/appium_lib_core/common/base/driver_settings.rb +51 -0
- data/lib/appium_lib_core/common/base/has_location.rb +80 -0
- data/lib/appium_lib_core/common/base/has_network_connection.rb +56 -0
- data/lib/appium_lib_core/common/base/http_default.rb +15 -38
- data/lib/appium_lib_core/{ios/uiautomation/bridge.rb → common/base/remote_status.rb} +9 -8
- data/lib/appium_lib_core/common/base/rotable.rb +62 -0
- data/lib/appium_lib_core/common/base/screenshot.rb +8 -8
- data/lib/appium_lib_core/common/base/search_context.rb +13 -17
- data/lib/appium_lib_core/common/base.rb +1 -5
- data/lib/appium_lib_core/common/command.rb +244 -4
- data/lib/appium_lib_core/common/device/app_management.rb +2 -26
- data/lib/appium_lib_core/common/device/context.rb +1 -5
- data/lib/appium_lib_core/common/device/image_comparison.rb +12 -4
- data/lib/appium_lib_core/common/device/keyevent.rb +4 -4
- data/lib/appium_lib_core/common/device/{touch_actions.rb → orientation.rb} +6 -10
- data/lib/appium_lib_core/common/error.rb +4 -5
- data/lib/appium_lib_core/common/log.rb +5 -4
- data/lib/appium_lib_core/common/wait.rb +38 -6
- data/lib/appium_lib_core/device.rb +3 -9
- data/lib/appium_lib_core/driver.rb +189 -158
- data/lib/appium_lib_core/{patch.rb → element.rb} +62 -25
- data/lib/appium_lib_core/ios/xcuitest/device.rb +2 -0
- data/lib/appium_lib_core/{common/base/command.rb → mac2/bridge.rb} +9 -8
- data/lib/appium_lib_core/mac2/device/screen.rb +48 -0
- data/lib/appium_lib_core/mac2/device.rb +92 -0
- data/lib/appium_lib_core/{ios.rb → mac2.rb} +2 -5
- data/lib/appium_lib_core/support/event_firing_bridge.rb +57 -0
- data/lib/appium_lib_core/version.rb +2 -2
- data/lib/appium_lib_core.rb +23 -10
- metadata +28 -94
- data/.github/ISSUE_TEMPLATE/issue-report.md +0 -29
- data/.github/contributing.md +0 -26
- data/.github/issue_template.md +0 -20
- data/.github/workflows/unittest.yml +0 -68
- data/.gitignore +0 -18
- data/.rubocop.yml +0 -58
- data/azure-pipelines.yml +0 -15
- data/ci-jobs/functional/android_setup.yml +0 -3
- data/ci-jobs/functional/ios_setup.yml +0 -7
- data/ci-jobs/functional/publish_test_result.yml +0 -18
- data/ci-jobs/functional/run_appium.yml +0 -25
- data/ci-jobs/functional/start-emulator.sh +0 -26
- data/ci-jobs/functional_test.yml +0 -298
- data/docs/mobile_command.md +0 -34
- data/lib/appium_lib_core/common/base/bridge/mjsonwp.rb +0 -81
- data/lib/appium_lib_core/common/base/bridge/w3c.rb +0 -252
- data/lib/appium_lib_core/common/command/common.rb +0 -110
- data/lib/appium_lib_core/common/command/w3c.rb +0 -56
- data/lib/appium_lib_core/common/device/value.rb +0 -52
- data/lib/appium_lib_core/common/touch_action/multi_touch.rb +0 -56
- data/lib/appium_lib_core/common/touch_action/touch_actions.rb +0 -203
- data/lib/appium_lib_core/ios/uiautomation/device.rb +0 -44
- data/lib/appium_lib_core/ios/uiautomation/patch.rb +0 -34
- data/release_notes.md +0 -816
- data/script/commands.rb +0 -200
@@ -16,52 +16,69 @@ module Appium
|
|
16
16
|
module Core
|
17
17
|
class Base
|
18
18
|
class Bridge < ::Selenium::WebDriver::Remote::Bridge
|
19
|
+
include Device::DeviceLock
|
20
|
+
include Device::Keyboard
|
21
|
+
include Device::ImeActions
|
22
|
+
include Device::Setting
|
23
|
+
include Device::Context
|
24
|
+
include Device::FileManagement
|
25
|
+
include Device::KeyEvent
|
26
|
+
include Device::ImageComparison
|
27
|
+
include Device::AppManagement
|
28
|
+
include Device::AppState
|
29
|
+
include Device::ScreenRecord::Command
|
30
|
+
include Device::Device
|
31
|
+
include Device::ExecuteDriver
|
32
|
+
include Device::Orientation
|
33
|
+
|
19
34
|
# Prefix for extra capability defined by W3C
|
20
35
|
APPIUM_PREFIX = 'appium:'
|
21
36
|
|
22
|
-
#
|
23
|
-
|
37
|
+
# No 'browserName' means the session is native appium connection
|
38
|
+
APPIUM_NATIVE_BROWSER_NAME = 'appium'
|
39
|
+
|
40
|
+
attr_reader :available_commands
|
41
|
+
|
42
|
+
def browser
|
43
|
+
@browser ||= begin
|
44
|
+
name = @capabilities&.browser_name
|
45
|
+
name ? name.tr(' ', '_').downcase.to_sym : 'unknown'
|
46
|
+
rescue KeyError
|
47
|
+
APPIUM_NATIVE_BROWSER_NAME
|
48
|
+
end
|
49
|
+
end
|
24
50
|
|
25
|
-
#
|
51
|
+
# Appium only.
|
52
|
+
# Attach to an existing session.
|
26
53
|
#
|
27
|
-
#
|
54
|
+
# @param [String] The session id to attach to.
|
55
|
+
# @param [String] platform_name The platform name to keep in the dummy capabilities
|
56
|
+
# @param [String] platform_name The automation name to keep in the dummy capabilities
|
57
|
+
# @return [::Appium::Core::Base::Capabilities]
|
28
58
|
#
|
29
|
-
#
|
30
|
-
# 2. Sniffs response.
|
31
|
-
# 3. Based on the response, understands which dialect we should use.
|
59
|
+
# @example
|
32
60
|
#
|
33
|
-
#
|
61
|
+
# new_driver = ::Appium::Core::Driver.attach_to(
|
62
|
+
# driver.session_id,
|
63
|
+
# url: 'http://127.0.0.1:4723/wd/hub', automation_name: 'UiAutomator2', platform_name: 'Android'
|
64
|
+
# )
|
34
65
|
#
|
35
|
-
def
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
end
|
45
|
-
|
46
|
-
bridge = new(opts)
|
47
|
-
capabilities = bridge.create_session(desired_capabilities)
|
48
|
-
|
49
|
-
case bridge.dialect
|
50
|
-
when :oss # for MJSONWP
|
51
|
-
Bridge::MJSONWP.new(capabilities, bridge.session_id, **opts)
|
52
|
-
when :w3c
|
53
|
-
Bridge::W3C.new(capabilities, bridge.session_id, **opts)
|
54
|
-
else
|
55
|
-
raise CoreError, 'cannot understand dialect'
|
56
|
-
end
|
66
|
+
def attach_to(session_id, platform_name, automation_name)
|
67
|
+
@available_commands = ::Appium::Core::Commands::COMMANDS.dup
|
68
|
+
@session_id = session_id
|
69
|
+
|
70
|
+
# generate a dummy capabilities instance which only has the given platformName and automationName
|
71
|
+
@capabilities = ::Appium::Core::Base::Capabilities.new(
|
72
|
+
'platformName' => platform_name,
|
73
|
+
'automationName' => automation_name
|
74
|
+
)
|
57
75
|
end
|
58
76
|
|
59
77
|
# Override
|
60
|
-
# Creates session handling
|
61
|
-
# Copy from Selenium::WebDriver::Remote::Bridge to keep using +merged_capabilities+ for Appium
|
78
|
+
# Creates session handling.
|
62
79
|
#
|
63
|
-
# @param [::
|
64
|
-
# @return [::
|
80
|
+
# @param [::Appium::Core::Base::Capabilities, Hash] capabilities A capability
|
81
|
+
# @return [::Appium::Core::Base::Capabilities]
|
65
82
|
#
|
66
83
|
# @example
|
67
84
|
#
|
@@ -79,44 +96,31 @@ module Appium
|
|
79
96
|
# }
|
80
97
|
# }
|
81
98
|
# core = ::Appium::Core.for(caps)
|
82
|
-
# driver = core.start_driver
|
99
|
+
# driver = core.start_driver
|
83
100
|
#
|
84
|
-
def create_session(
|
85
|
-
|
86
|
-
|
87
|
-
@session_id = response['sessionId']
|
88
|
-
oss_status = response['status'] # for compatibility with Appium 1.7.1-
|
89
|
-
value = response['value']
|
90
|
-
|
91
|
-
if value.is_a?(Hash) # include for W3C format
|
92
|
-
@session_id = value['sessionId'] if value.key?('sessionId')
|
101
|
+
def create_session(capabilities)
|
102
|
+
@available_commands = ::Appium::Core::Commands::COMMANDS.dup
|
93
103
|
|
94
|
-
|
95
|
-
|
96
|
-
elsif value.key?('value')
|
97
|
-
value = value['value']
|
98
|
-
end
|
99
|
-
end
|
104
|
+
always_match = add_appium_prefix(capabilities)
|
105
|
+
response = execute(:new_session, {}, { capabilities: { alwaysMatch: always_match, firstMatch: [{}] } })
|
100
106
|
|
107
|
+
@session_id = response['sessionId']
|
101
108
|
raise ::Selenium::WebDriver::Error::WebDriverError, 'no sessionId in returned payload' unless @session_id
|
102
109
|
|
103
|
-
json_create(
|
110
|
+
@capabilities = json_create(response['capabilities'])
|
104
111
|
end
|
105
112
|
|
106
113
|
# Append +appium:+ prefix for Appium following W3C spec
|
107
114
|
# https://www.w3.org/TR/webdriver/#dfn-validate-capabilities
|
108
115
|
#
|
109
|
-
# @param [::
|
110
|
-
# @return [::
|
116
|
+
# @param [::Appium::Core::Base::Capabilities, Hash] capabilities A capability
|
117
|
+
# @return [::Appium::Core::Base::Capabilities]
|
111
118
|
def add_appium_prefix(capabilities)
|
112
|
-
w3c_capabilities = ::
|
119
|
+
w3c_capabilities = ::Appium::Core::Base::Capabilities.new
|
113
120
|
|
114
|
-
capabilities = capabilities.
|
121
|
+
capabilities = capabilities.send(:capabilities) unless capabilities.is_a?(Hash)
|
115
122
|
|
116
123
|
capabilities.each do |name, value|
|
117
|
-
next if value.nil?
|
118
|
-
next if value.is_a?(String) && value.empty?
|
119
|
-
|
120
124
|
capability_name = name.to_s
|
121
125
|
w3c_name = extension_prefix?(capability_name) ? name : "#{APPIUM_PREFIX}#{capability_name}"
|
122
126
|
|
@@ -128,61 +132,273 @@ module Appium
|
|
128
132
|
|
129
133
|
private
|
130
134
|
|
131
|
-
def camel_case(
|
132
|
-
|
135
|
+
def camel_case(str_or_sym)
|
136
|
+
str_or_sym.to_s.gsub(/_([a-z])/) { Regexp.last_match(1).upcase }
|
133
137
|
end
|
134
138
|
|
135
139
|
def extension_prefix?(capability_name)
|
136
|
-
snake_cased_capability_names = ::
|
140
|
+
snake_cased_capability_names = ::Appium::Core::Base::Capabilities::KNOWN.map(&:to_s)
|
137
141
|
camel_cased_capability_names = snake_cased_capability_names.map { |v| camel_case(v) }
|
138
142
|
|
143
|
+
# Check 'EXTENSION_CAPABILITY_PATTERN'
|
139
144
|
snake_cased_capability_names.include?(capability_name) ||
|
140
145
|
camel_cased_capability_names.include?(capability_name) ||
|
141
|
-
capability_name.match(
|
146
|
+
capability_name.match(':')
|
142
147
|
end
|
143
148
|
|
144
|
-
def json_create(
|
145
|
-
|
146
|
-
::Selenium::WebDriver.logger.info 'Detected OSS dialect.'
|
147
|
-
@dialect = :oss
|
148
|
-
::Selenium::WebDriver::Remote::Capabilities.json_create(value)
|
149
|
-
else
|
150
|
-
::Selenium::WebDriver.logger.info 'Detected W3C dialect.'
|
151
|
-
@dialect = :w3c
|
152
|
-
::Selenium::WebDriver::Remote::W3C::Capabilities.json_create(value)
|
153
|
-
end
|
149
|
+
def json_create(value)
|
150
|
+
::Appium::Core::Base::Capabilities.json_create(value)
|
154
151
|
end
|
155
152
|
|
156
|
-
|
157
|
-
w3c_capabilities = ::Selenium::WebDriver::Remote::W3C::Capabilities.new
|
153
|
+
public
|
158
154
|
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
155
|
+
# command for Appium 2.0.
|
156
|
+
def add_command(method:, url:, name:, &block)
|
157
|
+
::Appium::Logger.info "Overriding the method '#{name}' for '#{url}'" if @available_commands.key? name
|
158
|
+
|
159
|
+
@available_commands[name] = [method, url]
|
160
|
+
|
161
|
+
::Appium::Core::Device.add_endpoint_method name, &block
|
162
|
+
end
|
163
|
+
|
164
|
+
def commands(command)
|
165
|
+
@available_commands[command]
|
166
|
+
end
|
167
|
+
|
168
|
+
def status
|
169
|
+
execute :status
|
170
|
+
end
|
171
|
+
|
172
|
+
# Perform 'touch' actions for W3C module.
|
173
|
+
# Generate +touch+ pointer action here and users can use this via +driver.action+
|
174
|
+
# - https://www.selenium.dev/documentation/webdriver/actions_api/
|
175
|
+
# - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/ActionBuilder.html
|
176
|
+
# - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/PointerActions.html
|
177
|
+
# - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/KeyActions.html
|
178
|
+
#
|
179
|
+
# The pointer type is 'touch' by default in the Appium Ruby client.
|
180
|
+
#
|
181
|
+
# @example
|
182
|
+
#
|
183
|
+
# element = @driver.find_element(:id, "some id")
|
184
|
+
# @driver.action.click(element).perform # The 'click' is a part of 'PointerActions'
|
185
|
+
#
|
186
|
+
def action(_deprecated_async = nil, async: false, devices: nil)
|
187
|
+
::Selenium::WebDriver::ActionBuilder.new(
|
188
|
+
self,
|
189
|
+
devices: devices || [::Selenium::WebDriver::Interactions.pointer(:touch, name: 'touch')],
|
190
|
+
async: async,
|
191
|
+
duration: 50 # milliseconds
|
192
|
+
)
|
193
|
+
end
|
194
|
+
|
195
|
+
# Port from MJSONWP
|
196
|
+
def get_timeouts
|
197
|
+
execute :get_timeouts
|
198
|
+
end
|
199
|
+
|
200
|
+
# For Appium
|
201
|
+
# override
|
202
|
+
def element_displayed?(element)
|
203
|
+
# For W3C
|
204
|
+
# https://github.com/SeleniumHQ/selenium/commit/b618499adcc3a9f667590652c5757c0caa703289
|
205
|
+
# execute_atom :isDisplayed, element
|
206
|
+
execute :is_element_displayed, id: element.id
|
207
|
+
end
|
208
|
+
|
209
|
+
# For Appium
|
210
|
+
# override
|
211
|
+
def element_attribute(element, name)
|
212
|
+
# For W3C in Selenium Client
|
213
|
+
# execute_atom :getAttribute, element, name.
|
214
|
+
# 'dom_attribute' in the WebDriver Selenium.
|
215
|
+
execute :get_element_attribute, id: element.id, name: name
|
216
|
+
end
|
164
217
|
|
165
|
-
|
218
|
+
# For Appium
|
219
|
+
# override
|
220
|
+
def active_element
|
221
|
+
::Appium::Core::Element.new self, element_id_from(execute(:get_active_element))
|
222
|
+
end
|
223
|
+
alias switch_to_active_element active_element
|
224
|
+
|
225
|
+
# For Appium
|
226
|
+
# override
|
227
|
+
def find_element_by(how, what, parent_ref = [])
|
228
|
+
how, what = convert_locator(how, what)
|
229
|
+
|
230
|
+
return execute_atom(:findElements, Support::RelativeLocator.new(what).as_json).first if how == 'relative'
|
231
|
+
|
232
|
+
parent_type, parent_id = parent_ref
|
233
|
+
id = case parent_type
|
234
|
+
when :element
|
235
|
+
execute :find_child_element, { id: parent_id }, { using: how, value: what.to_s }
|
236
|
+
when :shadow_root
|
237
|
+
execute :find_shadow_child_element, { id: parent_id }, { using: how, value: what.to_s }
|
238
|
+
else
|
239
|
+
execute :find_element, {}, { using: how, value: what.to_s }
|
240
|
+
end
|
241
|
+
|
242
|
+
::Appium::Core::Element.new self, element_id_from(id)
|
243
|
+
end
|
244
|
+
|
245
|
+
# For Appium
|
246
|
+
# override
|
247
|
+
def find_elements_by(how, what, parent_ref = [])
|
248
|
+
how, what = convert_locator(how, what)
|
249
|
+
|
250
|
+
return execute_atom :findElements, Support::RelativeLocator.new(what).as_json if how == 'relative'
|
251
|
+
|
252
|
+
parent_type, parent_id = parent_ref
|
253
|
+
ids = case parent_type
|
254
|
+
when :element
|
255
|
+
execute :find_child_elements, { id: parent_id }, { using: how, value: what.to_s }
|
256
|
+
when :shadow_root
|
257
|
+
execute :find_shadow_child_elements, { id: parent_id }, { using: how, value: what.to_s }
|
258
|
+
else
|
259
|
+
execute :find_elements, {}, { using: how, value: what.to_s }
|
260
|
+
end
|
261
|
+
|
262
|
+
ids.map { |id| ::Appium::Core::Element.new self, element_id_from(id) }
|
263
|
+
end
|
264
|
+
|
265
|
+
# For Appium
|
266
|
+
# @param [Hash] id The id which can get as a response from server
|
267
|
+
# @return [::Appium::Core::Element]
|
268
|
+
def convert_to_element(id)
|
269
|
+
::Appium::Core::Element.new self, element_id_from(id)
|
270
|
+
end
|
271
|
+
|
272
|
+
# For Appium
|
273
|
+
# override
|
274
|
+
# called in 'extend DriverExtensions::HasNetworkConnection'
|
275
|
+
def network_connection
|
276
|
+
execute :get_network_connection
|
277
|
+
end
|
278
|
+
|
279
|
+
# For Appium
|
280
|
+
# override
|
281
|
+
# called in 'extend DriverExtensions::HasNetworkConnection'
|
282
|
+
def network_connection=(type)
|
283
|
+
execute :set_network_connection, {}, { parameters: { type: type } }
|
284
|
+
end
|
285
|
+
|
286
|
+
# For Appium
|
287
|
+
# No implementation for W3C webdriver module
|
288
|
+
# called in 'extend DriverExtensions::HasLocation'
|
289
|
+
def location
|
290
|
+
obj = execute(:get_location) || {}
|
291
|
+
::Appium::Location.new obj['latitude'], obj['longitude'], obj['altitude']
|
292
|
+
end
|
293
|
+
|
294
|
+
# For Appium
|
295
|
+
# No implementation for W3C webdriver module
|
296
|
+
def set_location(lat, lon, alt = 0.0, speed: nil, satellites: nil)
|
297
|
+
loc = { latitude: lat, longitude: lon, altitude: alt }
|
298
|
+
loc[:speed] = speed unless speed.nil?
|
299
|
+
loc[:satellites] = satellites unless satellites.nil?
|
300
|
+
execute :set_location, {}, { location: loc }
|
301
|
+
end
|
302
|
+
|
303
|
+
#
|
304
|
+
# logs
|
305
|
+
#
|
306
|
+
# For Appium
|
307
|
+
# No implementation for W3C webdriver module
|
308
|
+
def available_log_types
|
309
|
+
types = execute :get_available_log_types
|
310
|
+
Array(types).map(&:to_sym)
|
311
|
+
end
|
312
|
+
|
313
|
+
# For Appium
|
314
|
+
# No implementation for W3C webdriver module
|
315
|
+
def log(type)
|
316
|
+
data = execute :get_log, {}, { type: type.to_s }
|
317
|
+
|
318
|
+
Array(data).map do |l|
|
319
|
+
::Selenium::WebDriver::LogEntry.new l.fetch('level', 'UNKNOWN'), l.fetch('timestamp'), l.fetch('message')
|
320
|
+
rescue KeyError
|
321
|
+
next
|
166
322
|
end
|
323
|
+
end
|
167
324
|
|
168
|
-
|
325
|
+
# For Appium
|
326
|
+
def log_event(vendor, event)
|
327
|
+
execute :post_log_event, {}, { vendor: vendor, event: event }
|
328
|
+
end
|
329
|
+
|
330
|
+
# For Appium
|
331
|
+
def log_events(type = nil)
|
332
|
+
args = {}
|
333
|
+
args['type'] = type unless type.nil?
|
334
|
+
|
335
|
+
execute :get_log_events, {}, args
|
336
|
+
end
|
337
|
+
|
338
|
+
def viewport_screenshot
|
339
|
+
execute_script('mobile: viewportScreenshot')
|
340
|
+
end
|
341
|
+
|
342
|
+
def element_screenshot(element_id)
|
343
|
+
execute :take_element_screenshot, id: element_id
|
344
|
+
end
|
345
|
+
|
346
|
+
# for selenium-webdriver compatibility in chrome browser session.
|
347
|
+
# This may be needed in selenium-webdriver 4.8 or over? (around the version)
|
348
|
+
# when a session starts browserName: 'chrome' for bridge.
|
349
|
+
# This method is not only for Android, but also chrome desktop browser as well.
|
350
|
+
# So this bridge itself does not restrict the target module.
|
351
|
+
def send_command(command_params)
|
352
|
+
execute :chrome_send_command, {}, command_params
|
169
353
|
end
|
170
354
|
|
171
|
-
|
172
|
-
|
173
|
-
|
355
|
+
private
|
356
|
+
|
357
|
+
def unwrap_script_result(arg)
|
358
|
+
case arg
|
359
|
+
when Array
|
360
|
+
arg.map { |e| unwrap_script_result(e) }
|
361
|
+
when Hash
|
362
|
+
element_id = element_id_from(arg)
|
363
|
+
return ::Appium::Core::Element.new(self, element_id) if element_id
|
174
364
|
|
175
|
-
|
365
|
+
shadow_root_id = shadow_root_id_from(arg)
|
366
|
+
return ::Selenium::WebDriver::Remote::ShadowRoot.new self, shadow_root_id if shadow_root_id
|
176
367
|
|
177
|
-
|
178
|
-
|
368
|
+
arg.each { |k, v| arg[k] = unwrap_script_result(v) }
|
369
|
+
else
|
370
|
+
arg
|
371
|
+
end
|
372
|
+
end
|
373
|
+
|
374
|
+
def element_id_from(id)
|
375
|
+
id['ELEMENT'] || id['element-6066-11e4-a52e-4f735466cecf']
|
376
|
+
end
|
179
377
|
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
}
|
378
|
+
# Don't convert locators for Appium in native context
|
379
|
+
def convert_locator(how, what)
|
380
|
+
# case how
|
381
|
+
# when 'class name'
|
382
|
+
# how = 'css selector'
|
383
|
+
# what = ".#{escape_css(what)}"
|
384
|
+
# when 'id'
|
385
|
+
# how = 'css selector'
|
386
|
+
# what = "##{escape_css(what)}"
|
387
|
+
# when 'name'
|
388
|
+
# how = 'css selector'
|
389
|
+
# what = "*[name='#{escape_css(what)}']"
|
390
|
+
# when 'tag name'
|
391
|
+
# how = 'css selector'
|
392
|
+
# end
|
393
|
+
#
|
394
|
+
# if what.is_a?(Hash)
|
395
|
+
# what = what.each_with_object({}) do |(h, w), hash|
|
396
|
+
# h, w = convert_locator(h.to_s, w)
|
397
|
+
# hash[h] = w
|
398
|
+
# end
|
399
|
+
# end
|
400
|
+
|
401
|
+
[how, what]
|
186
402
|
end
|
187
403
|
end # class Bridge
|
188
404
|
end # class Base
|
@@ -15,14 +15,27 @@
|
|
15
15
|
module Appium
|
16
16
|
module Core
|
17
17
|
class Base
|
18
|
-
|
19
|
-
#
|
20
|
-
#
|
21
|
-
#
|
22
|
-
#
|
23
|
-
#
|
24
|
-
|
25
|
-
|
18
|
+
class Capabilities < ::Selenium::WebDriver::Remote::Capabilities
|
19
|
+
# TODO: Move to 'Options' way instead of 'Capabilities'.
|
20
|
+
# Selenium 5 will have Options instead of 'Capabilities'.
|
21
|
+
# https://github.com/SeleniumHQ/selenium/blob/trunk/rb/lib/selenium/webdriver/common/options.rb
|
22
|
+
# Then, Ruby client also shoud move to the Options way.
|
23
|
+
# Appium's capabilities could change by depending on Appium versions. So it does not have
|
24
|
+
# standard options like chrome and firefox etc. So, the implementation should differ from
|
25
|
+
# other browsers. But here should inherit `Options` to follow Selenium.
|
26
|
+
|
27
|
+
# Method override
|
28
|
+
# FIXME: when we drop "symbolize_keys", this can be removed.
|
29
|
+
def convert_key(key)
|
30
|
+
case key
|
31
|
+
when String
|
32
|
+
key.to_s
|
33
|
+
when Symbol
|
34
|
+
# here do not convert to camel case
|
35
|
+
key.to_s
|
36
|
+
else
|
37
|
+
raise TypeError, "expected String or Symbol, got #{key.inspect}:#{key.class}"
|
38
|
+
end
|
26
39
|
end
|
27
40
|
end
|
28
41
|
end
|
@@ -14,15 +14,36 @@
|
|
14
14
|
|
15
15
|
module Appium
|
16
16
|
module Core
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
17
|
+
class Base
|
18
|
+
#
|
19
|
+
# @api private
|
20
|
+
#
|
21
|
+
class DeviceIME
|
22
|
+
# @private this class is private
|
23
|
+
def initialize(bridge)
|
24
|
+
@bridge = bridge
|
25
|
+
end
|
26
|
+
|
27
|
+
def activate(ime_name)
|
28
|
+
@bridge.ime_activate(ime_name)
|
29
|
+
end
|
30
|
+
|
31
|
+
def available_engines
|
32
|
+
@bridge.ime_available_engines
|
33
|
+
end
|
34
|
+
|
35
|
+
def active_engine
|
36
|
+
@bridge.ime_active_engine
|
37
|
+
end
|
38
|
+
|
39
|
+
def activated?
|
40
|
+
@bridge.ime_activated
|
41
|
+
end
|
42
|
+
|
43
|
+
def deactivate
|
44
|
+
@bridge.ime_deactivate
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|