笔记:Activity 中 获取View 实际的宽/高

Question

在Activity 已启动的时候需要获取某个View 宽/高?

Analysis

通常操作会在onCreate或者onResume里面去获取这个View的宽/高,但是实际上无法正确获得某个View的宽/高的信息。这是因为View 的measure过程和Activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate、onResume时某个View已经测量完毕了,如果View还没有测量完毕了,那么获得 宽/高就是0.

Resolve

  • Activity/View.onWindowFocusChanged

该方法的含义是:View 已经初始化完毕了,宽/高已经准备好了,这个时候去获取宽/高是没问题的。注意: onWindowFocusChanged会被调用多次,当Activity 的窗口得到焦点和失去焦点时均会被调用一次。具体来说,如果频繁地进行onResume和onPause,那么onWindowFocusChanged也会频繁地调用。

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

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

view.post(
    new Runnable(){
    public void run(){
       int width=view.getMeasuredWidth();
       int height=vieew.getMeasuredHeight();
    }
  }
);
  • ViewTreeObserver

使用ViewTreeObserver 的众多回调可以完成这个功能,比如使用OnGlobalLayoutListener 这个接口,当View 树的状态发生改变或者View 树内部的View的可见性发生改变时,onGlobalLayout 方法将被回调,因此这是获取View的宽/高一个很好的时机。注意: 伴随着View 树的状态改变等,onLayoutGlobal 会被调用多次。

ViewTreeObserver observer=view.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener(){
   public void onGlobalLayout(){
    view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
    int width=view.getMeasuredWidth();
    int height=vieew.getMeasuredHeight();    
   }
});
  • view.measure(int widthMeasureSpec, int heightMeasureSpec)

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

  1. match_parent

这种情况无法measure出具体的高度。原因是:根据View 的measure过程,构造此种MeasureSpec 需要知道parentSize(父容器的剩余空间)。而这个时候我们无法知道parentSize 的大小,所以理论上不可能测量出View 的大小。

  1. 具体的数值(dp/px)

假设宽/ 高都是100px

int widthMeasureSpec=MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
int heightMeasureSpec=MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
view.measure(widthMeasureSpec,heightMeasureSpec);
  1. 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);

View 的尺寸使用30位二进制表示,也就是说最大是30个1(即2^30-1),也就是(1<<30)-1,在最大化模式下,用View 理论上能支持的最大值去构造MeasureSpec是合理的。


版权声明:本文为xufei5789651原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。