使用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版不想改了。。)
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。