获取控件的宽高解决方法

在Android开发中,有时需要对控件进行测量,得到的控件宽度和高度可以用来做一些计算。在需要自适应屏幕的情况下,这种计算就显得特别重要。另一方便,由于需求的原因,希望一进入界面后,就能得到控件的宽度和高度。

实际上在onCreate、onStart、onResume中无法正确得到某个view的宽高信息,这是因为View的measure过程和activity的生命周期方法不是同步执行的,因此无法保证Activity执行了上述方法时某个view已经测量完毕了,如果view还没有测

量完毕,那么获取的宽高就是0,下面提供四种方法来解决这个问题:


(1)   Activity/View---onWindowFoucusChanged:


onWindowFoucsChanged这个方法的含义是:view已经初始化完毕,宽/高已经准备好了,这个时候去获取是没问题

的。需要注意的是onWindowFocusChanged会被调用多次,当activity的窗口得到焦点和失去焦点时均会被调用。

代码如下:

   @Override
    public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        if(hasFocus){
            int width = view.getMeasureWidth();
            int height = view.getMeausreHeight();
        }
}


(2)      view.post(runnable):


通过post可以将一个runnable投递到消息队列的尾部,然后等待looper调用次runnable的时候,view也已经初始化好了:

代码如下:
@Override
    protected void onStart() {
        super.onStart();
        view.post(new Runnable(){
            @Override
            public void run() {
                int width = view.getMEasureWidth();
                int height = view.getMeasureHeight();
            }
        });
}

(3)      ViewTreeObserver:


使用ViewTreeObserver的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener这个接口,当view树状态发

生改变或者view树内部的view的可见性发生改变时,onGlobalLayout方法将被回调,因此这是获取view的宽高的一个

很好的时机。同样,当view树的状态改变时,onGlobalLayout会被调用多次。

代码如下:

   @Override
    protected void onStart() {
        super.onStart();
        ViewTreeObserver observer = view.getViewTreeObserver();
        observer.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                view.getViewTreeObserver().removeGolbalLayoutListener(this);
                int width = view.getMeasureWidth();
                int height = view.getMeasureHeight();
            }
        });
}

(4)view.measure(int widthMeasureSpec,int heightMeasureSpec)


通过手动对view进行measure来得到view的宽高,这里要分情况处理,根据View的LayoutParams来分。


match_parent:

直接放弃,无法measure出具体宽高,构造此种measureSpec需要知道parentSize,即父容器的剩余空间,而这个时候我们无法知道parentSize的大小,所以理论上不可能测量出View的大小。


dp/px:

 int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
    int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(100, View.MeasureSpec.EXACTLY);
    view.measure(widthMeasureSpec,heightMeasureSpec);

wrap_content:

如下measure:

int widthMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) -1, View.MeasureSpec.AT_MOST);
    int heightMeasureSpec = View.MeasureSpec.makeMeasureSpec((1 << 30) -1 , View.MeasureSpec.AT_MOST);
    view.measuew(widthMeasureSpec,heightMeasureSpec);

注意到(1<<30)-1,通过分析MeasureSpec的实现可以知道,view的尺寸使用30位二进制表示,也就是说最大是30个1(2^30-1),也就是(1<<30)-1,在最大化模式下,我们用view理论上能支持最大值去构造measureSpec是合理的.






你可能感兴趣的:(Android)