美文网首页Android
Android自定义开机引导最强篇

Android自定义开机引导最强篇

作者: 小智在不在 | 来源:发表于2021-11-24 12:15 被阅读0次

    本文基于Android 10

    Andoid开机引导应的本质是一个具有android.intent.category.HOME属性的Launcher,在Pixel手机上,作为开机向导的应用是Google的com.google.android.setupwizard应用,这是谷歌的应用,代码不在AOSP中,在AOSP中有一个包名为com.android.provision的应用供厂商定制开机向导,该应用在源码中的位置是packages\apps\Provision,所以我们注意Provision应用做了些什么就好了,这个应用只做了两件事,第一:设置相关属性让自己早于Launcher起来;第二:设置开机引导已经走完的标记位。
    具体代码实现,让自己的应用比Launcher先起来的方式,Provision在Manifest中做的:

    <application>
            <activity android:name="DefaultActivity" android:excludeFromRecents="true">
                <!--设置priority属性让自己的优先级比默认Launcher高-->
                <intent-filter android:priority="1">
                    <action android:name="android.intent.action.MAIN" />
                    <!--设置android.intent.category.HOME属性让自己成为一个Launcher-->
                    <category android:name="android.intent.category.HOME" />
                    <category android:name="android.intent.category.DEFAULT" />
                    <category android:name="android.intent.category.SETUP_WIZARD" />
                </intent-filter>
            </activity>
        </application>
    

    除了Manifest中的内容,系统还做了一步才让Provision比Launcher先启动,就是将Provision内置到/system/product/priv-app/目录下,这是因为不在这个目录下的应用设置android:priority属性会被重置为0,至于启动优先级的细节可以再详细阅读Android系统启动Launcher的源码,切入点在com.android.server.am.ActivityManagerService#systemReady()方法中,参考文章:HomeLauncher启动Launcher的启动过程 等,这里不做展开叙述,画重点:

    • 当自己写Demo代替Provision在Android Studio上跑起来而不是以系统应用集成在Rom的时候,自己Demo的Activity设置如上属性后并不会比Launcher先启动,这是因为不在/system/product/priv-app/目录下的应用设置了android:priority后依然会被置为0,优先级也不会高于Launcher,解决方法:
      1. 直接将Demo打进系统/system/product/priv-app/目录下(正式编译Rom时用)
      2. 如果Launcher的代码在自己手上就把Launcher的android:priority设置为-1。(仅在Android Studio上编译做验证时用)
    • 添加 android:sharedUserId=“android.uid.system” 不生效
    • 手动安装方式不生效(这个要注意,即使已经打进Rom了,这个时候再添加一个不同android:priority等级的Activity重新安装,这个新添加的Activity也不会有相应的android:priority等级)

    再来看启动之后ProvisionDefaultActivity中做了什么,代码非常简单:

    public class DefaultActivity extends Activity {
    
        @Override
        protected void onCreate(Bundle icicle) {
            super.onCreate(icicle);
    
            // 添加持久设置以允许其他应用程序知道设备已配置。
            Settings.Global.putInt(getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 1);
            //这个标记位标识当前用户已经走完引导流程,如果不设置这个值,Home键、锁屏等将不可用
            Settings.Secure.putInt(getContentResolver(), Settings.Secure.USER_SETUP_COMPLETE, 1);
    
            // 从PackageManager中禁用该Activity。
            PackageManager pm = getPackageManager();
            ComponentName name = new ComponentName(this, DefaultActivity.class);
            pm.setComponentEnabledSetting(name, PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
                    PackageManager.DONT_KILL_APP);
    
            finish();
        }
    }
    

    可以看到DefaultActivity中具体做了如下操作:

    1. 设置相关标记位
    2. 将该Activity禁用
    3. finish自己

    设置相关标记位可以让其他服务知道设备可用,如锁屏服务可用,启用Home键功能等,将该Activity禁用可以让下次开机时我们的应用不会再起来而直接启动桌面,finish就不用做解释,开机引导走完了就该销毁自己了。
    自己写Demo测试时这一步需要注意的点,设置Settings.Global.DEVICE_PROVISIONEDSettings.Secure.USER_SETUP_COMPLETE两个属性需要添加如下两个权限:

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

    如果引用不到Settings.Global.DEVICE_PROVISIONEDSettings.Secure.USER_SETUP_COMPLETE就直接写字符串device_provisioneduser_setup_complete

    如上需要实现我们自己业务的Android开机向导就只需要将Provision的代码移到自己的项目,走完我们自己的引导流程后设置相关属性,然后把Provision从编译的Rom移除就行了,或者直接在Provision应用里写自己的业务,另外调试的时候因为开机向导只会走一次,所以调试起来会比较麻烦,我们可以通过adb命令重置属性方便调试:

    1.通过如下命令使能进入开机向导
    adb shell
    settings put global device_provisioned 0
    settings put secure user_setup_complete 0
    //开启Provision应用的DefaultActivity
    pm enable com.android.provision/com.android.provision.DefaultActivity
    //或者
    //开启Demo的MainActivity
    pm enable com.xzzbz.setupdemo/com.xzzbz.setupdemo.MainActivity
    sync
    //重启
    reboot
    
    2.查询settings的值
    settings get global device_provisioned
    settings get secure user_setup_complete
    
    3.通过代码实现
    Settings.Global.putInt(mContext.getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 0);
    Settings.Secure.putInt(mContext.getContentResolver(), Settings.Secure.USER_SETUP_COMPLETE, 0);
    ComponentName name = new ComponentName("com.android.provision", "com.android.provision.DefaultActivity");
    mContext.getPackageManager().setComponentEnabledSetting(name,PackageManager.COMPONENT_ENABLED_STATE_ENABLED, PackageManager.DONT_KILL_APP);
    

    这里通过adb命令设置属性的时候有一点需要注意,每一步adb命令执行后需要等几秒,比如执行pm enable com.xzzbz.setupdemo/com.xzzbz.setupdemo.MainActivity后没有等几秒直接执行syncreboot的话开机向导还是会起来,应该是需要时间同步状态。

    在有的应用中可能需要对用户是否走完开机引导流程做判断,例如语音助手中判断用户走完了开机引导流程才响应语音唤醒,我们可以取开机引导中设置的标记位做判断,这是个系统标记位,可以在不同应用中取到值,示例如下:

    if (Settings.Secure.getInt(mContext.getContentResolver(), Settings.Global.DEVICE_PROVISIONED, 0) == 1) {
      //开机引导走完了,走正常业务逻辑
    } else {
      Log.e(TAG, "收到了唤醒,但是开机引导没走完,不做通知");
    }
    

    援引:
    Google开机向导解析
    android开机向导的实现
    Android 自定义开机向导踩坑
    Android10定制Google开机向导
    Android 9.1 定制开机向导
    Android 8.1自定义开机向导
    Android7.1 应用组件添加intent-filter priority(优先级)不生效
    另:
    android系统开机向导无法启动数据进行上网

    相关文章

      网友评论

        本文标题:Android自定义开机引导最强篇

        本文链接:https://www.haomeiwen.com/subject/mfpjtrtx.html