参考(感谢作者):http://www.jianshu.com/p/cd1cd53909d7

http://blog.csdn.net/jiangwei0910410003/article/details/48895153

实现原理

  通过利用AccessibilityService辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能。关于AccessibilityService辅助服务,可以自行百度了解更多。

代码基础:

1.首先声明一个RedPacketService继承自AccessibilityService,该服务类有两个方法必须重写,如下:

/*** Created by cxk on 2017/2/3.* email:471497226@qq.com** 抢红包服务类*/public class RedPacketService extends AccessibilityService {/*** 必须重写的方法:此方法用了接受系统发来的event。在你注册的event发生是被调用。在整个生命周期会被调用多次。*/@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {}/*** 必须重写的方法:系统要中断此service返回的响应时会调用。在整个生命周期会被调用多次。*/@Overridepublic void onInterrupt() {Toast.makeText(this, "我快被终结了啊-----", Toast.LENGTH_SHORT).show();}/*** 服务已连接*/@Overrideprotected void onServiceConnected() {Toast.makeText(this, "抢红包服务开启", Toast.LENGTH_SHORT).show();super.onServiceConnected();}/*** 服务已断开*/@Overridepublic boolean onUnbind(Intent intent) {Toast.makeText(this, "抢红包服务已被关闭", Toast.LENGTH_SHORT).show();return super.onUnbind(intent);}
}

2.对我们的RedPacketService进行一些配置,这里配置方法可以选择代码动态配置(onServiceConnected里配置),也可以直接在res/xml下新建.xml文件,没有xml文件夹就新建。这里我们将文件命名为redpacket_service_config.xml,代码如下:

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

accessibilityEventTypes:

响应哪一种类型的事件,typeAllMask就是响应所有类型的事件了,另外还有单击、长按、滑动等。

accessibilityFeedbackType:

用什么方式反馈给用户,有语音播出和振动。可以配置一些TTS引擎,让它实现发音。

packageNames:

指定响应哪个应用的事件。这里我们是写抢红包助手,就写微信的包名:com.tencent.mm,这样就可以监听微信产生的事件了。

notificationTimeout:

响应时间

description:

辅助服务的描述信息。

3.service是四大组件之一,需要在AndroidManifest进行配置,注意这里稍微有些不同:

 <!--抢红包服务--><serviceandroid:name=".RedPacketService"android:enabled="true"android:exported="true"android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"><intent-filter><action android:name="android.accessibilityservice.AccessibilityService" /></intent-filter><meta-dataandroid:name="android.accessibilityservice"android:resource="@xml/redpacket_service_config"></meta-data></service>
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"  权限申请
android:resource="@xml/redpacket_service_config"  引用刚才的配置文件

核心代码:我们的红包助手,核心思路分为三步走:监听通知栏微信消息,如果弹出[微信红包]字样,模拟手指点击状态栏跳转到微信聊天界面→在微信聊天界面查找红包,如果找到则模拟手指点击打开,弹出打开红包界面→模拟手指点击红包“開”

