Skip to content

Commit b095c4a

Browse files
committedApr 25, 2013
Rewrite as a real lib
1 parent 71628ff commit b095c4a

32 files changed

+1353
-1314
lines changed
 

‎Rakefile

+10-3
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ require 'date'
55
# Defines gem name.
66
def repo_name; 'appium_lib'; end # ruby_lib published as appium_lib
77
def gh_name; 'ruby_lib'; end # the name as used on github.com
8-
def version_file; "lib/#{repo_name}/version.rb"; end
8+
def version_file; "lib/#{repo_name}/common/version.rb"; end
99
def version_rgx; /VERSION = '([^']+)'/m; end
1010

1111
def version
@@ -79,9 +79,16 @@ desc 'Build a new gem (same as gem task)'
7979
task :build => :gem do
8080
end
8181

82+
desc 'Uninstall gem'
83+
task :uninstall do
84+
cmd = "gem uninstall -aIx #{repo_name}"
85+
puts cmd
86+
# rescue on gem not installed error.
87+
begin; `cmd`; rescue; end
88+
end
89+
8290
desc 'Install gem'
83-
task :install => :gem do
84-
`gem uninstall -aIx #{repo_name}`
91+
task :install => [ :gem, :uninstall ] do
8592
sh "gem install --no-rdoc --no-ri #{repo_name}-#{version}.gem"
8693
end
8794

‎appium_lib.gemspec

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,15 +8,15 @@ end
88

99
add_to_path 'lib'
1010

11-
require 'appium_lib/version'
11+
require 'appium_lib/common/version'
1212

1313
Gem::Specification.new do |s|
1414
# 1.8.x is not supported
1515
s.required_ruby_version = '>= 1.9.3'
1616

1717
s.name = 'appium_lib'
18-
s.version = AppiumLib::VERSION
19-
s.date = AppiumLib::DATE
18+
s.version = Appium::VERSION
19+
s.date = Appium::DATE
2020
s.license = 'http://www.apache.org/licenses/LICENSE-2.0.txt'
2121
s.description = s.summary = 'Ruby lib for use with Appium'
2222
s.description += '.' # avoid identical warning

‎lib/appium_lib.rb

+10-7
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,13 @@
1-
def self.add_to_path file, path=false
2-
path = path ? "../#{path}/" : '..'
3-
path = File.expand_path path, file
1+
# encoding: utf-8
2+
module Appium
3+
def self.add_to_path file, path=false
4+
path = path ? "../#{path}/" : '..'
5+
path = File.expand_path path, file
46

5-
$:.unshift path unless $:.include? path
6-
end
7+
$:.unshift path unless $:.include? path
8+
end
79

8-
add_to_path __FILE__, 'appium_lib'
10+
add_to_path __FILE__
911

