注册

android 微信抢红包工具 AccessibilityService(上)

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

一、目标

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

二、实现流程

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

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

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

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

2、实现步骤

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

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

Ⅰ、AccessibilityService

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

private fun handleNotification(eventAccessibilityEvent) {
   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 notificationNotification? = event.parcelableData as Notification?
                       val pendingIntentPendingIntent = notification!!.contentIntent
                       try {
                           pendingIntent.send()
                      } catch (eCanceledException) {
                           e.printStackTrace()
                      }
                  }
              }
          }
      }

}
Ⅱ、NotificationListenerService

这是监听通知栏的另一种方式,记得要获取权限哦

class MyNotificationListenerService : NotificationListenerService() {

   override fun onNotificationPosted(sbnStatusBarNotification?) {
       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(sbnStatusBarNotification?) {
       super.onNotificationRemoved(sbn)
  }
}

2、点击红包

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

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

4bba5ce0c06b469f9850a8b2bb0ac4f2~tplv-k3u1fbpfcp-zoom-in-crop-mark:4536:0:0:0.awebp

我本地的jdk是11,我怀疑是不兼容,毕竟Android Device Monitor太老了。换新的layout Inspector,也就看看本地的debug应用,无法查看微信的呀。要么就反编译,这个就先不考虑了,或者在配置文件中设置android:accessibilityFlags="flagReportViewIds",然后暴力遍历Node树,打印相应的viewId和className,找到目标id即可。当然也可以换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[indexis AccessibilityNodeInfo && (index == 0 || caches[index-1!is String )){
               val node = caches[indexas AccessibilityNodeInfo
               var parent = node.parent
               while (parent != null) {
                   if (parent.isClickable) {
                       parent.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                       break
                  }
                   parent = parent.parent
              }
               break
          }
      }
  }

}

private fun recycle(nodeAccessibilityNodeInfo,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 && 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 && child.isClickable) {
                       child.performAction(AccessibilityNodeInfo.ACTION_CLICK)
                  }
              }

          }

      }
  }
}

三、遇到问题

1、AccessibilityService收不到AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED事件

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

2、需要点击View的定位

简单是就是到页面应该点哪个View,找到相应的规则,来过滤出对应的View,这个规则是随着微信的改变而变化的,findAccessibilityNodeInfosByViewId最直接,但是奈何工具问题,有点麻烦,遍历打印可以获取,但是id每个版本可能会变。还有就是通过文案来获取,即findAccessibilityNodeInfosByText,获取一些固定文案的View,这个相对而言在不改版,可能不会变,相对稳定些,如果这个文案的View本身没点击事件,可获取它的parent,尝试点击,或者遍历parent树,根据isClickable来判断是否可以点击。

划重点:

这里还有一种就是钉钉的开红包按钮,折腾了半天,始终拿不到,各种递归遍历,一直没有找到,最后换了个方式,通过AccessibilityService的模拟点击来做,也就是通过坐标来模拟点击,当然要在配置中开启android:canPerformGestures="true", 然后通过 accessibilityService.dispatchGesture() 来处理,具体坐标可以拿一个其他的View,然后通过比例来确定大概得位置,或者,看看能不能拿到外层的Layout也是一样的

object AccessibilityClick {
   fun click(accessibilityServiceAccessibilityServicexFloatyFloat) {
       val builder = GestureDescription.Builder()
       val path = Path()
       path.moveTo(xy)
       path.lineTo(xy)
       builder.addStroke(GestureDescription.StrokeDescription(path010))
       accessibilityService.dispatchGesture(builder.build(), object : AccessibilityService.GestureResultCallback() {
           override fun onCancelled(gestureDescriptionGestureDescription) {
               super.onCancelled(gestureDescription)
          }

           override fun onCompleted(gestureDescriptionGestureDescription) {
               super.onCompleted(gestureDescription)
          }
      }, null)
  }
}

续:android 微信抢红包工具 AccessibilityService(下)

作者:我有一头小毛驴你有吗
来源:juejin.cn/post/7196949524061339703

0 个评论

要回复文章请先登录注册