1. 程式人生 > >利用AccessibilityService自動獲取微信號(Android)

利用AccessibilityService自動獲取微信號(Android)

perm 問題: 能夠 all 更多 原因 override 改變 第一個

前言:

最近遇到一個需求,要求寫一個小插件,能夠自動在微信的頁面彈出一個窗口,展示用戶的相關信息(與我們公司有關的信息,方便運營快速了解用戶信息)。

當時我第一反應是不可能,如果能夠在別的app中獲取對應的信息,那豈不是太不安全了。直到我知道了AccessibilityService這個東西。

基本思路:

利用AccessibilityService服務來獲取到微信頁面的頁面信息,並獲取到用戶的微信號,有了微信號一切都好辦了。

由於獲取用戶好友微信號和獲取本人微信號的方法相同,因此此篇文章主要介紹的是如何通過AccessibilityService來獲取本人的微信號。

過程:

AccessibilityService
是什麽?

在你的手機更多設置或者高級設置中,我們會發現有個無障礙的功能,很多人不知道這個功能具體是幹嘛的,其實這個功能是為了增強用戶界面以幫助殘障人士,或者可能暫時無法與設備充分交互的人們。

它的具體實現是通過AccessibilityService服務運行在後臺中,通過AccessibilityEvent接收指定事件的回調。這樣的事件表示用戶在界面中的一些狀態轉換,例如:焦點改變了,一個按鈕被點擊,等等。這樣的服務可以選擇請求活動窗口的內容的能力。簡單的說AccessibilityService就是一個後臺監控

服務,當你監控的內容發生改變時,就會調用後臺服務的回調方法。

如何創建一個AccessibilityService

實現一個自己的AccessibilityService,需要繼承AccessibilityService類,並至少實現onAccessibilityEventonInterrupt方法:

 1 public class MyAccessibilityService extends AccessibilityService {
 2 
 3     final String TAG = "MyAccessibilityService";
 4 
 5     /**
 6      * 當服務啟動的時候會被調用
 7      */
 8     @Override
 9     protected
void onServiceConnected() { 10 super.onServiceConnected(); 11 Log.d(TAG, "connected"); 12 } 13 14 /** 15 * 監聽窗口變化的回調 16 */ 17 @Override 18 public void onAccessibilityEvent(AccessibilityEvent event) { 19 Log.d(TAG, event.getPackageName() + ""); 20 } 21 22 /** 23 * 中斷服務的回調 24 */ 25 @Override 26 public void onInterrupt() { 27 Log.d(TAG, "onInterrupt"); 28 } 29 }

AccessibilityService中的一些常用方法:

  1. disableSelf():禁用當前服務,服務可以通過該方法停止運行;
  2. findFocus(int focus):查找擁有特定焦點類型的控件;
  3. getRootInActiveWindow():如果配置能夠獲取窗口內容,則會返回當前活動窗口的根結點;
  4. getServiceInfo():獲取當前服務的配置信息;
  5. onAccessibilityEvent(AccessibilityEvent event):有關AccessibilityEvent事件的回調函數,系統通過sendAccessibiliyEvent()不斷的發送AccessibilityEvent到此處;
  6. performGlobalAction(int action):執行全局操作,比如返回,回到主頁,打開最近等操作;
  7. setServiceInfo(AccessibilityServiceInfo info):設置當前服務的配置信息;
  8. onServiceConnected():系統成功綁定該服務時被觸發,也就是當你在設置中開啟相應的服務,系統成功的綁定了該服務時會觸發,通常我們可以在這裏做一些初始化操作;
  9. onInterrupt():服務中斷時的回調。

聲明該服務:

 1 <service
 2     android:name=".MyAccessibilityService"
 3     android:enabled="true"
 4     android:exported="true"
 5     android:label="這是一個用戶測試的無障礙服務"
 6     android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
 7     <intent-filter>
 8         <action android:name="android.accessibilityservice.AccessibilityService" />
 9     </intent-filter>
10 </service>

配置服務參數:

主要是用於聲明該服務的一些配置參數,現在有兩種配置服務參數的方法:在安卓4.0之後可以通過meta-data標簽來在xml中配置,也可以通過動態代碼直接配置。這裏我們通過xml進行配置。

首先在res下的xml文件夾下創建配置文件,

