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跳转到系统的安装界面,之后所有的安装操作都会自动完成了。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值