【问题标题】:Android getMeasuredHeight returns wrong values !Android getMeasuredHeight 返回错误值!
【发布时间】:2011-09-03 16:47:13
【问题描述】:

我正在尝试确定某些 UI 元素的实际尺寸(以像素为单位)!

这些元素是从 .xml 文件中扩展而来的,并使用倾斜宽度和高度进行初始化,以便 GUI 最终支持多种屏幕尺寸和 dpi (as recommended by android specs)。

<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="150dip"
android:orientation="vertical">
    
<ImageView
    android:id="@+id/TlFrame" 
    android:layout_width="110dip" 
    android:layout_height="90dip"
    android:src="@drawable/timeline_nodrawing"
    android:layout_margin="0dip"
    android:padding="0dip"/></LinearLayout>

这个前面的 xml 代表一帧。但我确实在此处描述的水平布局中动态添加了许多:

<HorizontalScrollView 
        android:id="@+id/TlScroller"
        android:layout_width="wrap_content" 
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        android:layout_alignParentBottom="true"
        android:layout_alignParentLeft="true"
        android:layout_margin="0dip"
        android:padding="0dip"
        android:scrollbars="none"
        android:fillViewport="false"
        android:scrollbarFadeDuration="0"
        android:scrollbarDefaultDelayBeforeFade="0"
        android:fadingEdgeLength="0dip"
        android:scaleType="centerInside">
        
        <!-- HorizontalScrollView can only host one direct child -->
        <LinearLayout 
            android:id="@+id/TimelineContent" 
            android:layout_width="fill_parent" 
            android:layout_height="wrap_content"
            android:orientation="horizontal"
            android:layout_alignParentTop="true"
            android:layout_alignParentLeft="true"
            android:layout_margin="0dip"
            android:padding="0dip"
            android:scaleType="centerInside"/>
   
    </HorizontalScrollView > 

定义为在我的 java 代码中添加一帧的方法:

private void addNewFrame()
{       
    LayoutInflater inflater     = (LayoutInflater) _parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    ViewGroup root      = (ViewGroup) inflater.inflate(R.layout.tl_frame, null);
    TextView frameNumber = (TextView) root.findViewById(R.id.FrameNumber);
    Integer val = new Integer(_nFramesDisplayed+1); //+1 to display ids starting from one on the user side 
    frameNumber.setText(val.toString());
    
    ++_nFramesDisplayed;
    _content.addView(root);
// _content variable is initialized like this in c_tor
// _content = (LinearLayout) _parent.findViewById(R.id.TimelineContent);
}

然后在我的代码中,我尝试获取以像素为单位的实际实际大小,因为我需要它来在其上绘制一些 opengl 内容。

LayoutInflater inflater     = (LayoutInflater) _parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    ViewGroup root      = (ViewGroup) inflater.inflate(R.layout.tl_frame, null);
    ImageView frame = (ImageView) root.findViewById(R.id.TlFrame);
    
    frame.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
    frame.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);

    final int w = frame.getMeasuredWidth();
    final int h = frame.getMeasuredHeight();

除了这些值比 ImageView 的实际像素大小大得多之外,一切似乎都正常。

来自 getWindowManager().getDefaultDisplay().getMetrics(metrics) 的报告信息; 以下是: 密度 = 1,5 密度Dpi = 240 宽度像素 = 600 高度像素 = 1024

现在,我知道 android 的规则是:pixel = dip * (dpi /160)。但是返回的值没有任何意义。对于 (90dip X 110dip) 的 ImageView,measure() 方法的返回值是 (270 x 218),我假设它以像素为单位!

有人知道为什么吗? 返回的值是像素吗?

顺便说一句:我一直在测试相同的代码,但使用的是 TextView 而不是 ImageView,一切似乎都运行良好!为什么!?!?

