返回顶部
首页 > 资讯 > 移动开发 >android 微信抢红包工具AccessibilityService实现详解
  • 510
分享到

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

android AccessibilityServiceandroid 微信抢红包工具 2023-02-07 12:02:34 510人浏览 薄情痞子
摘要

目录1、目标2、实现流程1、流程分析(这里只分析在桌面的情况)2、实现步骤1、收到通知 以及 点击通知栏2、点击红包3、点击开红包4、退出红包详情页3、遇到问题4、完整代码MyNot

你有因为手速不够快抢不到红包而沮丧? 你有因为错过红包而懊恼吗? 没错,它来了。。。

1、目标

使用AccessibilityService的方式,实现微信自动抢红包(吐槽一下,网上找了许多文档,由于各种原因,无法实现对应效果,所以先给自己整理下),关于AccessibilityService的文章,网上有很多(没错,多的都懒得贴链接那种多),可自行查找。

2、实现流程

1、流程分析(这里只分析在桌面的情况)

我们把一个抢红包发的过程拆分来看,可以分为几个步骤

收到通知 -> 点击通知栏 -> 点击红包 -> 点击开红包 -> 退出红包详情页

以上是一个抢红包的基本流程。

2、实现步骤

1、收到通知 以及 点击通知栏

接收通知栏的消息,介绍两种方式

1、AccessibilityService

即通过AccessibilityService的AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED事件来获取到Notification

private fun handleNotification(event: AccessibilityEvent) {
    val texts = event.text
    if (!texts.isEmpty()) {
            for (text in texts) {
                val content = text.toString()
                //如果微信红包的提示信息,则模拟点击进入相应的聊天窗口
                if (content.contains("[微信红包]")) {
                    if (event.parcelableData != null && event.parcelableData is Notification) {
                        val notification: Notification? = event.parcelableData as Notification?
                        val pendingIntent: PendingIntent = notification!!.contentIntent
                        try {
                            pendingIntent.send()
                        } catch (e: CanceledException) {
                            e.printStackTrace()
                        }
                    }
                }
            }
        }
}
2、NotificationListenerService
class MyNotificationListenerService : NotificationListenerService() {
    override fun onNotificationPosted(sbn: StatusBarNotification?) {
        super.onNotificationPosted(sbn)
        val extras = sbn?.notification?.extras
        // 获取接收消息APP的包名
        val notificationPkg = sbn?.packageName
        // 获取接收消息的抬头
        val notificationTitle = extras?.getString(Notification.EXTRA_TITLE)
        // 获取接收消息的内容
        val notificationText = extras?.getString(Notification.EXTRA_TEXT)
        if (notificationPkg != null) {
            Log.d("收到的消息内容包名:", notificationPkg)
            if (notificationPkg == "com.tencent.mm"){
                if (notificationText?.contains("[微信红包]") == true){
                    //收到微信红包了
                    val intent = sbn.notification.contentIntent
                    intent.send()
                }
            }
        }
        Log.d("收到的消息内容", "Notification posted $notificationTitle & $notificationText")
    }
    override fun onNotificationRemoved(sbn: StatusBarNotification?) {
        super.onNotificationRemoved(sbn)
    }
}

2、点击红包

通过上述的跳转,可以进入聊天详情页面,到达详情页之后,接下来就是点击对应的红包卡片,那么问题来了,怎么点?肯定不是手动点。。。

我们来分析一下,一个聊天列表中,我们怎样才能识别到红包卡片,我看网上有通过findAccessibilityNodeInfosByViewId来获取对应的View,这个也可以,只是我们获取id的方式需要借助工具,可以用Android Device Monitor,但是这玩意早就废废弃了,虽然在sdk的目录下存在monitor,奈何本人太菜,点击就是打不开

我本地的jdk是11,我怀疑是不兼容,毕竟Android Device Monitor太老了。换新的layout Inspector,也就看看本地的debug应用,无法查看微信的呀。要么就反编译,这个就先不考虑了,换findAccessibilityNodeInfosByText这个方法试试。

