AccessibilityService是一个辅助类,可以监听我们手机的焦点,窗口变化,按钮点击等等。实现它的服务需要在手机设置里面->辅助功能在这里面找到你自己实现的辅助类,然后打开它就可以进行我们一系列的监听,这个功能google的出发点是给那些肢体上有障碍的人使用的,比如手指不健全的用户,怎么才能滑动屏幕,然后打开一个应用呢?那么辅助功能就是干这些事,他的功能其实就是可以概括两句话:

第一、寻找到我们想要的View节点

第二、然后模拟点击,实现特定功能

我们知道Android中的View体系是一个树形结构,那么每一个View就是一个节点。所以我们可以查找到指定的节点,那么我们该如何查找到我们想要的节点呢?这里我们先看一下辅助功能(AccessibilityService)的用法。

第一步、我们需要集成AccessibilityService类

我们需要自定一个Service然后继承AccessibilityService,当然还需要在AndroidManifest.xml中声明这个服务:

<application>  
  <service android:name=".MyAccessibilityService"  
      android:label="@string/accessibility_service_label">  
    <intent-filter>  
      <action android:name="android.accessibilityservice.AccessibilityService" />  
    </intent-filter>
    <mate-data
     android:name="android.accessibilityservice"
     anroid:resource="@xml/accessibility"/>
  </service>  
<uses-permission android:name="android.permission.BIND_ACCESSIBILITY_SERVICE" />  
</application>

在<application>标签下添加指定了AccessibilityService的子类MyAccessibilityService,同时加入相应的权限。

<uses-permission android:name="android.permission.BIND_ACCESSIBILITY_SERVICE" /> 

当然还要一个meta-data的声明,这个声明是对这个AccessibilityService的配置。我们看一下配置文件内容:

(注:从Android4.0开始,开发者可以通过在AndroidManifest里添加<meta-data>标签,在标签里指出配置文件的位置)

<?xml version="1.0" encoding="utf-8"?>  
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"  
    android:accessibilityEventTypes="typeNotificationStateChanged|typeWindowStateChanged"  
    android:accessibilityFeedbackType="feedbackGeneric"  
   android:accessibilityFlags="flagDefault"  
    android:canRetrieveWindowContent="true"  
    android:description="@string/desc"  
    android:notificationTimeout="100"  
    android:packageNames="com.tencent.mm" />  

这里我们看到有很多选项,我们看一下常用的几个属性:

1、android:accessibilityEventTypes="typeAllMask"
看属性名也差不多可以明白,这个是用来设置响应事件的类型,typeAllMask当然就是响应所有类型的事件了。当然还有单击、长按、滑动等。

2、android:accessibilityFeedbackType="feedbackSpoken"
设置回馈给用户的方式,有语音播出和振动。可以配置一些TTS引擎,让它实现发音。

3、android:notificationTimeout="100"
响应时间的设置就不用多说了

4、android:packageNames="com.example.android.apis"
可以指定响应某个应用的事件,这里因为要响应所有应用的事件,所以不填,默认就是响应所有应用的事件。比如我们写一个微信抢红包的辅助程序,就可以在这里填写微信的包名,便可以监听微信产生的事件了。
我们这些配置信息除了在xml中定义,同样也可以在代码中定义,我们一般都是在onServiceConnected()方法里进行

@Override  
protected void onServiceConnected() {  
    AccessibilityServiceInfo info = getServiceInfo();  
    info.eventTypes = AccessibilityEvent.TYPES_ALL_MASK;  
    info.feedbackType = AccessibilityServiceInfo.FEEDBACK_SPOKEN;    
    info.notificationTimeout = 100;    
    setServiceInfo(info);   
    info.packageNames = new String[]{"xxx.xxx.xxx", "yyy.yyy.yyy","...."};  
    setServiceInfo(info);  
    super.onServiceConnected();  
}  

