欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2024-02-13 19:37:46
实现原理 通过利用accessibilityservice辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能。关于accessibilit...

实现原理

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

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

 @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.未做熄屏自动抢红包处理,想要熄屏能自动抢红包的同学直接把开屏代码写在第一步即可。

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