-
代碼未行玛界,效果先上
-
如何實(shí)現(xiàn)
在 [KITKAT][null-link] 之后棵逊,Android Window支持了一些新的屬性伤疙,其中有兩個(gè)是這樣的 .
WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS
WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION
正如它們的變量名的意思,使用這兩個(gè)屬性辆影,可以使得狀態(tài)欄和導(dǎo)航欄變?yōu)橥该魍较瘢瑢?dǎo)航欄指的就是Android下方的三大按鍵,當(dāng)然只使用第一個(gè)屬性也可以達(dá)到今天所要完成的效果蛙讥。下面的示例代碼將使?fàn)顟B(tài)欄和導(dǎo)航欄變得透明
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
initWindow();
}
@TargetApi(19)
private void initWindow(){
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT){
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
}
直接運(yùn)行之后锯蛀,狀態(tài)欄直接透明了,但是并不是我們想要的效果次慢,狀態(tài)欄原本應(yīng)該占有的位置沒了旁涤。
這個(gè)問題也很好解決,在 style theme 添加
<item name="android:fitsSystemWindows">true</item>
之后我們?cè)龠\(yùn)行迫像,卻又發(fā)現(xiàn)劈愚,狀態(tài)欄的位置出來了,但是闻妓。菌羽。。
實(shí)際上由缆,狀態(tài)欄已經(jīng)透明了注祖,只是狀態(tài)欄底下沒有顏色呀!
Google 了之后在 Github 找到了一個(gè)開源項(xiàng)目 SystemBarTint 均唉,代碼就變成下面這個(gè)樣子:
private SystemBarTintManager tintManager;
@TargetApi(19)
private void initWindow(){
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT){
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
tintManager = new SystemBarTintManager(this);
tintManager.setStatusBarTintColor(getColor(R.color.app_main_color));
tintManager.setStatusBarTintEnabled(true);
}
}
運(yùn)行之后是晨,發(fā)現(xiàn)運(yùn)行效果跟第一張圖一樣,達(dá)到我們想要的效果了舔箭。
跟蹤進(jìn)去查看 SystemBarTint 的源代碼罩缴,會(huì)發(fā)現(xiàn) SystemBarTintManager 的構(gòu)造方法里面除了獲取 ActionBar 的高度等等這些配置之外,還有一個(gè)重要的方法 setupStatusBarView
@TargetApi(19)
public SystemBarTintManager(Activity activity) {
Window win = activity.getWindow();
ViewGroup decorViewGroup = (ViewGroup) win.getDecorView();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
//省去部分代碼...
if (mStatusBarAvailable) {
setupStatusBarView(activity, decorViewGroup);
}
if (mNavBarAvailable) {
setupNavBarView(activity, decorViewGroup);
}
}
于是接著查看 setupStatusBarView 的代碼
private void setupStatusBarView(Context context, ViewGroup decorViewGroup) {
mStatusBarTintView = new View(context);
LayoutParams params = new LayoutParams(LayoutParams.MATCH_PARENT, mConfig.getStatusBarHeight());
params.gravity = Gravity.TOP;
if (mNavBarAvailable && !mConfig.isNavigationAtBottom()) {
params.rightMargin = mConfig.getNavigationBarWidth();
}
mStatusBarTintView.setLayoutParams(params);
mStatusBarTintView.setBackgroundColor(DEFAULT_TINT_COLOR);
mStatusBarTintView.setVisibility(View.GONE);
decorViewGroup.addView(mStatusBarTintView);
}
可以發(fā)現(xiàn)這個(gè)開源項(xiàng)目能夠解決我們的問題的原因在這,就是往 DecorView 加入一個(gè) View靴庆, 而在代碼中我們將這個(gè) View 的背景設(shè)置成 ActionBar 一樣的顏色时捌,所以就達(dá)到了沉浸式的效果。到這里炉抒,基本也就分析完成了奢讨。
-
源代碼
附上源代碼地址: Github 源代碼 .
項(xiàng)目是我最近剛建不久的,主要放了一些 Android 方面的練習(xí)焰薄,最近更新了[RecycleView滾動(dòng)時(shí)隱藏Toolbar和狀態(tài)欄拿诸,類似pocket][null-link] 和 [滑動(dòng)菜單欄,支持設(shè)置左右滑動(dòng)呼出][null-link]
[null-link]: chrome://not-a-link