web-dev-qa-db-ja.com

Android getMeasuredHeightは間違った値を返します!

私はいくつかのUI要素の実際の寸法をピクセルで決定しようとしています!

これらの要素は.xmlファイルから拡張され、最終的にGUIが複数の画面サイズとdpiをサポートするように、ディップの幅と高さで初期化されます( 推奨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は1つのフレームを表します。しかし、私はここで説明する水平レイアウト内に多くを動的に追加します:

<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 > 

My Java code内に1つのフレームを追加するために定義されたメソッド:

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);から報告された情報次のとおりです:density = 1,5 densityDpi = 240 widthPixel = 600 heightPixel = 1024

今、私はAndroid isからのルールを知っています:pixel = dip *(dpi/160)。しかし、返される値には何の意味もありません。その(90dip X 110dip)のImageViewではmeasure()メソッドの戻り値は(270 x 218)です。

誰にも理由はありますか?値はピクセルで返されますか?

ちなみに、私は同じコードをテストしてきましたが、ImageViewではなくTextViewを使用しており、すべてが正常に機能しているようです!なぜ !?!?

37
oberthelot

measureを間違って呼び出しています。

measureMeasureSpecMeasureSpec.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));
53
adamp

それを行う:

frame.measure(0, 0);

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

解決しました!

17
Derzu

OK !ここで自分の質問に答えるのは...しかし完全ではありません

1-一部のデバイスでは、ImageView測定では正確な値が得られないようです。たとえば、NexusやGalaxyデバイスでこのような出来事に関する多くのレポートを見てきました。

2-私が思いついた回避策

ImageViewの幅と高さをxmlコード内で「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の計算 に基づいて、画像ビューの独自の比率を計算します

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

計算されたサイズを使用して、コード内でImageViewを動的に設定します

frame.getLayoutParams().width = pixeWidth;

そして出来上がり! ImageViewに必要なDipサイズが追加されました;)

5
oberthelot
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);
    }
  }
 });

ビューの幅/高さを取得する方法

4
Amit Yadav

カスタム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());

    }

}
2
Pramod

おそらく、AndroidManifest.xml( link )ファイルにあるものと、xmlファイルがどのdrawable-XXXディレクトリから来るのか、Androidはスケーリング操作でリソースをロードします。 「dip」( link )を使用することにしました。これは仮想であり、実際の値(px)は異なる場合があります。

1

残念ながら、 Activity ライフサイクルメソッド Activity#onCreate(Bundle) などでは、レイアウトパスがまだ実行されていないため、ビュー階層内のビューのサイズをまだ取得できません。ただし、 View#measure(int、int) を使用して、Androidを使用してビューを測定するように明示的に要求できます。

@adampの answer が指摘しているように、 View#measure(int、int)MeasureSpec の値ですが、正しい MeasureSpec を見つけるのは少々面倒です。

次のメソッドtriesは、正しい 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はディメンション内の階層内の任意のビューです。

1
Travis