Android Activity中获取View和屏幕的宽高
在Activity已经启动完之后,如果需要获取一个View的宽和高,大部分初学者都试过在onCreate、onStart、onResume中获取,然而获取的结果均为 0。这是因为View的measure过程和Activity的生命周期并不是同步执行的,因此无法保证在onCreate、onStart、onResume中获取宽高时View已经测量完了,如果没有测量完,获得的宽高就是0。
我们用以下几个方法解决这个问题:
一、Activity/View#onWindowFocusChanged
onWindowFocusChanged的含义:View已经初始化完毕了,宽高已经准备好了,这个时候获取宽高是没有问题的。当Activity的当前Window获得或失去焦点时会回调此方法,也就是说当Activity暂停执行和继续执行都会回调此方法,即这个方法会被频繁调用。我们一般在第一次获取焦点时获取宽高,代码如下:
private boolean isFirstFocus = true;
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
if (hasFocus&&isFirstFocus){
isFirstFocus = false;
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
}
二、view.post(runnable)
利用 Handler 通信机制,通过post将添加一个 Runnable到message queue的队尾,当View初始化完成之后,Looper会调用此runnable,然后通知UI线程。代码如下:
@Override
protected void onStart() {
super.onStart();
view.post(new Runnable() {
@Override
public void run() {
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
});
}
三、ViewTreeObserver
当View树状态发生改变,或者View树内部的view的可见性发生改变时,onGlobalLayout会被回调,所以这也是获取宽高的一个很好的时机。伴随着View树的状态的改变,onGlobalLayout会被调用多次,因此可在第一次调用完后,移除监听事件。代码如下:
@Override
protected void onStart() {
Logger.e("onStart");
super.onStart();
ViewTreeObserver viewTreeObserver = view.getViewTreeObserver();
viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
view.removeOnLayoutChangeListener(this);
int width = view.getMeasuredWidth();
int height = view.getMeasuredHeight();
}
});
}
四、View#addOnLayoutChangeListener
监听 View的onLayout()的绘制过程,一旦宽高发生变化就会回调onLayoutChange方法。因此可在第一次调用完后,移除监听事件。代码如下:
view.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
@Override
public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
view.removeOnLayoutChangeListener(this);
Logger.e("w/h:" + view.getWidth() + "-" + view.getHeight());
}
});
有时候我们并不需要View的宽高,而是想获取屏幕的宽高,通常有以下几种方式可以获取:
方法一:通过WindowManager获取
WindowManager wm = (WindowManager) this.getSystemService(Context.WINDOW_SERVICE);
int width = wm.getDefaultDisplay().getWidth();
int height = wm.getDefaultDisplay().getHeight();
方法二:通过WindowManager获取
WindowManager wm1 = this.getWindowManager();
int width1 = wm1.getDefaultDisplay().getWidth();
int height1 = wm1.getDefaultDisplay().getHeight();
方法三:通过WindowManager获取
WindowManager manager = this.getWindowManager();
DisplayMetrics outMetrics = new DisplayMetrics();
manager.getDefaultDisplay().getMetrics(outMetrics);
int width = outMetrics.widthPixels;
int height = outMetrics.heightPixels;
方法四:通过Resources 获取
Resources resources = this.getResources();
DisplayMetrics dm = resources.getDisplayMetrics();
float density = dm.density;
int width = dm.widthPixels;
int height = dm.heightPixels;