使用AccessibilityService实现自动遍历点赞功能
作者:化作孤岛的瓜 发布时间:2023-10-18 16:22:53
概述:
利用AccessibilityService机制实现了一个比较好玩儿的功能,微信朋友圈自动遍历点赞。即通过不断的滚动+点赞实现把每一条朋友圈都赞一次。
当然其中还要涉及一些判断算法,比如如果这条朋友圈已经赞过就跳过去,以及当前界面没有可赞的朋友圈时执行翻页。其实做起来试错是个很繁冗的过程,这个效果也差不多做了两天。
使用方式:
运行程序-开启无障碍服务,再切换到微信主界面,进入朋友圈,就会自动执行点赞程序了。
效果图如下:
实现原理步骤以及难点:
1.首先要获取到微信朋友圈这个界面的ListView结点,或者通过根节点描述判断是否进入该界面。
2.到了朋友圈界面之后可以执行程序方法体了,但是要有个boolean值判断只能执行一次。
为什么该方法体只能执行一次呢?(代码在下面有),因为如果被动地让onAccessibilityEvent调用我们的方法,会出现很多问题,比如结点刷新过快,多次触发方法导致点赞步骤同时执行N次然后无限死循环,因为onAccessibilityEvent触发太快了,大概0.几毫秒触发一次,所以我最后让方法体只触发一次,再每秒钟休眠1次确保结点有足够的时间刷新,也保证了执行的稳定性。
3.记录下用户自己的名字,比如我的是“至秦的瓜”,然后我在下面每个item的结点里去找到点赞区域,然后找是否有“至秦的瓜”这个字段,有的话说明这条朋友圈已经赞过了,跳过去,没有则执行点赞。
4.点赞程序的执行,则没什么难度了,代码都看得懂,这里就一带而过了。
代码实现:
/**
* Created by jiangzn on 17/2/6.
*/
public class MyAccessibilityService extends AccessibilityService {
@Override
protected void onServiceConnected() {
LogUtils.d("onServiceConnected");
}
String description;
ArrayList<Integer> topList = new ArrayList<>();
List<AccessibilityNodeInfo> lvs;
@Override
public void onAccessibilityEvent(AccessibilityEvent event) {
try {
//微信UI界面的根节点,开始遍历节点
AccessibilityNodeInfo rootNodeInfo = getRootInActiveWindow();
if (rootNodeInfo == null) {
return;
}
description = "";
if (rootNodeInfo.getContentDescription() != null) {
description = rootNodeInfo.getContentDescription().toString();
}
//自动点赞流程
if (mUserName.equals("")) {
//Lv
lvs = rootNodeInfo.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cn0");
LogUtils.d("找到的Lv数量: " + lvs.size());
//如果size不为0,证明当前在朋友圈页面下,开始执行逻辑
if (lvs.size() != 0) {
//1.先记录用户名
List<AccessibilityNodeInfo> userNames =
rootNodeInfo.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/afa");
if (userNames.size() != 0) {
if (userNames.get(0).getParent() != null && userNames.get(0).getParent().getChildCount() == 4) {
mUserName = userNames.get(0).getText().toString();
if (!mUserName.equals("") && !ifOnce) {
LogUtils.d("初始化,只会执行一次");
LogUtils.d("当前的用户名:" + mUserName);
ifOnce = true;
//测试朋友圈点赞
test3(rootNodeInfo);
}
}
}
} else {
ifOnce = false;
mUserName = "";
}
}
} catch (Exception e) {
if (e != null && e.getMessage() != null) {
LogUtils.d("报错:" + e.getMessage().toString());
}
}
}
String mUserName = "";
private boolean ifOnce = false;
/**
* com.tencent.mm:id/cn0
* 朋友圈点赞 (目前实现手动滚动全部点赞)
* 上方固定显示的名字:com.tencent.mm:id/afa
* 下方点赞:显示id:com.tencent.mm:id/cnn
* 每发现一个【评论按钮】,就去搜索当前同父组件下的点赞区域有没有自己的ID。
* 如果有就不点赞,如果没有就点赞
* 这里要改成不通过Id抓取提高稳定性
*
* @param rootNodeInfo
*/
private synchronized void test3(AccessibilityNodeInfo rootNodeInfo) {
LogUtils.d("当前线程:" + Thread.currentThread());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
topList.clear();
if (!mUserName.equals("")) {
//测试获得评论按钮的父节点,再反推出点赞按钮
List<AccessibilityNodeInfo> fuBtns =
rootNodeInfo.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/co0");
LogUtils.d("fuBtns数量:" + fuBtns.size());
if (fuBtns.size() != 0) {
//删掉超出屏幕的fuBtn
AccessibilityNodeInfo lastFuBtn = fuBtns.get(fuBtns.size() - 1);
Rect lastFuBtnOutBound = new Rect();
lastFuBtn.getBoundsInScreen(lastFuBtnOutBound);
if (lastFuBtnOutBound.top > Config.height) {
fuBtns.remove(lastFuBtn);
}
for (int i = 0; i < fuBtns.size(); i++) {
AccessibilityNodeInfo fuBtn = fuBtns.get(i);
LogUtils.d("fuBtn的子节点数量:" + fuBtn.getChildCount());//3-4个
List<AccessibilityNodeInfo> plBtns = fuBtn.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cj9");
LogUtils.d("从这里发现评论按钮:" + plBtns.size());
if (plBtns.size() == 0) {
if (lvs.get(0).performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD)) {
test3(getRootInActiveWindow());
}
return;
}
AccessibilityNodeInfo plbtn = plBtns.get(0); //评论按钮
List<AccessibilityNodeInfo> zanBtns = fuBtn.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cnn");
LogUtils.d("从这里发现点赞文字显示区域:" + zanBtns.size());
if (zanBtns.size() != 0) {
//2.如果不为空,则查找有没有自己点过赞,有则不点,没有则点
AccessibilityNodeInfo zanbtn = zanBtns.get(0);
LogUtils.d("点赞的人是:" + zanbtn.getText().toString());
if (zanbtn != null && zanbtn.getText() != null &&
zanbtn.getText().toString().contains(mUserName)) {
LogUtils.d("*********************这一条已经被赞过辣");
//判断是否需要翻页,如果当前所有页面的父节点都没点过了,就需要翻页
boolean ifxuyaofanye = false;
LogUtils.d("O(≧口≦)O: i=" + i + " fuBtns.size():" + fuBtns.size());
if (i == fuBtns.size() - 1) {
ifxuyaofanye = true;
}
if (ifxuyaofanye) {
//滑动前检测一下是否还有没有点过的点
if (jianceIfLou()) {
LogUtils.d("还有遗漏的点!!!!再检查一遍!!!!!!!!!!");
test3(getRootInActiveWindow());
} else {
if (lvs.get(0).performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD)) {
test3(getRootInActiveWindow());
return;
}
}
}
} else {
LogUtils.d("**************************:自己没有赞过!");
//开始执行点赞流程
if (plBtns.size() != 0) {
Rect outBounds = new Rect();
plbtn.getBoundsInScreen(outBounds);
int top = outBounds.top;
//根据top判断如果已经点开了就不重复点开了
if (topList.contains(top)) {
return;
}
//com.tencent.mm:id/cj5 赞
if (plbtn.performAction(AccessibilityNodeInfo.ACTION_CLICK)) {
List<AccessibilityNodeInfo> zanlBtns = rootNodeInfo.
findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cj3");
if (zanlBtns.size() != 0) {
if (!topList.contains(top) && zanlBtns.get(0).performAction(AccessibilityNodeInfo.ACTION_CLICK)) {
topList.add(top);
LogUtils.d("topList:" + topList.toString());
//判断是否需要翻页,如果当前所有页面的父节点都没点过了,就需要翻页
boolean ifxuyaofanye = false;
LogUtils.d("O(≧口≦)O: i=" + i + " fuBtns.size():" + fuBtns.size());
if (i == fuBtns.size() - 1) {
ifxuyaofanye = true;
}
if (ifxuyaofanye) {
//滑动前检测一下是否还有没有点过的点
if (jianceIfLou()) {
LogUtils.d("还有遗漏的点!!!!再检查一遍!!!!!!!!!!");
test3(getRootInActiveWindow());
} else {
if (lvs.get(0).performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD)) {
test3(getRootInActiveWindow());
return;
}
}
}
}
}
}
}
}
} else {
LogUtils.d("**************************:点赞区域为空!plBtns.size() :" + plBtns.size());
//开始执行点赞流程
if (plBtns.size() != 0) {
Rect outBounds = new Rect();
plbtn.getBoundsInScreen(outBounds);
int top = outBounds.top;
//根据top判断如果已经点开了就不重复点开了
if (topList.contains(top)) {
return;
}
//com.tencent.mm:id/cj5 赞
if (plbtn.performAction(AccessibilityNodeInfo.ACTION_CLICK)) {
List<AccessibilityNodeInfo> zanlBtns = rootNodeInfo.
findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cj3");
if (zanlBtns.size() != 0) {
if (!topList.contains(top) && zanlBtns.get(0).performAction(AccessibilityNodeInfo.ACTION_CLICK)) {
topList.add(top);
LogUtils.d("topList:" + topList.toString());
//判断是否需要翻页,如果当前所有页面的父节点都没点过了,就需要翻页
boolean ifxuyaofanye = false;
LogUtils.d("O(≧口≦)O: i=" + i + " fuBtns.size():" + fuBtns.size());
if (i == fuBtns.size() - 1) {
ifxuyaofanye = true;
}
if (ifxuyaofanye) {
//滑动前检测一下是否还有没有点过的点
if (jianceIfLou()) {
LogUtils.d("还有遗漏的点!!!!再检查一遍!!!!!!!!!!");
test3(getRootInActiveWindow());
} else {
if (lvs.get(0).performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD)) {
test3(getRootInActiveWindow());
return;
}
}
}
}
}
}
}
}
}
}
}
}
private boolean jianceIfLou() {
boolean result = false;
List<AccessibilityNodeInfo> fuBtns =
getRootInActiveWindow().findAccessibilityNodeInfosByViewId("com.tencent.mm:id/co0");
LogUtils.d("检查的父节点数量:" + fuBtns.size());
if (fuBtns.size() != 0) {
for (AccessibilityNodeInfo fuBtn : fuBtns) {
//点赞区域
List<AccessibilityNodeInfo> zanBtns = fuBtn.findAccessibilityNodeInfosByViewId("com.tencent.mm:id/cnn");
LogUtils.d("检查的父节点的点赞区域数量:" + zanBtns.size());
if (zanBtns.size() != 0) {
AccessibilityNodeInfo zanbtn = zanBtns.get(0);
LogUtils.d(" zanbtn.getText().toString():" + zanbtn.getText().toString());
if (zanbtn != null && zanbtn.getText() != null &&
zanbtn.getText().toString().contains(mUserName)) {
result = false;
} else {
result = true;
}
} else {
result = true;
}
}
}
return result;
}
@Override
public void onInterrupt() {
LogUtils.d("onInterrupt");
}
}
辅助服务类的配置方法可以参考上文AccessibilityService——实现微信切换账号功能。
目前的代码有两段几乎重复的,这里没有抽离出来了因为之后我还要进一步优化(恩这就是个demo版不想改了。。)
来源:https://blog.csdn.net/qq_22770457/article/details/55688322


