3

簡単なタスク: ボタンを押すと、X の値が 0 にスケーリングされ、アニメーションが終了すると、X を 0 から 1 にスケーリングする 2 番目のビューで別のアニメーションが開始されます。1 秒後にリバース アニメーションが再生され、それだけです。以下のコードを実行すると、アニメーションの最初の部分で無限のアニメーション ループが発生します。

nineoldandroids lib を使用しましたが、少なくともジェリー ビーン デバイスでは、これがネイティブ アニメーション フレームワークと大きく異なるとは思いません。

public class MainActivity extends Activity
{

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    final View mybutton = findViewById(R.id.mybutton);
    final View myprogress = findViewById(R.id.myprogress);

    mybutton.setOnClickListener(new View.OnClickListener()
    {
        @Override
        public void onClick(View v)
        {
            animate(mybutton).scaleX(0).setListener(new AnimatorListenerAdapter()
            {
                @Override
                public void onAnimationEnd(Animator animation)
                {
                    mybutton.setVisibility(View.INVISIBLE);
                    myprogress.setVisibility(View.VISIBLE);
                    ViewHelper.setScaleX(myprogress, 0f);

                    animate(myprogress).scaleX(1).setListener(new AnimatorListenerAdapter()
                    {
                        @SuppressWarnings("ConstantConditions")
                        @Override
                        public void onAnimationEnd(Animator animation)
                        {
                            mybutton.getHandler().postDelayed(new Runnable()
                            {
                                @Override
                                public void run()
                                {
                                    animate(myprogress).scaleX(0).setListener(new AnimatorListenerAdapter()
                                    {
                                        @Override
                                        public void onAnimationEnd(Animator animation)
                                        {
                                            myprogress.setVisibility(View.INVISIBLE);
                                            mybutton.setVisibility(View.VISIBLE);

                                            ViewHelper.setScaleX(mybutton, 0);
                                            animate(mybutton).scaleX(1);
                                        }
                                    });
                                }
                            }, 1000);
                        }
                    });
                }
            });
        }
    });
}

}

レイアウトはシンプルです:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:paddingBottom="@dimen/activity_vertical_margin"
>

    <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:id="@+id/mycontainer">


            <Button
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:text="string text"
                    android:id="@+id/mybutton"
                    />

            <ProgressBar
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_alignLeft="@id/mybutton"
                    android:layout_alignTop="@id/mybutton"
                    android:layout_alignRight="@id/mybutton"
                    android:layout_alignBottom="@id/mybutton"
                    android:visibility="invisible"
                    android:id="@+id/myprogress"
                    />


            </RelativeLayout>
</RelativeLayout>

どこが間違っていたのですか?

4

1 に答える 1

21

狂気!調査の結果、ソースが見つかりました。animate() メソッドが呼び出されると、ViewPropertyAnimator が作成され、マップに保存されます。animate() を使用してこのビューを再度アニメーション化しようとすると、マップから既に作成されている viewproperyanimator が取得され、新しいアニメーション パラメータを設定する前であってもすぐに onAnimationStart() が呼び出されます。初アニメ化(後編)開始。これは無限ループを作成します。

それを停止するには、ビューを 2 回目にアニメーション化しようとするときに古いリスナーをクリアする必要があるため

animate(mybutton).setListener(null).scaleX(1);

無限ループを停止します。ドキュメンテーションは、間違いなくそれについて警告する必要があります!

于 2014-04-08T10:42:55.527 に答える