这个方法从字面意思能看出来,是通过text来匹配的,我们可以知道红包卡片上面是有“微信红包”的固定字样的,是不是可以通股票这个来匹配呢,这还有个其他问题,并不是所有的红包都需要点,比如已过期,已领取的是不是要过滤下,咋一看挺好过滤的,一个循环就好,仔细想,这是棵树,不太好剔除,所以换了个思路。

最终方案就是递归一棵树,往一个列表里面塞值,“已过期”和“已领取”的塞一个字符串“#”,匹配到“微信红包”的塞一个AccessibilitynodeInfo,这样如果这个红包不能抢,那肯定一前一后分别是一个字符串和一个AccessibilityNodeInfo,因此,我们读到一个AccessibilityNodeInfo,并且前一个值不是字符串,就可以执行点击事件,代码如下

private fun getPacket() {
    val rootNode = rootInActiveWindow
    val caches:ArrayList<Any> = ArrayList()
    recycle(rootNode,caches)
    if(caches.isNotEmpty()){
        for(index in 0 until caches.size){
            if(caches[index] is AccessibilityNodeInfo && (index == 0 || caches[index-1] !is String )){
                val node = caches[index] as AccessibilityNodeInfo
                node.perforMaction(AccessibilityNodeInfo.ACTION_CLICK)
                var parent = node.parent
                while (parent != null) {
                    if (parent.isClickable) {
                        parent.perfORMAction(AccessibilityNodeInfo.ACTION_CLICK)
                        break
                    }
                    parent = parent.parent
                }
                break
            }
        }
    }
}
private fun recycle(node: AccessibilityNodeInfo,caches:ArrayList<Any>) {
        if (node.childCount == 0) {
            if (node.text != null) {
                if ("已过期" == node.text.toString() || "已被领完" == node.text.toString() || "已领取" == node.text.toString()) {
                    caches.add("#")
                }
                if ("微信红包" == node.text.toString()) {
                    caches.add(node)
                }
            }
        } else {
            for (i in 0 until node.childCount) {
                if (node.getChild(i) != null) {
                    recycle(node.getChild(i),caches)
                }
            }
        }
    }

以上只点击了第一个能点击的红包卡片,想点击所有的可另行处理。

3、点击开红包

这里思路跟上面类似,开红包页面比较简单,但是奈何开红包是个按钮,在不知道id的前提下,我们也不知道则呢么获取它,所以采用迂回套路,找固定的东西,我这里发现每个开红包的页面都有个“xxx的红包”文案,然后这个页面比较简单,只有个关闭,和开红包,我们通过获取“xxx的红包”对应的View来获取父View,然后递归子View,判断可点击的,执行点击事件不就可以了吗

private fun openPacket() {
    val nodeInfo = rootInActiveWindow
    if (nodeInfo != null) {
        val list = nodeInfo.findAccessibilityNodeInfosByText ("的红包")
        for ( i in 0 until list.size) {
            val parent = list[i].parent
            if (parent != null) {
                for ( j in 0 until  parent.childCount) {
                    val child = parent.getChild (j)
                    if (child != null &amp;&amp; child.isClickable) {
                        child.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                    }
                }
            }
        }
    }
}

4、退出红包详情页

这里回退也是个按钮,我们也不知道id,所以可以跟点开红包一样,迂回套路,获取其他的View,来获取父布局,然后递归子布局,依次执行点击事件,当然关闭事件是在前面的,也就是说关闭会优先执行到

private fun close() {
    val nodeInfo = rootInActiveWindow
    if (nodeInfo != null) {
        val list = nodeInfo.findAccessibilityNodeInfosByText ("的红包")
        if (list.isNotEmpty()) {
            val parent = list[0].parent.parent.parent
            if (parent != null) {
                for ( j in 0 until  parent.childCount) {
                    val child = parent.getChild (j)
                    if (child != null &amp;&amp; child.isClickable) {
                        child.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                    }
                }
            }
        }
    }
}

3、遇到问题

1、AccessibilityService收不到AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED事件

android碎片问题很正常,我这边是使用NotificationListenerService来替代的。

2、需要点击View的定位

