地址:http://blog.csdn.net/sk719887916/article/details/46746991

作者: skay

 

最近在做apk自我静默更新,在获取内置情况下,或者已root过的手机可以完美实现自我静默安装功能,但是发布到市场的apk非内置(非system apk) 也非root,所以自我静默安装做起来不太靠谱。因此借助辅助去实现了一个apk辅助自动装功能,辅助功能可以参考谷歌官网指南:

https://developer.android.com/reference/android/accessibilityservice/package-summary.html

Accessibilityservice

一 简介


accessibilityservice是用户可选服务,

AccessibilityService由系统在后台运行,并接收回调函数AccessibilityEvents。此类事件表示一些状态转换的用户界面,

例如,界面已经改变, 点击一个按钮,等等。这种服务可以选择请求的能力查询活动窗口的内容。

开发一个可访问性服务需要扩展这个类并实现其抽象方法。

AccessibilityService由 AccessibilityServiceInfo来描述。

系统通知的AccessibilityService AccessibilityEvents的节点信息封装在这个类中。

用法

生命周期

AccessibilityService的生命周期管理体系和专门的遵循既定的服务生命周期。

开始触发一个AccessibilityService完全由用户显式地将服务在设备上设置中辅助功能中打开。这样在系统绑定到一个服务,调用callsonServiceConnected()。

此方法可以被重载, 客户想要执行post绑定设置。

AccessibilityService停止或者当用户在设备设置关闭后,会调用disableSelf()。

声明AccessibilityService

AndroidManifest声明AccessibilityService。xml,

但是它必须做两件事:

指定意图处理“android.accessibilityservice.AccessibilityService”。

请求允许BIND_ACCESSIBILITY_SERVICE确保只有系统可以绑定到它。

下面是一个例子声明:

  1. <service android:name=".MyAccessibilityService"
  2. android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
  3. <intent-filter>
  4. <action android:name="android.accessibilityservice.AccessibilityService" />
  5. </intent-filter>
  6. . . .

配置

AccessibilityService可以配置为接收特定类型的辅助的事件,监听特定的包,给定的时间内得到每种事件,检索窗口内容, 指定一个设置的activiy,等等。

配置一个可访问性服务有两种方法:

提供元数据条目在清单申报服务。服务声明和一个元数据标记下面:

  1. <service android:name=".MyAccessibilityService">
  2. <intent-filter>
  3. <action android:name="android.accessibilityservice.AccessibilityService" />
  4. </intent-filter>
  5. <meta-data android:name="android.accessibilityservice" android:resource="@xml/accessibilityservice" />

注意:这个方法设置所有属性。

调用setServiceInfo(AccessibilityServiceInfo)。

注意:这个方法可以调用任何时间动态更改服务配置。

注意:这种方法只允许设置动态可配置属性:eventTypes, feedbackType, flags, notificationTimeout, packageNames

检索视图内容

服务可以指定在其声明, 它可以检索窗口的内容, 返回的为一个AccessibilityWindowInfo对象。

注意:申明此功能要求辅助必须声明其通过SERVICE_META_DATA引用的XML资源配置。

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

三 用法

onServiceConnected()

服务连接时,也就是第一次打开时调用,这里我们可以初始化常量和标签等

onCreate()

服务创建时调用,初始化一些数据

onDestroy()

服务消亡是,或者用户关闭时,调用,这里我们可以去做些业务相关的释放任务,

onAccessibilityEvent()

监测到内容节点时调用

disableSelf()

自身关闭时主动调用

四 实现思路部分

效果图

我们可以指定监测的某个包,这里我为了实现自动安装,因此监测com.android.packageinstaller“包下的com.android.packageinstaller.InstallerActivity界面。

在遍历到需要的安装按钮调用api来实现自动点击功能,微信抢红包是监测微信红白的activity:

  1. aAccessibilityNodeInfo.performAction(targetAction)

关键部分

重写processAccessibilityEvent方法, 用来获得系统当前的AccessibilityEvent信息,找出要监测的包名,和监测的view类型以及节点内容

是安装apk,并且界面是安装acitvity,文本信息是安装的 我们可以帮他执行点击事件。

