对于Android而言,查找appUI界面元素属性的工具有三种:appium desktop,uiautomatorviewer.bat,weditor。之前已经介绍过了weditor的使用,这里我将通过使用uiautomatorview工具查看元素的属性值,来介绍Android app通过appium进行元素定位常用的几种方法。
一,使用uiautomatorview对app页面截图,查看元素属性
- 连接Android手机
- 在Android SDK的目录里的tools文件夹中找到uiautomatorviewer.bat文件,双击启动,会出现如下页面:(未安装Android SDK的请参考Android SDK 安装与配置)
建议创建uiautomatorviewer.bat的快捷方式到桌面,下次使用时就不需要进入到android-sdk/tools里打开。
- 如下图,1所示位置从左至右依次为:Open(打开之前保存的布局),Device Screenshot(获取详细布局),Device Screenshot -compressed(获取简单布局),Save(保存当前页面的布局);2所示位置展示app当前页面的UI树;3所示位置展示当前app页面被选择元素的属性值。
-
确认手机连接后,点击页面的Device Screenshot(获取详细布局)按钮,获取app当前页面的截图。此时工具页面可能会报错,请参考以下方法解决:
1,手机系统为Android 8.0以上版本请参考Android8.0以上版本使用uiautomatorviewer定位报错解决方法,替换SDK中相应的文件。
2,参考教程替换SDK中相应的文件后,仍然报错。请确认appium是否关闭,appium开启的状态下,uiautomatorviewer会与之冲突。
查看app当前页面的元素及对应的元素属性值。
以今日头条app为例,Node Detail里展示的便是【发布】按钮的各种属性值,后续在自动化代码中,我们就可以通过查找元素对应的属性来定位该元素并做相关操作。
二,appium移动端自动化常用的元素定位方法
appium继承了selenium框架中webdriver提供的元素定位方法,接下介绍几种常用的方法。
- find_element_by_id()
如果某个元素的resource-id对应的值在app当前页面是唯一,那么就可以优先通过id定位该元素。以上图中点击【发布】按钮为例,写法如下
driver.find_element_by_id("com.ss.android.article.news:id/cde").click()
- find_element_by_class_name()
元素的class属性值唯一,可以通过class name定位。如下图中所指的元素,其class属性值在当前页面唯一
写法如下:
find_element_by_class_name("android.widget.HorizontalScrollView")
-
find_element_by_xpath()
如果不是原生app,这大概是元素定位过程中用的最多的方法,使用比较灵活。1,如果当前页面某元素的text属性值唯一,就可以使用text文本构造xpath,如【发布】按钮
# "//*[@text='发布']"该字符串表示:当前页面下所有元素里面text='发布'的元素
drvier.find_element_by_xpath("//*[@text='发布']").click()
2,class唯一的话可以使用find_element_by_class_name()定位,但大部分情况向class都不是唯一的,此时可以用class与其他属性结合构成xpath,如:
# class与text
driver.find_element_by_xpath("//android.widget.TextView[@text='消息通知']").click()
# class、text、resource-id结合,注意text与id之间使用and
driver.find_element_by_xpath("//android.widget.TextView[@text='消息通知' and @resource-id='com.ss.android.article.news:id/c_3']").click()
3,通过父元素定位子元素
在实际项目中,我们可能通过UI的层级结果进行定位。如下图,我们要定位【收藏】按钮,除了使用2方外,还可以它的上一级元素定位。
# 父元素下面有4个class相同的子元素,子元素后面不加索引,则默认定位的是第一个子元素
driver.find_element_by_xpath("//androidx.recyclerview.widget.RecyclerView/android.widget.RelativeLayout")
子元素有多个,通过class的索引定位
# 后面的2表示索引,注意uiautomatorviewer里显示的是从0开始的,而xpath里面是从1开始
driver.find_element_by_xpath("//androidx.recyclerview.widget.RecyclerView/android.widget.RelativeLayout[2]")
4,通过元素子元素定位父元素
即通过下一级元素定位上一级元素。
# 后面的/..表示当前级的上一级,此处即整个下载管理按钮区域
driver.find_element_by_xpath("//android.widget.TextView[@text='下载管理']/..")
同样定位元素的上一级的上一级(孙子定位爷爷),如下
driver.find_element_by_xpath("//android.widget.TextView[@text='下载管理']/../..")
5,定位兄弟元素
通过同级元素定位,下面以定位【下载管理】上的图标为例
# 兄->父->兄
driver.find_element_by_xpath("//android.widget.TextView[@text='下载管理']/../android.widget.ImageView")
三,总结。
- 具体使用什么样的方式定位,与app开发框架有关,一般使用id与xpath比较多。需要注意的是,能使用id定位元素的话就不要使用xpath定位,因为id定位元素方便维护的同时,效率也相对xpath更高。
- xpath语法的基础知识需要有所了解,这样有助于能更好的理解元素定位。
-
当然appium还有其他的定位元素方法,这里不做过多的介绍,有兴趣的话可以去百度其他方法。实际使用过程中,常用的就是以上几种方法。