Android ViewTreeObserver使用总结

我们知道在oncreate中View.getWidth和View.getHeight无法获得一个view的高度和宽度,这是因为View组件布局要在onResume回调后完成。所以现在需要使用getViewTreeObserver().addOnGlobalLayoutListener()来获得宽度或者高度。这是获得一个view的宽度和高度的方法之一。

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

除了OnGlobalLayoutListener ,ViewTreeObserver还有如下内部类:



interface ViewTreeObserver.OnGlobalFocusChangeListener

当在一个视图树中的焦点状态发生改变时,所要调用的回调函数的接口类


interface ViewTreeObserver.OnGlobalLayoutListener

当在一个视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变时,所要调用的回调函数的接口类


interface ViewTreeObserver.OnPreDrawListener

当一个视图树将要绘制时,所要调用的回调函数的接口类


interface ViewTreeObserver.OnScrollChangedListener

当一个视图树中的一些组件发生滚动时,所要调用的回调函数的接口类


interface ViewTreeObserver.OnTouchModeChangeListener

当一个视图树的触摸模式发生改变时,所要调用的回调函数的接口类


其中,我们可以利用OnGlobalLayoutListener来获得一个视图的真实高度。

int mHeaderViewHeight;
mHeaderView.getViewTreeObserver().addOnGlobalLayoutListener(
        new OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {

                mHeaderViewHeight = mHeaderView.getHeight();
                getViewTreeObserver()
                        .removeGlobalOnLayoutListener(this);
            }
        });

但是需要注意的是OnGlobalLayoutListener可能会被多次触发,因此在得到了高度之后,要将OnGlobalLayoutListener注销掉。
有时候需要在onCreate方法中知道某个View组件的宽度和高度等信息,而直接调用View组件的getWidth()、getHeight()、getMeasuredWidth()、getMeasuredHeight()、getTop()、getLeft()等方法是无法获取到真实值的,只会得到0。这是因为View组件布局要在onResume回调后完成。下面提供实现方法,onGlobalLayout回调会在view布局完成时自动调用:

类似:

// This listener is used to get the final width of the GridView and then calculate the
// number of columns and the width of each column. The width of each column is variable
// as the GridView has stretchMode=columnWidth. The column width is used to set the height
// of each view so we get nice square thumbnails.
mGridView.getViewTreeObserver().addOnGlobalLayoutListener( 
//view 布局完成时调用,每次view改变时都会调用
        new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                if (mAdapter.getNumColumns() == 0) {
                        final int numColumns = (int) Math.floor(
                                 mGridView.getWidth() / (mImageThumbSize + mImageThumbSpacing));
                    if (numColumns > 0) {
                        final int columnWidth =
                            (mGridView.getWidth() / numColumns) - mImageThumbSpacing;
                        mAdapter.setNumColumns(numColumns);   //设置 列数
                            mAdapter.setItemHeight(columnWidth);  //设置 高度
                    }
                }
            }
        });

在gridview布局完成后,根据girdview的宽和高设置adapter列数和每个item高度

举例

做一个界面,当列表中的条目没有全部显示出来时,底部添加一个提醒加载更多的标识。当用户拉到最后一个条目时,隐藏提醒

  //当在一个视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变时,所要调用的回调函数的接口类
        rlvOrderDetail.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {

            @Override
            public void onGlobalLayout() {
                boolean is = rlvOrderDetail.canScrollVertically(1);
                if (is) {
                    imMoreLogo.setVisibility(VISIBLE);
                    rlvOrderDetail.setBackgroundResource(R.drawable.no_corner_preparation_bottom);
                } else {
                    imMoreLogo.setVisibility(GONE);

                }
                rlvOrderDetail.getViewTreeObserver().removeOnGlobalLayoutListener(this);
            }

        });

        //当一个视图树中的一些组件发生滚动时,所要调用的回调函数的接口类
        rlvOrderDetail.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
            @Override
            public void onScrollChanged() {
                RecyclerView.LayoutManager layoutManager = rlvOrderDetail.getLayoutManager();
                //判断是当前layoutManager是否为LinearLayoutManager
                // 只有LinearLayoutManager才有查找第一个和最后一个可见view位置的方法
                if (layoutManager instanceof LinearLayoutManager) {
                    LinearLayoutManager linearManager = (LinearLayoutManager) layoutManager;
                    //获取最后一个可见view的位置
                    int lastItemPosition = linearManager.findLastVisibleItemPosition();
                    /*LogManager.get().getLogger(GridOrderItem.class).info("最后一个可见view的位置:"+lastItemPosition+",条目总数"+(adapter.getItemCount()-1));*/
                    if (lastItemPosition==(adapter.getItemCount()-1)){
                        imMoreLogo.setVisibility(GONE);
                    }else{
                        imMoreLogo.setVisibility(VISIBLE);
                    }
                }
            }
            });

转自大神的博客:https://blog.csdn.net/linghu_java/article/details/46544811

并且,以下链接都是很好的参考

https://www.jianshu.com/p/5b026ffc36f1

https://www.cnblogs.com/baiqiantao/p/5631050.html

https://www.cnblogs.com/dongdong230/p/4691467.html

猜你喜欢

转载自blog.csdn.net/changhuzichangchang/article/details/81007278
今日推荐