1.监听通知栏消息,查看是否有[微信红包]字样,代码如下:
    @Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) {//通知栏来信息,判断是否含有微信红包字样,是的话跳转case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:List<CharSequence> texts = event.getText();for (CharSequence text : texts) {String content = text.toString();if (!TextUtils.isEmpty(content)) {//判断是否含有[微信红包]字样if (content.contains("[微信红包]")) {//如果有则打开微信红包页面
                            openWeChatPage(event);}}}break;}}/*** 开启红包所在的聊天页面*/private void openWeChatPage(AccessibilityEvent event) {//A instanceof B 用来判断内存中实际对象A是不是B类型,常用于强制转换前的判断if (event.getParcelableData() != null && event.getParcelableData() instanceof Notification) {Notification notification = (Notification) event.getParcelableData();//打开对应的聊天界面PendingIntent pendingIntent = notification.contentIntent;try {pendingIntent.send();} catch (PendingIntent.CanceledException e) {e.printStackTrace();}}}
2.判断当前是否在微信聊天页面,是的话遍历当前页面各个控件,找到含有微信红包或者领取红包的textview控件,然后逐层找到他的可点击父布局(图中绿色部分),模拟点击跳转到含有“開”的红包界面,代码如下:

 @Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) {//窗口发生改变时会调用该事件case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:String className = event.getClassName().toString();//判断是否是微信聊天界面if ("com.tencent.mm.ui.LauncherUI".equals(className)) {//获取当前聊天页面的根布局AccessibilityNodeInfo rootNode = getRootInActiveWindow();//开始找红包
                    findRedPacket(rootNode);}}}/*** 遍历查找红包*/private void findRedPacket(AccessibilityNodeInfo rootNode) {if (rootNode != null) {//从最后一行开始找起for (int i = rootNode.getChildCount() - 1; i >= 0; i--) {AccessibilityNodeInfo node = rootNode.getChild(i);//如果node为空则跳过该节点if (node == null) {continue;}CharSequence text = node.getText();if (text != null && text.toString().equals("领取红包")) {AccessibilityNodeInfo parent = node.getParent();//while循环,遍历"领取红包"的各个父布局,直至找到可点击的为止while (parent != null) {if (parent.isClickable()) {//模拟点击
                            parent.performAction(AccessibilityNodeInfo.ACTION_CLICK);//isOpenRP用于判断该红包是否点击过isOpenRP = true;break;}parent = parent.getParent();}}//判断是否已经打开过那个最新的红包了,是的话就跳出for循环,不是的话继续遍历if (isOpenRP) {break;} else {findRedPacket(node);}}}}

3.点击红包后,在模拟手指点击“開”以此开启红包,跳转到红包详情界面,方法与步骤二类似:

 @Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) {//窗口发生改变时会调用该事件case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:String className = event.getClassName().toString();//判断是否是显示‘开’的那个红包界面if ("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI".equals(className)) {AccessibilityNodeInfo rootNode = getRootInActiveWindow();//开始抢红包
                    openRedPacket(rootNode);}break;}}/*** 开始打开红包*/private void openRedPacket(AccessibilityNodeInfo rootNode) {for (int i = 0; i < rootNode.getChildCount(); i++) {AccessibilityNodeInfo node = rootNode.getChild(i);if ("android.widget.Button".equals(node.getClassName())) {node.performAction(AccessibilityNodeInfo.ACTION_CLICK);}openRedPacket(node);}}

结合以上三步,下面是完整代码,注释已经写的很清楚,直接看代码:

package com.cxk.redpacket;import android.accessibilityservice.AccessibilityService;
import android.app.Instrumentation;
import android.app.KeyguardManager;
import android.app.Notification;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.os.IBinder;
import android.os.PowerManager;
import android.text.TextUtils;
import android.util.Log;
import android.view.KeyEvent;
import android.view.accessibility.AccessibilityEvent;
import android.view.accessibility.AccessibilityNodeInfo;
import android.widget.Toast;import java.util.List;/*** 抢红包Service,继承AccessibilityService*/
public class RedPacketService extends AccessibilityService {/*** 微信几个页面的包名+地址。用于判断在哪个页面* LAUCHER-微信聊天界面* LUCKEY_MONEY_RECEIVER-点击红包弹出的界面* LUCKEY_MONEY_DETAIL-红包领取后的详情界面*/private String LAUCHER = "com.tencent.mm.ui.LauncherUI";private String LUCKEY_MONEY_DETAIL = "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyDetailUI";private String LUCKEY_MONEY_RECEIVER = "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI";/*** 用于判断是否点击过红包了*/private boolean isOpenRP;private boolean isOpenDetail = false;/*** 用于判断是否屏幕是亮着的*/private boolean isScreenOn;/*** 获取PowerManager.WakeLock对象*/private PowerManager.WakeLock wakeLock;/*** KeyguardManager.KeyguardLock对象*/private KeyguardManager.KeyguardLock keyguardLock;@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();switch (eventType) {//通知栏来信息,判断是否含有微信红包字样,是的话跳转case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED:List<CharSequence> texts = event.getText();for (CharSequence text : texts) {String content = text.toString();if (!TextUtils.isEmpty(content)) {//判断是否含有[微信红包]字样if (content.contains("[微信红包]")) {if (!isScreenOn()) {wakeUpScreen();}//如果有则打开微信红包页面
                            openWeChatPage(event);isOpenRP = false;}}}break;//界面跳转的监听case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:String className = event.getClassName().toString();//判断是否是微信聊天界面if (LAUCHER.equals(className)) {//获取当前聊天页面的根布局AccessibilityNodeInfo rootNode = getRootInActiveWindow();//开始找红包
                    findRedPacket(rootNode);}//判断是否是显示‘开’的那个红包界面if (LUCKEY_MONEY_RECEIVER.equals(className)) {AccessibilityNodeInfo rootNode = getRootInActiveWindow();//开始抢红包
                    openRedPacket(rootNode);}//判断是否是红包领取后的详情界面if (isOpenDetail && LUCKEY_MONEY_DETAIL.equals(className)) {isOpenDetail = false;//返回桌面
                    back2Home();//如果之前是锁着屏幕的则重新锁回去
                    release();}break;}}/*** 开始打开红包*/private void openRedPacket(AccessibilityNodeInfo rootNode) {for (int i = 0; i < rootNode.getChildCount(); i++) {AccessibilityNodeInfo node = rootNode.getChild(i);if ("android.widget.Button".equals(node.getClassName())) {node.performAction(AccessibilityNodeInfo.ACTION_CLICK);isOpenDetail = true;}openRedPacket(node);}}/*** 遍历查找红包*/private void findRedPacket(AccessibilityNodeInfo rootNode) {if (rootNode != null) {//从最后一行开始找起for (int i = rootNode.getChildCount() - 1; i >= 0; i--) {AccessibilityNodeInfo node = rootNode.getChild(i);//如果node为空则跳过该节点if (node == null) {continue;}CharSequence text = node.getText();if (text != null && text.toString().equals("领取红包")) {AccessibilityNodeInfo parent = node.getParent();//while循环,遍历"领取红包"的各个父布局,直至找到可点击的为止while (parent != null) {if (parent.isClickable()) {//模拟点击
                            parent.performAction(AccessibilityNodeInfo.ACTION_CLICK);//isOpenRP用于判断该红包是否点击过isOpenRP = true;break;}parent = parent.getParent();}}//判断是否已经打开过那个最新的红包了,是的话就跳出for循环,不是的话继续遍历if (isOpenRP) {break;} else {findRedPacket(node);}}}}/*** 开启红包所在的聊天页面*/private void openWeChatPage(AccessibilityEvent event) {//A instanceof B 用来判断内存中实际对象A是不是B类型,常用于强制转换前的判断if (event.getParcelableData() != null && event.getParcelableData() instanceof Notification) {Notification notification = (Notification) event.getParcelableData();//打开对应的聊天界面PendingIntent pendingIntent = notification.contentIntent;try {pendingIntent.send();} catch (PendingIntent.CanceledException e) {e.printStackTrace();}}}/*** 服务连接*/@Overrideprotected void onServiceConnected() {Toast.makeText(this, "抢红包服务开启", Toast.LENGTH_SHORT).show();super.onServiceConnected();}/*** 必须重写的方法:系统要中断此service返回的响应时会调用。在整个生命周期会被调用多次。*/@Overridepublic void onInterrupt() {Toast.makeText(this, "我快被终结了啊-----", Toast.LENGTH_SHORT).show();}/*** 服务断开*/@Overridepublic boolean onUnbind(Intent intent) {Toast.makeText(this, "抢红包服务已被关闭", Toast.LENGTH_SHORT).show();return super.onUnbind(intent);}/*** 返回桌面*/private void back2Home() {Intent home = new Intent(Intent.ACTION_MAIN);home.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);home.addCategory(Intent.CATEGORY_HOME);startActivity(home);}/*** 判断是否处于亮屏状态** @return true-亮屏,false-暗屏*/private boolean isScreenOn() {PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);isScreenOn = pm.isScreenOn();Log.e("isScreenOn", isScreenOn + "");return isScreenOn;}/*** 解锁屏幕*/private void wakeUpScreen() {//获取电源管理器对象PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);//后面的参数|表示同时传入两个值,最后的是调试用的TagwakeLock = pm.newWakeLock(PowerManager.ACQUIRE_CAUSES_WAKEUP | PowerManager.FULL_WAKE_LOCK, "bright");//点亮屏幕
        wakeLock.acquire();//得到键盘锁管理器KeyguardManager km = (KeyguardManager) getSystemService(Context.KEYGUARD_SERVICE);keyguardLock = km.newKeyguardLock("unlock");//解锁
        keyguardLock.disableKeyguard();}/*** 释放keyguardLock和wakeLock*/public void release() {if (keyguardLock != null) {keyguardLock.reenableKeyguard();keyguardLock = null;}if (wakeLock != null) {wakeLock.release();wakeLock = null;}}}

使用方法:

设置-辅助功能-无障碍-点击RedPacket开启即可(或者直接在设置搜索辅助功能or RedPacket)

注:因为AccessibilityService服务很容易断开,所以我们需要将我们的App设置为白名单,防止被系统KO掉。这样他就能一直跑在我们的后台啦。

已知问题:

1.聊天列表或者聊天界面中无法直接自动抢红包

Demo下载地址:https://github.com/CKTim/RedPacket