10-
require 'console'
12+
require 'appium_lib/driver'
13+
end
+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
# encoding: utf-8
2+
module Appium::Android
3+
# Tap the alert button identified by value.
4+
# @param value [Integer, String] either an integer index of the button or the button's name
5+
# @return [void]
6+
def alert_click value
7+
button(value).click
8+
end
9+
10+
# Get the alert message text.
11+
# @return [String]
12+
def alert_text
13+
get_page
14+
end
15+
16+
# Accept the alert.
17+
# The last button is considered "accept."
18+
# @return [void]
19+
def alert_accept
20+
last_button.click
21+
end
22+
23+
# Get the text of the alert's accept button.
24+
# The last button is considered "accept."
25+
# @return [String]
26+
def alert_accept_text
27+
last_button.text
28+
end
29+
30+
# Dismiss the alert.
31+
# The first button is considered "dismiss."
32+
# @return [void]
33+
def alert_dismiss
34+
first_button.click
35+
end
36+
37+
# Get the text of the alert's dismiss button.
38+
# The first button is considered "dismiss."
39+
# @return [String]
40+
def alert_dismiss_text
41+
first_button.text
42+
end
43+
end # module Appium::Android
+94
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
# encoding: utf-8
2+
module Appium::Android
3+
=begin
4+
name, names, text, text should match substring and case insensitive.
5+
6+
In Android //* is used to find partial case insensitive text matches.
7+
//* is not currently implemented in iOS.
8+
9+
find_element :name by default uses a partial case insensitive match.
10+
On iOS the default is an exact name match.
11+
=end
12+
13+
=begin
14+
// iOS version
15+
// https://github.com/appium/ruby_lib/blob/37bb4e90b29e5adb4438b287b6387a504c94b5c4/lib/appium_lib/element/ios/generic.rb#L23
16+
var search = "name contains[c] '#{text}' || label contains[c] '#{text}' || value contains[c] '#{text}'";
17+
var a = w.secureTextFields().firstWithPredicate(search);
18+
if ( isNil(a) ) {
19+
a = w.textFields().firstWithPredicate(search);
20+
if ( isNil(a) ) {
21+
a = w.buttons().firstWithPredicate(search);
22+
if ( isNil(a) ) {
23+
a = w.elements().firstWithPredicate(search);
24+
}
25+
}
26+
}
27+
28+
Android considers both a textfield and a secure textfield to be "EditText".
29+
Name (the content desc) is searched first and then we search for value (text).
30+
There's no label in Android.
31+
32+
Android buttons have different class names (android.widget.Button, android.widget.ImageButton)
33+
so we consider the element a button if the class name contains the word button.
34+
35+
After looking for textfields and buttons, then we search all elements. Find will return
36+
the first element that matches.
37+
=end
38+
def find val
39+
# s.className('android.widget.EditText').descriptionContains(value);
40+
args = [ [4, 'android.widget.EditText'], [7, val] ],
41+
# s.className('android.widget.EditText').textContains(value);
42+
[ [4, 'android.widget.EditText'], [3, val] ],
43+
44+
# s.className('android.widget.Button').descriptionContains(value);
45+
[ [4, 'android.widget.Button'], [7, val] ],
46+
# s.className('android.widget.Button').textContains(value);
47+
[ [4, 'android.widget.Button'], [3, val] ],
48+
49+
# s.className('android.widget.ImageButton').descriptionContains(value);
50+
[ [4, 'android.widget.ImageButton'], [7, val] ],
51+
# s.className('android.widget.ImageButton').textContains(value);
52+
[ [4, 'android.widget.ImageButton'], [3, val] ],
53+
54+
# s.descriptionContains(value);
55+
[ [7, val] ],
56+
# s.textContains(value);
57+
[ [3, val] ]
58+
mobile :find, args
59+
end
60+
61+
# Return the first element matching text.
62+
# @param text [String] the text to search for
63+
# @return [Element] the first matching element
64+
def text text
65+
# Return the first element matching selector.
66+
# s.textContains(value)
67+
mobile :find, [ [ [3, text] ] ]
68+
end
69+
70+
# Return all elements matching text.
71+
# @param text [String] the text to search for
72+
# @return [Array<Element>] all matching elements
73+
def texts text
74+
@driver.find_elements :xpath, "//*[contains(@text, '#{text}')]"
75+
end
76+
77+
# Return the first element matching name.
78+
# on Android name is content description
79+
# on iOS name is the accessibility label or the text.
80+
# @param name [String] the name to search for
81+
# @return [Element] the first matching element
82+
def name name
83+
@driver.find_element :name, name
84+
end
85+
86+
# Return all elements matching name.
87+
# on Android name is content description
88+
# on iOS name is the accessibility label or the text.
89+
# @param name [String] the name to search for
90+
# @return [Array<Element>] all matching elements
91+
def names name
92+
@driver.find_elements :name, name
93+
end
94+
end # module Appium::Android
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
# encoding: utf-8
2+
module Appium::Android
3+
# UIATextField methods
4+
5+
# Get an array of textfield texts.
6+
# @return [Array<String>]
7+
def textfields
8+
find_eles_attr :textfield, :text
9+
end
10+
11+
# Get an array of textfield elements.
12+
# @return [Array<Textfield>]
13+
def e_textfields
14+
find_eles :textfield
15+
end
16+
17+
# Get the first textfield element.
18+
# @return [Textfield]
19+
def first_textfield
20+
first_ele :textfield
21+
end
22+
23+
# Get the last textfield element.
24+
# @return [Textfield]
25+
def last_textfield
26+
last_ele :textfield
27+
end
28+
29+
# Get the first textfield that matches text.
30+
# @param text [String, Integer] the text to match exactly. If int then the textfield at that index is returned.
31+
# @return [Textfield]
32+
def textfield text
33+
return ele_index :textfield, text if text.is_a? Numeric
34+
find_ele_by_text :textfield, text
35+
end
36+
37+
# Get the first textfield that includes text.
38+
# @param text [String] the text the textfield must include
39+
# @return [Textfield]
40+
def textfield_include text
41+
find_ele_by_text_include :textfield, text
42+
end
43+
end # module Appium::Android

