Android开发笔记——如何正确实现App启动页

启动页在当前的Android APP中有着广泛的使用,在Material Design规范出来之前,Android官方是不推荐使用启动页的,理由是让用户在启动APP后干等是一种浪费时间的行为。Material Design 则制定了LaunchScreen 规范,用于在APP启动的时候合理的利用从启动到实际渲染出主界面之间这段时间,显示一个LauncherScreen用来展示一些品牌元素等信息,避免显示一个枯燥的空白界面。

APP启动过程

APP的启动过程可以参考这里。这里主要简单介绍下APP冷启动的过程。
APP冷启动开始时,系统会做以下三个任务:

  1. 加载并启动app
  2. 展示一个空白Window
  3. 创建app进程

在上面的第二个步骤中,WindowManager 使用Activity主题中定义的背景和状态栏颜色的属性来绘制一个临时的展示UI。
当APP进程创建后,APP进程会做以下几件事:

  1. 创建app对象
  2. 创建app的主进程
  3. 创建MainActivity
  4. inflating views
  5. 执行Layout过程
  6. 执行界面绘制

当首次绘制完成后,系统会将之前展示的空白window移除,并展示创建好的MainActivity。在某些大型APP中,这个过程可能会有1-5秒。

启动页实现原理

从上面的启动分析可以知道,WindowManager是根据Activity主题中定义的背景等属性来创建一个临时preview window的。我们在Activity的主题style中设置windowBackground属性来设置这个preview window的显示内容。使用layer-list可以创建一个包含logo的drawable对象:

<layer-list xmlns:android="http://schemas.android.com/apk/res/android" android:opacity="opaque">
  <item android:drawable="@android:color/white"/>
  <item>
    <bitmap
      android:src="@drawable/logo"
      android:gravity="center"/>
  </item>
</layer-list>

创建一个自定义Theme:

<style name=“LaunchScreenTheme" parent="Theme.AppCompat.NoActionBar">
    <item name="windowNoTitle">true</item>
    <item name="android:windowContentOverlay">@null</item>
    <item name="android:windowFullscreen">true</item>
    <item name="android:windowBackground”>@drawable/launchscreen</item>
</style>

最后在Manifest中将这个主题赋值给待启动的Activity即可

<activity ...
android:theme="@style/AppTheme.Launcher" />

实现方式

谷歌官方推荐的启动页实现方式是在将LauncherScreen和MainActivity合二为一。首先创建按照之前说的原理创建一个启动页Theme,并在Manifest中将其作为MainActivity的默认Theme。然后在MainActivity的onCreate方法中,在onCreate()和setContentView()之前使用setTheme方法,重新设置主界面的主题。

public class MainActivity extends AppCompatActivity {
 @Override
  protected void onCreate(Bundle savedInstanceState) {
    // Make sure this is before calling super.onCreate
    setTheme(R.style.Theme_MyApp);
    super.onCreate(savedInstanceState);
    // ...
  }
}

这样做的好处有以下几点:

  1. 不需要额外的Activity实现启动页
  2. 没有overdraw的问题,因为重新设置主题后,之前的绘画层也一并移除了。

当然,如果你需要根据schema或者其他方式从启动页路由到不同的界面的功能,使用单独的启动页Activity也是一个不错的选择。

相关推荐