前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >免Root实现Apk静默安装,覆盖兼容市场主流的98%的机型

免Root实现Apk静默安装,覆盖兼容市场主流的98%的机型

作者头像
开发者技术前线
发布2020-11-23 11:36:07
2.2K0
发布2020-11-23 11:36:07
举报
文章被收录于专栏:开发者技术前线

眼看2016年工作日只剩两天了,几家欢喜几家愁!有的人这一年买房了,有的人买车了,有的人结婚了,有的人第一次出国,也有的人换了理想的工作,有的人可能未有实质性的东西兑现。不管怎样,只要这一年进步了就是收获,成长毕竟是要付出代价的。当然技术硬实力也不能落下,技术还是得跟上。毕竟这是大部分人的饭碗。

依然来首经典的歌曲,或许歌声里有你的一段往事,也许你会心痛,毕竟你入心了。好,伴随歌声继续鸡汤。

写在最前

前篇文章介绍了静默安装的三种基本方案(静默安装从入门到转行!(点我))。

同样这篇文章也是从一年前的博客中移植过来的。觉得用的地方比较多,因此决定再发一下。

最近在做APP自我静默更新,在获取内置情况下,或者已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确保只有系统可以绑定到它。

下面是一个例子声明:

代码语言:javascript
复制
<service android:name=".MyAccessibilityService"
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
<intent-filter>
<action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>

配置

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

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

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

代码语言:javascript
复制
<service android:name=".MyAccessibilityService">
<intent-filter>
<action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>
<meta-data android:name="android.accessibilityservice"  android:resource="@xml/accessibilityservice" />

注意:这个方法设置所有属性。 调用setServiceInfo(AccessibilityServiceInfo)。 注意:这个方法可以调用任何时间动态更改服务配置。

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

检索视图内容

服务可以指定在其声明, 它可以检索窗口的内容, 返回的为一个AccessibilityWindowInfo对象。 注意:申明此功能要求辅助必须声明其通过SERVICE_META_DATA引用的XML资源配置。

代码语言:javascript
复制
<?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/description_auto_install_accessibility_service"
android:notificationTimeout="100" />

三 用法

  • onServiceConnected() 服务连接时,也就是第一次打开时调用,这里我们可以初始化常量和标签等
  • onCreate() 服务创建时调用,初始化一些数据
  • onDestroy() 服务消亡是,或者用户关闭时,调用,这里我们可以去做些业务相关的释放任务,
  • onAccessibilityEvent() 监测到内容节点时调用
  • disableSelf() 自身关闭时主动调用

四 实现思路部分

效果图:

由于不是GIF动图可能看起来不是很明了,因此我在描述一下,我在一个Activity中某个按钮点击,就指定安装我放在sdk下面的UcAPk, 当系统安装界面起来时候,我在上面盖了一层黑色view,遮挡住系统安装界面,当apk安装完成后,移除当前view从而实现伪静默安装。

逻辑:

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

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

代码语言:javascript
复制
aAccessibilityNodeInfo.performAction(targetAction)

关键部分

重写processAccessibilityEvent方法, 用来获得系统当前的AccessibilityEvent信息,找出要监测的包名和监测的view类型以及节点(node)内容 如果是安装apk,并且当前界面是属于安装acitivity,文本信息是“安装”的 ,我们可以帮他执行点击事件。

代码片段如下:

代码语言:javascript
复制
 private void processAccessibilityEvent (AccessibilityEvent aAccessibilityEvent) { 
      if(aAccessibilityEvent.getSource() != null) {
 String packageName = aAccessibilityEvent.getPackageName().toString();
 String className = aAccessibilityEvent.getClassName().toString();
 String nodeText = aAccessibilityEvent.getSource().getText() == null ? "" :
       aAccessibilityEvent.getSource().getText().toString().trim();  
     if(packageName.equals("com.android.packageinstaller")) {  
       if(className.equalsIgnoreCase("android.app.AlertDialog")) { 
       // handleAlertDialog(aAccessibilityEvent, className, nodeText); 
//should for uninstall
 Log.e("test", "onAccessibilityEvent alert dialog");  return;
 }AccessibilityNodeInfo targetNode = extractNode(aAccessibilityEvent, aClassName,
“安装”); 
    if (targetNode != null) { 
       int targetAction = AccessibilityNodeInfo.ACTION_CLICK;  
       if ((aAccessibilityNodeInfo != null)  && aAccessibilityNodeInfo.isEnabled()
  && aAccessibilityNodeInfo.isClickable()
  && aAccessibilityNodeInfo.performAction(targetAction)) {   
       return true;
      }return;
     }  return;
    }
   }
}

