美文网首页安卓
获取Android手机屏幕各个区域的宽高尺寸

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

作者: ei_chinn | 来源:发表于2016-12-24 19:17 被阅读557次

    屏幕物理宽高

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

        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();
        }
    
    • 上面的方法稍微修改一下就可以添加对软键盘显示/消失的监听

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

    相关文章

      网友评论

        本文标题:获取Android手机屏幕各个区域的宽高尺寸

        本文链接:https://www.haomeiwen.com/subject/ckdlmttx.html