详解Android沉浸式实现兼容解决办法

自android5.0开始,沉浸式状态栏似乎成为一种潮流,应用里缺少沉浸式总感觉少些什么。于是乎,我开始到处找如何兼容低版本的沉浸式,由于Android平台跨度问题,总遇到一些不如人意的问题。终于,皇天不负有心人,通过参考一些网络上的资料以及开发的一些经验,总结出一个可行的且良好的解决方案!

先介绍下,什么是沉浸式状态栏?

沉浸式,要求在应用中Android状态栏(StatusBar)与标题栏(ActionBar/Toolbar)要拥有相同的颜色,或者使用同一张图的连续背景。


话不多说,亮剑吧!

具体实现需要针对不同Android版本做处理,还有针对DecorView做处理以及做activity的xml布局文件根布局控件做属性处理。

java代码,设置沉浸式方法

  /**
   * 设置沉浸式状态栏颜色
   *
   * @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
  }

activity布局xml根布局添加以下属性

 android:fitsSystemWindows="true"
 android:clipToPadding="false"

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。

相关文章

Android性能优化——之控件的优化 前面讲了图像的优化,接下...
前言 上一篇已经讲了如何实现textView中粗字体效果,里面主要...
最近项目重构,涉及到了数据库和文件下载,发现GreenDao这个...
WebView加载页面的两种方式 一、加载网络页面 加载网络页面,...
给APP全局设置字体主要分为两个方面来介绍 一、给原生界面设...
前言 最近UI大牛出了一版新的效果图,按照IOS的效果做的,页...