OnCreate()里得到控件宽高----监听视图树 OnGlobalLayoutListener(转载)

来源:互联网 发布:农村淘宝宣传视频下载 编辑:程序博客网 时间:2024/06/03 17:45

背景:
我们都知道在onCreate()里面获取控件的高度是0,这是为什么呢?我们来看一下示例:
首先我们写一个控件

public class MyImageView extends android.support.v7.widget.AppCompatImageView {    public MyImageView(Context context) {        super(context);    }    public MyImageView(Context context, @Nullable AttributeSet attrs) {        super(context, attrs);    }    @Override    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        super.onMeasure(widthMeasureSpec, heightMeasureSpec);        System.out.println("onMeasure 我被调用了" + System.currentTimeMillis());    }    @Override    protected void onDraw(Canvas canvas) {        super.onDraw(canvas);        System.out.println("onDraw 我被调用了" + System.currentTimeMillis());    }}

布局文件:

<?xml version="1.0" encoding="utf-8"?><LinearLayout    xmlns:android="http://schemas.android.com/apk/res/android"    android:id="@+id/layout"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:orientation="vertical">    <com.bsofts.ongloballayoutlistenertest.view.MyImageView        android:id="@+id/myImageView"        android:layout_width="200dp"        android:layout_height="300dp"        android:src="@mipmap/ic_launcher"        android:background="#ff0"/></LinearLayout>

测试的Activity:

public class MainActivity extends AppCompatActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        System.out.println("onCreate方法执行完毕.."+System.currentTimeMillis());    }}

运行结果:

I/System.out: onCreate方法执行完毕..1490255501304I/System.out: onMeasure 我被调用了1490255501343I/System.out: onMeasure 我被调用了1490255501363I/System.out: onDraw 我被调用了1490255501400

说明等onCreate方法执行完了,我们定义的控件才会被测量(measure),所以我们在onCreate方法里面通过view.getHeight()获取控件的高度或者宽度肯定是0,因为它自己还没有被测量,也就是说他自己都不知道自己有多高,而你这时候去获取它的尺寸,肯定是不行的。

解决方法一:
使用view的post()方法,这种方法适用于需要在onCreate完成之前就获得一个view的宽和高的情况。
比如获得一个MyImageView的宽和高:

public class MainActivity extends AppCompatActivity {    private MyImageView mMyImageView;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        mMyImageView = (MyImageView) findViewById(R.id.myImageView);        mMyImageView.post(new Runnable() {            @Override            public void run() {                System.out.println("MyImageView的宽高分别为:"+ mMyImageView.getHeight()                        + "和" + mMyImageView.getHeight());            }        });    }}

运行结果:

I/System.out: MyImageView的宽高分别为====:600和900

解决方法二:

利用ViewTreeObserver的OnGlobalLayoutListener

  1. 优点:不需要额外的测量过程
  2. 缺点:只有在布局加载完成后,才能得到宽和高

OnGlobalLayoutListener是ViewTreeObserver的内部类,当一个视图树的布局发生改变时,可以被ViewTreeObserver监听到,这是一个注册监听视图树的观察者(observer),在视图树的全局事件改变时得到通知。ViewTreeObserver不能直接实例化,而是通过getViewTreeObserver()获得。

其中,我们可以利用OnGlobalLayoutListener来获得一个视图的真实高度,但是需要注意的是OnGlobalLayoutListener会被多次触发,因此在得到了高度之后,要将OnGlobalLayoutListener清除掉。

public class MainActivity extends AppCompatActivity {    private LinearLayout mLinearLayout;    private MyImageView mMyImageView;    private int mLayoutWidth;    private int mLayoutHeight;    private int mMyImageViewWidth;    private int mMyImageViewHeight;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        mMyImageView = (MyImageView) findViewById(R.id.myImageView);        mLinearLayout = (LinearLayout) findViewById(R.id.layout);        getWidthAndHeight();    }    /**     * 通过是视图树得到view的宽高     */    private void getWidthAndHeight() {        mMyImageView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {            @Override            public void onGlobalLayout() { //view 布局完成时调用,每次view改变时都会调用                //得到宽高                mLayoutWidth = mLinearLayout.getWidth();                mLayoutHeight = mLinearLayout.getHeight();                mMyImageViewWidth = mMyImageView.getWidth();                mMyImageViewHeight = mMyImageView.getHeight();                System.out.println("MyImageView的宽高分别为====:"+ mMyImageViewWidth                        + "和" + mMyImageViewHeight);                System.out.println("根布局LinearLayout的宽高分别为=====:"+ mLayoutWidth                        + "和" + mLayoutHeight);                //使用完必须撤销监听(只测量一次),否则,会一直不停的不定时的测量,这比较耗性能                mMyImageView.getViewTreeObserver().removeOnGlobalLayoutListener(this);            }        });    }}

运行结果:

I/System.out: MyImageView的宽高分别为====:600和900I/System.out: 根布局LinearLayout的宽高分别为=====:1080和1692

这种方法无法像第一种方法那样通过一个函数返回值,因为他是基于listener的,OnGlobalLayoutListener的onGlobalLayout被回调之前是没有值的。由于布局状态可能会发生多次改变,因此OnGlobalLayoutListener的onGlobalLayout可能被回调多次,所以我们在 第一次获得值之后就将listener注销掉。

其他相关回调接口:

ViewTreeObserver的内部回调接口:
OnGlobalLayoutListener 当在一个视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变时回调
OnPreDrawListener 当一个视图树将要绘制时,所要调用的回调函数的接口类
OnScrollChangedListener 当一个视图树中的一些组件发生滚动时回调
OnGlobalFocusChangeListener 当在一个视图树中的焦点状态发生改变时回调
OnTouchModeChangeListener 当一个视图树的触摸模式发生改变时回调
回调方法:
addOnGlobalFocusChangeListener 当在一个视图树中的焦点状态发生改变时调用
addOnGlobalLayoutListener 当在一个视图树中某个视图的可视状态发生改变时调用
addOnPreDrawListener 当一个视图树将要绘制时调用
addOnScrollChangedListener 当一个视图发生滚动时调用
addOnTouchModeChangeListener 当一个触摸模式发生改变时调用
dispatchOnGlobalLayout 当整个布局发生改变时通知相应的注册监听器
dispatchOnPreDraw 当一个视图树将要绘制时通知相应的注册监听器
isAlive 指示当前的ViewTreeObserver是否可用
removeGlobalOnLayoutListener 移除之前已经注册的全局布局回调函数
removeOnGlobalFocusChangeListener 移除之前已经注册的焦点改变回调函数
removeOnPreDrawListener 移除之前已经注册的预绘制回调函数
removeOnScrollChangedListener 移除之前已经注册的滚动改变回调函数
removeOnTouchModeChangeListener 移除之前已经注册的触摸模式改变回调函数

0 0
原创粉丝点击