相信很多人使用AccessibilityService获取WebView内容的时候是为空的,使用AccessibilityNodeInfo的findAccessibilityNodeInfosByText方法是获取不到WebView里面的内容的。
解决方法:
给AccessibilityService服务添加一个Flag,下面的所有代码是Kotlin的写法
// 设置标志位,读取webView内容
serviceInfo.flags = serviceInfo.flags or AccessibilityServiceInfo.FLAG_REQUEST_ENHANCED_WEB_ACCESSIBILITY
serviceInfo 是AccessibilityService服务的配置信息,可以使用AccessibilityService的getServiceInfo()方法获取,因为Kotlin写法简化了。
如果想获取WebView的子控件信息的封装类AccessibilityNodeInfo,使用findAccessibilityNodeInfosByText是不行的,只能先找出WebView控件的AccessibilityNodeInfo再遍历它的子控件来获取。
下面是查找页面的WebView控件的代码:
/**
* 递归遍历出WebView节点
*/
private var accessibilityNodeInfoWebView: AccessibilityNodeInfo? = null
private fun findWebViewNode(rootNode: AccessibilityNodeInfo) {
for (i in 0 until rootNode.childCount) {
val child = rootNode.getChild(i)
if ("android.webkit.WebView" == child.className) {
accessibilityNodeInfoWebView = child
Log.d("findWebViewNode--", "找到webView")
return
}
if (child.childCount > 0) {
findWebViewNode(child)
}
}
}
传入的rootNode 是AccessibilityService的rootInActiveWindow,就是页面视图的根
现在找出了WebView的AccessibilityNodeInfo信息accessibilityNodeInfoWebView,然后我们就可以遍历它的子控件获取想要的内容了。
private fun getRecordNode(webviewNode: AccessibilityNodeInfo) {
val count = webviewNode.childCount
Log.e("getRecordNode--", "孩子数:$count")
...
val child = webviewNode.getChild(i)
child.contentDescription.toString()
...
}
注意的地方:
WebView的子控件是没有ID的,只能获取contentDescription等有限的信息。
目前测试WebView里面的控件也不可以模拟点击,滚动等操作,估计非原生控件的缘故,找到好方法的可以告诉我。
还有WebView的内容在加载完成之后整个页面的内容都可以获取,包括屏幕不可见的部分,区别于原生控件的列表视图只可以获取屏幕可见的控件,原生控件的列表视图是经过优化的,为达到减少内存消耗的目的,只创建满足屏幕高度的有限个数Item,然后在滚动的时候重用不可见的Item。
系统自带有一个叫Talkback的AccessibilityService,它是可以获取WebView内容的,可以参考它的源码:
参考:
How to access html content of AccessibilityNodeInfo of a WebView element using Accessibility Service in Android?
https://stackoverflow.com/questions/40522043/how-to-access-html-content-of-accessibilitynodeinfo-of-a-webview-element-using-a