‎lib/appium_lib/android/helper.rb

+120
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,120 @@
1+
# encoding: utf-8
2+
module Appium::Android
3+
# Returns an array of android classes that match the tag name
4+
def tag_name_to_android tag_name
5+
tag_name = tag_name.to_s.downcase.strip
6+
7+
def prefix *tags
8+
tags.map!{ |tag| "android.widget.#{tag}" }
9+
end
10+
# note that 'secure' is not an allowed tag name on android
11+
# because android can't tell what a secure textfield is
12+
# they're all edittexts.
13+
14+
# must match names in AndroidElementClassMap (Appium's Java server)
15+
case tag_name
16+
when 'button'
17+
prefix 'Button', 'ImageButton'
18+
when 'text'
19+
prefix 'TextView'
20+
when 'list'
21+
prefix 'ListView'
22+
when 'window', 'frame'
23+
prefix 'FrameLayout'
24+
when 'grid'
25+
prefix 'GridView'
26+
when 'relative'
27+
prefix 'RelativeLayout'
28+
when 'linear'
29+
prefix 'LinearLayout'
30+
when 'textfield'
31+
prefix 'EditText'
32+
else
33+
raise "Invalid tag name #{tag_name}"
34+
end # return result of case
35+
end
36+
37+
# on android, assume the attr is name (which falls back to text).
38+
def find_eles_attr tag_name
39+
=begin
40+
sel1 = [ [4, 'android.widget.Button'], [100] ]
41+
sel2 = [ [4, 'android.widget.ImageButton'], [100] ]
42+
43+
args = [ 'all', sel1, sel2 ]
44+
45+
mobile :find, args
46+
=end
47+
array = ['all']
48+
49+
tag_name_to_android(tag_name).each do |name|
50+
# sel.className(name).getStringAttribute("name")
51+
array.push [ [4, name], [100] ]
52+
end
53+
54+
mobile :find, array
55+
end
56+
57+
58+
# Android only.
59+
def get_inspect
60+
def run node
61+
r = []
62+
63+
run_internal = lambda do |node|
64+
if node.kind_of? Array
65+
node.each { |node| run_internal.call node }
66+
return
67+
end
68+
69+
keys = node.keys
70+
return if keys.empty?
71+
72+
obj = {}
73+
obj.merge!( { desc: node['@content-desc'] } ) if keys.include?('@content-desc') && !node['@content-desc'].empty?
74+
obj.merge!( { text: node['@text'] } ) if keys.include?('@text') && !node['@text'].empty?
75+
obj.merge!( { class: node['@class'] } ) if keys.include?('@class') && !obj.empty?
76+
77+
r.push obj if !obj.empty?
78+
run_internal.call node['node'] if keys.include?('node')
79+
end
80+
81+
run_internal.call node
82+
r
83+
end
84+
85+
json = JSON.parse(@driver.page_source)
86+
node = json['hierarchy']
87+
results = run node
88+
89+
out = ''
90+
results.each { |e|
91+
out += e[:class].split('.').last + "\n"
92+
93+
out += " class: #{e[:class]}\n"
94+
if ( e[:text] == e[:desc] )
95+
out += " text, name: #{e[:text]}\n" unless e[:text].nil?
96+
else
97+
out += " text: #{e[:text]}\n" unless e[:text].nil?
98+
out += " name: #{e[:desc]}\n" unless e[:desc].nil?
99+
end
100+
}
101+
out
102+
end
103+
104+
# Android only. Intended for use with console.
105+
# Inspects and prints the current page.
106+
def page
107+
puts get_inspect
108+
nil
109+
end
110+
111+
# JavaScript code from https://github.com/appium/appium/blob/master/app/android.js
112+
#
113+
# Math.round((duration * 1000) / 200)
114+
# (.20 * 1000) / 200 = 1
115+
#
116+
# We want steps to be exactly 1. If it's zero then a tap is used instead of a swipe.
117+
def fast_duration
118+
0.20
119+
end
120+
end # module Appium::Android

‎lib/appium_lib/android/patch.rb

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
# encoding: utf-8
2+
module Appium::Android
3+
# Implement useful features for element.
4+
class Selenium::WebDriver::Element
5+
# Cross platform way of entering text into a textfield
6+
def type text
7+
self.send_keys text
8+
end
9+
end
10+
end

0 commit comments

Comments
 (0)
Please sign in to comment.