AccessibilityService——实现自动遍历点赞功能

Posted 化作孤岛的瓜

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了AccessibilityService——实现自动遍历点赞功能相关的知识,希望对你有一定的参考价值。

概述:

利用AccessibilityService机制实现了一个比较好玩儿的功能,微信朋友圈自动遍历点赞。即通过不断的滚动+点赞实现把每一条朋友圈都赞一次。

当然其中还要涉及一些判断算法,比如如果这条朋友圈已经赞过就跳过去,以及当前界面没有可赞的朋友圈时执行翻页。其实做起来试错是个很繁冗的过程,这个效果也差不多做了两天。


使用方式:

运行程序-开启无障碍服务,再切换到微信主界面,进入朋友圈,就会自动执行点赞程序了。


效果图如下:



实现原理步骤以及难点:

1.首先要获取到微信朋友圈这个界面的ListView结点,或者通过根节点描述判断是否进入该界面。

2.到了朋友圈界面之后可以执行程序方法体了,但是要有个boolean值判断只能执行一次。

为什么该方法体只能执行一次呢?(代码在下面有),因为如果被动地让onAccessibilityEvent调用我们的方法,会出现很多问题,比如结点刷新过快,多次触发方法导致点赞步骤同时执行N次然后无限死循环,因为onAccessibilityEvent触发太快了,大概0.几毫秒触发一次,所以我最后让方法体只触发一次,再每秒钟休眠1次确保结点有足够的时间刷新,也保证了执行的稳定性。

3.记录下用户自己的名字,比如我的是“至秦的瓜”,然后我在下面每个item的结点里去找到点赞区域,然后找是否有“至秦的瓜”这个字段,有的话说明这条朋友圈已经赞过了,跳过去,没有则执行点赞。

3.点赞程序的执行,则没什么难度了,代码都看得懂,这里就一带而过了。 

代码实现:

/**
 * 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版不想改了。。)。 就说这么多,有不懂得可以问我哈~

以上是关于AccessibilityService——实现自动遍历点赞功能的主要内容,如果未能解决你的问题,请参考以下文章

Android中微信抢红包助手的实现(代码整理)

AccessibilityService——实现微信切换账号功能

Android自动化之AccessibilityService

AccessibilityService——实现自动遍历点赞功能

Android AccessibilityService 事件分发原理

安卓开发之基于AccessibilityService实现聊天机器人对其他应用的调起