Android中微信抢红包助手的实现相关推荐

  1. Android】中微信抢红包助手的实现(代码整理)

    实现原理 通过利用AccessibilityService辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能.关于AccessibilityService辅助服务,可以 ...

  2. 【Android】中微信抢红包助手的实现(代码整理)

     实现原理 通过利用AccessibilityService辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能.关于AccessibilityService辅助服务 ...

  3. 2016微信抢红包助手及源码

    代码名称:Android微信抢红包源码  微信抢红包助手一款全能抢红包软件,支持微信和QQ群内有抢红包自动抢.只需要根据提示在手机上安装我们的软件,当微信和QQ群里里出现红包时,我们的程序将自动将红包 ...

  4. android群聊红包原理,Android之微信抢红包实现原理分析

    微信抢红包主要用到AccessibilityService,该服务需要用户手动到辅助功能里面授权,下面直接粘贴源码,先看Activity,主要是跳转到辅助功能界面,让用户开启辅助功能的服务: publ ...

  5. 微信 助手 android版,微信恢复助手下载-微信恢复助手 安卓版v1.3.39-PC6安卓网

    需要调用以下重要权限 - 查看更多权限列表[微信恢复助手] ● 获取定位允许应用通过网络或卫星对设备进行定位 ● 查看网络状态允许应用程序查看所有网络的状态. ● 开关WLAN允许应用开启或关闭WLA ...

  6. Android QQ微信抢红包apk的实现

    临近元旦,即将有一大波红包向我们袭来,有个轻松抢红包的工具岂不是更好,那就写个抢红包的应用吧,至于原理,网上有一大堆分享,我就不再详述了,只是大致将流程整理下,以作笔记. 一.界面 二 .功能实现原理 ...

  7. Android中微信主界面菜单栏的布局实现代码

    最近一段时间进了Android的大坑,一门心思的学习Android大法,读了两三本书,就萌生了模仿微信界面的想法. 分析Android版本的微信界面需求就会发现它的核心,不得不佩服微信界面的简洁和灵活 ...

  8. 拆红包robmoney.java,android 模拟微信抢红包 实例源码下载

    [实例简介]需要 开启该应用的 辅助功能 [实例截图] [核心代码] package krelve.demo.rob; import java.util.List; import android.ac ...

  9. 微信抢红包软件-android

    微信红包不错的分析: 附带源码 并包含了源码 参考: Android中微信抢红包助手的实现 (1) https://www.jianshu.com/p/19ddd41aa349 (2) http:// ...

最新文章

  1. java 中文符号占位_java – ‘占位符’字符以避免积极比较?
  2. 在CentOS 6.9 x86_64的nginx 1.12.2上开启标准模块ngx_http_auth_request_module实录
  3. Flex 布局:语法篇
  4. SSRS报表连接超时的问题
  5. Ajax Control Toolkit 34个服务器端控件 详解
  6. Oracle学习笔记:备份、还原、恢复的一点理解
  7. scala命令行运行spark代码
  8. MOXy的@XmlVariableNode – JSON模式示例
  9. SliceProceduralMesh的使用
  10. 【算法】剑指 Offer 25. 合并两个排序的链表
  11. 中国计算机信息系统集成行业协会有含金量吗,系统集成工程师证书的含金量怎么样...
  12. Hibenate映射属性
  13. 学生学籍管理系统页面源代码html_学生信息管理系统-系统源代码
  14. 【Lingo 18.0及其安装教程】
  15. 前端小白奋斗史03(css)
  16. 陀螺仪mpu6050的使用(附带HAL的使用)
  17. vce数学考试能用计算机,准备VCE数学考试的诀窍,从一本出色的笔记开始!
  18. 计算机毕业设计php+vue基于微信小程序的音乐云系统-音乐播放系统
  19. python编程图文_深入Python多进程编程基础——图文版
  20. 关于c4d在给模型添加FFD时,遇到拉升FFD里的点,而模型没有任何变化的解决方法

热门文章

  1. jodaTime 的使用说明
  2. 基于Qt的收银点餐系统之小票打印(二)
  3. 【数据结构与算法】二叉树题目很难?一句话秒杀基础二叉树题目
  4. 计算机网络-因特网的组成
  5. MySQL无法启动并提示file could not be found的解决方法
  6. springboot毕设项目养老平台的设计与实现u8sua(java+VUE+Mybatis+Maven+Mysql)
  7. 大数据第一阶段学习笔记
  8. JAVA环境配置多环境(全,细,简单)
  9. Python神经网络识别手写数字-MNIST数据集
  10. Java根据模板生成PDF文件