假如我们在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());
}
});