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

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

程序员文章站 2022-09-02 08:19:52
概述: 利用accessibilityservice机制实现了一个比较好玩儿的功能,微信朋友圈自动遍历点赞。即通过不断的滚动+点赞实现把每一条朋友圈都赞一次。 当然其中还要涉及一些...

概述:

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

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