Android AccessibilityService实现微信抢红包插件

在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人们

它的具体实现是通过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中常用的方法的介绍

disableSelf():禁用当前服务,也就是在服务可以通过该方法停止运行
findFoucs(int falg):查找拥有特定焦点类型的控件
getRootInActiveWindow():如果配置能够获取窗口内容,则会返回当前活动窗口的根结点
getSeviceInfo():获取当前服务的配置信息
onAccessibilityEvent(AccessibilityEvent event):有关AccessibilityEvent事件的回调函数,系统通过sendAccessibiliyEvent()不断的发送AccessibilityEvent到此处
performGlobalAction(int action):执行全局操作,比如返回,回到主页,打开最近等操作
setServiceInfo(AccessibilityServiceInfo info):设置当前服务的配置信息
getSystemService(String name):获取系统服务
onKeyEvent(KeyEvent event):如果允许服务监听按键操作,该方法是按键事件的回调,需要注意,这个过程发生了系统处理按键事件之前
onServiceConnected():系统成功绑定该服务时被触发,也就是当你在设置中开启相应的服务,系统成功的绑定了该服务时会触发,通常我们可以在这里做一些初始化操作
onInterrupt():服务中断时的回调

1.2 声明服务

既然是个后台服务,那么就需要我们在manifests中配置该服务信息

<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:label:在无障碍列表中显示该服务的名字

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参数的介绍

accessibilityEventTypes:表示该服务对界面中的哪些变化感兴趣,即哪些事件通知,比如窗口打开,滑动,焦点变化,长按等。具体的值可以在AccessibilityEvent类中查到,如typeAllMask表示接受所有的事件通知
accessibilityFeedbackType:表示反馈方式,比如是语音播放,还是震动
canRetrieveWindowContent:表示该服务能否访问活动窗口中的内容。也就是如果你希望在服务中获取窗体内容,则需要设置其值为true
description:对该无障碍功能的描述,具体体现在下图

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等等

1.4 启动服务

这里我们需要在无障碍功能里面手动打开该项功能,否则无法继续进行,通过下面代码可以打开系统的无障碍功能列表

Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS); startActivity(intent);

1.5 处理事件信息

由于我们监听了事件的通知栏和界面等信息,当我们指定packageNames的通知栏或者界面发生变化时,会通过onAccessibilityEvent回调我们的事件,接着进行事件的处理

@Override public 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监听,执行里面的内容

AccessibilityEvent的方法

getEventType():事件类型
getSource():获取事件源对应的结点信息
getClassName():获取事件源对应类的类型,比如点击事件是有某个Button产生的,那么此时获取的就是Button的完整类名
getText():获取事件源的文本信息,比如事件是有TextView发出的,此时获取的就是TextView的text属性。如果该事件源是树结构,那么此时获取的是这个树上所有具有text属性的值的集合
isEnabled():事件源(对应的界面控件)是否处在可用状态
getItemCount():如果事件源是树结构,将返回该树根节点下子节点的数量

1.6 获取节点信息

获取了界面窗口变化后,这个时候就要获取控件的节点。整个窗口的节点本质是个树结构,通过以下操作节点信息

1.6.1 获取窗口节点(根节点)

AccessibilityNodeInfo nodeInfo = getRootInActiveWindow();

1.6.2 获取指定子节点(控件节点)

//通过文本找到对应的节点集合 List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByText(text); //通过控件ID找到对应的节点集合,如com.tencent.mm:id/gd List<AccessibilityNodeInfo> list = nodeInfo.findAccessibilityNodeInfosByViewId(clickId);

1.7 模拟节点点击

当我们获取了节点信息之后,对控件节点进行模拟点击、长按等操作,AccessibilityNodeInfo类提供了performAction()方法让我们执行模拟操作,具体操作可看官方文档介绍,这里列举常用的操作

//模拟点击 accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK); //模拟长按 accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_LONG_CLICK); //模拟获取焦点 accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_FOCUS); //模拟粘贴 accessibilityNodeInfo.performAction(AccessibilityNodeInfo.ACTION_PASTE);

抢红包插件实现

2.1 原理分析

1、收到微信红包的推送信息,在推送信息中判断是否出现”[微信红包]”的消息提示,如果出现则点击进入聊天界面
2、通过遍历窗口树节点,发现带有”领取红包”字样的节点,则点击进入,即红包,弹出抢红包界面
3、在抢红包界面,通过ID获取”开”按钮的节点,则打开红包
4、在红包详情页面,通过ID获取返回键按钮的节点,点击并返回微信聊天界面

2.2 注意事项

1、由于微信每个版本的按钮ID都是不一样的,在我们的程序中是需要去修改按钮ID,以达到版本的适配
2、在获取控件ID的时候,注意其布局是否可点击,否则获取不可点击的控件,会使程序无反应

2.3 获取控件ID

当我们手机接入USB线时,在Android Device Monitor中的选择设备并开启Dump View Hierarchy for UI Automator工具,通过它可以获取控件信息

获取”开”按钮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的打印信息

11-21 13:53:06.275 2909-2909/com.handsome.boke2 E/demo: 进入微信
11-21 13:53:06.921 2909-2909/com.handsome.boke2 E/demo: 点击红包
11-21 13:53:07.883 2909-2909/com.handsome.boke2 E/demo: 开红包
11-21 13:53:08.732 2909-2909/com.handsome.boke2 E/demo: 退出红包

你可能会想到做一些窃取信息的软件,比如获取QQ密码、支付宝密码等等,哈哈,凡是EditText中设置inputType为password类型的,都无法获取其输入值