简单是就是到页面应该点哪个View,找到相应的规则,来过滤出对应的View,这个规则是随着微信的改变而变化的,findAccessibilityNodeInfosByViewId最直接,但是奈何工具问题,有点麻烦,于是采用取巧的办法,通过找到其他View来定位自身

4、完整代码

MyNotificationListenerService

class MyNotificationListenerService : NotificationListenerService() {
    override fun onNotificationPosted(sbn: StatusBarNotification?) {
        super.onNotificationPosted(sbn)
        val extras = sbn?.notification?.extras
        // 获取接收消息APP的包名
        val notificationPkg = sbn?.packageName
        // 获取接收消息的抬头
        val notificationTitle = extras?.getString(Notification.EXTRA_TITLE)
        // 获取接收消息的内容
        val notificationText = extras?.getString(Notification.EXTRA_TEXT)
        if (notificationPkg != null) {
            Log.d("收到的消息内容包名:", notificationPkg)
            if (notificationPkg == "com.tencent.mm"){
                if (notificationText?.contains("[微信红包]") == true){
                    //收到微信红包了
                    val intent = sbn.notification.contentIntent
                    intent.send()
                }
            }
        } Log.d("收到的消息内容", "Notification posted $notificationTitle &amp; $notificationText")
    }
    override fun onNotificationRemoved(sbn: StatusBarNotification?) {
        super.onNotificationRemoved(sbn)
    }
}

MyAccessibilityService

class RobService : AccessibilityService() {
    override fun onAccessibilityEvent(event: AccessibilityEvent) {
        val eventType = event.eventType
        when (eventType) {
            AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED -&gt; handleNotification(event)
            AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED, AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED -&gt; {
                val className = event.className.toString()
                Log.e("测试无障碍id",className)
                if (className == "com.tencent.mm.ui.LauncherUI") {
                    getPacket()
                } else if (className == "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI") {
                    openPacket()
                }  else if (className == "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyDetailUI") {
                    close()
                }
            }
        }
    }
    
    private fun handleNotification(event: AccessibilityEvent) {
        val texts = event.text
        if (!texts.isEmpty()) {
            for (text in texts) {
                val content = text.toString()
                //如果微信红包的提示信息,则模拟点击进入相应的聊天窗口
                if (content.contains("[微信红包]")) {
                    if (event.parcelableData != null &amp;&amp; event.parcelableData is Notification) {
                        val notification: Notification? = event.parcelableData as Notification?
                        val pendingIntent: PendingIntent = notification!!.contentIntent
                        try {
                            pendingIntent.send()
                        } catch (e: CanceledException) {
                            e.printStackTrace()
                        }
                    }
                }
            }
        }
    }
    
    @Targetapi(Build.VERSION_CODES.JELLY_BEAN_MR2)
    private fun close() {
        val nodeInfo = rootInActiveWindow
        if (nodeInfo != null) {
            val list = nodeInfo.findAccessibilityNodeInfosByText ("的红包")
            if (list.isNotEmpty()) {
                val parent = list[0].parent.parent.parent
                if (parent != null) {
                    for ( j in 0 until  parent.childCount) {
                        val child = parent.getChild (j)
                        if (child != null &amp;&amp; child.isClickable) {
                            child.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                        }
                    }
                }
            }
        }
    }
    