在子类MyAccessibilityService里实现几个重要的重载方法:
         onServiceConnected() - 可选。系统会在成功连接上你的服务的时候调用这个方法,在这个方法里你可以做一下初始化工作,例如设备的声音震动管理,也可以调用setServiceInfo()进行配置工作。
         onAccessibilityEvent() - 必须。通过这个函数可以接收系统发送来的AccessibilityEvent,接收来的AccessibilityEvent是经过过滤的,过滤是在配置工作时设置的。
         onInterrupt() - 必须。这个在系统想要中断AccessibilityService返给的响应时会调用。在整个生命周期里会被调用多次。

         onUnbind() - 可选。在系统将要关闭这个AccessibilityService会被调用。在这个方法中进行一些释放资源的工作。

2、这里我们一般都会在这里写上我们需要监听的应用的包名,但是有时候我们需要监听多个应用,那么这时候我们该怎么办呢?

第一种:我们在代码中注册多个应用的包名,从而可以监听多个应用:

 

@Override  
protected void onServiceConnected() {  
    AccessibilityServiceInfo info = getServiceInfo();  
    //这里可以设置多个包名,监听多个应用  
    info.packageNames = new String[]{"xxx.xxx.xxx", "yyy.yyy.yyy","...."};  
    setServiceInfo(info);  
    super.onServiceConnected();  
} 

 

第二种:我们在onAccessibilityEvent事件监听的方法中做包名的过滤(这种方式最常用)

@Override  
public void onAccessibilityEvent(AccessibilityEvent event) {  
    String pkgName = event.getPackageName().toString();  
    if("xxx.xxx.xxx".equals(pkgName)){  
  
    }else if("yyy.yyy.yyy".equals(pkgName)){  
  
    }else if("....".equals(pkgName)){  
  
    }  
} 

第三步、在onAccessibilityEvent方法中监听指定的事件

这里面最重要的部分就是onAccessibilityEvent这个回调函数,当我们注册了监听事件的时候,当有事件发生就会通知我们这个函数,但是一定要注意这个函数通知是异步的,当然很多朋友就会问这个是怎么通知到这里来的呢?他是通过AccessibilityDelegate这个代理类,发送出来的,这个类有个方法sendAccessibilityEvent可以发送事件。那这个类又怎么和我们的窗口联系呢?这里举个例子。比如我们的View类里面有个setAccessibilityDelegate这个方法,是不是这下一切都明了了呢?然后就是调用我们的find函数去当前节点里面找到我们需要的节点信息。

@Override
    public void onAccessibilityEvent(AccessibilityEvent event) {
        int eventType = event.getEventType();
        switch (eventType) {
        case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:
            //.......
        }
}

这个事件类型很多的,我们可以查看AccessibilityEvent类的源码

 

第四步、查找到我们想要处理的节点View

这里系统提供了两个方法让我们来进行查找想要的节点View

第一种是通过节点View的Text内容来查找

findAccessibilityNodeInfosByText("查找内容")

这种方式查找,就是像TextView,Button等View有文本内容的,可以使用这种方式快速的找到。

第二种是通过节点View在xml布局中的id名称

findAccessibilityNodeInfosByViewId("@id/xxx")

这个一般很难知道,但是我们在查找系统控件的时候还是可以做的,因为系统的控件的id是可以知道的,而且是统一的。
(关于这两个方法我们在写网页爬虫程序的时候可能知道,在html中通过tag/name/id等信息可以找到一个节点,原理都类似)

 

第五步、模拟点击指定事件

 

我们找到我们想要的View节点,调用方法模拟事件:

 

performAction(AccessibilityNodeInfo.ACTION_CLICK)

 

调用这个方法即可,当然这里的参数就是指定事件的名称,这个和AccessibilityEvent中监听的那些事件是一一对应的,这里是模拟点击事件,我们当然可以模拟View的滚动事件,长按事件等。

 

posted on 2016-07-08 23:41  ip小子  阅读(32306)  评论(0编辑  收藏  举报