2.5 源码下载:http://xiazai.jb51.net/201611/yuanma/Androidwxpackage(jb51.net).rar

本文已被整理到了《Android微信开发教程汇总》,欢迎大家学习阅读。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

时间: 2024-10-12 12:40:21

Android AccessibilityService实现微信抢红包插件的相关文章

Android AccessibilityService实现微信抢红包插件_Android

在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人们 它的具体实现是通过AccessibilityService服务运行在后台中,通过AccessibilityEvent接收指定事件的回调.这样的事件表示用户在界面中的一些状态转换,例如:焦点改变了,一个按钮被点击,等等.这样的服务可以选择请求活动窗口的内容的能力.简单的说AccessibilityService就是一个后

Android中微信抢红包插件原理解析及开发思路_Android

一.前言 自从去年中微信添加抢红包的功能,微信的电商之旅算是正式开始正式火爆起来.但是作为Android开发者来说,我们在抢红包的同时意识到了很多问题,就是手动去抢红包的速度慢了,当然这些有很多原因导致了.或许是网络的原因,而且这个也是最大的原因.但是其他的不可忽略的因素也是要考虑到进去的,比如在手机充电锁屏的时候,我们并不知道有人已经开始发红包了,那么这时候也是让我们丧失了一大批红包的原因.那么关于网络的问题,我们开发者可能用相关技术无法解决(当然在Google和Facebook看来的话,他们

Android中微信抢红包插件原理解析及开发思路

一.前言 自从去年中微信添加抢红包的功能,微信的电商之旅算是正式开始正式火爆起来.但是作为Android开发者来说,我们在抢红包的同时意识到了很多问题,就是手动去抢红包的速度慢了,当然这些有很多原因导致了.或许是网络的原因,而且这个也是最大的原因.但是其他的不可忽略的因素也是要考虑到进去的,比如在手机充电锁屏的时候,我们并不知道有人已经开始发红包了,那么这时候也是让我们丧失了一大批红包的原因.那么关于网络的问题,我们开发者可能用相关技术无法解决(当然在Google和Facebook看来的话,他们

Android微信自动抢红包插件优化和实现_Android

又是兴趣系列 网上有很多自动强红包的例子和代码,笔者也是做了一些优化. 先说说自己的两个个优势 1.可以在聊天界面自动强不依赖于通知栏推送 2.可以在屏幕熄灭的时候的时候点亮屏幕自动抢(目前只测过flyme) 先上图: 代码传送门: https://github.com/AndroidMsky/WXhongbao 欢迎点星星~ 原理: 通过AccessibilityService监听到状态栏通知,进行模拟点击,获取屏幕中view节点为领取红包的list并且点击最后一个. 如果不通过状态栏通知,通

Android微信自动抢红包插件优化和实现

又是兴趣系列 网上有很多自动强红包的例子和代码,笔者也是做了一些优化. 先说说自己的两个个优势 1.可以在聊天界面自动强不依赖于通知栏推送 2.可以在屏幕熄灭的时候的时候点亮屏幕自动抢(目前只测过flyme) 先上图: 代码传送门: https://github.com/AndroidMsky/WXhongbao 欢迎点星星~ 原理: 通过AccessibilityService监听到状态栏通知,进行模拟点击,获取屏幕中view节点为领取红包的list并且点击最后一个. 如果不通过状态栏通知,通

Android抢红包插件实现原理浅析_Android

抢红包,先看效果图~ 实现自动抢红包,解决问题有两点: 一:如何实时监听发红包的事件 二:如何在红包到来的时候自动进入页面并自动点击红包 一.如何获取红包到来的事件 为了获取红包到来状态栏的变化,我们要用到一个类:Accessibility 许多Android使用者因为各种情况导致他们要以不同的方式与手机交互. 这包括了有些用户由于视力上,身体上,年龄上的问题致使他们不能看完整的屏幕或者使用触屏,也包括了无法很好接收到语音信息和提示的听力能力比较弱的用户. Android提供了Accessibi

微信/QQ抢红包插件安装使用教程

  (温馨提示:该资源由未经认证的个人源发布,可能存在安全风险,请谨慎下载.) iOS9越狱微信/QQ抢红包插件安装使用教程: 1.下载微信/QQ抢红包插件deb文件 2.设备连接PP助手电脑版,在[文件-文件系统(越狱)]中找到路径[/var/root/Media/Cydia/AutoInstall],导入已下载的微信/QQ插件deb文件,然后重启设备即可自动安装完成. 安装了微信/QQ抢红包插件之后怎么抢红包: 安装好微信抢红包插件之后,你可以在微信中"我"的界面看到[红包开关]按

分享Android微信红包插件_Android

本文实例为大家分享了Android微信红包插件,供大家参考,具体内容如下 效果图: 具体代码 @TargetApi(Build.VERSION_CODES.JELLY_BEAN) private void handleWindowChange(AccessibilityNodeInfo nodeInfo) { switch (Stage.getInstance().getCurrentStage()) { case Stage.OPENING_STAGE: // 调试信息,打印TTL // Lo

分享Android微信红包插件

本文实例为大家分享了Android微信红包插件,供大家参考,具体内容如下 效果图: 具体代码 @TargetApi(Build.VERSION_CODES.JELLY_BEAN) private void handleWindowChange(AccessibilityNodeInfo nodeInfo) { switch (Stage.getInstance().getCurrentStage()) { case Stage.OPENING_STAGE: // 调试信息,打印TTL // Lo