最近有需求需要动态设置第三个button位于前两个按钮位置中间,在activity中需要获取view.getTopMargin()获取顶边距,我们知道在onCreate()和onResume()生命周期内获取无法获取到view 的Width, Height,Magin等参数信息,因为view的绘制是在onResume()之后执行的,在这里可以用ViewTreeObserver 提供的内部类获取到View的宽高和margin信息。
ViewTreeObserver 是一个注册监听视图树的观察者(observer),会监听视图树发生全局变化时发出的通知。这个全局事件包括整个树的布局,从绘画过程开始,触摸模式的改变等等。ViewTreeObserver不能够被应用程序实例化,因为它是由视图提供的。
ViewTreeObserver主要提供了以下几个内部类:
ViewTreeObserver内部类
ps:偷懒把官网介绍截屏了
从上面可以看出 我们可以在OnGlobalLayoutListener,OnPreDrawListener,OnDrawListener中获取到View宽高和margin值,代码如下:
btn02.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
LogUtil.d("btn02.width" + btn02.getWidth());
RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(btn02.getLayoutParams());
params.topMargin = (btn01.getTop() + btn03.getTop()) / 2;
params.leftMargin = btn02.getLeft();
params.rightMargin = btn02.getRight();
btn02.setLayoutParams(params);
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) {
btn02.getViewTreeObserver().removeOnGlobalLayoutListener(this);
} else {
btn02.getViewTreeObserver().removeGlobalOnLayoutListener(this);
}
}
});
btn02.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
@Override
public boolean onPreDraw() {
LogUtil.d("btn02.width" + btn02.getWidth());
btn02.getViewTreeObserver().removeOnPreDrawListener(this);
return true;
}
});
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) {
btn02.getViewTreeObserver().addOnDrawListener(new ViewTreeObserver.OnDrawListener() {
@Override
public void onDraw() {
LogUtil.d("btn02.width" + btn02.getWidth());
btn02.getViewTreeObserver().removeOnDrawListener(this);
}
});
需要注意的是这些listener可能会重复触发,因此在获取到view宽高后需要调用remove移除相应listener。通过设置Button02的LayoutParams让它居于Button01和Button03中间。如图:
运行效果图
其实除了ViewTreeObserver这个观察者类,还可以通过View#Post获取到View宽高margin数值,想了解的可以移步到下一篇博客:
Android获取View的宽和高(二)