【问题讨论】:

    标签: android android-layout android-widget android-ui android-xml


    【解决方案1】:

    你打错了measure

    measure 采用MeasureSpec 值,这些值由MeasureSpec.makeMeasureSpec 专门打包。 measure 忽略 LayoutParams。进行测量的父级应根据自己的测量和布局策略以及子级的 LayoutParams 创建一个 MeasureSpec。

    如果您想衡量WRAP_CONTENT 通常在大多数布局中的工作方式,请像这样调用measure

    frame.measure(MeasureSpec.makeMeasureSpec(maxWidth, MeasureSpec.AT_MOST),
            MeasureSpec.makeMeasureSpec(maxHeight, MeasureSpec.AT_MOST));
    

    如果您没有最大值(例如,如果您正在编写具有无限空间的 ScrollView),您可以使用 UNSPECIFIED 模式:

    frame.measure(MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED),
            MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
    

    【讨论】:

    • 似乎也不起作用....我已经尝试了最大值和 MesureSpec 的所有组合,结果要么与以前相同,要么与我使用的“假”maxValues 相同用于测试或归零。
    • 我不知道它是否与此有关...但是上面的LinearLayout是动态添加的(在java代码中)...在一个Horizo​​ntalScrollView中逐帧添加,在一个LinearLayout中.. ..目标是创建一个动态时间线 GUI 并使用 openGl 在里面绘制!
    • 有没有办法动态设置ImageView的宽高???我正在考虑根据屏幕尺寸和 dpi 自己进行计算...
    • 澄清一下,如果你想测量环绕的文本,你不应该对两个维度都使用MeasureSpec.UNSPECIFIED。这基本上会给你无限的水平滚动,并导致文本展开。您需要将垂直尺寸设置为MeasureSpec.UNSPECIFIED,并将水平尺寸设置为最大宽度,以便触发文本换行并获得准确的高度测量。记得根据需要考虑填充。
    • @BobAman 非常感谢您的评论。这完全清除了我在使用 getMeasuredHeight/Width 时遇到的问题。
    【解决方案2】:

    这样做:

    frame.measure(0, 0);
    
    final int w = frame.getMeasuredWidth();
    final int h = frame.getMeasuredHeight();
    

    解决了!

    【讨论】:

    • @Simon,你的设备是什么?
    • 对我来说,这个解决了神秘的措施,通过指定MeasureSpec.UNSPECIFIED(双关语不是故意的),我在第一次和后来的调用中返回不同的值,谢谢!
    • @Pavlus 与使用MeasureSpec.UNSPECIFIED 调用相同,因为它是一个值为0 的常量
    【解决方案3】:

    好的!在这里回答我自己的问题......但不完全

    1 - 似乎在某些设备上,ImageView 测量不提供精确值。例如,我在 Nexus 和 Galaxy 设备上看到了很多关于这种情况的报告。

    2 - 我想出的解决方法

    在 xml 代码中将 ImageView 的宽度和高度设置为“wrap_content”。

    在你的代码中扩展布局(我想通常是在 UI 初始化中)。

    LayoutInflater inflater     = (LayoutInflater) 
    _parent.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    ViewGroup root      = (ViewGroup) inflater.inflate(R.layout.tl_frame, null);
    ImageView frame = (ImageView) root.findViewById(R.id.TlFrame);
    

    根据典型的Android calculation 计算您自己的图像视图比例

    //ScreenDpi can be acquired by getWindowManager().getDefaultDisplay().getMetrics(metrics);
     pixelWidth = wantedDipSize * (ScreenDpi / 160)
    

    使用计算出的大小在代码中动态设置 ImageView

    frame.getLayoutParams().width = pixeWidth;
    

    瞧!您的 ImageView 现在具有所需的 Dip 大小;)

    【讨论】:

      【解决方案4】:
      view.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
       @SuppressLint("NewApi")
       @SuppressWarnings("deprecation")
       @Override
        public void onGlobalLayout() {
         //now we can retrieve the width and height
         int width = view.getWidth();
         int height = view.getHeight();
      
      
         //this is an important step not to keep receiving callbacks:
         //we should remove this listener
         //I use the function to remove it based on the api level!
      
          if(android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN){
              view.getViewTreeObserver().removeOnGlobalLayoutListener(this);
          }else{
              view.getViewTreeObserver().removeGlobalOnLayoutListener(this);
          }
        }
       });
      

      应该选择How to get width/height of a View

      【讨论】:

        【解决方案5】:

        不幸的是,在Activity 生命周期方法(例如Activity#onCreate(Bundle))中,尚未执行布局传递,因此您还无法检索视图层次结构中的视图大小。但是,您可以使用 View#measure(int, int) 明确要求 Android 测量视图。

        正如@adamp 的answer 指出的那样,您必须为View#measure(int, int) 提供MeasureSpec 值,但找出正确的MeasureSpec 可能有点令人生畏。

        以下方法尝试确定正确的MeasureSpec 值并测量传入的view

        public class ViewUtil {
        
            public static void measure(@NonNull final View view) {
                final ViewGroup.LayoutParams layoutParams = view.getLayoutParams();
        
                final int horizontalMode;
                final int horizontalSize;
                switch (layoutParams.width) {
                    case ViewGroup.LayoutParams.MATCH_PARENT:
                        horizontalMode = View.MeasureSpec.EXACTLY;
                        if (view.getParent() instanceof LinearLayout
                                && ((LinearLayout) view.getParent()).getOrientation() == LinearLayout.VERTICAL) {
                            ViewGroup.MarginLayoutParams lp = (ViewGroup.MarginLayoutParams) view.getLayoutParams();
                            horizontalSize = ((View) view.getParent()).getMeasuredWidth() - lp.leftMargin - lp.rightMargin;
                        } else {
                            horizontalSize = ((View) view.getParent()).getMeasuredWidth();
                        }
                        break;
                    case ViewGroup.LayoutParams.WRAP_CONTENT:
                        horizontalMode = View.MeasureSpec.UNSPECIFIED;
                        horizontalSize = 0;
                        break;
                    default:
                        horizontalMode = View.MeasureSpec.EXACTLY;
                        horizontalSize = layoutParams.width;
                        break;
                }
                final int horizontalMeasureSpec = View.MeasureSpec
                        .makeMeasureSpec(horizontalSize, horizontalMode);
        
                final int verticalMode;
                final int verticalSize;
                switch (layoutParams.height) {
                    case ViewGroup.LayoutParams.MATCH_PARENT:
                        verticalMode = View.MeasureSpec.EXACTLY;
                        if (view.getParent() instanceof LinearLayout
                                && ((LinearLayout) view.getParent()).getOrientation() == LinearLayout.HORIZONTAL) {
                            ViewGroup.MarginLayoutParams lp = (ViewGroup.MarginLayoutParams) view.getLayoutParams();
                            verticalSize = ((View) view.getParent()).getMeasuredHeight() - lp.topMargin - lp.bottomMargin;
                        } else {
                            verticalSize = ((View) view.getParent()).getMeasuredHeight();
                        }
                        break;
                    case ViewGroup.LayoutParams.WRAP_CONTENT:
                        verticalMode = View.MeasureSpec.UNSPECIFIED;
                        verticalSize = 0;
                        break;
                    default:
                        verticalMode = View.MeasureSpec.EXACTLY;
                        verticalSize = layoutParams.height;
                        break;
                }
                final int verticalMeasureSpec = View.MeasureSpec.makeMeasureSpec(verticalSize, verticalMode);
        
                view.measure(horizontalMeasureSpec, verticalMeasureSpec);
            }
        
        }
        

        那么你可以简单地调用:

        ViewUtil.measure(view);
        int height = view.getMeasuredHeight();
        int width = view.getMeasuredWidth();
        

        或者,作为@Amit Yadav suggested,您可以使用OnGlobalLayoutListener 在布局传递完成后调用一个侦听器。以下是处理注销监听器和跨版本的方法命名更改的方法:

        public class ViewUtil {
        
            public static void captureGlobalLayout(@NonNull final View view,
                                                   @NonNull final ViewTreeObserver.OnGlobalLayoutListener listener) {
                view.getViewTreeObserver()
                        .addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
                            @Override
                            public void onGlobalLayout() {
                                final ViewTreeObserver viewTreeObserver = view.getViewTreeObserver();
                                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
                                    viewTreeObserver.removeOnGlobalLayoutListener(this);
                                } else {
                                    //noinspection deprecation
                                    viewTreeObserver.removeGlobalOnLayoutListener(this);
                                }
                                listener.onGlobalLayout();
                            }
                        });
            }
        
        }
        

        那么你可以:

        ViewUtil.captureGlobalLayout(rootView, new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                int width = view.getMeasureWidth();
                int height = view.getMeasuredHeight();
            }
        });
        

        rootView 可以是视图层次结构的根视图,view 可以是层次结构中您想知道其维度的任何视图。

        【讨论】:

        • 感谢measure 方法。当使用new TextView() 创建视图时,它不包含 LayoutParams 但有一个宽度。所以你会在switch (layoutParams.width)上得到一个空指针异常。
        • 对不起,这是一个错误的方法,view.getMeasuredWidth() 在很多情况下都会给出 0。
        • 使用getViewTreeObserver方法获取大小。
        • @CoolMind,我提供的measure 方法适用于许多常见布局,但不幸的是并非全部。因此,正如您所提到的,使用ViewTreeObserver 是一种更可靠的解决方案。提供measure 方法是因为有时依赖ViewTreeObserver 会有点混乱。
        • 是的,你是对的,可能 ViewTreeObserver 不是 100% 可靠的方法。最好检查 isAlive(),见stackoverflow.com/questions/12867760/…(但我可能会犯错误)。
        【解决方案6】:

        您必须创建自定义 Textview 并在您的布局中使用它,并使用 getActual height 函数在运行时设置高度

        public class TextViewHeightPlus extends TextView {
            private static final String TAG = "TextViewHeightPlus";
            private int actualHeight=0;
        
        
            public int getActualHeight() {
                return actualHeight;
            }
        
            public TextViewHeightPlus(Context context) {
                super(context);
            }
        
            public TextViewHeightPlus(Context context, AttributeSet attrs) {
                super(context, attrs);
                setCustomFont(context, attrs);
            }
        
            public TextViewHeightPlus(Context context, AttributeSet attrs, int defStyle) {
                super(context, attrs, defStyle);
        
            }
        
           @Override
            protected void onSizeChanged(int w, int h, int oldw, int oldh) {
                super.onSizeChanged(w, h, oldw, oldh);
                actualHeight=0;
        
                actualHeight=(int) ((getLineCount()-1)*getTextSize());
        
            }
        
        }
        

        【讨论】:

          【解决方案7】:

          可能是因为您在AndroidManifest.xml (link) 文件中的内容以及xml 文件来自哪个drawable-XXX 目录,Android 会通过缩放操作加载资源。您决定使用“dip”(link)维度单位,它是虚拟的,实际值(px)可能不同。

          【讨论】:

          • Manifest 仅指定活动、意图和 sdk 版本。与屏幕尺寸或支持无关。另外,我所有的可绘制对象都在基本的 res/drawable 文件夹中...没有 mdpi 或 hdpi 前缀/后缀/子文件夹...
          • 以及您在 emu 上的哪个设备上运行应用程序?
          • ImageView 的比例类型如何?
          • ScaleType 设置为“centerInside”...我试过删除它,结果是一样的...
          • “与屏幕尺寸或支持无关。”这不是真的
          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 2012-01-29
          • 2013-06-01
          • 1970-01-01
          • 1970-01-01
          • 2014-08-17
          • 2019-02-03
          • 2011-04-06
          相关资源
          最近更新 更多