在开发中我们需要用着这种场景:在一个Activity启动的时候获取某个View的宽高,但是在OnCreate、onStart、onResume中都无法正确的获取该View的宽高。这是由于View的measure过程和Activity生命周期方法并不是同步执行的,因此无法在Activity生命周期的方法中获取到正确的View的宽高,那该如何获取呢?下面给出常用的四个在Activity启动时获取View宽高的方法。
1. 使用onWindowFocusChanged
onWindowFocusChanged()这个方法是View已经初始化完毕,宽高已经确定时候可以在此方法中获取到指定View准确的宽高,需要注意的是onWindowFocusChanged会被调用多次,在Activity的窗口获得焦点和失去焦点的时候均会被调用一次,使用代码如下:
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
if (hasFocus){
int width=text.getMeasuredWidth();
int height=text.getMeasuredHeight();
}
}
2.view.post(runnable)
通过post可以讲一个runnable投递到消息队列的尾部,然后等待Looper调用此runnable时,View已经初始化完毕,因此此时可以获取到View的准确宽高,使用代码如下:
@Override
protected void onStart() {
super.onStart();
text.post(new Runnable() {
@Override
public void run() {
int width=text.getMeasuredWidth();
int height=text.getMeasuredHeight();
}
});
}
3.使用ViewTreeObserver
使用ViewTreeObserver的很多回调都可以完成这个功能,比如使用onGlobalLayoutListener接口,当VIew树的状态发生改变或者View树内部的View的可见性发生变化时,onGlobalLayout 方法将会被回调,因此此时可以获取到View的宽高,onGlobalLayout会被调用多次。使用方法如下:
@Override
protected void onStart() {
super.onStart();
ViewTreeObserver observer=text.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
text.getViewTreeObserver().removeOnGlobalLayoutListener(this);
int width = text.getMeasuredWidth();
int height = text.getMeasuredHeight();
}
});
}
4.view.measure(int widthMeasureSpec, int heightMeasureSpec)
通过手动对View进行measure来得到View的宽高,这种方法要根据View的LayoutParams来分:
1.wrap_parent
使用代码如下:
int widthMeasureSpec= View.MeasureSpec.makeMeasureSpec((1<<30)-1, View.MeasureSpec.AT_MOST);
int heightMeasureSpec= View.MeasureSpec.makeMeasureSpec((1<<30)-1, View.MeasureSpec.AT_MOST);
text.measure(widthMeasureSpec,heightMeasureSpec);
其中(1<<30)-1 通过MeasureSpec的实现可以知道,View的尺寸使用30位二进制表示,也就是说最大是30个1(2^30-1或者(1<<30)-1),在最大化模式下我们用View理论上能支持的最大值去构造MeasureSpec是合理的。
2.具体的数值 代码如下:
int widthMeasureSpec= View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
int heightMeasureSpec= View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
text.measure(widthMeasureSpec,heightMeasureSpec);