广告位联系
返回顶部
分享到

详解使用AccessibilityService实现自动遍历点赞的功能

Android 来源:互联网搜集 作者:秩名 发布时间:2019-12-15 08:49:09 人浏览
摘要

概述: 利用AccessibilityService机制实现了一个比较好玩儿的功能,微信朋友圈自动遍历点赞。即通过不断的滚动+点赞实现把每一条朋友圈都赞一次。 当然其中还要涉及一些判断算法,比如如果这条朋友圈已经赞过就跳过去,以及当前界面没有可赞的朋友圈时执行翻

概述:

利用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版不想改了。。)


版权声明 : 本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务和不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权, 违法违规的内容, 请发送邮件至2530232025#qq.cn(#换@)举报,一经查实,本站将立刻删除。
原文链接 : https://blog.csdn.net/qq_22770457/article/details/55688322
    Tag :
相关文章
  • Kotlin的Collection与Sequence操作异同点介绍

    Kotlin的Collection与Sequence操作异同点介绍
    在Android开发中,集合是我们必备的容器,Kotlin的标准库中提供了很多处理集合的方法,而且还提供了两种基于容器的工作方式:Collection 和
  • 实现一个Kotlin函数类型方法

    实现一个Kotlin函数类型方法
    接口与函数类型 业务开发中,经常会有实现一个函数式接口(即接口只有一个方法需要实现)的场景,大家应该都会不假思索的写出如下代
  • Android10 App启动Activity源码分析
    ActivityThread的main方法 让我们把目光聚焦到ActivityThread的main方法上。 ActivityThread的源码路径为/frameworks/base/core/java/android/app/ActivityThread。 1 2
  • Android10客户端事务管理ClientLifecycleManager源码解析

    Android10客户端事务管理ClientLifecycleManager源码解析
    在Android 10 App启动分析之Activity启动篇(二)一文中,简单地介绍了Activity的生命周期管理器是如何调度Activity进入onCreate生命周期的流程。这
  • Kotlin对象的懒加载方式by lazy与lateinit异同介绍

    Kotlin对象的懒加载方式by lazy与lateinit异同介绍
    属性或对象的延时加载是我们相当常用的,一般我们都是使用 lateinit 和 by lazy 来实现。 他们两者都是延时初始化,那么在使用时那么他们两
  • Android类加载流程分析

    Android类加载流程分析
    本文分析的代码基于Android8.1.0源码。 流程分析 从loadClass开始,我们来看下Android中类加载的流程 /libcore/ojluni/src/main/java/java/lang/ClassLoader.ja
  • Android实现读写USB串口数据的代码

    Android实现读写USB串口数据的代码
    最近在研究USB方面的内容;先后做了关于Android读写HID、串口设备的DEMO。本文比较简单,主要介绍的是Android实现读取串口数据的功能 废话不
  • Epoxy - 在RecyclerView中构建复杂界面
    Diffing 对于复杂数据结构支持的多个视图类型展示在屏幕上, Epoxy此时是尤其有用的. 在这些场景中, 数据可能会被网络请求, 异步 Observable, 用
  • Android性能优化的详细介绍

    Android性能优化的详细介绍
    性能优化是一个app很重要的一部分,一个性能优良的app从被下载到启动到使用都能给用户到来很好的体验。自然我们做性能优化也是从被下
  • Android进阶宝典-插件化2(Hook启动插件中四大组件

    Android进阶宝典-插件化2(Hook启动插件中四大组件
    在上一节,我们主要介绍了如果通过反射来加载插件中的类,调用类中的方法;既然插件是一个apk,其实最重要的是启动插件中的Activity、
  • 本站所有内容来源于互联网或用户自行发布,本站仅提供信息存储空间服务,不拥有版权,不承担法律责任。如有侵犯您的权益,请您联系站长处理!
  • Copyright © 2017-2022 F11.CN All Rights Reserved. F11站长开发者网 版权所有 | 苏ICP备2022031554号-1 | 51LA统计