Android中微信抢红包助手的实现详解

实现原理

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

代码基础:

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

/**
 * Created by cxk on 2017/2/3.
 *
 * 抢红包服务类
 */

public class RedPacketService extends AccessibilityService {

  /**
   * 必须重写的方法:此方法用了接受系统发来的event。在你注册的event发生是被调用。在整个生命周期会被调用多次。
   */
  @Override
  public void onAccessibilityEvent(AccessibilityEvent event) {

  }

  /**
   * 必须重写的方法:系统要中断此service返回的响应时会调用。在整个生命周期会被调用多次。
   */
  @Override
  public void onInterrupt() {
    Toast.makeText(this, "我快被终结了啊-----", Toast.LENGTH_SHORT).show();
  }

  /**
   * 服务已连接
   */
  @Override
  protected void onServiceConnected() {
    Toast.makeText(this, "抢红包服务开启", Toast.LENGTH_SHORT).show();
    super.onServiceConnected();
  }

  /**
   * 服务已断开
   */
  @Override
  public 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进行配置,注意这里稍微有些不同:

 <!--抢红包服务-->
    <service
      android: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-data
        android: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.监听通知栏消息,查看是否有[微信红包]字样,代码如下:

  @Override
  public 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控件,然后逐层找到他的可点击父布局(图中绿色部分),模拟点击跳转到含有“開”的红包界面,代码如下:

 @Override
  public 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.点击红包后,在模拟手指点击“開”以此开启红包,跳转到红包详情界面,方法与步骤二类似:

 @Override
  public 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.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.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-点击红包弹出的界面
   */
  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;

  @Override
  public 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);

              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(LUCKEY_MONEY_DETAIL.equals(className)){
          //返回桌面
          back2Home();
        }
        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);
    }
  }

  /**
   * 遍历查找红包
   */
  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();
      }
    }
  }

  /**
   * 服务连接
   */
  @Override
  protected void onServiceConnected() {
    Toast.makeText(this, "抢红包服务开启", Toast.LENGTH_SHORT).show();
    super.onServiceConnected();
  }

  /**
   * 必须重写的方法:系统要中断此service返回的响应时会调用。在整个生命周期会被调用多次。
   */
  @Override
  public void onInterrupt() {
    Toast.makeText(this, "我快被终结了啊-----", Toast.LENGTH_SHORT).show();
  }

  /**
   * 服务断开
   */
  @Override
  public 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);
  }

}

使用方法:

设置-辅助功能-无障碍-点击RedPacket开启即可

已知问题:

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

2.未做熄屏自动抢红包处理,想要熄屏能自动抢红包的同学直接把开屏代码写在第一步即可。

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

(0)