代码如下:

  1. private void processAccessibilityEvent(AccessibilityEvent aAccessibilityEvent) {
  2. if(aAccessibilityEvent.getSource() != null) {
  3. String packageName = aAccessibilityEvent.getPackageName().toString();
  4. String className = aAccessibilityEvent.getClassName().toString();
  5. String nodeText = aAccessibilityEvent.getSource().getText() == null ? "" : aAccessibilityEvent.getSource().getText().toString().trim();
  6. if(packageName.equals("com.android.packageinstaller")) {
  7. if(className.equalsIgnoreCase("android.app.AlertDialog")) {
  8. // handleAlertDialog(aAccessibilityEvent, className, nodeText); //should for uninstall
  9. Log.e("test", "onAccessibilityEvent alert dialog");
  10. return;
  11. }
  12. AccessibilityNodeInfo targetNode = extractNode(aAccessibilityEvent, aClassName,
  13. “安装”);
  14. if (targetNode != null) {
  15. int targetAction = AccessibilityNodeInfo.ACTION_CLICK;
  16. if ((aAccessibilityNodeInfo != null) && aAccessibilityNodeInfo.isEnabled()
  17. && aAccessibilityNodeInfo.isClickable()
  18. && aAccessibilityNodeInfo.performAction(targetAction)) {
  19. return true;
  20. }
  21. return;
  22. }
  23. return;
  24. }
  25. }
  26. }

五 后续

目前用辅助来实现自动装,用户还是能看到系统安装的界面的,我们则在当前界面添加一个window,用来遮盖系统的安装界面,在安装成功后移除目前的window

,但是会遇到机型问题,有的机子安装界面的activity,有的是dialog, 况且内容也不一样,有安装,有下一步,有我知道了,有同意,因此后期做兼容时废了很大力气,

不仅判断机型还要判断rom版本,此项目中几乎覆盖了市场上主流机型,不兼容的请读者自我加入if语言判断,由于首次点击安装按钮时,用户为打开辅助服务,因此

我做了是否开启本服务判断,如果未打开则跳到设置页面开启此服务,开启了则直接安装,

  1. /**
  2. * isAccessibilitySettingsOn
  3. */
  4. public static boolean isAccessibilityOn(Context mContext) {
  5. int accessibilityEnabled = 0;
  6. final String service = mContext.getPackageName() + "/" + TamicInstallService.class.getCanonicalName();
  7. try {
  8. accessibilityEnabled = Settings.Secure.getInt(
  9. mContext.getApplicationContext().getContentResolver(),
  10. android.provider.Settings.Secure.ACCESSIBILITY_ENABLED);
  11. Log.v(LOG_TAG, "accessibilityEnabled = " + accessibilityEnabled);
  12. } catch (Settings.SettingNotFoundException e) {
  13. Log.e(LOG_TAG, "Error finding setting, default accessibility to not found: "
  14. + e.getMessage());
  15. }
  16. TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':');
  17. if (accessibilityEnabled == 1) {
  18. Log.v(LOG_TAG, "***ACCESSIBILITY IS ENABLED*** -");
  19. String settingValue = Settings.Secure.getString(
  20. mContext.getApplicationContext().getContentResolver(),
  21. Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);
  22. if (settingValue != null) {
  23. mStringColonSplitter.setString(settingValue);
  24. while (mStringColonSplitter.hasNext()) {
  25. String accessibilityService = mStringColonSplitter.next();
  26. Log.v(LOG_TAG, " accessibilityService :: " + accessibilityService + " " + service);
  27. if (accessibilityService.equalsIgnoreCase(service)) {
  28. Log.v(LOG_TAG, "We've found the correct setting - accessibility is switched on!");
  29. return true;
  30. }
  31. }
  32. }
  33. } else {
  34. Log.v(LOG_TAG, "---ACCESSIBILITY IS DISABLED--");
  35. }
  36. return false;
  37. }

Activity代码做了业务判断

  1. installButton.setText("Install apk");
  2. installButton.setOnClickListener(new View.OnClickListener() {
  3. @Override
  4. public void onClick(View view) {
  5. if (!TamicInstallService.isAccessibilitySettingsOn(MainActivity.this)) {
  6. // TODO Auto-generated method stub
  7. Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
  8. startActivity(intent);
  9. return;
  10. }
  11. TamicInstallService.setInvokeType(TamicInstallService.TYPE_INSTALL_APP);
  12. TamicWindowManager.makeWatingWiew(MainActivity.this, "安装中...").show();
  13. installApk();
  14. }
  15. });

结束语

通过辅助实现了自动装,root的静默安装请看: Android PackageManager源码浅析以及静默安装实现方式

一文

很多时候我们可以根据用户当前系统的情况分别处理,如果是内置渠道的,直接用静默安装实现,如果非root则申请root权限

在root后可以用静默安装,用户拒绝root情况下,可以采用辅助的自动装来实现apk的快速安装,以用来增大apk升级转换率。

如果的需要增量更新的话,可以用插件免安装实现。

辅助自动装:https://github.com/NeglectedByBoss/Autoinstall

插件免安装:https://github.com/NeglectedByBoss/PluginLoader

静默安装:http://blog.csdn.net/sk719887916/article/details/50314017

