图凌闪屏页及Android彩蛋探究

原文:http://blog.fiftykg.com/

前言

通过本文,你可以
1、了解一种特别的闪屏
2、了解android版本彩蛋的实现原理
3、获得一个android各版本彩蛋的demo

特殊的闪屏

在体验App时发现了一款叫‘图凌’的app,闪屏页非常特别。从下图可以看到,是一个以桌面壁纸为背景的页面。
图凌闪屏页及Android彩蛋探究

这种闪屏效果让人眼前一亮,所以非常好奇他的实现原理。在不**apk的情况下(**失败,有腾讯乐固加固==),猜想了几种实现方式:

1、通过Api获取壁纸,然后设置Activity的背景
2、特殊的Activity theme

在逐个验证猜想之前,想到了一个页面的实现与‘图凌’的闪屏非常相似,就是Android的版本彩蛋(设置-关于手机-Android版本点击3下)。而这个页面是可以找到源码的。

扒源码

打开彩蛋页面,执行以下命令,可以得知Activity的名字是PlatLogoActivity。

adb shell dumpsys activity | grep "Focus"

mFocusedActivity: ActivityRecord{2829baa u0 android/com.android.internal.app.PlatLogoActivity t4844}
  mFocusedStack=ActivityStack{d93bf0d stackId=1, 4 tasks} mLastFocusedStack=ActivityStack{d93bf0d stackId=1, 4 tasks}

(或者 adb shell dumpsys activity top)

通过在线源码平台Xref,找到PlatLogoActivity,源码就不贴了,因为重点并不在PlatLogoActivity.java,而是在AndroidManifest,theme才是关键!也就是开始的第二个猜想。

<activity android:name="com.android.internal.app.PlatLogoActivity"
    android:theme="@style/Theme.Wallpaper.NoTitleBar.Fullscreen"
    android:configChanges="orientation|keyboardHidden"
    android:process=":ui">
</activity>

试一试

通过源码基本了解了实现原理,但没有跑一下Demo是不靠谱的。
PlatLogoActivity的实现非常独立,没有太多依赖。所以copy了一下各个版本的PlatLogoActivity做了一个Demo。

Github: https://github.com/PortgasAce/PlatLogoCollections

图凌闪屏页及Android彩蛋探究

图凌闪屏页及Android彩蛋探究

探究

现在我们已经知道theme可以方便的实现‘图凌’的闪屏效果。那么代码可以实现吗?官方有开放相应的api吗?

这些问题的答案可以从theme的实现原理入手。通过google或者xref搜索theme的一些关键字,最终可以找到PhoneWindow#generateLayout有这样一段解析theme标签的代码:

        ... //省略了很多标签解析
        if (a.getBoolean(R.styleable.Window_windowSwipeToDismiss, false)) {
            requestFeature(FEATURE_SWIPE_TO_DISMISS);
        }

        if (a.getBoolean(R.styleable.Window_windowFullscreen, false)) {
            setFlags(FLAG_FULLSCREEN, FLAG_FULLSCREEN & (~getForcedWindowFlags()));
        }
        ...//省略了很多标签解析

theme的主要实现就是通过window#requestFeaturewindow#setFlags方法改变样式。

临摹着写了Theme.Wallpaper.NoTitleBar.Fullscreen的java实现,见github:PlatLogoActivityNoStyle
主要代码:

    requestWindowFeature(Window.FEATURE_NO_TITLE);
    getWindow().getDecorView().setWillNotDraw(true);
    getWindow().setBackgroundDrawable(new ColorDrawable(android.graphics.Color.TRANSPARENT));
    getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
    getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
    getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WALLPAPER);

到此为止,猜想2通过theme和java实现都是验证了。那么其他猜想是否可行呢?
答案当然是可以的啦。通过sdk提供的WallpaperManager可以获取桌面壁纸。

Drawable bg = WallpaperManager.getInstance(context).getDrawable();
rootView.setBackground(bg);

以上代码就可以给Activity设置背景为桌面壁纸。

但是有一个问题。动态壁纸(live wallpapewr)时通过该方法获取的drawable不但不会动,而且是错误的图片。

其实liveWallpaper获取的正确姿势是通过wallpaperManager#getWallpaperInfo#loadThumbnail

  private Drawable getWallpaperDrawable() {
    Drawable wallpaperDrawable;
    PackageManager pm = getApplicationContext().getPackageManager();
    WallpaperManager wallpaperManager = WallpaperManager.getInstance(this);
    if (wallpaperManager.getWallpaperInfo() != null) {
      /*
       * Wallpaper info is not equal to null, that is if the live wallpaper
       * is set, then get the drawable image from the package for the
       * live wallpaper
       */
      wallpaperDrawable = wallpaperManager
          .getWallpaperInfo().loadThumbnail(pm);
    } else {
      /*
       * Else, if static wallpapers are set, then directly get the
       * wallpaper image
       */
      wallpaperDrawable = wallpaperManager.getDrawable();
    }
    return wallpaperDrawable;
  }

但是问题还是没有完全解决,背景不会动!每次调用loadThumbnail返回的图片都是一样的,因此猜想1只使用于静态壁纸。

以上。

巨人的肩膀

Andriod中Style/Theme原理以及Activity界面文件选取过程浅析:https://blog.****.net/qinjuning/article/details/8829877