获取View的宽高

来源:互联网 发布:网络电视必须买盒子么 编辑:程序博客网 时间:2024/05/17 20:27

实际上在onCreate、onStart、onResume中均无法正确得到某个View的宽高信息,这是因为View的measure过程和Activity的生命周期方式不是同步执行的,因此无法保证Activity执行了onCreate、onStart、onResume时某个View已经测量完毕了,如果没有测量完毕,获得的宽高就是0。
给出四种方法来解决这个问题:
(1)Activity/View#onWindowFocusChanged。
方法的含义是:View已经初始化完毕,宽高已经准备好了。需要注意的是,onWindowFocusChanged会被调用多次,当Activity的窗口得到焦点和失去焦点时均会被调用一次。

public void onWindowFocusChanged(boolean hasFocus) {        super.onWindowFocusChanged(hasFocus);        if(hasFocus) {            int width = view.getMeasuredWidth();            int height = view.getMeasuredHeight();        }    }

(2)view.post(runnable)。
通过post可以将一个runnable投递到消息队列的尾部,然后等待Looper调用此runnable的时候,View也已经初始化好了。

protected void onStart() {        super.onStart();        view.post(new Runnable() {            @Override            public void run() {                int width = view.getMeasuredWidth();                int height = view.getMeasuredHeight();            }        });    }

(3)ViewTreeObserver。
使用ViewTreeObserver的回调接口OnGlobalLayoutListener可以完成这个功能,当View树的状态发生改变或者View树内部的可见性发生改变时,onGlobalLayout方法将被回调。

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

(4)view.measure(int widthMeasureSpec, int heightMeasureSpec)。
通过手动对View进行measure来得到View的宽高,这里要分情况处理,根据View的LayoutParams来分:
match_parent
直接放弃,无法measure出具体的宽高,构造此种MeasureSpec需要知道parentSize,即父容器剩余空间,而这个时候我们无法知道parentSize的大小。
具体的数值(dp/dx)
比如宽高都是100px,如下measure:

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

wrap_content

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

(1 << 30) -1是在最大化模式下,View理论上能支持的最大值。

Android开发艺术探索笔记

0 0
原创粉丝点击