Android 沉浸式状态栏

Android 沉浸式状态栏 已经算是出来比较久的技术了,只不过我们公司项目最近要优化一下UI,就研究了一下,在此特别感谢简书作者扬州慢_提供的解决方案,原文链接戳这里

言归正传,Android沉浸式分为以下几个维度去处理

1.API < 19(无法进行沉浸式)

2.API = 19(添加与状态栏同等宽高的View)

3.API > 19   (设置状态栏颜色) 「 API > 22 可以修改状态栏的字体和图标颜色,避免因设置颜色相近看不清状态栏原有的字体」

接下来逐条阐明具体解决方案,

(获取状态栏高度以及获取actionbar的高度的函数会在最后给出,大家先看懂原理)

首先,对于API < 19(即安卓4.4系统以下)的没办法做处理,而且市场占有率很低,可以忽略;

其次,对于API = 19(即安卓4.4系统),核心代码如下:

两种情况处理:1.有ActionBar(直接调用系统的才算,toolbar不算在这个分类里面)

                         2.没有ActionBar

扫描二维码关注公众号,回复: 1211418 查看本文章

//增加占位状态栏

ViewGroup decorView = (ViewGroup) mActivity.getWindow().getDecorView();

View statusBarView = new View(activity);

//为statusBarView准备的布局参数,全屏宽,与状态栏一样的高度

ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, getStatusBarHeight(activity));

ViewGroup rootView = mActivity.findViewById(android.R.id.content);

//通过外面设置是否有actionbar

if (isActionBar()) {

//要增加内容视图的 paddingTop,否则内容被 ActionBar 遮盖

rootView.setPadding(0, getStatusBarHeight(mActivity) + getActionBarHeight(mActivity), 0, 0);

} else {

rootView.setPadding(0, getStatusBarHeight(mActivity), 0, 0);

}

//为占位状态栏设置我们想要的颜色

statusBarView.setBackgroundColor(color);

//为DecorView添加占位状态栏

decorView.addView(statusBarView, lp);

以上是代码逻辑的阐述,接下来就是针对API = 19的style的特殊处理(文件路径 values-v19/styles.xml)

//为你的app的主题加入以下两个item

<item name="android:windowTranslucentStatus">true</item>

<item name="android:windowTranslucentNavigation">false</item>

那么这两个属性是用来干什么的呢?(后续补充)

最后,对于 API > 19(即安卓5.0及以上),核心代码如下:

//直接设置状态栏颜色

Window window = mActivity.getWindow();

//取消设置透明状态栏,使 ContentView 内容不再沉浸到状态栏下

window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

//需要设置这个 flag 才能调用 setStatusBarColor 来设置状态栏颜色

window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);

activity.getWindow().setStatusBarColor(color);

补充,对于 API > 22(即安卓6.0及以上),设置状态栏的字体颜色,核心代码如下:

/** * 设置状态栏文字颜色 bDark为TRUE代表沉浸式 */

private static void setStatusBarMode(Activity activity, boolean bDark) {

    //6.0以上

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {

        View decorView = activity.getWindow().getDecorView();

        if (decorView != null) {

              int vis = decorView.getSystemUiVisibility();

              if (bDark) {

                vis |= View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR;

                } else {

                vis &= ~View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR;

                }

        decorView.setSystemUiVisibility(vis);

        }

    }

}

什么原理呢:????(后续补充)

getStatusBarHeight获取状态栏高度的函数如下:

/** * 利用反射获取状态栏高度 */

private static int getStatusBarHeight(Activity activity) {

        int result = 0; //获取状态栏高度的资源id

        int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");

        if (resourceId > 0) {

            result = activity.getResources().getDimensionPixelSize(resourceId);

       }

        Log.e("getStatusBarHeight", result + "");

        return result;

}

getActionBarHeight获取ActionBar高度的函数如下:

/** * 获得 ActionBar 的高度 */

public static int getActionBarHeight(Context context) {

        int result = 0;

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {

                TypedValue tv = new TypedValue();

                //划重点!!!!!!之前在4.4手机上出现了部分UI会被遮住一点点就是在这里出的问题,一定要看你的actionbar用的是哪个包里面的,不同包下面大小有些差异

                context.getTheme().resolveAttribute(android.support.v7.appcompat.R.attr.actionBarSize, tv, true);

                result = TypedValue.complexToDimensionPixelSize(tv.data, context.getResources().getDisplayMetrics());

        }

return result;

}

在设置沉浸式的同时,我们要对actionbar做去除阴影处理,代码如下:

/*** 去除 ActionBar 阴影 */

public void clearActionBarShadow() {

        if (Build.VERSION.SDK_INT >= 21) {

                ActionBar supportActionBar = ((AppCompatActivity) mActivity).getSupportActionBar();

                if (supportActionBar != null) {

                        supportActionBar.setElevation(0);

                }

        }

        return this;

}

猜你喜欢

转载自blog.csdn.net/Android_Glimmer/article/details/80496630