getWindowVisibleDisplayFrame()使用总结

im类项目的聊天界面中需要在键盘上显示一个输入控制框,所以需要获取到软键盘的高度,这里就需要使用到android中的getWindowVisibleDisplayFrame()方法。收集整理了一些getWindowVisibleDisplayFrame()的相关资料,这里记录一下备忘。

getWindowVisibleDisplayFrame()是View类下的一个方法,用来获取当前窗口可视区域的大小。该方法原型为:

public void getWindowVisibleDisplayFrame(Rect outRect);

outRect中保存了可视区域的范围,如left, top, right, bottom。

该方法使用注意事项
  1. 调用该方法的view对象必须在有效的window中,比如activity,fragment或者dialog的layout中。类似new TextView(context).getWindowVisibleDisplayFrame(rect)无法得到正确的结果。
  2. 该方法必须在view已经attach到window时调用才能得到期望的正确结果。比如我们可以在Activity、Fragment和Dialog的onWindowFocusChanged()方法中执行,在view的onAttachedToWindow()中可能无法获得正确结果
  3. outRect所表示的只是窗体可见范围,其会受到系统状态栏,虚拟键盘和导航栏的影响,状态栏主要影响outRect的top值,虚拟键盘和导航栏会影响outRect的bottom值
getWindowVisibleDisplayFrame()的应用场景

由于Android系统并没有提供api来获取系统状态栏,软键盘和导航栏的高度,所以我们通常使用getWindowVisibleDisplayFrame()来得到这些模块的值。对系统状态栏高度,获取一个非全屏,且窗口的LayoutParams的height设置为WindowManager.LayoutParams.MATCH_PARENT的窗口可视区域大小,其top值就是状态栏的高度。对系统软键盘,获取一个高度是MATCH_PARENT的窗口在软键盘显示和隐藏两种不同状态下的可视区域大小,将bottom值相减就可以得到软键盘的高度。对系统虚拟按键栏,获取一个高度是MATCH_PARENT的窗口在虚拟按键显示和隐藏两种不同状态下的可视区域大小,将bottom值相减就可以得到虚拟按键的高度。

下面提供获取状态栏,虚拟键盘和导航栏的几种方法:
public static int getStatusBarHeight(Context context) {
    int statusbarheight = 0;
    try {
            Class<?> c = Class.forName("com.android.internal.R$dimen");
            Object o = c.newInstance();
            Field field = c.getField("status_bar_height");
            int x = (Integer) field.get(o);
            statusbarheight = context.getResources().getDimensionPixelSize(x);
        } catch (Exception e) {
            e.printStackTrace();
        }
        
    return statusbarheight;
}
    
public static int getStatusBarHeight(View view) {
    Rect outRect = new Rect();
    view.getWindowVisibleDisplayFrame(outRect);
    statusbarheight = outRect.top;  
    return outRect.top;
}

/**
* 获取软键盘高度
*/
view.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            Rect rect = new Rect();
            int windowVisibleBottomWithoutKeyboard;
            
            @Override
            public void onGlobalLayout() {
                rect.setEmpty();
                view.getWindowVisibleDisplayFrame(rect);
                int detal = windowVisibleBottomWithoutKeyboard - rect.bottom;
                if (detal > screenHeight / 3) {
                    keyboardHeight = detal;
                } else {
                    windowVisibleBottomWithoutKeyboard = rect.bottom;
                } 
            }
        });
        
public static int getNavBarHeight(Context context){
    if (isMeizu()) {
        return getSmartBarHeight(context);
    }
    if (checkDeviceHasNavigationBar(context)) {
        Resources resources = context.getResources();
        int resourceId = resources.getIdentifier("navigation_bar_height", "dimen", "android");
        if (resourceId > 0) {
            return resources.getDimensionPixelSize(resourceId);
        }
    }
    return 0;
}

public static boolean checkDeviceHasNavigationBar(Context context) {
        boolean hasNavigationBar = false;
        Resources rs = context.getResources();
        int id = rs.getIdentifier("config_showNavigationBar", "bool", "android");
        if (id > 0) {
            hasNavigationBar = rs.getBoolean(id);
        }
        
        String navBarOverride = getSystemProperty("qemu.hw.mainkeys", "");
        if ("1".equals(navBarOverride)) {
            hasNavigationBar = false;
        } else if ("0".equals(navBarOverride)) {
            hasNavigationBar = true;
        }
        
        return hasNavigationBar;

}

public static boolean isMeizu() {
    /* 获取魅族系统操作版本标识*/
    String meizuFlymeOSFlag = getSystemProperty("ro.build.display.id", "");
    if (TextUtils.isEmpty(meizuFlymeOSFlag)) {
        return false;
    } else if (meizuFlymeOSFlag.contains("flyme") || meizuFlymeOSFlag.toLowerCase().contains("flyme")) {
        return true;
    } else {
        return false;
    }
}

public static int getSmartBarHeight(Context context) {
    try {
        Class c = Class.forName("com.android.internal.R$dimen");
        Object obj = c.newInstance();
        Field field = c.getField("mz_action_button_min_height");
        int height = Integer.parseInt(field.get(obj).toString());
        return context.getResources().getDimensionPixelSize(height);
    } catch (Exception e) {
        e.printStackTrace();
    }
    return 0;

参考文档

Android获取窗口可视区域大小: getWindowVisibleDisplayFrame()

    原文作者:codeKeeper
    原文地址: https://www.jianshu.com/p/0ecfb3e3fa28
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