如何在Activity启动时,获取某个View的宽高

来源:互联网 发布:伊斯兰 知乎 编辑:程序博客网 时间:2024/05/21 17:34

如何在Activity启动时,获取某个View的宽高

四种方法:

(1)Activity/View#onWindowFocusChanged

  onWindowFocusChanged这个方法的含义是:View已经初始化完毕了,宽/高已经准备好了,这个时候去获取宽/高是没有问题的。

  注意:

    onWindowsFocusChange会被调用多次。

        当Activity的窗口得到焦点和失去焦点时均会被调用一次。 具体来说,当Activity继续执行和暂停执行时,onWindowFocusChanged均会被调用。

       比如频繁地进行onResume和onPause,那么onWindowsFocusChanged会被频繁的调用。

      public void onWindowsFocusChanged(boolean hasFocus) {

            super.onWindowsFocusChanged(hasFocus);

           if (hasFocus) {

                   int width = view.getMeasuredWidth();

                   int height = view.getMeasuredHeight();

           }

      }  

  (2) view.postRunnable()

      通过post可以将一个runnable投递到消息队列的尾部,然后等待Loooper调用次runnable的时候,View已经初始化好了。典型代码如下:

      protected void onStart() {

       support.onStart();

       view.post(new Runnable()) {

         @Override

         public void run() {

               int width = view.getMeasuredWidth();

               int height = view.getMeasureHeight();

         }

       }

     }

       

 (3) ViewTreeObserver

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

    注意:伴随着View树的状态改变等,onGlobalLayout会被调用多次。

      protected void onStart() {

          super.onStart();

          ViewTreeObserver observer = view.getViewTreeObserver();

         observer.addOnGlobalLayoutListener(new onGlobalLayoutListener() {

              @Override

              public void onGlobalLayout() {

                     view.getViewTreeObserver().removeGlobalLayoutListener(this);

                     int width = view.getMeasuredWidth();

                    int height = view.getMeasuredHeight();

              }

         });

     }

  (4) view.measure(int widthMeasureSpec, int heightMeasureSpec)

       通过手动队View进行measure来得到View的宽/高。

       这种方法比价复杂,这里要分情况处理,根据View的LayoutParams来分:

      match_parent

     直接放弃,无法measure出具体的宽/高。原因很简单,根据View的measure过程,如表4-1所示,构造此种MeasureSpec需要知道parentSize,即父容器的剩余空间,而这个时候我们无法知道parentSize的大小,所以理论上不可能测量出View的大小。

    

      具体数值

     比如宽/高都是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 , 通过分析MeasureSpec的实现可以知道,View的尺寸使用30位二进制表示,也就是说最大是30个1 (即 2^  30 -1) 也就是 (1 << 30) -1, 在最大化模式下,我们用View理论上能支持的最大值去构造MeasureSpec是合理的。

  关于View的measure,网上有两个错误的方法。为什么说是错误的,首先其违背了系统的内部实现规范(因为无法通过错误的MeasureSpec去得出合法的SpecMode,从而导致measure过程出错),其次不能保证一定能measure出正确的结果。

   第一种错误用法:

      int widthMeasureSpec = MeasureSpec.makeMeasureSpec( -1, MeasureSpec.UNSPECIFIED);

      int heightMeasureSpec = MeasureSpec.makeMeasureSpec( -1, MeasureSpec.UNSPECIFIED);

     view.measure(widthMeasureSpec, heightMeasureSpec);

   第二种错误用法:

     view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);

参考文献:

   Android 开发艺术探索 任玉刚


0 0
原创粉丝点击