网上的沉浸式状态栏有很多方式,总结一下,进行备份学习。
关于沉浸式大概可以分成三个阶段:
Android4.4(API 19) - Android 5.0(API 21): 这个阶段可以实现沉浸式,但是表现得还不是很好,实现方式为: 通过FLAG_TRANSLUCENT_STATUS设置状态栏为透明并且为全屏模式,然后通过添加一个与StatusBar 一样大小的View,将View 的 background 设置为我们想要的颜色,从而来实现沉浸式。
Android 5.0(API 21)以上版本: 在Android 5.0的时候,加入了一个重要的属性和方法 android:statusBarColor (对应方法为 setStatusBarColor),通过这个方法我们就可以轻松实现沉浸式。也就是说,从Android5.0开始,系统才真正的支持沉浸式。
Android 6.0(API 23)以上版本:其实Android6.0以上的实现方式和Android 5.0 +是一样,为什么要将它归为一个单独重要的阶段呢?是因为从Android 6.0(API 23)开始,我们可以改状态栏的绘制模式,可以显示白色或浅黑色的内容和图标(除了魅族手机,魅族自家有做源码更改,6.0以下就能实现)
一些基础的代码可如下所示:
1、首先在xml的根节点下加入添加属性android:fitsSystemWindows="true"
android:fitsSystemWindows这个属性,主要是通过调整当前设置这个属性的view的padding去为我们的status_bar留下空间。
2、代码布局:
if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.LOLLIPOP) {
getWindow().addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
getWindow().setStatusBarColor(Color.BLUE);
}
else if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.KITKAT) {
//透明状态栏
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
//透明导航栏
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
}
之后还想说的是,如果状态栏为白色或者浅色系,很难看清,这就需要我们让状态栏的颜色变黑色,在android里的6.0之后可以设置,在android6.0以前则是不可以的。
/**
* 设置状态栏黑色字体图标,
* 适配4.4以上版本MIUIV、Flyme和6.0以上版本其他Android
*
*@paramactivity
*@return1:MIUUI 2:Flyme 3:android6.0
*/
public static intStatusBarLightMode(Activity activity) {
int result =0;
if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.KITKAT) {
if(MIUISetStatusBarLightMode(activity.getWindow(),true)) {
result =1;
}else if(FlymeSetStatusBarLightMode(activity.getWindow(),true)) {
result =2;
}else if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.M) {
activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN|View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
result =3;
}
}
return result;
}
/**
* 已知系统类型时,设置状态栏黑色字体图标。
* 适配4.4以上版本MIUIV、Flyme和6.0以上版本其他Android
*
*@paramactivity
*@paramtype1:MIUUI 2:Flyme 3:android6.0
*/
public static voidStatusBarLightMode(Activity activity,inttype) {
if(type ==1) {
MIUISetStatusBarLightMode(activity.getWindow(),true);
}else if(type ==2) {
FlymeSetStatusBarLightMode(activity.getWindow(),true);
}else if(type ==3) {
activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN| View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
}
}
/**
* 清除MIUI或flyme或6.0以上版本状态栏黑色字体
*/
public static voidStatusBarDarkMode(Activity activity,inttype) {
if(type ==1) {
MIUISetStatusBarLightMode(activity.getWindow(),false);
}else if(type ==2) {
FlymeSetStatusBarLightMode(activity.getWindow(),false);
}else if(type ==3) {
activity.getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE);
}
}
/**
* 设置状态栏图标为深色和魅族特定的文字风格
* 可以用来判断是否为Flyme用户
*@paramwindow需要设置的窗口
*@paramdark是否把状态栏字体及图标颜色设置为深色
*@returnboolean 成功执行返回true
*/
public static booleanFlymeSetStatusBarLightMode(Window window,booleandark) {
boolean result =false;
if(window !=null) {
try{
WindowManager.LayoutParams lp = window.getAttributes();
Field darkFlag = WindowManager.LayoutParams.class.getDeclaredField("MEIZU_FLAG_DARK_STATUS_BAR_ICON");
Field meizuFlags = WindowManager.LayoutParams.class.getDeclaredField("meizuFlags");
darkFlag.setAccessible(true);
meizuFlags.setAccessible(true);
intbit = darkFlag.getInt(null);
intvalue = meizuFlags.getInt(lp);
if(dark) {
value |= bit;
}else{
value &= ~bit;
}
meizuFlags.setInt(lp, value);
window.setAttributes(lp);
result =true;
}catch(Exception e) {
}
}
return result;
}
/**
* 设置状态栏字体图标为深色,需要MIUIV6以上
*@paramwindow需要设置的窗口
*@paramdark是否把状态栏字体及图标颜色设置为深色
*@returnboolean 成功执行返回true
*/
public static booleanMIUISetStatusBarLightMode(Window window,booleandark) {
boolean result =false;
if(window !=null) {
Class clazz = window.getClass();
try{
intdarkModeFlag =0;
Class layoutParams = Class.forName("android.view.MiuiWindowManager$LayoutParams");
Field field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE");
darkModeFlag = field.getInt(layoutParams);
Method extraFlagField = clazz.getMethod("setExtraFlags",int.class,int.class);
if(dark) {
extraFlagField.invoke(window, darkModeFlag, darkModeFlag);//状态栏透明且黑色字体
}else{
extraFlagField.invoke(window,0, darkModeFlag);//清除黑色字体
}
result =true;
}catch(Exception e) {
}
}
return result;
}
使用的方式,在把颜色值设置为白色的时候进行设置:
StatusBarLightMode(activity,StatusBarLightMode(activity));
最后分享一个开源的状态栏的库,方便我们在开发的时候进行集成使用:
https://github.com/laobie/StatusBarUtil