最近在做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 2 3 4 5 6
| <service android:name=".MyAccessibilityService" android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"> <intent-filter> <action android:name="android.accessibilityservice.AccessibilityService" /> </intent-filter>
|
配置
AccessibilityService可以配置为接收特定类型的辅助的事件,监听特定的包,给定的时间内得到每种事件,检索窗口内容, 指定一个设置的activity,等等。
配置一个可访问性服务有两种方法:
提供元数据条目在清单申报服务。服务声明和一个元数据标记下面:
1 2 3 4 5
| <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资源配置。
1 2 3 4 5 6 7 8 9
| <?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" />
|
三 用法
四 实现思路部分
效果图;
我们可以指定监测的某个包,这里我为了实现自动安装,因此监测com.android.packageinstaller
“包下的com.android.packageinstaller.InstallerActivity界面。
在遍历到需要的安装按钮调用api来实现自动点击功能,微信抢红包是监测微信红白的activity:
aAccessibilityNodeInfo.performAction(targetAction)
关键部分
重写processAccessibilityEvent方法, 用来获得系统当前的AccessibilityEvent
信息,找出要监测的包名,和监测的view类型以及节点内容
如果是安装apk,并且当前界面是是属于安装acitvity,文本信息是“安装”的 ,我们可以帮他执行点击事件。
代码片段如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| 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")) { 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语言判断,由于首次点击安装按钮时,用户为打开辅助服务,因此
我做了是否开启本服务判断,如果未打开则跳到设置页面开启此服务,开启了则直接安装,
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
| 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.equalsIgnoreCase(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代码做了业务判断
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| installButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { if (!TamicInstallService.isAccessibilitySettingsOn (MainActivity.this)) { 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 PackageManager
源码浅析以及静默安装实现方式
一文
很多时候我们可以根据用户当前系统的情况分别处理,如果是内置渠道的,直接用静默安装实现,如果非root则申请root权限
在root后可以用静默安装,用户拒绝root情况下,可以采用辅助的自动装来实现apk的快速安装,以用来增大apk升级转换率。
如果的需要增量更新的话,可以用插件免安装实现。
第一时间获资讯请关注微信公众号!