屏幕物理宽高
一般来说计算屏幕宽高都会使用以下方法:
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();
}
- 上面的方法稍微修改一下就可以添加对软键盘显示/消失的监听
以上搜集的大部分都是获取各个区域的高度,其宽度我们一般都不关心,因为一般都是屏幕的宽度,如果不是的情况下,稍微修改一下代码也就可以获得了。