Android在4.4版本才实现沉浸式标题栏方案,所以要适配沉浸式标题栏要大于等于Android4.4版本

类型一:单独一个Activity

这种很简单,需要在Activity的onCreate方法中加入

ViewColor.setColor(this, R.color.red); // 设置颜色


然后在布局文件的根节点加入

android:fitsSystemWindows="true"


ViewColor工具类的代码如下:

public class ViewColor {  
    /** * 生成一个和状态栏大小相同的矩形条 * * @param activity 需要设置的activity *  
     * @param color 状态栏颜色值 *  
     * @return 状态栏矩形条 */  
    private static View createStatusView(Activity activity, int color) {  
        // 获得状态栏高度  
        int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");  
        int statusBarHeight = activity.getResources().getDimensionPixelSize(resourceId);  

        // 绘制一个和状态栏一样高的矩形  
        View statusView = new View(activity);  
        LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,  
                statusBarHeight);  
        statusView.setLayoutParams(params);  
        statusView.setBackgroundColor(color);  
        return statusView;  
    }  

    /** * 设置状态栏颜色 * * @param activity 需要设置的activity * @param color 状态栏颜色值 */  
    @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH)  
    public static void setColor(Activity activity, int color) {  
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {  
            // 设置状态栏透明  
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);  
            // 生成一个状态栏大小的矩形  
            View statusView = createStatusView(activity, color);  
            // 添加 statusView 到布局中  
            ViewGroup decorView = (ViewGroup) activity.getWindow().getDecorView();  
            decorView.addView(statusView);  
            // 设置根布局的参数  
            ViewGroup rootView = (ViewGroup) ((ViewGroup) activity.findViewById(android.R.id.content)).getChildAt(0);  
            rootView.setFitsSystemWindows(true);  
            rootView.setClipToPadding(true);  
        }  
    }  

}

类型二:一个Activity中有多个Fragment,每个Fragment都有不同的状态栏颜色

1.此方法适合每个fragment的状态栏颜色是一样的

android:fitsSystemWindows="true"


2.此方法适合每个fragment的状态栏颜色都是不一样的

  1. 在Activity的布局中根节点下第一个控件上加上属性:
android:fitsSystemWindows="true"


一般情况下布局第一个控件位自己的标题栏RelativeLayout或者LinearLayout
可以在title的上面加一个空的view,这个空的view就是用来填充状态栏的

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
    android:id="@+id/title_bar"  
    android:gravity="center"  
    android:orientation="vertical">  

    <TextView  
        android:layout_width="match_parent"  
        android:layout_height="25dp"  
        android:background="@color/base_bar_color"  
        android:fitsSystemWindows="true" />  

    <LinearLayout  
        android:layout_width="fill_parent"  
        android:layout_height="wrap_content"  
        android:gravity="center"  
        android:orientation="vertical"

第一个LinearLayout是根节点,下面的TextView就是用来填充状态栏的,下面的LinearLayout就是我自己的title栏,填充什么样的颜色,自己决定。
如果其中的一个Fragment是一张图片,那么就不需要这个空的TextView了,直接在图片的ImageView中添加属性:

android:fitsSystemWindows="true"


在onCreateview方法中加入透明状态栏的方法

getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);


需要注意的是,一个Activity中有多个Fragment,而且每个Fragment的状态栏颜色都不一样,在低版本的Android手机上状态栏会出现刚才添加25dp的textview,需要进行版本判断隐藏这个textview

// 获得状态栏高度
int resourceId = getActivity().getResources().getIdentifier("status_bar_height", "dimen", "android");
int statusBarHeight = getActivity().getResources().getDimensionPixelSize(resourceId);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            tv_top.setHeight(statusBarHeight);
            tv_top.setVisibility(View.VISIBLE);
        }else {
            tv_top.setVisibility(View.GONE);
        }

最后在Android5.0版本可能会出现状态栏上面有黑色背景条的情况,需要加入版本判断

if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            Window window = getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS
                    | WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                    | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
                    | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setStatusBarColor(Color.TRANSPARENT);
            window.setNavigationBarColor(Color.TRANSPARENT);
        }

类型三、有的需要沉浸式的状态栏颜色可能是很浅的颜色(在此非常鄙视那种在Android上设计标题栏为灰不辣鸡的那种很浅的颜色的设计师,因为在IOS中可以随意设置状态栏的字体颜色,而安卓手机众多没有标准),设置状态栏字体颜色。

由于部分安卓手机厂商才提供沉浸式状态栏字体颜色修改API

小米:

public void setStatusBarDarkMode(boolean darkmode, Activity activity) {
    Class<? extends Window> clazz = activity.getWindow().getClass();
    try {
        int darkModeFlag = 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);
        extraFlagField.invoke(activity.getWindow(), darkmode ? darkModeFlag : 0, darkModeFlag);
    } catch (Exception e) {
        e.printStackTrace();
    }
}

魅族:

public static boolean setStatusBarDarkIcon(Window window, boolean dark) {
    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);
            int bit = darkFlag.getInt(null);
            int value = meizuFlags.getInt(lp);
            if (dark) {
                value |= bit;
            } else {
                value &= ~bit;
            }
            meizuFlags.setInt(lp, value);
            window.setAttributes(lp);
            result = true;
        } catch (Exception e) {
            Log.e("MeiZu", "setStatusBarDarkIcon: failed");
        }
    }
    return result;
}
  • 25

更新工具类!封装为utils

/**
 * Created by kxx 
 */
public class StatusBarUtil {

/**
 * 修改状态栏为全透明
 * @param activity
 */
@TargetApi(19)
public static void transparencyBar(Activity activity){
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        Window window = activity.getWindow();
        window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
        window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
        window.setStatusBarColor(Color.TRANSPARENT);

    } else
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        Window window =activity.getWindow();
        window.setFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS,
                WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
    }
}

   /**
 * 修改状态栏颜色,支持4.4以上版本
 * @param activity
 * @param colorId
 */
public static void setStatusBarColor(Activity activity,int colorId) {

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        Window window = activity.getWindow();
//      window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);    
        window.setStatusBarColor(activity.getResources().getColor(colorId));
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        //使用SystemBarTint库使4.4版本状态栏变色,需要先将状态栏设置为透明
        transparencyBar(activity);
        SystemBarTintManager tintManager = new SystemBarTintManager(activity);
        tintManager.setStatusBarTintEnabled(true);
        tintManager.setStatusBarTintResource(colorId);
    }
}

/**
 *设置状态栏黑色字体图标,
 * 适配4.4以上版本MIUIV、Flyme和6.0以上版本其他Android
 * @param activity
 * @return 1:MIUUI 2:Flyme 3:android6.0
 */
public static int StatusBarLightMode(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
 * @param activity
 * @param type 1:MIUUI 2:Flyme 3:android6.0
 */
public static void StatusBarLightMode(Activity activity,int type){
    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 void StatusBarDarkMode(Activity activity,int type){
    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用户
 * @param window 需要设置的窗口
 * @param dark 是否把状态栏字体及图标颜色设置为深色
 * @return  boolean 成功执行返回true
 *
 */
public static boolean FlymeSetStatusBarLightMode(Window window, boolean dark) {
    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);
            int bit = darkFlag.getInt(null);
            int value = 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以上
 * @param window 需要设置的窗口
 * @param dark 是否把状态栏字体及图标颜色设置为深色
 * @return  boolean 成功执行返回true
 *
 */
public static boolean MIUISetStatusBarLightMode(Window window, boolean dark) {
    boolean result = false;
    if (window != null) {
        Class clazz = window.getClass();
        try {
            int darkModeFlag = 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;
}

}