Android中微信搶紅包助手的實(shí)現(xiàn)詳解
實(shí)現(xiàn)原理
通過利用AccessibilityService輔助服務(wù),監(jiān)測屏幕內(nèi)容,如監(jiān)聽狀態(tài)欄的信息,屏幕跳轉(zhuǎn)等,以此來實(shí)現(xiàn)自動(dòng)拆紅包的功能。關(guān)于AccessibilityService輔助服務(wù),可以自行百度了解更多。
代碼基礎(chǔ):
1.首先聲明一個(gè)RedPacketService繼承自AccessibilityService,該服務(wù)類有兩個(gè)方法必須重寫,如下:
/** * Created by cxk on 2017/2/3. * * 搶紅包服務(wù)類 */ public class RedPacketService extends AccessibilityService { /** * 必須重寫的方法:此方法用了接受系統(tǒng)發(fā)來的event。在你注冊的event發(fā)生是被調(diào)用。在整個(gè)生命周期會(huì)被調(diào)用多次。 */ @Override public void onAccessibilityEvent(AccessibilityEvent event) { } /** * 必須重寫的方法:系統(tǒng)要中斷此service返回的響應(yīng)時(shí)會(huì)調(diào)用。在整個(gè)生命周期會(huì)被調(diào)用多次。 */ @Override public void onInterrupt() { Toast.makeText(this, "我快被終結(jié)了啊-----", Toast.LENGTH_SHORT).show(); } /** * 服務(wù)已連接 */ @Override protected void onServiceConnected() { Toast.makeText(this, "搶紅包服務(wù)開啟", Toast.LENGTH_SHORT).show(); super.onServiceConnected(); } /** * 服務(wù)已斷開 */ @Override public boolean onUnbind(Intent intent) { Toast.makeText(this, "搶紅包服務(wù)已被關(guān)閉", Toast.LENGTH_SHORT).show(); return super.onUnbind(intent); } }
2.對(duì)我們的RedPacketService進(jìn)行一些配置,這里配置方法可以選擇代碼動(dòng)態(tài)配置(onServiceConnected里配置),也可以直接在res/xml下新建.xml文件,沒有xml文件夾就新建。這里我們將文件命名為redpacket_service_config.xml,代碼如下:
<?xml version="1.0" encoding="utf-8"?> <accessibility-service xmlns:android="http://schemas.android.com/apk/res/android" android:accessibilityEventTypes="typeAllMask" android:accessibilityFeedbackType="feedbackGeneric" android:accessibilityFlags="flagDefault" android:canRetrieveWindowContent="true" android:description="@string/desc" android:notificationTimeout="100" android:packageNames="com.tencent.mm" />
accessibilityEventTypes:
響應(yīng)哪一種類型的事件,typeAllMask就是響應(yīng)所有類型的事件了,另外還有單擊、長按、滑動(dòng)等。
accessibilityFeedbackType:
用什么方式反饋給用戶,有語音播出和振動(dòng)。可以配置一些TTS引擎,讓它實(shí)現(xiàn)發(fā)音。
packageNames:
指定響應(yīng)哪個(gè)應(yīng)用的事件。這里我們是寫搶紅包助手,就寫微信的包名:com.tencent.mm,這樣就可以監(jiān)聽微信產(chǎn)生的事件了。
notificationTimeout:
響應(yīng)時(shí)間
description:
輔助服務(wù)的描述信息。
3.service是四大組件之一,需要在AndroidManifest進(jìn)行配置,注意這里稍微有些不同:
<!--搶紅包服務(wù)--> <service android:name=".RedPacketService" android:enabled="true" android:exported="true" android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"> <intent-filter> <action android:name="android.accessibilityservice.AccessibilityService" /> </intent-filter> <meta-data android:name="android.accessibilityservice" android:resource="@xml/redpacket_service_config"></meta-data> </service>
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE" 權(quán)限申請
android:resource="@xml/redpacket_service_config" 引用剛才的配置文件
核心代碼:
我們的紅包助手,核心思路分為三步走:
監(jiān)聽通知欄微信消息,如果彈出[微信紅包]字樣,模擬手指點(diǎn)擊狀態(tài)欄跳轉(zhuǎn)到微信聊天界面→在微信聊天界面查找紅包,如果找到則模擬手指點(diǎn)擊打開,彈出打開紅包界面→模擬手指點(diǎn)擊紅包“開”
1.監(jiān)聽通知欄消息,查看是否有[微信紅包]字樣,代碼如下:
@Override public void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); switch (eventType) { //通知欄來信息,判斷是否含有微信紅包字樣,是的話跳轉(zhuǎn) case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED: List<CharSequence> texts = event.getText(); for (CharSequence text : texts) { String content = text.toString(); if (!TextUtils.isEmpty(content)) { //判斷是否含有[微信紅包]字樣 if (content.contains("[微信紅包]")) { //如果有則打開微信紅包頁面 openWeChatPage(event); } } } break; } } /** * 開啟紅包所在的聊天頁面 */ private void openWeChatPage(AccessibilityEvent event) { //A instanceof B 用來判斷內(nèi)存中實(shí)際對(duì)象A是不是B類型,常用于強(qiáng)制轉(zhuǎn)換前的判斷 if (event.getParcelableData() != null && event.getParcelableData() instanceof Notification) { Notification notification = (Notification) event.getParcelableData(); //打開對(duì)應(yīng)的聊天界面 PendingIntent pendingIntent = notification.contentIntent; try { pendingIntent.send(); } catch (PendingIntent.CanceledException e) { e.printStackTrace(); } } }
2.判斷當(dāng)前是否在微信聊天頁面,是的話遍歷當(dāng)前頁面各個(gè)控件,找到含有微信紅包或者領(lǐng)取紅包的textview控件,然后逐層找到他的可點(diǎn)擊父布局(圖中綠色部分),模擬點(diǎn)擊跳轉(zhuǎn)到含有“開”的紅包界面,代碼如下:
@Override public void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); switch (eventType) { //窗口發(fā)生改變時(shí)會(huì)調(diào)用該事件 case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED: String className = event.getClassName().toString(); //判斷是否是微信聊天界面 if ("com.tencent.mm.ui.LauncherUI".equals(className)) { //獲取當(dāng)前聊天頁面的根布局 AccessibilityNodeInfo rootNode = getRootInActiveWindow(); //開始找紅包 findRedPacket(rootNode); } } } /** * 遍歷查找紅包 */ private void findRedPacket(AccessibilityNodeInfo rootNode) { if (rootNode != null) { //從最后一行開始找起 for (int i = rootNode.getChildCount() - 1; i >= 0; i--) { AccessibilityNodeInfo node = rootNode.getChild(i); //如果node為空則跳過該節(jié)點(diǎn) if (node == null) { continue; } CharSequence text = node.getText(); if (text != null && text.toString().equals("領(lǐng)取紅包")) { AccessibilityNodeInfo parent = node.getParent(); //while循環(huán),遍歷"領(lǐng)取紅包"的各個(gè)父布局,直至找到可點(diǎn)擊的為止 while (parent != null) { if (parent.isClickable()) { //模擬點(diǎn)擊 parent.performAction(AccessibilityNodeInfo.ACTION_CLICK); //isOpenRP用于判斷該紅包是否點(diǎn)擊過 isOpenRP = true; break; } parent = parent.getParent(); } } //判斷是否已經(jīng)打開過那個(gè)最新的紅包了,是的話就跳出for循環(huán),不是的話繼續(xù)遍歷 if (isOpenRP) { break; } else { findRedPacket(node); } } } }
3.點(diǎn)擊紅包后,在模擬手指點(diǎn)擊“開”以此開啟紅包,跳轉(zhuǎn)到紅包詳情界面,方法與步驟二類似:
@Override public void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); switch (eventType) { //窗口發(fā)生改變時(shí)會(huì)調(diào)用該事件 case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED: String className = event.getClassName().toString(); //判斷是否是顯示‘開'的那個(gè)紅包界面 if ("com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI".equals(className)) { AccessibilityNodeInfo rootNode = getRootInActiveWindow(); //開始搶紅包 openRedPacket(rootNode); } break; } } /** * 開始打開紅包 */ private void openRedPacket(AccessibilityNodeInfo rootNode) { for (int i = 0; i < rootNode.getChildCount(); i++) { AccessibilityNodeInfo node = rootNode.getChild(i); if ("android.widget.Button".equals(node.getClassName())) { node.performAction(AccessibilityNodeInfo.ACTION_CLICK); } openRedPacket(node); } }
結(jié)合以上三步,下面是完整代碼,注釋已經(jīng)寫的很清楚,直接看代碼:
package com.cxk.redpacket; import android.accessibilityservice.AccessibilityService; import android.app.KeyguardManager; import android.app.Notification; import android.app.PendingIntent; import android.app.Service; import android.content.Context; import android.content.Intent; import android.os.IBinder; import android.os.PowerManager; import android.text.TextUtils; import android.util.Log; import android.view.accessibility.AccessibilityEvent; import android.view.accessibility.AccessibilityNodeInfo; import android.widget.Toast; import java.util.List; /** * 搶紅包Service,繼承AccessibilityService */ public class RedPacketService extends AccessibilityService { /** * 微信幾個(gè)頁面的包名+地址。用于判斷在哪個(gè)頁面 LAUCHER-微信聊天界面,LUCKEY_MONEY_RECEIVER-點(diǎn)擊紅包彈出的界面 */ private String LAUCHER = "com.tencent.mm.ui.LauncherUI"; private String LUCKEY_MONEY_DETAIL = "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyDetailUI"; private String LUCKEY_MONEY_RECEIVER = "com.tencent.mm.plugin.luckymoney.ui.LuckyMoneyReceiveUI"; /** * 用于判斷是否點(diǎn)擊過紅包了 */ private boolean isOpenRP; @Override public void onAccessibilityEvent(AccessibilityEvent event) { int eventType = event.getEventType(); switch (eventType) { //通知欄來信息,判斷是否含有微信紅包字樣,是的話跳轉(zhuǎn) case AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED: List<CharSequence> texts = event.getText(); for (CharSequence text : texts) { String content = text.toString(); if (!TextUtils.isEmpty(content)) { //判斷是否含有[微信紅包]字樣 if (content.contains("[微信紅包]")) { //如果有則打開微信紅包頁面 openWeChatPage(event); isOpenRP=false; } } } break; //界面跳轉(zhuǎn)的監(jiān)聽 case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED: String className = event.getClassName().toString(); //判斷是否是微信聊天界面 if (LAUCHER.equals(className)) { //獲取當(dāng)前聊天頁面的根布局 AccessibilityNodeInfo rootNode = getRootInActiveWindow(); //開始找紅包 findRedPacket(rootNode); } //判斷是否是顯示‘開'的那個(gè)紅包界面 if (LUCKEY_MONEY_RECEIVER.equals(className)) { AccessibilityNodeInfo rootNode = getRootInActiveWindow(); //開始搶紅包 openRedPacket(rootNode); } //判斷是否是紅包領(lǐng)取后的詳情界面 if(LUCKEY_MONEY_DETAIL.equals(className)){ //返回桌面 back2Home(); } break; } } /** * 開始打開紅包 */ private void openRedPacket(AccessibilityNodeInfo rootNode) { for (int i = 0; i < rootNode.getChildCount(); i++) { AccessibilityNodeInfo node = rootNode.getChild(i); if ("android.widget.Button".equals(node.getClassName())) { node.performAction(AccessibilityNodeInfo.ACTION_CLICK); } openRedPacket(node); } } /** * 遍歷查找紅包 */ private void findRedPacket(AccessibilityNodeInfo rootNode) { if (rootNode != null) { //從最后一行開始找起 for (int i = rootNode.getChildCount() - 1; i >= 0; i--) { AccessibilityNodeInfo node = rootNode.getChild(i); //如果node為空則跳過該節(jié)點(diǎn) if (node == null) { continue; } CharSequence text = node.getText(); if (text != null && text.toString().equals("領(lǐng)取紅包")) { AccessibilityNodeInfo parent = node.getParent(); //while循環(huán),遍歷"領(lǐng)取紅包"的各個(gè)父布局,直至找到可點(diǎn)擊的為止 while (parent != null) { if (parent.isClickable()) { //模擬點(diǎn)擊 parent.performAction(AccessibilityNodeInfo.ACTION_CLICK); //isOpenRP用于判斷該紅包是否點(diǎn)擊過 isOpenRP = true; break; } parent = parent.getParent(); } } //判斷是否已經(jīng)打開過那個(gè)最新的紅包了,是的話就跳出for循環(huán),不是的話繼續(xù)遍歷 if (isOpenRP) { break; } else { findRedPacket(node); } } } } /** * 開啟紅包所在的聊天頁面 */ private void openWeChatPage(AccessibilityEvent event) { //A instanceof B 用來判斷內(nèi)存中實(shí)際對(duì)象A是不是B類型,常用于強(qiáng)制轉(zhuǎn)換前的判斷 if (event.getParcelableData() != null && event.getParcelableData() instanceof Notification) { Notification notification = (Notification) event.getParcelableData(); //打開對(duì)應(yīng)的聊天界面 PendingIntent pendingIntent = notification.contentIntent; try { pendingIntent.send(); } catch (PendingIntent.CanceledException e) { e.printStackTrace(); } } } /** * 服務(wù)連接 */ @Override protected void onServiceConnected() { Toast.makeText(this, "搶紅包服務(wù)開啟", Toast.LENGTH_SHORT).show(); super.onServiceConnected(); } /** * 必須重寫的方法:系統(tǒng)要中斷此service返回的響應(yīng)時(shí)會(huì)調(diào)用。在整個(gè)生命周期會(huì)被調(diào)用多次。 */ @Override public void onInterrupt() { Toast.makeText(this, "我快被終結(jié)了啊-----", Toast.LENGTH_SHORT).show(); } /** * 服務(wù)斷開 */ @Override public boolean onUnbind(Intent intent) { Toast.makeText(this, "搶紅包服務(wù)已被關(guān)閉", Toast.LENGTH_SHORT).show(); return super.onUnbind(intent); } /** * 返回桌面 */ private void back2Home() { Intent home=new Intent(Intent.ACTION_MAIN); home.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); home.addCategory(Intent.CATEGORY_HOME); startActivity(home); } }
使用方法:
設(shè)置-輔助功能-無障礙-點(diǎn)擊RedPacket開啟即可
已知問題:
1.聊天列表或者聊天界面中無法直接自動(dòng)搶紅包
2.未做熄屏自動(dòng)搶紅包處理,想要熄屏能自動(dòng)搶紅包的同學(xué)直接把開屏代碼寫在第一步即可。
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Flutter自定義實(shí)現(xiàn)神奇動(dòng)效的卡片切換視圖的示例代碼
這篇文章主要介紹了Flutter自定義實(shí)現(xiàn)神奇動(dòng)效的卡片切換視圖的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-04-04Android技巧一之啟動(dòng)屏+新功能左右導(dǎo)航邏輯
這篇文章主要介紹了Android技巧一之啟動(dòng)屏+新功能左右導(dǎo)航邏輯的相關(guān)資料,需要的朋友可以參考下2016-01-01Android編程實(shí)現(xiàn)WebView添加進(jìn)度條的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)WebView添加進(jìn)度條的方法,涉及Android WebView界面及控件功能相關(guān)操作技巧,需要的朋友可以參考下2017-02-02android studio logcat 無篩選 顯示全部日志 無應(yīng)用包名區(qū)分方式
這篇文章主要介紹了android studio logcat 無篩選 顯示全部日志 無應(yīng)用包名區(qū)分方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-04-04Android實(shí)例代碼理解設(shè)計(jì)模式SOLID六大原則
程序設(shè)計(jì)領(lǐng)域, SOLID (單一功能、開閉原則、里氏替換、接口隔離以及依賴反轉(zhuǎn))是由羅伯特·C·馬丁在21世紀(jì)早期 引入的記憶術(shù)首字母縮略字,指代了面向?qū)ο缶幊毯兔嫦驅(qū)ο笤O(shè)計(jì)的基本原則2021-10-10Android Force Close 出現(xiàn)的異常原因分析及解決方法
本文給大家講解Android Force Close 出現(xiàn)的異常原因分析及解決方法,forceclose意為強(qiáng)行關(guān)閉,當(dāng)前應(yīng)用程序發(fā)生了沖突。對(duì)android force close異常分析感興趣的朋友一起通過本文學(xué)習(xí)吧2016-08-08Android自定義View實(shí)現(xiàn)拖拽效果
這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)拖拽效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11