最近使用AccessibilityService完成业务,感觉还是蛮有意思的,顺手写一下这个类的用法,将来要是再有需要用到的时候也比较方便复习查阅。
AccessibilityService是用于开发无障碍功能应用的api类,帮助残障人士使用app。同样的,使用它可以帮助我们对用户体验进行提升,例如手机助手中的一键安装,免去了我们多次点击的麻烦,它还能帮助我们完成一些看似外挂般的插件,比如抢红包插件。
如何使用AccessibilityService:
首先,创建子类继承AccessibilityService,AccessibilityService是服务的一种,那么我们需要在Mainifest注册,同时添加相对应的权限和过滤条件如下图。
AccessibilityService可以在xml中配置它的辅助信息即
我们需要在res文件夹中增加对应的xml文件res--xml--rob_service_config.xml,内容如下
eventTypes代表该服务关注的事件类型,例如:
typeNotificationStateChanged 通知栏状态改变typeViewClicked 点击事件typeWindowStateChanged 窗体状态变化typeAllMask 拦截所有的事件
等等等,这里我们可以根据我们的需求去选择拦截类型。
packageNames即我们想要监听的应用包名,可以监听多个应用,包名之间以","隔开。
同样的我们可以在代码中配置这些信息:
@Override public void onCreate() { super.onCreate(); // AccessibilityServiceInfo info = new AccessibilityServiceInfo(); // info.packageNames = installPackge; //监听过滤的包名 // info.eventTypes = AccessibilityEvent.TYPES_ALL_MASK; //监听哪些行为 // info.feedbackType = AccessibilityServiceInfo.FEEDBACK_SPOKEN; //反馈 // info.notificationTimeout = 100; //通知的时间 // setServiceInfo(info); }
做完这些以后,开启服务,打开设置中的辅助选项,选择你的辅助服务并开启,你就可以享受拦截其他app各种操作的快感了~那么,拦截到的事件在什么地方呢,AccessibilityService是一个抽象类,它的核心方法就是他的抽象方法
public abstract void onAccessibilityEvent(AccessibilityEvent event);
这个event就是我们拦截到的事件,这个方法是异步执行的(当然了,万一这事件被拦截并处理了半天,人家的app还让不让人用了),这个时间有view中的accessibilityDelegate对象发出。
好了,事件拿到了,就能对里面的内容大做文章了,AccessibilityEvent最重要的就是它的eventType了,
/** * Gets the event type. * * @return The event type. */ public int getEventType() { return mEventType; }
毕竟我们需要知道这是什么类型时间才好继续往下做嘛,然后就是
String className = event.getClassName().toString()
因为监听的是别人家的app,我们不知道我们想要知道的页面类名是什么,所以想要在指定页面做指定事情那就需要这个页面的类名,有了这个方法,你还会不知道别人家的活动页面的类名叫什么了吗?
接下来就是getSource了获取事件的节点信息,又或者我们可以使用
/** * Gets the root node in the currently active window if this service * can retrieve window content. The active window is the one that the user * is currently touching or the window with input focus, if the user is not * touching any window. ** Note: In order to access the root node your service has * to declare the capability to retrieve window content by setting the * {@link android.R.styleable#AccessibilityService_canRetrieveWindowContent} * property in its meta-data. For details refer to {@link #SERVICE_META_DATA}. *
* * @return The root node if this service can retrieve window content. */ public AccessibilityNodeInfo getRootInActiveWindow() { return AccessibilityInteractionClient.getInstance().getRootInActiveWindow(mConnectionId); }
获取该事件触发时的活动窗口,从这个活动窗口拿到我们想要的信息,比如包含某文字的控件,或者执行想要做的辅助事件,比如点击、滚动等。AccessibilityNodeIfo对象包含了树状父子节点信息,
public ListfindAccessibilityNodeInfosByViewId(String viewId){...}public List findAccessibilityNodeInfosByText(String text){...}
这两个方法可以找到包含某文字或者某控件id名称的节点
public boolean performAction(int action) {...}public boolean performAction(int action, Bundle arguments){...}
这两个方法可以使该节点执行某个动作比如
AccessibilityNodeInfoA.CTION_CLICK
此外对于部分action,可以携带额外的参数
Bundle bundle = new Bundle(); bundle.putInt(AccessibilityNodeInfo.ACTION_ARGUMENT_SELECTION_START_INT,0); bundle.putInt(AccessibilityNodeInfo.ACTION_ARGUMENT_SELECTION_END_INT,1); nodeInfo.performAction(AccessibilityNodeInfo.ACTION_SET_SELECTION,bundle);
此外AccessibilityService本身有一个方法
* @see #GLOBAL_ACTION_BACK * @see #GLOBAL_ACTION_HOME * @see #GLOBAL_ACTION_NOTIFICATIONS * @see #GLOBAL_ACTION_RECENTS */ public final boolean performGlobalAction(int action)
让我们去执行全局的动作,如回退,返回home页等。
AccessibilityService的使用大体就是这样了,至于再详细的东西就一个个的去文档中找,并一个个使用它们吧。
此外,再记一下使用AccessibilityService过程中用到的tool
一个是D:\Users\XXX\Android\sdk\build-tools\23.0.0目录下的aapt.exe
使用cmd切换到aapt目录执行aapt dump badging <file_path.apk>可以查看指定apk的包名等详细内容;这里有十分详细的介绍,对于这里来说,我们只需要知道目标应用的包名即可;
另一个是D:\Users\XXX\Android\sdk\tools目录下的uiautomatorviewer.bat
打开此文件连接手机,点击工具的截屏按钮,稍后你就能看到该页面的布局层次以及各控件的信息(有我AccessibilityService需要的resId,这样妈妈在不用担心我找不到想要的那个nodeInfo了)。
Ok,this is the end.