Android 免Root实现Apk静默安装,覆盖兼容市场主流的98%的机型的更多相关文章

  1. android apk静默安装

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/47803149 之前有很多朋友都问过我,在Android系统中怎样才能实现静默安装呢 ...

  2. Android为TV端助力 apk静默安装

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/47803149 之前有很多朋友都问过我,在Android系统中怎样才能实现静默安装呢 ...

  3. Android免Root无侵入AOP框架Dexposed

    Dexposed框架是阿里巴巴无线事业部近期开源的一款在Android平台下的免Root无侵入运行期AOP框架,该框架基于AOP思想,支持经典的AOP使用场景,可应用于日志记录,性能统计,安全控制,事 ...

  4. apk 静默安装

    老大要我弄个自动更新,要用到静默安装,网上找到了些大拿的代码,我拿去改吧改吧,先贴出来: /** * 软件静默安装 * @param apkAbsolutePath apk文件所在路径 * @retu ...

  5. APP流氓大法之apk 静默安装

    老大要我弄个自动更新,要用到静默安装,网上找到了些大拿的代码,我拿去改吧改吧,先贴出来: /** * 软件静默安装 * @param apkAbsolutePath apk文件所在路径 * @retu ...

  6. android免root hook框架legend

    一.前言 Android中hook框架已经非常多了,最优秀的当属Xposed和Substrate了,这两个框架我在之前的文章都详细介绍过了,不了解的同学,可以转战这里:http://www.wjdia ...

  7. android免root兼容所有版本ui调试工具

    SwissArmyKnife是什么 SwissArmyKnife 是一款方便调试android UI的工具,可以兼容所有android版本,不需要root权限.可以直接在android手机屏幕上显示当 ...

  8. Dexposed:android免Root无侵入Aop框架

    在网上看到了阿里推出的一个android开源项目,名为Dexposed, 是一个Android平台下的无侵入运行期AOP框架.旨在解决像性能监控.在线热补丁等移动开发常见难题,典型使用场景为: AOP ...

  9. Android照片库选择图片裁剪闪退(兼容小米以及7.0以上机型)

    未经允许,禁止

随机推荐

  1. PHP 实例 AJAX 与 MySQL

    AJAX 数据库实例 下面的实例将演示网页如何通过 AJAX 从数据库读取信息: 实例   Person info will be listed here... 实例解释 - MySQL 数据库 在上 ...

  2. git 撤销没有提交的变化

    参考: https://stackoverflow.com/questions/5807137/how-to-revert-uncommitted-changes-including-files-an ...

  3. 一小时入门PHP

    [版权申明:本文系作者原创,转载请注明出处] 文章出处:[http://blog.csdn.net/sdksdk0/article/details/52332296](http://blog.csdn ...

  4. OpenResty 执行阶段的概念和用途

    主要还是 Nginx 的执行阶段知识了,都是因为 OR 才会那么深刻, 它有些自己的阶段. 主要还是参照 春哥的 Nginx 教程 请多读几遍,如果不清楚nginx的执行阶段就无法充分利用 openr ...

  5. 安卓高级8 SurfaceView (1)

    文章转载:http://www.cnblogs.com/xuling/archive/2011/06/06/android.html 首先我们先来看下官方API对SurfaceView的介绍 Surf ...

  6. React Native之ScrollView控件详解

    概述 ScrollView在Android和ios原生开发中都比较常见,是一个 滚动视图控件.在RN开发中,系统也给我们提供了这么一个控件.不过在RN开发中 ,使用ScrollView必须有一个确定的 ...

  7. JavaMail API 概述

    JavaMail API提供了一种与平台无关和协议独立的框架来构建邮件和消息应用程序. JavaMail API提供了一组抽象类定义构成一个邮件系统的对象.它是阅读,撰写和发送电子信息的可选包(标准扩 ...

  8. Android Studio 使用wifi调试插件

    由于手机亦或是数据线的问题,在应用开发过程中会时不时地遇到手机突然连不上电脑的尴尬时刻,于是就学习了如何使用wifi进行应用调试.下面就具体介绍一下adb wifi插件的安装和使用.其实我们只需要安装 ...

  9. [struts2学习笔记] 第六节 struts2依赖的jar包还有Could not find action or result 错误解决

    本文地址:http://blog.csdn.net/sushengmiyan/article/details/43272061 本文作者:sushengmiyan ------------------ ...

  10. Dynamics CRM2015 Update1 新功能之表单增强功能

    CRM2015 Update 1发布后,系统的界面的变化很大,仔细观察后会发现表单窗体也有些不同了,在CRM2015 Update1的官方介绍中对此变化的解释是起用了新的窗体呈现引擎,让界面更好看加载 ...