猜你喜欢
- mybatis-plus今天遇到一个问题,就是mybatis 没有读取到mapper.xml 文件。特此记录一下,问题如下:at com.b
- Set系列集合介绍Set集合概述Set系列集合特点:无序:存取数据的顺序是不一定的, 当数据存入后, 集合的顺序就固定下来了不重复:可以去除
- 某些Google Play服务(例如Google登录和App Invites)要求我们提供签名证书的SHA-1,以便google paly为
- 方法一:效果如下图所示:代码如下:using System;using System.Collections.Generic;using S
- 本文实例为大家分享了Java图片验证码代码,供大家参考,具体内容如下网页显示效果:index.jsp 使用两种方式强制图片更新: 1、设置图
- 前言Unity在运行时可以将一些物体进行合并,从而用一个绘制调用来渲染他们。这一操作,我们称之为“批处理”,能得到越好的渲染性能。Unity
- 1.使用ASCII码判断您可以使用ASCII码来进行判断字符串中的内容是否为纯数字。步骤如下:先判断字符串是否为空的情况,保证代码运行的稳定
- 一、C++11智能指针概述在C++中,动态内存的使用时有一定的风险的,因为它没有垃圾回收机制,很容易导致忘记释放内存的问题,具体体现在异常的
- 1. 最小生成树连通图中的每一棵生成树 , 都是原图的极大无环子图 , 即: 从中删去任何一条边 , 生成树就不再连通;反之 , 在其中引入
- 先看一下Android悬浮按钮点击回到顶部的效果:FloatingActionButton是Design Support库中提供的一个控件,
- 一、依赖注入方式思考:向一个类中传递数据的方式有几种?普通方法(set方法)构造方法思考:依赖注入描述了在容器中建立bean与bean之间依
- AOP我想大家都很清楚,有时候我们需要处理一些请求日志,或者对某些方法进行一些监控,如果出现例外情况应该进行怎么样的处理,现在,我们从spr
- 异常捕获机制 C#1.示意图2.异常捕获机制,代码:3.异常捕获机制,结果:4.求几周,剩余几天?代码:5.结果:6.求几月几周零几天 设一
- 前言公司的邮件系统用的是 * 的 Lotus notes, 你敢信?最近要实现一个功能,邮件提醒功能,就是通过自动发送提醒邮件 前
- 从SD卡中获取图片资源,或者拍一张新的图片。 先贴代码 获取图片: 注释:拍照获取的话,可以指定图片的保存地址,在此不说明。 CharSeq
- 1. 介绍结合上面的ReentrantLock类图,ReentrantLock实现了Lock接口,它的内部类Sync继承自AQS,绝大部分使
- 引言:前面几个专题对委托进行了详细的介绍的,然后我们在编写代码过程中经常会听到“事件”这个概念的,尤其是写UI的时候,当我们点击一个按钮后V
- IO流基本概念IO流用来处理设备之间的数据传输Java对数据的操作是通过流的方式Java用于操作流的对象都是在IO包上流按操作数
- 一、运行class文件执行带main方法的class文件,命令行为:java <CLASS文件名>注意:CLASS文件名不要带文
- 利用Javaweb开发的一个校园服务系统,通过发布自己的任务并设置悬赏金额,有些类似于赏金猎人,在这里分享给大家,有需要可以联系我:2186