UiAutomator源码分析之获取控件信息

发布时间:2020-08-07 14:52:23 作者:zhukev
来源:网络 阅读:1837

根据上一篇文章《UiAutomator源码分析之注入事件》开始时提到的计划,这一篇文章我们要分析的是第二点:

我们在测试脚本中初始化一个UiObject的时候通常是像以下这个样子:
UiObject appsTab = new UiObject(new UiSelector().text("Apps")); appsTab.click()
那么这个过程发生了什么呢?这就是我们接下来要说的事情了。

1. 获取控件信息顺序图

这里依然是一个手画的不规范的顺序图,描述了UiObject尝试获得一个控件的过程中与相关的类的交互,这些类的关系在《UiAutomator源码分析之UiAutomatorBridge框架》中已经进行了描述。
UiAutomator源码分析之获取控件信息
这里整一个过程并不复杂,简单说明下就这几点:

2.触发控件查找真正发生的地方

在我没有去分析uiautomator的源代码之前,我一直以为空间查找是在通过UiSelector初始化一个UiObject的时候发生的:

UiObject appsTab = new UiObject(new UiSelector().text("Apps"));

这让我有一种先入为主的感觉,一个控件对象初始化好后应该就已经得到了该控件所代表的节点的所有信息了,但看了源码后发现事实并非如此,以上所做的事情只是以一定的格式准备好UiSelector选择子而已,真正触发uiautomator去获取控件节点信息的是在触发控件事件的时候,比如:

appsTab.click()
我们进入到代表一个控件的UiObject对应的操作控件的方法去看下就清楚了,以上面的click为例:

/*      */   public boolean click() /*      */     throws UiObjectNotFoundException /*      */   { /*  389 */     Tracer.trace(new Object[0]); /*  390 */     AccessibilityNodeInfo node = findAccessibilityNodeInfo(this.mConfig.getWaitForSelectorTimeout()); /*  391 */     if (node == null) { /*  392 */       throw new UiObjectNotFoundException(getSelector().toString()); /*      */     } /*  394 */     Rect rect = getVisibleBounds(node); /*  395 */     return getInteractionController().clickAndSync(rect.centerX(), rect.centerY(), this.mConfig.getActionAcknowledgmentTimeout()); /*      */   }
正式290行的调用触发uiautomator去调用UiAutomation去获取到我们想要的控件节点AccessibilityNodeInfo信息的。


3.获得根节点

下面我们看下uiautomator是怎么去获取到代表窗口所有控件的根的Root Node的,我们进入UiObject的findAccessibilityNodeInfo这个方法:

/*      */   protected AccessibilityNodeInfo findAccessibilityNodeInfo(long timeout) /*      */   { /*  164 */     AccessibilityNodeInfo node = null; /*  165 */     long startMills = SystemClock.uptimeMillis(); /*  166 */     long currentMills = 0L; /*  167 */     while (currentMills <= timeout) { /*  168 */       node = getQueryController().findAccessibilityNodeInfo(getSelector()); /*  169 */       if (node != null) { /*      */         break; /*      */       } /*      */        /*  173 */       UiDevice.getInstance().runWatchers(); /*      */        /*  175 */       currentMills = SystemClock.uptimeMillis() - startMills; /*  176 */       if (timeout > 0L) { /*  177 */         SystemClock.sleep(1000L); /*      */       } /*      */     } /*  180 */     return node; /*      */   }
UiObject对象会首先去获得一个QueryController对象,然后调用该对象的findAccessibilityNodeInfo同名方法:

/*     */   protected AccessibilityNodeInfo findAccessibilityNodeInfo(UiSelector selector, boolean isCounting) /*     */   { /* 143 */     this.mUiAutomatorBridge.waitForIdle(); /* 144 */     initializeNewSearch(); /*     */      /* 146 */     if (DEBUG) { /* 147 */       Log.d(LOG_TAG, "Searching: " + selector); /*     */     } /* 149 */     synchronized (this.mLock) { /* 150 */       AccessibilityNodeInfo rootNode = getRootNode(); /* 151 */       if (rootNode == null) { /* 152 */         Log.e(LOG_TAG, "Cannot proceed when root node is null. Aborted search"); /* 153 */         return null; /*     */       } /*     */        /*     */  /* 157 */       UiSelector uiSelector = new UiSelector(selector); /* 158 */       return translateCompoundSelector(uiSelector, rootNode, isCounting); /*     */     } /*     */   }
这里做了两个重要的事情:

