版权声明:本文为博主原创文章,转载请注明出处+地址
hello,大家好~第一次在简书上发表文章,如果有什么不对的地方,大家留言指正。
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
//增加占位状态栏
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;
}