1:
android:fitsSystemWindows="true" -- 可以解决大部分的适配

2:监听虚拟键盘的变化
在主activity的oncreate中调用

Workaround.assistActivity(findViewById(android.R.id.content));
// Workaround  类的方法
 package com.qianyue.game.darkworld;


import android.graphics.Rect;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewTreeObserver;
public class Workaround {
    // For more information, see  https://code.google.com/p/android/issues/detail?id=5497
    // To use this class, simply invoke assistActivity() on an Activity that already has its content view set.

    /**
     * 关联要监听的视图
     *
     * @param viewObserving
     */
    public static void assistActivity(View viewObserving) {
        new Workaround(viewObserving);
    }

    private View mViewObserved;//被监听的视图
    private int usableHeightPrevious;//视图变化前的可用高度
    private ViewGroup.LayoutParams frameLayoutParams;

    private Workaround(View viewObserving) {
        mViewObserved = viewObserving;
        //给View添加全局的布局监听器
        mViewObserved.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            public void onGlobalLayout() {
                resetLayoutByUsableHeight(computeUsableHeight());
            }
        });
        frameLayoutParams = mViewObserved.getLayoutParams();
    }

    private void resetLayoutByUsableHeight(int usableHeightNow) {
        //比较布局变化前后的View的可用高度
        if (usableHeightNow != usableHeightPrevious) {
            //如果两次高度不一致
            //将当前的View的可用高度设置成View的实际高度
            frameLayoutParams.width = usableHeightNow;
            mViewObserved.requestLayout();//请求重新布局
            usableHeightPrevious = usableHeightNow;
        }
    }

    /**
     * 计算视图可视高度
     *
     * @return
     */
    private int computeUsableHeight() {
        Rect r = new Rect();
        mViewObserved.getWindowVisibleDisplayFrame(r);
        return (r.right - r.left);
    }
}

3:虚拟按键的透明化
主activity oncreate
Window window = getWindow();
 // 状态栏
 window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
 // 虚拟导航键
 window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);

Android状态栏和虚拟导航栏是android提供的系统级UI,Android允许开发者通过控制这两个Bar来实现对应用页面的适配。可是我在适配的过程中却发现Android提供的API对某些手机不可用,并不像官方文档中写的那样,因此,在这里总结一下我在适配过程中发现的一些问题。一般谈到对系统UI的控制,就涉及到两个方面:布局和颜色,在平时的开发中也只需要控制这两点就足够了。

布局

控制状态栏和虚拟导航栏的布局,表示控制它们的显示和隐藏,Android提供了这方面的API,开发者只需要设置以下几行代码即可实现显示或隐藏状态栏和虚拟导航栏:

View decorView = getWindow().getDecorView();int uiOptions = View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_FULLSCREEN;decorView.setSystemUiVisibility(uiOptions);

  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

uiOptions,Android提供了十几种标志位,在平时的开发中,我们的APP一般而言只有一下3种状态:

  1. 隐藏状态栏或虚拟导航栏
  2. 状态栏或虚拟导航栏半透明,遮掩APP
  3. 状态栏或虚拟导航栏正常显示,不遮掩APP

根据以上3种状态,对应地,我们有以下的设置:

uiOption

功能

应用场景

备注

SYSTEM_UI_FLAG_VISIBLE

默认状态

主页等大部分页面

呼出虚拟导航栏会自动resize

SYSTEM_UI_FLAG_FULLSCREEN

隐藏状态栏

录制视频,播放电影等

API >= 16

不可单独使用,否则部分手机顶部有白条

SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

半透明状态栏

直播,视频播放页等

API >= 16

SYSTEM_UI_FLAG_HIDE_NAVIGATION

隐藏虚拟导航栏

录制视频,播放电影等

API >= 16

自动resize,触摸屏幕会自动显示虚拟导航栏

SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION

半透明虚拟导航栏,会自动设置半透明状态栏

直播,视频播放页等

API >= 16

SYSTEM_UI_FLAG_IMMERSIVE

自动隐藏状态栏和虚拟导航栏,并且在bar出现的位置滑动可以呼出bar

播放电影等

API >= 19

SYSTEM_UI_FLAG_IMMERSIVE_STIKY

和上面不同的是,呼出的bar会自动再隐藏掉

电子书等

API >= 19

SYSTEM_UI_FLAG_LAYOUT_STABLE

保持整个View稳定,使View不会因为System UI的变化而重新layout

 

API >= 16没发现作用



在这个列表中,有几个需要注意的地方:

  1. 在某些手机上隐藏状态栏的标志

SYSTEM_UI_FLAG_FULLSCREEN

  1. 会失效,因为我用华为、Nexus和小米分别设置了这个标志,发现只有小米的设置才起作用,其它两类手机当隐藏掉状态栏时状态栏的位置会显示白条,APP的内容不会顶上去,如果有知道原因的同学麻烦告知下。因此,当需要隐藏掉状态栏时,推荐使用以下代码进行设置,或者结合

SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

  1. 来设置: 

window.setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN)

  1. 在设置隐藏虚拟导航栏的标志的时候,如果结合 

SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION

  1. 可保持原来的布局,如果结合

SYSTEM_UI_FLAG_IMMERSIVE

  1. 触摸屏幕不会弹出虚拟导航栏

SYSTEM_UI_FLAG_IMMERSIVE

SYSTEM_UI_FLAG_IMMERSIVE_STIKY

  1.  
    需要结合

SYSTEM_UI_FLAG_FULLSCREEN

  1. 或者

SYSTEM_UI_FLAG_HIDE_NAVIGATION

  1. 一起使用,才能得到列表中展示的效果

SYSTEM_UI_FLAG_LAYOUT_STABLE

  1. 没发现具体的效果,如果仅仅是为了保持布局,那么 

SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN

SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION

  1. 也可以做到

fitSystemWindows只有在设置了半透明状态栏或者半透明虚拟导航栏中才有意义,它的作用是为你的APP腾出空间,将界面展示在状态栏下方和虚拟导航栏上方,从而不被状态栏和虚拟导航栏遮挡。

颜色

状态栏和虚拟导航栏的颜色相对于布局设置来说比较容易,如下表所示:

API版本

状态栏

虚拟导航栏

<19

无法设置

无法设置

19<=API<21

使用开源项目SystemBarTintManager

无法设置

>= 21

getWindow().setStatusBarColor(color);

getWindow().setNavigationBarColor(color);



这里同样有几点需要注意的地方:

  1. 在19到21的API中,Android还没有提供设置状态栏颜色的API,因此我们使用了SystemBarTintManager这个开源项目,但是这个项目却不适用于设置虚拟导航栏颜色,因为该项目实际上是设置了透明状态栏和虚拟导航栏后再

Decorview

  1. 中添加了上下两个

view

  1. ,通过这两个设置这两个

view

  1. 来达到设置状态栏和虚拟导航栏颜色的目的。在很多国产手机,虚拟导航栏是可手动隐藏的,当手动隐藏掉虚拟导航栏后,导航栏原来的位置会显示一条你设置的颜色的矩形区域
  2. 在API>=21的版本中,设置状态栏和虚拟导航栏的颜色的同时,还需要保证没有进行对它们进行过透明化的操作,通过以下设置可清楚透明设置:

window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);

  • 1
  • 2


  • 1
  • 2

以上便是我在控制状态栏和虚拟导航栏的显示和隐藏以及颜色方面时总结出来的经验,希望帮到遇到这方面问题的朋友。如果有总结得不正确或者不全面的地方,请不吝赐教!