    @TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR2)
    private fun openPacket() {
        val nodeInfo = rootInActiveWindow
        if (nodeInfo != null) {
            val list = nodeInfo.findAccessibilityNodeInfosByText ("的红包")
            for ( i in 0 until list.size) {
                val parent = list[i].parent
                if (parent != null) {
                    for ( j in 0 until  parent.childCount) {
                        val child = parent.getChild (j)
                        if (child != null &amp;&amp; child.isClickable) {
                            child.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                        }
                    }
                }
            }
        }
    }
    
    @TargetApi(Build.VERSION_CODES.JELLY_BEAN)
    private fun getPacket() {
        val rootNode = rootInActiveWindow
        val caches:ArrayList&lt;Any&gt; = ArrayList()
        recycle(rootNode,caches)
        if(caches.isNotEmpty()){
            for(index in 0 until caches.size){
                if(caches[index] is AccessibilityNodeInfo &amp;&amp; (index == 0 || caches[index-1] !is String )){
                    val node = caches[index] as AccessibilityNodeInfo
                    node.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                    var parent = node.parent
                    while (parent != null) {
                        if (parent.isClickable) {
                            parent.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                            break
                        }
                        parent = parent.parent
                    }
                    break
                }
            }
        }
    }
    
    private fun recycle(node: AccessibilityNodeInfo,caches:ArrayList&lt;Any&gt;) {
        if (node.childCount == 0) {
            if (node.text != null) {
                if ("已过期" == node.text.toString() || "已被领完" == node.text.toString() || "已领取" == node.text.toString()) {
                    caches.add("#")
                }
                if ("微信红包" == node.text.toString()) {
                    caches.add(node)
                }
            }
        } else {
            for (i in 0 until node.childCount) {
                if (node.getChild(i) != null) {
                    recycle(node.getChild(i),caches)
                }
            }
        }
    }
    override fun onInterrupt() {}
    override fun onServiceConnected() {
        super.onServiceConnected()
        Log.e("测试无障碍id","启动")
        val info: AccessibilityServiceInfo = serviceInfo
        info.packageNames = arrayOf("com.tencent.mm")
        serviceInfo = info
    }
}

5、总结

此文是对AccessibilityService的使用的一个梳理,这个功能其实不麻烦,主要是一些细节问题,像自动领取支付宝红包,自动领取QQ红包或者其他功能等也都可以用类似方法实现。

以上就是android 微信抢红包工具AccessibilityService实现详解的详细内容,更多关于android AccessibilityService的资料请关注编程网其它相关文章!

--结束END--

本文标题: android 微信抢红包工具AccessibilityService实现详解

