Skip to content

Commit e0f3683

Browse files
authored
Enhance espresso adaptation for text, button, finds wrappers (#844)
* rename name to text for both uia2 and espresso * add warning for uiautomator if it is espresso context * tweak tests, add workaround for espresso * extract espresso * tweak comment * update changelog * back to uiautomator2
1 parent fb49333 commit e0f3683

File tree

15 files changed

+313
-31
lines changed

15 files changed

+313
-31
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@ Release tags are https://github.com/appium/ruby_lib/releases .
55

66
## Unreleased
77
### 1. Enhancements
8+
- Enhance Espresso automation name adaptation
9+
- Call `xpath` locator strategy instead of `uiautomator` locator strategy in various wrapper methods like `text/s`, `button/s`, etc
10+
- [#884](https://github.com/appium/ruby_lib/pull/844/)
811

912
### 2. Bug fixes
1013

android_tests/lib/android/specs/android/element/button.rb

+12-8
Original file line numberDiff line numberDiff line change
@@ -13,39 +13,43 @@ def after_last
1313
end
1414

1515
def fade_in
16-
'FADE IN'
16+
if automation_name_is_espresso?
17+
'Fade in'
18+
else
19+
'FADE IN'
20+
end
1721
end
1822

1923
t { before_first }
2024

2125
t 'button' do
2226
# by index
23-
2.times { wait { button(1).name.must_equal fade_in } }
27+
2.times { wait { button(1).text.upcase.must_equal fade_in.upcase } }
2428

2529
# by name contains
26-
wait { button('ade').name.must_equal fade_in }
30+
wait { button('ade').text.upcase.must_equal fade_in.upcase }
2731
end
2832

2933
t 'buttons' do
3034
exp = ['ZOOM IN', 'MODERN ZOOM IN', 'THUMBNAIL ZOOM']
31-
wait { buttons('zoom').map(&:text).must_equal exp }
35+
wait { buttons('zoom').map(&:text).map(&:upcase).must_equal exp }
3236
wait { buttons.length.must_equal 6 }
3337
end
3438

3539
t 'first_button' do
36-
wait { first_button.name.must_equal fade_in }
40+
wait { first_button.text.upcase.must_equal fade_in.upcase }
3741
end
3842

3943
t 'last_button' do
40-
wait { last_button.name.must_equal 'THUMBNAIL ZOOM' }
44+
wait { last_button.text.upcase.must_equal 'THUMBNAIL ZOOM' }
4145
end
4246

4347
t 'button_exact' do
44-
wait { button_exact(fade_in).name.must_equal fade_in }
48+
wait { button_exact(fade_in).text.upcase.must_equal fade_in.upcase }
4549
end
4650

4751
t 'buttons_exact' do
48-
2.times { wait { buttons_exact(fade_in).first.name.must_equal fade_in } }
52+
2.times { wait { buttons_exact(fade_in).first.text.upcase.must_equal fade_in.upcase } }
4953
end
5054

5155
t { after_last }

android_tests/lib/android/specs/android/element/text.rb

+3-3
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ def must_raise_no_element
55
end
66

77
t 'text' do
8-
wait { text(1).text.must_equal 'API Demos' }
8+
wait { ['API Demos', "Access'ibility"].must_include text(1).text }
99
wait { text('mos').text.must_equal 'API Demos' }
1010
end
1111

@@ -15,11 +15,11 @@ def must_raise_no_element
1515
end
1616

1717
t 'first_text' do
18-
wait { first_text.text.must_equal 'API Demos' }
18+
wait { ['API Demos', "Access'ibility"].must_include first_text.text }
1919
end
2020

2121
t 'last_text' do
22-
wait { last_text.text.must_equal 'Views' }
22+
wait { ['API Demos', 'Views'].must_include last_text.text }
2323
end
2424

2525
t 'text_exact' do

android_tests/lib/android/specs/android/helper.rb

+19-5
Original file line numberDiff line numberDiff line change
@@ -11,11 +11,25 @@
1111

1212
act = get_page_class
1313
act.split("\n").length.must_be :>=, 5
14+
1415
act.must_include '13x android.widget.TextView'
15-
act.must_include '3x android.widget.FrameLayout'
16-
act.must_include '2x android.view.ViewGroup'
17-
act.must_include '1x android.widget.ListView'
18-
act.must_include '1x hierarchy'
16+
if automation_name_is_espresso?
17+
# 13x android.widget.TextView
18+
# 1x android.widget.ActionMenuView
19+
# 1x android.widget.Toolbar
20+
# 1x com.android.internal.widget.ActionBarContainer
21+
# 1x com.android.internal.widget.ActionBarContextView
22+
# 1x android.widget.ListView
23+
# 1x android.widget.FrameLayout
24+
# 1x com.android.internal.widget.ActionBarOverlayLayout
25+
# 1x com.android.internal.policy.DecorView
26+
act.must_include '1x android.widget.ActionMenuView'
27+
else
28+
act.must_include '3x android.widget.FrameLayout'
29+
act.must_include '2x android.view.ViewGroup'
30+
act.must_include '1x android.widget.ListView'
31+
act.must_include '1x hierarchy'
32+
end
1933
end
2034

2135
# t 'page_class' do # tested by get_page_class
@@ -64,7 +78,7 @@ def id_value
6478
wait { find('accessibility').click }
6579
wait { find('accessibility node provider').click }
6680

67-
if automation_name_is_uiautomator2?
81+
if automation_name_is_uiautomator2? || automation_name_is_espresso?
6882
wait { text 'Accessibility/Accessibility Node Provider' }
6983
else
7084
# With string.xml

android_tests/lib/android/specs/common/helper.rb

+8-4
Original file line numberDiff line numberDiff line change
@@ -77,15 +77,17 @@
7777
end
7878

7979
t 'xpath' do
80-
wait { xpath('//android.widget.TextView').name.must_equal 'API Demos' }
80+
# Access'ibility is for Espresso
81+
wait { ['API Demos', "Access'ibility"].must_include xpath('//android.widget.TextView').text }
8182
end
8283

8384
t 'xpaths' do
8485
wait { xpaths('//android.widget.TextView').length.must_equal 13 }
8586
end
8687

8788
t 'ele_index' do
88-
wait { ele_index('android.widget.TextView', 3).name.must_equal 'Accessibility' }
89+
# Animation is for Espresso
90+
wait { %w(Accessibility Animation).must_include ele_index('android.widget.TextView', 3).text }
8991
end
9092

9193
t 'tags' do
@@ -94,13 +96,15 @@
9496

9597
t 'first_ele' do
9698
wait do
97-
first_ele('android.widget.TextView').text.must_equal 'API Demos'
99+
# Access'ibility is for Espresso
100+
['API Demos', "Access'ibility"].must_include first_ele('android.widget.TextView').text
98101
end
99102
end
100103

101104
t 'last_ele' do
102105
wait do
103-
last_ele('android.widget.TextView').text.must_equal 'Views'
106+
# "API Demos" is for Espresso
107+
['API Demos', 'Views'].must_include last_ele('android.widget.TextView').text
104108
end
105109
end
106110

android_tests/lib/android/specs/common/patch.rb

+3-3
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212
# t 'value' do; end # Doesn't work on Android
1313

1414
t 'name' do
15-
wait { first_text.name.must_equal 'API Demos' }
15+
wait { first_text.text.must_equal 'API Demos' }
1616
end
1717

1818
# t 'tag_name' do; end # Doesn't work on Android
@@ -48,12 +48,12 @@
4848
if !automation_name_is_uiautomator2?
4949
wait do
5050
el = id 'autocomplete_3_button_7' # <string name="autocomplete_3_button_7">Text</string>
51-
el.name.must_equal 'Text'
51+
el.text.must_equal 'Text'
5252
end
5353
else
5454
wait do
5555
el = text 'text' # <string name="autocomplete_3_button_7">Text</string>
56-
el.name.must_equal 'Text'
56+
el.text.must_equal 'Text'
5757
end
5858
end
5959
end

android_tests/lib/android/specs/driver.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,7 @@ def sauce?
7979
diff = HashDiff.diff expected, dup_actual
8080
diff = "diff (expected, actual):\n#{diff}"
8181

82-
dup_actual[:caps][:app] = caps_app_for_teardown
82+
dup_actual[:caps][:app] = caps_app_for_teardown if dup_actual.key? :caps
8383
# example:
8484
# change :ios in expected to match 'ios' in actual
8585
# [["~", "caps.platformName", :ios, "ios"]]

lib/appium_lib/android/element/button.rb

+1-3
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ def button(value)
1414
index = value
1515
raise "#{index} is not a valid index. Must be >= 1" if index <= 0
1616

17+
# 1 indexed
1718
return find_element :uiautomator, _button_visible_selectors(index: index)
1819
end
1920

@@ -74,7 +75,6 @@ def raise_no_such_element_if_empty(elements)
7475
elements.first
7576
end
7677

77-
# @private
7878
def _button_visible_selectors(opts = {})
7979
button_index = opts.fetch :button_index, false
8080
image_button_index = opts.fetch :image_button_index, false
@@ -88,14 +88,12 @@ def _button_visible_selectors(opts = {})
8888
end
8989
end
9090

91-
# @private
9291
def _button_exact_string(value)
9392
button = string_visible_exact Button, value
9493
image_button = string_visible_exact ImageButton, value
9594
button + image_button
9695
end
9796

98-
# @private
9997
def _button_contains_string(value)
10098
button = string_visible_contains Button, value
10199
image_button = string_visible_contains ImageButton, value

lib/appium_lib/android/espresso.rb

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
1+
require_relative 'espresso/helper'
2+
require_relative 'espresso/element'
13
require_relative 'espresso/bridge'
24

35
module Appium
46
module Android
57
module Espresso
68
# parent
7-
end # module Uiautomator2
9+
end # module Espresso
810
end # module Android
911
end # module Appium

lib/appium_lib/android/espresso/bridge.rb

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ def self.for(target)
88
target.extend Appium::Android
99
target.extend Appium::Android::Command
1010
target.extend Appium::Android::Espresso
11+
target.extend Appium::Android::Espresso::Helper
12+
target.extend Appium::Android::Espresso::Element
1113
end
1214
end
1315
end
+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
require_relative 'element/generic'
2+
require_relative 'element/button'
3+
4+
module Appium
5+
module Android
6+
module Espresso
7+
module Element
8+
end # module Element
9+
end # module Espresso
10+
end # module Android
11+
end # module Appium
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,124 @@
1+
module Appium
2+
module Android
3+
module Espresso
4+
module Element
5+
# Find the first button that contains value or by index.
6+
# @param value [String, Integer] the value to exactly match.
7+
# If int then the button at that index is returned.
8+
# @return [Button]
9+
def button(value)
10+
# Don't use ele_index because that only works on one element type.
11+
# Android needs to combine button and image button to match iOS.
12+
if value.is_a? Numeric
13+
index = value
14+
raise "#{index} is not a valid index. Must be >= 1" if index <= 0
15+
16+
# zero index
17+
_button_visible_selectors_xpath(index: index - 1)
18+
end
19+
20+
i = find_elements :xpath, _button_contains_string_xpath(Button, value)
21+
e = find_elements :xpath, _button_contains_string_xpath(ImageButton, value)
22+
23+
raise_no_such_element_if_empty(i + e)
24+
25+
(i + e)[0]
26+
end
27+
28+
# Find all buttons containing value.
29+
# If value is omitted, all buttons are returned.
30+
# @param value [String] the value to search for
31+
# @return [Array<Button>]
32+
def buttons(value = false)
33+
return _button_visible_selectors_xpath unless value
34+
35+
i = find_elements :xpath, _button_contains_string_xpath(Button, value)
36+
e = find_elements :xpath, _button_contains_string_xpath(ImageButton, value)
37+
i + e
38+
end
39+
40+
# Find the first button.
41+
# @return [Button]
42+
def first_button
43+
_button_visible_selectors_xpath(button_index: 0, image_button_index: 0)
44+
end
45+
46+
# Find the last button.
47+
# @return [Button]
48+
def last_button
49+
# uiautomator index doesn't support last
50+
# and it's 0 indexed
51+
button_index = tags(::Appium::Android::Button).length
52+
button_index -= 1 if button_index > 0
53+
image_button_index = tags(::Appium::Android::ImageButton).length
54+
image_button_index -= 1 if image_button_index > 0
55+
56+
_button_visible_selectors_xpath(button_index: button_index,
57+
image_button_index: image_button_index)
58+
end
59+
60+
# Find the first button that exactly matches value.
61+
# @param value [String] the value to match exactly
62+
# @return [Button]
63+
def button_exact(value)
64+
i = find_elements :xpath, _button_exact_string_xpath(Button, value)
65+
e = find_elements :xpath, _button_exact_string_xpath(ImageButton, value)
66+
67+
raise_no_such_element_if_empty(i + e)
68+
69+
(i + e)[0]
70+
end
71+
72+
# Find all buttons that exactly match value.
73+
# @param value [String] the value to match exactly
74+
# @return [Array<Button>]
75+
def buttons_exact(value)
76+
i = find_elements :xpath, _button_exact_string_xpath(Button, value)
77+
e = find_elements :xpath, _button_exact_string_xpath(ImageButton, value)
78+
i + e
79+
end
80+
81+
private
82+
83+
# @private
84+
def raise_no_such_element_if_empty(elements)
85+
raise _no_such_element if elements.empty?
86+
87+
elements.first
88+
end
89+
90+
def _button_visible_selectors_xpath(opts = {})
91+
button_index = opts.fetch :button_index, false
92+
image_button_index = opts.fetch :image_button_index, false
93+
94+
index = opts.fetch :index, false
95+
96+
b = find_elements :xpath, "//#{Button}"
97+
i = find_elements :xpath, "//#{ImageButton}"
98+
99+
if index
100+
raise_no_such_element_if_empty(b + i)
101+
(b + i)[index]
102+
elsif button_index && image_button_index
103+
raise_no_such_element_if_empty(b + i)
104+
b_index = button_index + image_button_index
105+
(b + i)[b_index]
106+
else
107+
b + i
108+
end
109+
end
110+
111+
def _button_exact_string_xpath(class_name, value)
112+
r_id = resource_id(value, " or @resource-id='#{value}'")
113+
"//#{class_name}[@text='#{value}' or @content-desc='#{value}'#{r_id}]"
114+
end
115+
116+
def _button_contains_string_xpath(class_name, value)
117+
r_id = resource_id(value, " or @resource-id='#{value}'")
118+
"//#{class_name}[contains(translate(@text,'#{value.upcase}', '#{value}'), '#{value}')" \
119+
" or contains(translate(@content-desc,'#{value.upcase}', '#{value}'), '#{value}')#{r_id}]"
120+
end
121+
end # module Element
122+
end # module Espresso
123+
end # module Android
124+
end # module Appium

0 commit comments

Comments
 (0)