1. 程式人生 > >Android 智慧安裝

Android 智慧安裝

最近老大讓給專案中新增一個自動更新的功能,需求是這樣的,檢測到有新的版本,直接下載,下載完成後自動安裝,重啟之後直接啟動新版本,我合計了下,這不按套路出牌啊,這不就是靜默安裝嗎,而且還沒有ROOT許可權,沒辦法再難也得做啊,網上找了幾種方法,一種是讓應用成為系統應用,,但這方法不靠譜啊。後來看到了郭神寫的智慧安裝,才解決這個問題。傳送門

不多說了,上程式碼
首先在res/xml目錄下新建一個accessibility_service_config.xml檔案,程式碼如下所示:

<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:packageNames="com.android.packageinstaller" android:description="@string/accessibility_service_description" android:accessibilityEventTypes="typeAllMask" android:accessibilityFlags="flagDefault"
android:accessibilityFeedbackType="feedbackGeneric" android:canRetrieveWindowContent="true" />

其中,packageNames指定我們要監聽哪個應用程式下的視窗活動,這裡寫com.android.packageinstaller表示監聽Android系統的安裝介面。description指定在無障礙服務當中顯示給使用者看的說明資訊,上圖中360手機助手的一大段內容就是在這裡指定的。accessibilityEventTypes指定我們在監聽視窗中可以模擬哪些事件,這裡寫typeAllMask表示所有的事件都能模擬。accessibilityFlags可以指定無障礙服務的一些附加引數,這裡我們傳預設值flagDefault就行。accessibilityFeedbackType指定無障礙服務的反饋方式,實際上無障礙服務這個功能是Android提供給一些殘疾人士使用的,比如說盲人不方便使用手機,就可以藉助無障礙服務配合語音反饋來操作手機,而我們其實是不需要反饋的,因此隨便傳一個值就可以,這裡傳入feedbackGeneric。最後canRetrieveWindowContent指定是否允許我們的程式讀取視窗中的節點和內容,必須寫true。
記得在string.xml檔案中寫一下description中指定的內容,如下所示:

<resources>  
    <string name="app_name">InstallTest</string>  
    <string name="accessibility_service_description">智慧安裝服務,無需使用者的任何操作就可以自動安裝程式。</string>  
</resources>  

接下來修改AndroidManifest.xml檔案,在裡面配置無障礙服務:

<?xml version="1.0" encoding="utf-8"?>  
<manifest xmlns:android="http://schemas.android.com/apk/res/android"  
    package="com.example.installtest">  

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />  

    <application  
        android:allowBackup="true"  
        android:icon="@mipmap/ic_launcher"  
        android:label="@string/app_name"  
        android:supportsRtl="true"  
        android:theme="@style/AppTheme">  
        ......  

        <service  
            android:name=".MyAccessibilityService"  
            android:label="我的智慧安裝"  
            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/accessibility_service_config" />  
        </service>  
    </application>  

</manifest>  

這部分配置的內容多數是固定的,必須要宣告一個android.permission.BIND_ACCESSIBILITY_SERVICE的許可權,且必須要有一個值為android.accessibilityservice.AccessibilityService的action,然後我們通過將剛才建立的配置檔案指定進去。
接下來就是要去實現智慧安裝功能的具體邏輯了,建立一個MyAccessibilityService類並繼承自AccessibilityService,程式碼如下所示:

/** 
 * 智慧安裝功能的實現類。 
 * 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149 
 * @author guolin 
 * @since 2015/12/7 
 */  
public class MyAccessibilityService extends AccessibilityService {  

    Map<Integer, Boolean> handledMap = new HashMap<>();  

    public MyAccessibilityService() {  
    }  

    @Override  
    public void onAccessibilityEvent(AccessibilityEvent event) {  
        AccessibilityNodeInfo nodeInfo = event.getSource();  
        if (nodeInfo != null) {  
            int eventType = event.getEventType();  
            if (eventType== AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED ||  
                    eventType == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) {  
                if (handledMap.get(event.getWindowId()) == null) {  
                    boolean handled = iterateNodesAndHandle(nodeInfo);  
                    if (handled) {  
                        handledMap.put(event.getWindowId(), true);  
                    }  
                }  
            }  
        }  
    }  

