它的具体实现是通过AccessibilityService服务运行在后台中,通过AccessibilityEvent接收指定事件的回调。这样的事件表示用户在界面中的一些状态转换,例如:焦点改变了,一个按钮被点击,等等。这样的服务可以选择请求活动窗口的内容的能力。简单的说AccessibilityService就是一个后台监控服务,当你监控的内容发生改变时,就会调用后台服务的回调方法
AccessibilityService使用
1.1 创建服务类
编写自己的Service类,重写onServiceConnected()方法、onAccessibilityEvent()方法和onInterrupt()方法
public class QHBAccessibilityService extends AccessibilityService { /** * 当启动服务的时候就会被调用 */ @Override protected void onServiceConnected() { super.onServiceConnected(); } /** * 监听窗口变化的回调 */ @Override public void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); //根据事件回调类型进行处理 } /** * 中断服务的回调 */ @Override public void onInterrupt() { }}下面是对AccessibilityService中常用的方法的介绍
<service android:name=".AccessibilityService.QHBAccessibilityService" android:enabled="true" android:exported="true" android:label="@string/label" android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"> <intent-filter> <action android:name="android.accessibilityservice.AccessibilityService" /> </intent-filter></service>我们必须注意:任何一个信息配置错误,都会使该服务无反应
android:permission:需要指定BIND_ACCESSIBILITY_SERVICE权限,这是4.0以上的系统要求的
intent-filter:这个name是固定不变的
1.3 配置服务参数
配置服务参数是指:配置用来接受指定类型的事件,监听指定package,检索窗口内容,获取事件类型的时间等等。其配置服务参数有两种方法:
方法一:安卓4.0之后可以通过meta-data标签指定xml文件进行配置
方法二:通过代码动态配置参数
1.3.1 方法一
在原先的manifests中增加meta-data标签指定xml文件
<service android:name=".AccessibilityService.QHBAccessibilityService" android:enabled="true" android:exported="true" android:label="@string/label" android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"> <intent-filter> <action android:name="android.accessibilityservice.AccessibilityService" /> </intent-filter> <meta-data android:name="android.accessibilityservice" android:resource="@xml/accessibility_service_config" /></service>接下来是accessibility_service_config文件的配置
<?xml version="1.0" encoding="utf-8"?><accessibility-service xmlns:android="http://schemas.android.com/apk/res/android" android:accessibilityEventTypes="typeNotificationStateChanged|typeWindowStateChanged|typeWindowContentChanged|typeWindowsChanged" android:accessibilityFeedbackType="feedbackGeneric" android:accessibilityFlags="flagDefault" android:canRetrieveWindowContent="true" android:description="@string/description" android:notificationTimeout="100" android:packageNames="com.tencent.mm" />下面是对xml参数的介绍
notificationTimeout:接受事件的时间间隔,通常将其设置为100即可
packageNames:表示对该服务是用来监听哪个包的产生的事件,这里以微信的包名为例
1.3.2 方法二
通过代码为我们的AccessibilityService配置AccessibilityServiceInfo信息,这里我们可以抽取成一个方法进行设置
private void settingAccessibilityInfo() { String[] packageNames = {"com.tencent.mm"}; AccessibilityServiceInfo mAccessibilityServiceInfo = new AccessibilityServiceInfo(); // 响应事件的类型,这里是全部的响应事件(长按,单击,滑动等) mAccessibilityServiceInfo.eventTypes = AccessibilityEvent.TYPES_ALL_MASK; // 反馈给用户的类型,这里是语音提示 mAccessibilityServiceInfo.feedbackType = AccessibilityServiceInfo.FEEDBACK_SPOKEN; // 过滤的包名 mAccessibilityServiceInfo.packageNames = packageNames; setServiceInfo(mAccessibilityServiceInfo);}在这里涉及到了AccessibilityServiceInfo类,AccessibilityServiceInfo类被用于配置AccessibilityService信息,该类中包含了大量用于配置的常量字段及用来xml属性,常见的有:accessibilityEventTypes,canRequestFilterKeyEvents,packageNames等等
Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);startActivity(intent);1.5 处理事件信息
@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); //根据事件回调类型进行处理 switch (eventType) { //当通知栏发生改变时 case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:break; //当窗口的状态发生改变时 case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:break; }}当我们微信收到通知时,状态栏会有一条推送信息到达,这个时候就会被TYPE_NOTIFICATION_STATE_CHANGED监听,执行里面的内容,当我们切换微信界面时,或者使用微信时,这个时候就会被TYPE_WINDOW_STATE_CHANGED监听,执行里面的内容
//通过文本找到对应的节点集合List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByText(text);//通过控件ID找到对应的节点集合,如com.tencent.mm:id/gdList<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByViewId(clickId);1.7 模拟节点点击
//模拟点击accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK);//模拟长按accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_LONG_CLICK);//模拟获取焦点accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_FOCUS);//模拟粘贴accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_PASTE);抢红包插件实现
获取”开”按钮ID和返回按钮ID
2.4 代码实现
注意:这里使用的是微信最新6.3.30版本的控件ID,如果是其他版本的请自行适配
/** * =====作者===== * 许英俊 * =====时间===== * 2016/11/19. */public class QHBAccessibilityService extends AccessibilityService { private List<AccessibilityNodeInfo> parents; /*** 当启动服务的时候就会被调用*/ @Override protected void onServiceConnected() {super.onServiceConnected();parents = new ArrayList<>(); } /*** 监听窗口变化的回调*/ @Override public void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) { //当通知栏发生改变时 case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:List<CharSequence> texts = event.getText();if (!texts.isEmpty()) { for (CharSequence text : texts) {String content = text.toString();if (content.contains("[微信红包]")) { //模拟打开通知栏消息,即打开微信 if (event.getParcelableData() != null && event.getParcelableData() instanceof Notification) {Notification notification = (Notification) event.getParcelableData();PendingIntent pendingIntent = notification.contentIntent;try { pendingIntent.send(); Log.e("demo","进入微信");} catch (Exception e) { e.printStackTrace();} }} }}break; //当窗口的状态发生改变时 case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:String className = event.getClassName().toString();if (className.equals("com.tencent.mm.ui.LauncherUI")) { //点击最后一个红包 Log.e("demo","点击红包"); getLastPacket();} else if (className.equals("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI")) { //开红包 Log.e("demo","开红包"); inputClick("com.tencent.mm:id/bg7");} else if (className.equals("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyDetailUI")) { //退出红包 Log.e("demo","退出红包"); inputClick("com.tencent.mm:id/gd");}break;} } /*** 通过ID获取控件,并进行模拟点击* @param clickId*/ @TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR2) private void inputClick(String clickId) {AccessibilityNodeInfo nodeInfo = getRootInActiveWindow();if (nodeInfo != null) { List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByViewId(clickId); for (AccessibilityNodeInfo item : list) {item.performAction(AccessibilityNodeInfo.ACTION_CLICK); }} } /*** 获取List中最后一个红包,并进行模拟点击*/ private void getLastPacket() {AccessibilityNodeInfo rootNode = getRootInActiveWindow();recycle(rootNode);if(parents.size()>0){ parents.get(parents.size() - 1).performAction(AccessibilityNodeInfo.ACTION_CLICK);} } /*** 回归函数遍历每一个节点,并将含有"领取红包"存进List中** @param info*/ public void recycle(AccessibilityNodeInfo info) {if (info.getChildCount() == 0) { if (info.getText() != null) {if ("领取红包".equals(info.getText().toString())) { if (info.isClickable()) {info.performAction(AccessibilityNodeInfo.ACTION_CLICK); } AccessibilityNodeInfo parent = info.getParent(); while (parent != null) {if (parent.isClickable()) { parents.add(parent); break;}parent = parent.getParent(); }} }} else { for (int i = 0; i < info.getChildCount(); i++) {if (info.getChild(i) != null) { recycle(info.getChild(i));} }} } /*** 中断服务的回调*/ @Override public void onInterrupt() { }}当收到红包发送的时候,Log的打印信息