1 <?xml version="1.0" encoding="utf-8"?>
2 <accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
3     android:accessibilityEventTypes="typeAllMask"
4     android:accessibilityFeedbackType="feedbackAllMask"
5     android:canRetrieveWindowContent="true"
6     android:notificationTimeout="100"
7     android:packageNames="com.tencent.mm"
8     android:description="@string/description" />

然後將配置文件添加到清單文件中,

 1 <service
 2     android:name=".MyAccessibilityService"
 3     android:enabled="true"
 4     android:exported="true"
 5     android:label="這是一個用戶測試的無障礙服務"
 6     android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
 7     <intent-filter>
 8         <action android:name="android.accessibilityservice.AccessibilityService" />
 9     </intent-filter>
10     <meta-data
11         android:name="android.accessibilityservice"
12         android:resource="@xml/config_accessibility" />
13 </service>

下面對xml中的一些參數進行介紹:

  1. accessibilityEventTypes:表示該服務對界面中的哪些變化感興趣,即哪些事件通知,比如窗口打開,滑動,焦點變化,長按等。具體的值可以在AccessibilityEvent類中查到,如typeAllMask表示接受所有的事件通知;
  2. accessibilityFeedbackType:表示反饋方式,比如是語音播放,還是震動(此參數是必須的,不寫的話不會走回調方法)
  3. canRetrieveWindowContent:表示該服務能否訪問活動窗口中的內容.也就是如果你希望在服務中獲取窗體內容的化,則需要設置其值為true;
  4. notificationTimeout:接受事件的時間間隔,通常將其設置為100即可;
  5. packageNames:表示對該服務是用來監聽哪個包的產生的事件,這裏以微信的包名為例(如果要監聽的包有多個,則可以在代碼中設置;如若不寫,則監控的是所有的包)
  6. description:對該服務的無障礙描述。

如何開啟AccessibilityService呢?

以小米手機為例,在設置中打開更多設置,進入無障礙。然後打開之前聲明的服務即可。

技術分享圖片

如何獲取微信“我的”頁面的微信號呢?

這裏主要利用AccessibilityNodeInfofindAccessibilityNodeInfosByViewId(String viewId)方法,該方法用於根據控件標識來獲取到整個控件。

那麽問題來了,如何知道微信該控件的標識呢?這裏可以通過SDK的工具DDMS工具。

進入SDK目錄的tools目錄,找到monitor.bat文件,雙擊即可。

技術分享圖片

進入DDMS界面後,選中微信的包名,並點擊如下所示按鈕即可分析當前微信頁面的布局信息:

技術分享圖片

如下圖,可以發現該控件的標識為:com.tencent.mm:id/czz

技術分享圖片

因此,即可通過如下方法獲取到該控件的值:

 1 @Override
 2 public void onAccessibilityEvent(AccessibilityEvent event) {
 3     Log.d(TAG, event.getPackageName() + "");
 4 
 5     if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2) {
 6         // 通過id獲取到微信號的View
 7         List<AccessibilityNodeInfo> nodeInfoList = getRootInActiveWindow().findAccessibilityNodeInfosByViewId("com.tencent.mm:id/czz");
 8         String wxCode;
 9         if (nodeInfoList != null && nodeInfoList.size() > 0) {
10             wxCode = nodeInfoList.get(0).getText().toString();
11             Log.d(TAG, wxCode);
12         }
13     }
14 
15 }

運行結果如下:

技術分享圖片

總結:

  • 在過程中遇到了兩個問題:第一個問題是運行完了之後服務已經開啟,但是一直不走回調,經查是由於沒有寫accessibilityFeedbackType參數的原因;
  • 第二個問題是,一開始通過Android Studio自帶的Layout Inspector來獲取控件的標識,當時獲取的標識是czz,並不全,導致一直獲取不到相應的微信號;
  • 無障礙服務是一個很便攜但是也很危險的服務,所以輕易不要給別人無障礙服務的權限;
  • 不要輕易對一件事進行判斷,需要進行了解之後才進行判斷(在此之前我一直覺得該功能是無法實現的);
  • 對於一個功能,第一反應應該是如何實現而不是如何推脫。

參考博客:

  • 你真的理解AccessibilityService嗎
  • [Android進階]學習AccessibilityService實現微信搶紅包插件

大家如果有什麽疑問或者建議可以通過評論或者郵件的方式聯系我,歡迎大家的評論~

利用AccessibilityService自動獲取微信號(Android)