好,我们继续往下进入getRootNode:
/*     */   protected AccessibilityNodeInfo getRootNode() /*     */   { /* 168 */     int maxRetry = 4; /* 169 */     long waitInterval = 250L; /* 170 */     AccessibilityNodeInfo rootNode = null; /* 171 */     for (int x = 0; x < 4; x++) { /* 172 */       rootNode = this.mUiAutomatorBridge.getRootInActiveWindow(); /* 173 */       if (rootNode != null) { /* 174 */         return rootNode; /*     */       } /* 176 */       if (x < 3) { /* 177 */         Log.e(LOG_TAG, "Got null root node from accessibility - Retrying..."); /* 178 */         SystemClock.sleep(250L); /*     */       } /*     */     } /* 181 */     return rootNode; /*     */   }
172调用的是UiAutomatorBridge对象的方法,通过我们上面的几篇文章我们知道UiAutomatorBridge提供的方法大部分都是直接调用UiAutomation的方法的,我们进去看看是否如此:
/*     */   public AccessibilityNodeInfo getRootInActiveWindow() { /*  66 */     return this.mUiAutomation.getRootInActiveWindow(); /*     */   }
果不其然,最终简单明了的直接调用UiAutomation的getRootInActiveWindow来获得根AccessibilityNodeInfo.

4.遍历根节点获得选择子UiSelector指定的控件

如前所述,QueryController的方法findAccessibilityNodeInfo在获得根节点后下来做的第二个事情:
里面的算法细节我就不打算去研究了,里面考虑到选择子嵌套的情况,分析起来也比较费力,且了解了它的算法对我去立交uiautomator的运行原理并没有非常大的帮助,我只需要知道给定一棵树的根,然后制定了我想要的叶子的属性,那么我遍历整棵树肯定是可以找到我想要的那个/些满足要求的控件的。大家由兴趣了解其算法的话还是自行去研究吧。

5.最终还是通过坐标点来点击控件

上面UiObject的Click方法通过UiAutomation这个高大上的新框架获得了代表我们目标控件的AccessibilityNodeInfo后,跟着是不是就直接调用这个节点的Click方法进行点击了呢?其实不是的,首先AccessibilityNodeInfo并没有click这个方法,我们继续看代码:
/*      */   public boolean click() /*      */     throws UiObjectNotFoundException /*      */   { /*  389 */     Tracer.trace(new Object[0]); /*  390 */     AccessibilityNodeInfo node = findAccessibilityNodeInfo(this.mConfig.getWaitForSelectorTimeout()); /*  391 */     if (node == null) { /*  392 */       throw new UiObjectNotFoundException(getSelector().toString()); /*      */     } /*  394 */     Rect rect = getVisibleBounds(node); /*  395 */     return getInteractionController().clickAndSync(rect.centerX(), rect.centerY(), this.mConfig.getActionAcknowledgmentTimeout()); /*      */   }
从395行可以看到,最终还是把控件节点的信息转换成控件的坐标点进行点击的,至于怎么点击,大家可以参照上一篇文章,无非就是通过建立一个runnable的线程进行点击事件的注入了

6.系列结语

UiAutomator源码分析这个系列到了这篇文章算是完结了,从启动运行,到核心的UiAutomatorBridge架构,到实例解剖,通过这些文章我相信大家已经很清楚uiautomator这个运用了UiAutomation框架与AccessibilityService通信的测试框架是怎么回事了,置于uiautomator那5个专供测试用例调用的类是怎么回事,网上可获得的信息不少,我这里就没有必要做从新造轮子的事情了,况且这些已经不是uiautomator这个框架的核心了,它们只是运用了UiAutomatorBridge这个核心的一些类而已。


 

作者

自主博客

微信

CSDN

天地会珠海分舵

http://techgogogo.com


服务号:TechGoGoGo

扫描码:

UiAutomator源码分析之获取控件信息

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》