使用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版不想改了。。)
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。