通常在实际开发中,为了适配,我们会把View设置为match_parent或者wrap_content、又或者是设置weight权重来分配宽高,而不是使用具体值。那么就出现一个问题了,如果动态测量View的实际宽高。
其实这个问题网上有很多解决方法,无奈给出的方法众多,却没有解析,实际可能根本不管用。所以详细记录下这个问题。
测量宽高的API
我们以高度为例子:
android提供了两个API来动态获取View的高:
- view.getMeasuredHeight()
- view.getHeight()
那么这两个API的区别是什么呢?
简单来说就是:
- view.getMeasuredHeight() 是由view中的测量方法赋值的,这个值包含了隐藏的高度(比如一个view部分超出屏幕,他也会计算出来)。
- view.getHeight() 由view的底部位置减去顶部位置,即实际显示的View的高度,不包含隐藏了的高度。
如果是在View的onMeasure方法后执行上面两个方法,会发现可以得到正确的高度,但是如果是在onCreate等方法中执行就会发现返回的值不正确或者为0,这是为什么呢?
Activity和View的混合生命周期
我画了一个图来表明两者的生命周期先后问题:
可以看到的是onCreate()和onResume()是在onMeasure方法之前。
说明此时还没有测量出实际的宽高,还没有进行绘制,所以调用上述的两个API会出现值为0或者数值错误的情况。
经过验证的解决方法
下面的方法都是进过验证的,但是有使用的条件限制,务必注意。
1.主动测量
代码:
view.measure(0, 0);
view.getMeasuredWidth();
view.getMeasuredHeight();
说明:
onMeasure传入的两个参数是由父控件的大小,
也可以使用 View.MeasureSpec.makeMeasureSpec(0,mode);
设置值
其中mode可以选择
- MeasureSpec.UNSPECIFIED 未指定尺寸,比如listview中尺寸由父控件决定
- MeasureSpec.EXACTLY 适合match_parent或者具体值
- MeasureSpec.AT_MOST 适合wrap_content不确定值
注意:
这种方法不一定能测出正确的值,因为onMesure会多次调用(由于onMesure自上而下,父控件如果对于子控件的宽高不满意,即如子控件没有限制宽高,父控件会重新调用onMesure重新测量),所以测量结果不一定正确。
2.使用OnPreDrawListener
代码:
ViewTreeObserver observer = vi