本文链接: https://lsjlt.com/news/194433.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • android 微信抢红包工具AccessibilityService实现详解
    目录1、目标2、实现流程1、流程分析(这里只分析在桌面的情况)2、实现步骤1、收到通知 以及 点击通知栏2、点击红包3、点击开红包4、退出红包详情页3、遇到问题4、完整代码MyNot...
    99+
    2023-02-07
    android AccessibilityService android 微信抢红包工具
  • Android AccessibilityService实现微信抢红包插件
    在你的手机更多设置或者高级设置中,我们会发现有个无障碍的功能,很多人不知道这个功能具体是干嘛的,其实这个功能是为了增强用户界面以帮助残障人士,或者可能暂时无法与设备充分交互的人...
    99+
    2022-06-06
    插件 抢红包 Android
  • Android中微信抢红包助手的实现详解
    实现原理 通过利用AccessibilityService辅助服务,监测屏幕内容,如监听状态栏的信息,屏幕跳转等,以此来实现自动拆红包的功能。关于AccessibilitySe...
    99+
    2022-06-06
    抢红包 Android
  • Android实现微信自动抢红包的程序
     简单实现了微信自动抢红包的服务,原理就是根据关键字找到相应的View, 然后自动点击。主要是用到AccessibilityService这个辅助服务,基本可以满足自...
    99+
    2022-06-06
    自动 程序 抢红包 Android
  • Python+Appium实现自动抢微信红包
    目录环境准备实现思路code环境准备 appium环境 安卓手机 usb数据线 python环境 实现思路 我们收到红包和消息都是自动置顶到第一个,于是我们打开第一个判断是否有红...
    99+
    2022-06-02
    Python 抢微信红包 Python+Appium 抢红包
  • Android微信自动抢红包插件优化和实现
    又是兴趣系列 网上有很多自动强红包的例子和代码,笔者也是做了一些优化。 先说说自己的两个个优势 1.可以在聊天界面自动强不依赖于通知栏推送 2.可以在屏幕熄灭的时候的时候点亮...
    99+
    2022-06-06
    自动 插件 抢红包 优化 Android
  • 教你一步步实现Android微信自动抢红包
    本文介绍微信自动抢红包的实现方法,主要实现以下几个功能:       1.自动拆开屏幕上出现的红包   ...
    99+
    2022-06-06
    自动 抢红包 Android
  • Android微信抢红包功能的实现原理浅析
    快到过农历年了,微信红包也越来越多了,出现了好多红包外挂程序,就很好奇如何实现的,于是自己研究了一番,亲自写了个微信抢红包的APP。现在就一步一步来实现它。 实现思路 微信抢红...
    99+
    2022-06-06
    抢红包 Android
  • Java实现微信抢红包算法有哪些
    这期内容当中小编将会给大家带来有关Java实现微信抢红包算法有哪些,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。概述14年微信推出红包功能以后,很多公司开始上自己的红包功能,到现在为止仍然有很多红包开发的...
    99+
    2023-06-22
  • Python+Appium实现自动抢微信红包的方法
    这篇文章将为大家详细讲解有关Python+Appium实现自动抢微信红包的方法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。环境准备appium环境安卓手机usb数据线python环境实现思路我们收到红包...
    99+
    2023-06-15
  • Java实现4种微信抢红包算法(小结)
    目录概述 一、剩余金额随机法 二、二倍均值法(微信红包采用此法) 三、整体随机法 四、割线法 概述 14年微信推出红包功能以后,很多公司开始上自己的红包功能,到现在为止仍然有很多红...
    99+
    2024-04-02
  • asp.net开发微信派发现金红包/H5网页抢红包功能(思路详解)
    临近节日,公司里为了给广大老客户赠福利,以及来一波拉新活动:搞了一个营销活动,大概活动规则如下: 1、老客户通过网页抢红包,中奖会派发微信支付的红包; 2、新客户只要关注后就可以中奖...
    99+
    2024-04-02
  • Android实现QQ抢红包插件
    又想到快要过年了,到时候还不知道群里要发好多红包,所以我将之前在网上宕的一份微信抢红包的代码修改了一下,实现了QQ抢红包!可以支持抢QQ拼手气红包,普通红包,口令红包,现在再也...
    99+
    2022-06-06
    插件 Android
  • Redis分布式缓存怎么实现微信抢红包
    本篇内容主要讲解“Redis分布式缓存怎么实现微信抢红包”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Redis分布式缓存怎么实现微信抢红包”吧!一、场景分析微信抢红包已经在我们生活中很常见的场...
    99+
    2023-06-22
  • Android中微信抢红包插件原理解析及开发思路
    一、前言 自从去年中微信添加抢红包的功能,微信的电商之旅算是正式开始正式火爆起来。但是作为Android开发者来说,我们在抢红包的同时意识到了很多问题,就是手动去抢红包的速度...
    99+
    2022-06-06
    插件 抢红包 Android
  • asp.net如何开发微信派发现金红包/H5网页抢红包功能
    这篇文章将为大家详细讲解有关asp.net如何开发微信派发现金红包/H5网页抢红包功能,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一、网页版抢红包功能,前端先做好抢红包的页面;后台配置微信公众号的app...
    99+
    2023-06-14
  • Redis分布式缓存:微信抢红包解决方案
    目录一、场景分析二、技术方案三、案例实战一、场景分析 微信抢红包已经在我们生活中很常见的场景了,特别是年底公司开年会和春节2个时间段,长辈领导都发红包,手都点抽筋了,也没抢到多少。 ...
    99+
    2024-04-02
  • Android抢红包插件实现原理浅析
    抢红包,先看效果图~ 实现自动抢红包,解决问题有两点: 一:如何实时监听发红包的事件 二:如何在红包到来的时候自动进入页面并自动点击红包 一、如何获取红包到来的事件 为了获...
    99+
    2022-06-06
    插件 Android
  • Android辅助功如何实现自动抢红包
    这篇文章主要介绍Android辅助功如何实现自动抢红包,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、描述最近看到同事有用抢红包的软件,就想看看抢红包的具体实现是如何的,所以了解了一下,有用辅助功能实现的,所以在下...
    99+
    2023-05-30
    android
  • Android实现微信自动向附近的人打招呼(AccessibilityService)
    学习功能强大的AccessibilityService!!! 以下是本人根据自动抢红包的实现思路敲的用于微信自动向附近的人打招呼的核心代码 public class Aut...
    99+
    2022-06-06
    自动 Android
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作