    private boolean iterateNodesAndHandle(AccessibilityNodeInfo nodeInfo) {  
        if (nodeInfo != null) {  
            int childCount = nodeInfo.getChildCount();  
            if ("android.widget.Button".equals(nodeInfo.getClassName())) {  
                String nodeContent = nodeInfo.getText().toString();  
                Log.d("TAG", "content is " + nodeContent);  
                if ("安裝".equals(nodeContent)  
                        || "完成".equals(nodeContent)  
                        || "確定".equals(nodeContent)) {  
                    nodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK);  
                    return true;  
                }  
            } else if ("android.widget.ScrollView".equals(nodeInfo.getClassName())) {  
                nodeInfo.performAction(AccessibilityNodeInfo.ACTION_SCROLL_FORWARD);  
            }  
            for (int i = 0; i < childCount; i++) {  
                AccessibilityNodeInfo childNodeInfo = nodeInfo.getChild(i);  
                if (iterateNodesAndHandle(childNodeInfo)) {  
                    return true;  
                }  
            }  
        }  
        return false;  
    }  

    @Override  
    public void onInterrupt() {  
    }  

}  

程式碼並不複雜,我們來解析一下。每當視窗有活動時,就會有訊息回撥到onAccessibilityEvent()方法中,因此所有的邏輯都是從這裡開始的。首先我們可以通過傳入的AccessibilityEvent引數來獲取當前事件的型別,事件的種類非常多,但是我們只需要監聽TYPE_WINDOW_CONTENT_CHANGED和TYPE_WINDOW_STATE_CHANGED這兩種事件就可以了,因為在整個安裝過程中,這兩個事件必定有一個會被觸發。當然也有兩個同時都被觸發的可能,那麼為了防止二次處理的情況,這裡我們使用了一個Map來過濾掉重複事件。
接下來就是呼叫iterateNodesAndHandle()方法來去解析當前介面的節點了,這裡我們通過遞迴的方式將安裝介面中所有的子節點全部進行遍歷,當發現按鈕節點的時候就進行判斷,按鈕上的文字是不是“安裝”、“完成”、“確定”這幾種型別,如果是的話就模擬一下點選事件,這樣也就相當於幫使用者自動操作了這些按鈕。另外從Android 4.4系統開始,使用者需要將應用申請的所有許可權看完才可以點選安裝,因此如果我們在節點中發現了ScrollView,那就模擬一下滑動事件,將介面滑動到最底部,這樣安裝按鈕就可以點選了。
最後,回到MainActivity中,來增加對智慧安裝功能的呼叫,如下所示:

/** 
 * 仿360手機助手秒裝和智慧安裝功能的主Activity。 
 * 原文地址:http://blog.csdn.net/guolin_blog/article/details/47803149 
 * @author guolin 
 * @since 2015/12/7 
 */  
public class MainActivity extends AppCompatActivity {  

    ......  

    public void onForwardToAccessibility(View view) {  
        Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);  
        startActivity(intent);  
    }  

    public void onSmartInstall(View view) {  
        if (TextUtils.isEmpty(apkPath)) {  
            Toast.makeText(this, "請選擇安裝包!", Toast.LENGTH_SHORT).show();  
            return;  
        }  
        Uri uri = Uri.fromFile(new File(apkPath));  
        Intent localIntent = new Intent(Intent.ACTION_VIEW);  
        localIntent.setDataAndType(uri, "application/vnd.android.package-archive");  
        startActivity(localIntent);  
    }  

}  

當點選了開啟智慧安裝服務按鈕時,我們通過Intent跳轉到系統的無障礙服務介面,在這裡啟動智慧安裝服務。當點選了智慧安裝按鈕時,我們通過Intent跳轉到系統的安裝介面,之後所有的安裝操作都會自動完成了。