自android5.0开始,沉浸式状态栏似乎成为一种潮流,应用里缺少沉浸式总感觉少些什么。于是乎,我开始到处找如何兼容低版本的沉浸式,由于Android平台跨度问题,总遇到一些不如人意的问题。终于,皇天不负有心人,通过参考一些网络上的资料以及开发的一些经验,总结出一个可行的且良好的解决方案!
先介绍下,什么是沉浸式状态栏?
沉浸式,要求在应用中Android状态栏(StatusBar)与标题栏(ActionBar/Toolbar)要拥有相同的颜色,或者使用同一张图的连续背景。
话不多说,亮剑吧!
具体实现需要针对不同Android版本做处理,还有针对DecorView做处理以及做activity的xml布局文件根布局控件做属性处理。
/** * 设置沉浸式状态栏颜色 * * @param colorResId 状态栏颜色 */ protected void setImmersiveStatusBarColor(@ColorRes int colorResId) { int flags = View.SYstem_UI_FLAG_LAYOUT_STABLE | View.SYstem_UI_FLAG_LAYOUT_FULLSCREEN; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { int statusBarColor = ApkUtil.getColor(this,colorResId); //① float lightDegress = (Color.red(statusBarColor) + Color.green(statusBarColor) + Color.blue(statusBarColor)) / 3; //作色彩亮度判断,好针对颜色做相应的状态栏的暗色还是亮色。 if ((lightDegress > 200 || lightDegress == 0) && Build.VERSION.SDK_INT > Build.VERSION_CODES.M) rootView.setsystemUIVisibility(View.SYstem_UI_FLAG_LIGHT_STATUS_BAR); window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); window.setStatusBarColor(statusBarColor); } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); rootView.setsystemUIVisibility(flags | View.SYstem_UI_FLAG_IMMERSIVE | View.SYstem_UI_FLAG_IMMERSIVE_STICKY); } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { rootView.setsystemUIVisibility(flags); } if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.KITKAT) { //当API小于等于19,此时为了实现沉浸式状态栏,需要添加一个view来做statusbar背景控件 final boolean isHasstatusBarView = rootView.getTag() != null; View statusbarView = !isHasstatusBarView ? new View(this) : (View)rootView.getTag(); statusbarView.setBackgroundResource(colorResId); if(!isHasstatusBarView) { rootView.setTag(statusBarView); statusbarView.setLayoutParams(new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT,ViewUtil.getStatusBarHeight(this))); //② rootView.addView(statusbarView); } } }
注:此处针对rootView(即DecorView)、window的获取不再陈述!
①.ApkUtil.getColor(this,colorResId)
/** * 获取颜色资源 * @param context 上下文对象 * @param colorId 颜色ResId * @return */ @SuppressWarnings("deprecation") public static int getColor(Context context,int colorId) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { return context.getColor(colorId); } return context.getResources().getColor(colorId); }
②. 获取状态栏高度
/** * 获取状态栏高度 * @param context 上下文对象 */ @JvmStatic @SuppressLint("PrivateApi") fun getStatusBarHeight(context: Context): Int { val clazz = Class.forName("com.android.internal.R\$dimen") val obj = clazz?.newInstance() val field = clazz.getField("status_bar_height") field?.let { field.isAccessible = true val x = Integer.parseInt(field.get(obj).toString()) return context.resources.getDimensionPixelSize(x) } return 75 }
android:fitsSystemWindows="true" android:clipToPadding="false"