五 展望

目前用辅助来实现自动装,用户还是能看到系统安装的界面的,我们则在当前界面添加一个window,用来遮盖系统的安装界面,在安装成功后移除目前的window,但是会遇到机型适配问题,有的机子安装界面是activity,有的是dialog, 况且内容文本也不一样,“有安装”,有“下一步”,有“我知道了”,“有同意”,因此后期做兼容时废了很大力气,不仅需要判断机型还要判断rom版本,此项目中几乎覆盖了市场上主流机型,不兼容的请读者自我加入if语言判断,由于首次点击安装按钮时,用户未打开辅助服务,因此我做了是否开启本服务的判断,如果未打开则跳到设置页面开启此服务,开启了本服务的情况则直接安装;

代码语言:javascript
复制
  
public static boolean isAccessibilityOn(Context mContext) {   
  
  int accessibilityEnabled = 0;
  final String service = 
          mContext.getPackageName() + "/" + TamicInstallService.class.getCanonicalName();   try {
  accessibilityEnabled = Settings.Secure.getInt(
  mContext.getApplicationContext().getContentResolver(),
  android.provider.Settings.Secure.ACCESSIBILITY_ENABLED);
  Log.v(LOG_TAG, "accessibilityEnabled = " + accessibilityEnabled);
  } catch (Settings.SettingNotFoundException e) {
      Log.e(LOG_TAG, "Error finding setting, default accessibility to not found: "
     + e.getMessage());
  }
  TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':');   
    if (accessibilityEnabled == 1) {
     Log.v(LOG_TAG, "***ACCESSIBILITY IS ENABLED*** -");
     String settingValue = Settings.Secure.getString(
     mContext.getApplicationContext().getContentResolver(),
     Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);   
    if (settingValue != null) {
        mStringColonSplitter.setString(settingValue);  
     while (mStringColonSplitter.hasNext()) {
        String accessibilityService = mStringColonSplitter.next();      Log.v(LOG_TAG, " accessibilityService :: " + accessibilityService + " " + service);  
   if (accessibilityService.equalsIgnoreCa
     se(service)) {
      Log.v(LOG_TAG, "We've found the correct setting - accessibility is switched on!");  
       return true;
      }
    }
  }
 } else {
     Log.v(LOG_TAG, "---ACCESSIBILITY IS DISABLED--");
  }   return false;
}

Activity代码做了业务判断:如果未打开则去设置界面打开此服务。

代码语言:javascript
复制
installButton.setOnClickListener(new View.OnClickListener() {  
   @Override
  public void onClick(View view) {   
    if (!TamicInstallService.isAccessibilitySettingsOn (MainActivity.this)) {   
     // TODO Auto-generated method stub
     Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
     startActivity(intent);     
     return;
 }
  TamicInstallService.setInvokeType(TamicInstallService.TYPE_INSTALL_APP);
  TamicWindowManager.makeWatingWiew(MainActivity.this, "安装中...").show();
  installApk();
 }
 });

结束语

通过辅助实现了自动装,root的静默安装请看: Android 静默安装从入门到改行(戳我)

很多时候我们根据用户当前系统的情况分别处理:

  • 如果是内置渠道的,直接用静默安装实现。
  • 如果非root则申请root权限,在root后可以用静默安装。
  • 用户拒绝root情况下,可以采用辅助的自动装来实现apk的快速安装,以用来增大apk升级转换率。

如果有需要增量更新场景,可以用插件免安装实现。各种热更新技术也同样达到换量需求。

---我是分割线---

Tamic开发社区

非专业的移动社区

不只是干货,还有人生

长按二维码关注我们

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2016-12-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 开发者技术前线 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 写在最前
  • 二 用法
    • 生命周期
      • 声明AccessibilityService
        • 指定意图处理
          • 配置
            • 检索视图内容
            • 三 用法
            • 四 实现思路部分
            • 关键部分
              • 五 展望
              • 结束语
              领券
              问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档