获取Android手机屏幕各个区域的宽高尺寸

屏幕物理宽高

一般来说计算屏幕宽高都会使用以下方法:

    public static int getScreenWidth(Context context) {
        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        DisplayMetrics dm = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics(dm);
        return dm.widthPixels;
    }

    public static int getScreenHeight(Context context) {
        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        DisplayMetrics dm = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics(dm);
        return dm.heightPixels;
    }

但在有虚拟按键(NavigationBar)的屏幕上,不管是横屏还是竖屏,NavigationBar占了位置的那一边得到的结果就会不准确。获得的结果其实是真实长度 - NavigationBar的高度

于是为了在有NavigationBar的屏幕上也能准确的计算到屏幕的宽高,就有了以下方法:

    public static int getRealWidth(Context context) {
        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        Display display = wm.getDefaultDisplay();
        int screenWidth = 0;

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
            DisplayMetrics dm = new DisplayMetrics();
            display.getRealMetrics(dm);
            screenWidth = dm.widthPixels;

            //或者也可以使用getRealSize方法
//            Point size = new Point();
//            display.getRealSize(size);
//            screenWidth = size.x;
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
            try {
                screenWidth = (Integer) Display.class.getMethod("getRawWidth").invoke(display);
            } catch (Exception e) {
                DisplayMetrics dm = new DisplayMetrics();
                display.getMetrics(dm);
                screenWidth = dm.widthPixels;
            }
        }
        return screenWidth;
    }
    public static int getRealHeight(Context context) {
        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        Display display = wm.getDefaultDisplay();
        int screenHeight = 0;

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
            DisplayMetrics dm = new DisplayMetrics();
            display.getRealMetrics(dm);
            screenHeight = dm.heightPixels;

            //或者也可以使用getRealSize方法
//            Point size = new Point();
//            display.getRealSize(size);
//            screenHeight = size.y;
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
            try {
                screenHeight = (Integer) Display.class.getMethod("getRawHeight").invoke(display);
            } catch (Exception e) {
                DisplayMetrics dm = new DisplayMetrics();
                display.getMetrics(dm);
                screenHeight = dm.heightPixels;
            }
        }
        return screenHeight;
    }```

#状态栏的高度
状态栏(StatusBar)是屏幕顶部显示手机状态(如电池电量、网络状态、时间、运营商信息等)的区域。其高度可以通过读取定义在Android系统尺寸资源中的status_bar_height获得,所以不管当前Activity有没有隐藏StatusBar,获得的结果都是一样的。
```java
    public static int getStatusBarHeight(Context context) {
        int statusBarHeight = -1;
        Resources resources = context.getResources();
        int resourceId = resources.getIdentifier("status_bar_height", "dimen","android");
        if (resourceId > 0) {
            statusBarHeight = resources.getDimensionPixelSize(resourceId);
        }
        return statusBarHeight;
    }

虚拟按键的高度

虚拟按键(NavigationBar)是部分Android手机屏幕底部用以取代物理按键的区域,可隐藏。其高度可以通过读取定义在Android系统尺寸资源中的navigation_bar_height获得,所以不管当前Activity有没有隐藏NavigationBar,获得的结果都是一样的。

    public static int getNavigationBarHeight(Context context) {
        int navigationBarHeight = -1;
        Resources resources = context.getResources();
        int resourceId = resources.getIdentifier("navigation_bar_height","dimen", "android");
        if (resourceId > 0) {
            navigationBarHeight = resources.getDimensionPixelSize(resourceId);
        }
        return navigationBarHeight;
    }

ActionBar的高度

ActionBar就是在带有ActionBar的Theme中StatusBar下方、Activity顶部的类似于标题栏的区域。其高度可以通过读取定义在Android系统属性资源中的actionBarSize获得,所以不管当前Activity有没有使用ActionBar,获得的结果都是一样的。

    public static float getActoinBarHeight(Context context) {
        TypedArray actionbarSizeTypedArray = context.obtainStyledAttributes(new int[] {
                android.R.attr.actionBarSize
        });

        return actionbarSizeTypedArray.getDimension(0, 0);
    }

ContentView的高度

一般来说,ContentView就是我们为Activity设计的layout布局然后通过setContentView添加到Window上的那个View了。该方法推荐在onWindowFocusChanged()中执行,在onCreate()等方法中执行会返回0。

    public static int getContentViewHeight(Activity activity) {
        Rect rectangle= new Rect();
        activity.getWindow().findViewById(Window.ID_ANDROID_CONTENT).getDrawingRect(rectangle);
        return rectangle.height();
    }

WindowVisibleDisplay的高度

这个不太好解释,一般来说就是我们的程序能显示(或者说可见)的区域。返回结果受StatusBar、NavigationBar和软键盘等显示/消失的状态影响。当Activity隐藏了StatusBar、NavigationBar和软键盘处于全屏状态时,这个区域的大小就是屏幕的大小,即使Activity的Theme是Theme.Dialog之类。该方法推荐在onWindowFocusChanged()中执行,在onCreate()等方法中执行会返回0。

    public static int getWindowVisibleDisplayHeight(Activity activity) {
        Rect rectangle= new Rect();
        activity.getWindow().getDecorView().getWindowVisibleDisplayFrame(rectangle);
        return rectangle.height();
    }

软键盘的高度

上面说了软键盘的显示/消失可能会影响WindowVisibleDisplay的高度,可以据此来获得软键盘的高度。

    //ViewTree的状态发生改变或者ViewTree内部的View的可见性发生改变时均会调用onGlobalLayout方法,不用时记得移除掉监听
    getWindow().getDecorView().getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener(){


        @Override
        public void onGlobalLayout(){
            //rootViewHeight可以理解为屏幕的高度
            int rootViewHeight = getWindow().getDecorView().getRootView().getHeight();
            //程序可视区域的高度
            int windowVisibleDisplayHeight = ScreenSizeUtils.getWindowVisibleDisplayHeight(ScreenActivity.this);
            //判断软键盘是否显示的一个阀值
            boolean keyboardShow = (rootViewHeight - windowVisibleDisplayHeight) > rootViewHeight / 3;
            if (keyboardShow) {
                int keyboardHeight = rootViewHeight - windowVisibleDisplayHeight;
                Log.i("tag", "keyboardHeight: " + keyboardHeight);
            }

        }
    });
  • 因为是监听回调,上面的方法在Activity的onCreate方法中也有效
  • 当Activity的windowSoftInputMode 属性为"adjustNothing"模式时以上方法无效
  • 监听不仅在软键盘显示/消失时会被调用,ViewTree的状态发生改变或者ViewTree内部的View的可见性发生改变时均会调用,所以不用时记得移除掉监听,可以在onDestroy方法里面进行移除:
@Override
    protected void onDestroy() {
        //移除布局变化监听
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
            getWindow().getDecorView().getViewTreeObserver().removeOnGlobalLayoutListener(mLayoutChangeListener);
        } else {
            getWindow().getDecorView().getViewTreeObserver().removeGlobalOnLayoutListener(mLayoutChangeListener);
        }
        super.onDestroy();
    }
  • 上面的方法稍微修改一下就可以添加对软键盘显示/消失的监听

以上搜集的大部分都是获取各个区域的高度,其宽度我们一般都不关心,因为一般都是屏幕的宽度,如果不是的情况下,稍微修改一下代码也就可以获得了。

推荐阅读更多精彩内容