相关推荐

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

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

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

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

  • Android微信抢红包功能的实现原理浅析

    快到过农历年了,微信红包也越来越多了,出现了好多红包外挂程序,就很好奇如何实现的,于是自己研究了一番,亲自写了个微信抢红包的APP.现在就一步一步来实现它. 实现思路 微信抢红包程序开启时候,他就可以随时识别.捕获红包,服务可以实现正在功能,当我们开启服务的时候,服务就不停的在后台运行,不停地轮询着微信里面的消息,当发现红包时候就立即打开微信红包所在的界面.但是他怎识别红包呢?需要找到微信抢红包里面节点的view,当找到对应的view,在获取view的关键字或者id,根据关键字或者id,自动的模

  • 教你一步步实现Android微信自动抢红包

    本文介绍微信自动抢红包的实现方法,主要实现以下几个功能: 1.自动拆开屏幕上出现的红包 2.处于桌面或聊天列表时接收到红包信息时自动进入聊天界面并拆红包 3.日志功能,记录抢红包的详细日志 实现原理 1.利用AccessibilityService辅助服务,监测屏幕内容,实现自动拆红包的目的. 2.利用ActiveAndroid数据库简单记录红包日志 3.利用preference实现监控选项纪录 最终界面 抢红包核心代码 AccessibilityService配置 android:access

  • Android实现微信自动抢红包的程序

    简单实现了微信自动抢红包的服务,原理就是根据关键字找到相应的View, 然后自动点击.主要是用到AccessibilityService这个辅助服务,基本可以满足自动抢红包的功能,但是有些逻辑需要优化,比如,拆完一个红包后,必须手动点击返回键,才能进行下一次自动抢红包. AndroidManifest.xml <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="h

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

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

  • Android中微信抢红包助手的实现详解

    实现原理 通过利用AccessibilityService辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能.关于AccessibilityService辅助服务,可以自行百度了解更多. 代码基础: 1.首先声明一个RedPacketService继承自AccessibilityService,该服务类有两个方法必须重写,如下: /** * Created by cxk on 2017/2/3. * * 抢红包服务类 */ public class RedPack

  • android 微信抢红包工具AccessibilityService实现详解

    目录 1.目标 2.实现流程 1.流程分析(这里只分析在桌面的情况) 2.实现步骤 1.收到通知 以及 点击通知栏 2.点击红包 3.点击开红包 4.退出红包详情页 3.遇到问题 4.完整代码 MyNotificationListenerService MyAccessibilityService 5.总结 你有因为手速不够快抢不到红包而沮丧? 你有因为错过红包而懊恼吗? 没错,它来了... 1.目标 使用AccessibilityService的方式,实现微信自动抢红包(吐槽一下,网上找了许多

  • Android中FileProvider的各种场景应用详解

    目录 前言 一.常规使用与定义 二.能不能自定义接收文件? 三.能不能主动查询对方的沙盒? 总结 前言 有部分同学只要是上传或者下载,只要用到了文件,不管三七二十一写个 FileProvider 再说. 不是每一种情况都需要使用 FileProvider 的,啥?你问行不行?有没有毛病? 这... 写了确实可以,没毛病!但是这没有必要啊. 如果不需要FileProvider就不需要定义啊,如果定义了重复的 FileProvider,还会导致清单文件合并失败,需要处理冲突,从而引出又一个问题,解决

  • Android 中Fragment与Activity通讯的详解

    Android 中Fragment与Activity通讯的详解 与activity通讯 尽管fragment的实现是独立于activity的,可以被用于多个activity,但是每个activity所包含的是同一个fragment的不同的实例. Fragment可以调用getActivity()方法很容易的得到它所在的activity的对象,然后就可以查找activity中的控件们(findViewById()). 例如: ViewlistView =getActivity().findView

  • Android中asset和raw的区别详解

    *res/raw和assets的相同点: 1.两者目录下的文件在打包后会原封不动的保存在apk包中,不会被编译成二进制. *res/raw和assets的不同点: 1.res/raw中的文件会被映射到R.java文件中,访问的时候直接使用资源ID即R.id.filename:assets文件夹下的文件不会被映射到 R.java中,访问的时候需要AssetManager类. 2.res/raw不可以有目录结构,而assets则可以有目录结构,也就是assets目录下可以再建立文件夹 *读取文件资源

  • Android 中Lambda表达式的使用实例详解

     Android 中Lambda表达式的使用实例详解 Java8 中着实引入了一些非常有特色的功能,如Lambda表达式.streamAPI.接口默认实现等等.Lambda表达式在 Android 中最低兼容到 Android2.3 系统,兼容性还是不错的,Lambda表达式本质上是一种匿名方法,它既没有方法名,也没有访问修饰符和返回值类型,使用它编写的代码将更加简洁易读. 1.Lambda表达式的基本写法 如果想要在 Android 项目中使用 Lambda表达式 或者 Java8 的其他新特

  • Android中判断网络是否连接实例详解

    Android中判断网络是否连接实例详解 在android中,如何监测网络的状态呢,这个有的时候也是十分重要的,方法如下: public class ConnectionDetector { private Context _context; public ConnectionDetector(Context context){ this._context = context; } public boolean isConnectingToInternet(){ ConnectivityMana

  • Android仿微信图片选择器ImageSelector使用详解

    今天给大家介绍一个仿微信的图片选择器:ImageSelector.ImageSelector支持图片的单选.限数量的多选和不限数量的多选.支持图片预览和图片文件夹的切换.在上一篇文章 <Android 实现一个仿微信的图片选择器> 中我介绍了ImageSelector的实现思路和分析了它的核心代码,有兴趣的同学可以看一下.完整的代码放在了GitHub,欢迎大家下载和使用.本篇文章为大家介绍ImageSelector的具体使用方式. 先上效果图: 1.引入依赖 在Project的build.gr

  • Android中关于Notification及NotificationManger的详解

    Android状态栏提醒 在Android中提醒功能也可以用AlertDialog,但是我们要慎重的使用,因为当使用AlertDialog的时候,用户正在进行的操作将会被打断,因为当前焦点被AlertDialog得到.我们可以想像一下,当用户打游戏正爽的时候,这时候来了一条短信.如果这时候短信用AlertDialog提醒,用户必须先去处理这条提醒,从而才能继续游戏.用户可能会活活被气死.而使用Notification就不会带来这些麻烦事,用户完全可以打完游戏再去看这条短信.所以在开发中应根据实际

随机推荐