图凌闪屏页及Android彩蛋探究

前言

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

特殊的闪屏

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

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

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

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

扒源码

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

1
2
3
4
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才是关键!也就是开始的第二个猜想。

1
2
3
4
5
<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

探究

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

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

1
2
3
4
5
6
7
8
9
... //省略了很多标签解析
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
主要代码:

1
2
3
4
5
6
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可以获取桌面壁纸。

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

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

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

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
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.csdn.net/qinjuning/article/details/8829877