web-dev-qa-db-ja.com

切り離されたビューでこのアニメーターを起動できません!効果を明らかにする

私は自分のアプリケーションで公開効果を作成しようとしていますが、成功していません。フラグメントを開くと、カードビューが表示されます。私がこれまで試したのは:

_private void toggleInformationView(View view) {
        infoContainer = view.findViewById(R.id.contact_card);

        int cx = (view.getLeft() + view.getRight()) / 2;
        int cy = (view.getTop() + view.getBottom()) / 2;
        float radius = Math.max(infoContainer.getWidth(), infoContainer.getHeight()) * 2.0f;

        if (infoContainer.getVisibility() == View.INVISIBLE) {
            infoContainer.setVisibility(View.VISIBLE);
            ViewAnimationUtils.createCircularReveal(infoContainer, cx, cy, 0, radius).start();
        } else {
            Animator reveal = ViewAnimationUtils.createCircularReveal(
                    infoContainer, cx, cy, radius, 0);
            reveal.addListener(new AnimatorListenerAdapter() {
                @Override
                public void onAnimationEnd(Animator animation) {
                    infoContainer.setVisibility(View.INVISIBLE);
                }
            });
            reveal.start();
        }
    }
_

onCreateViewでメソッドを開始しました:

_toggleInformationView(view);
_

これがカードビューです:

_<Android.support.v7.widget.CardView
        xmlns:card_view="http://schemas.Android.com/apk/res-auto"
        Android:layout_width="match_parent"
        Android:id="@+id/contact_card"
        Android:visibility="invisible"
        Android:layout_height="wrap_content"
        Android:layout_margin="5dp"
        Android:elevation="4dp"
        Android:foreground="?android:attr/selectableItemBackground"
        Android:orientation="vertical"
        Android:padding="10dp"
        card_view:cardCornerRadius="2dp" >

        <LinearLayout   
            Android:orientation="vertical"
            Android:layout_width="match_parent"
            Android:layout_height="wrap_content"
            >
            <ImageView
                Android:id="@+id/bg_contact"
                Android:layout_width="match_parent"
                Android:layout_height="wrap_content"
                Android:layout_centerHorizontal="true"
                Android:layout_weight="0"
                Android:adjustViewBounds="true"
                Android:scaleType="centerCrop"
                Android:src="@drawable/banner" />

        </LinearLayout>
    </Android.support.v7.widget.CardView>
_

およびlogcat:

_11-08 17:25:48.541: E/AndroidRuntime(26925): Java.lang.IllegalStateException: Cannot start this animator on a detached view!
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.view.RenderNode.addAnimator(RenderNode.Java:817)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.view.RenderNodeAnimator.setTarget(RenderNodeAnimator.Java:277)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.view.RenderNodeAnimator.setTarget(RenderNodeAnimator.Java:261)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.animation.RevealAnimator.<init>(RevealAnimator.Java:37)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.view.ViewAnimationUtils.createCircularReveal(ViewAnimationUtils.Java:48)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at com.as.asapp.TFragment.toggleInformationView(TFragment.Java:64)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at com.as.asapp.TFragmentshowInformation(TFragment.Java:52)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at com.as.asapp.TFragment.onCreateView(TFragment.Java:37)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.Fragment.performCreateView(Fragment.Java:1786)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.Java:947)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.Java:1126)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.BackStackRecord.run(BackStackRecord.Java:739)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.Java:1489)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.support.v4.app.FragmentManagerImpl$1.run(FragmentManager.Java:454)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.os.Handler.handleCallback(Handler.Java:739)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.os.Handler.dispatchMessage(Handler.Java:95)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.os.Looper.loop(Looper.Java:135)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Android.app.ActivityThread.main(ActivityThread.Java:5221)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Java.lang.reflect.Method.invoke(Native Method)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at Java.lang.reflect.Method.invoke(Method.Java:372)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at com.Android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.Java:899)
11-08 17:25:48.541: E/AndroidRuntime(26925):    at com.Android.internal.os.ZygoteInit.main(ZygoteInit.Java:694)
_

確認すると、getWidth()getHeight()は0を返します。しかし、それが本当の問題なのかわかりません。ありがとう

63
End.Game

これが私がそれを解決した方法です.onCreateViewコールバックのビューにonLayoutChangeリスナーを追加したので、ビューに接続されて描画の準備ができたらいつでも、

@Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
    {
        // Inflate the layout for this fragment
        final View view = inflater.inflate(R.layout.fragment_map_list, container, false);
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.Lollipop) {
            view.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
                @TargetApi(Build.VERSION_CODES.Lollipop)
                @Override
                public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
                    v.removeOnLayoutChangeListener(this);
                    toggleInformationView(view);
                }
            });
        }
        return view;
    }
72

Runnableを使用して、アニメーションを実行できます。 runnableは、ビューの作成後に実行されます。

view.post(new Runnable() {
            @Override
            public void run() {
                //create your anim here
            }
        });
36
BennyKok

Hitesh's answerへの小さな追加。 https://developer.Android.com/reference/Android/support/v4/view/ViewCompat.html#isAttachedToWindow(Android.view.View) を使用することをお勧めします

Android.support.v4.view.ViewCompat

boolean isAttachedToWindow (View view)

例:

LinearLayout rootView = null;
rootView = (LinearLayout) findViewById(R.id.rootView);
boolean isAttachedToWindow = ViewCompat.isAttachedToWindow(rootView);
14
Alexander

メソッドをonViewCreated()メソッドに単純に配置する方法は次のとおりです。

@Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        toggleInformationView(view);
    }

できます

6
Ziv Kesten

質問の範囲外で簡素化

private void startAnimation(@NonNull final View view, @NonNull final Runnable onAttachedToWindow) {
    if (view.isAttachedToWindow()) {
        onAttachedToWindow.run();
    } else {
        view.addOnAttachStateChangeListener(new View.OnAttachStateChangeListener() {
            @Override
            public void onViewAttachedToWindow(View view) {
                onAttachedToWindow.run();
                view.removeOnAttachStateChangeListener(this)
            }

            @Override
            public void onViewDetachedFromWindow(View view) {

            }
        });
    }
}

1つまたは複数のアニメーションを念頭にstartAnimationを呼び出す

startAnimation(view, new Runnable() {
        @Override
        public void run() {
          //the code inside toggleInformationView(view)
        }
    });
2
Juan Mendez

ああ!!!だから、最後にこのエラーの解決策があります。実際、問題はビューをアニメーションにその高さと幅で割り当てていることであり、突然ビューにアニメーションを割り当てるときにそのビューが適切に読み込まれない場合があることに気付かなかったということです。そのため、ビューが適切にロードされるまで待つ必要があります。

LayoutInflater inflater = this.getLayoutInflater();
View rowView = inflater.inflate( R.layout.fileselecttype,null, true );

rowView   or [Your View]

[Your View].post(new Runnable() {
            @Override
            public void run() {
                //Do your animation work here.
            }
        });
2
Sohaib Aslam

カスタムビューを使用していたので、isAttachedToWindow()メソッドを使用して、ビューがアタッチされているかどうかを確認し、ビューがアタッチされていない間は公開しないようにしました。

1
Hitesh Sahu

onViewCreatedメソッドで呼び出してみてください

1

ViewTreeObserverを使用します。

 ViewTreeObserver viewTreeObserver = rootLayout.getViewTreeObserver();
    if (viewTreeObserver.isAlive()) {
        viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                revealActivity(revealX, revealY);
                rootLayout.getViewTreeObserver().removeOnGlobalLayoutListener(this);
            }
        });
    }
1
josedlujan

rootLayout.isAttachedToWindow()メソッドを使用して、レイアウトがウィンドウに接続されているかどうかを確認します。

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Lollipop) {
        if (rootLayout!= null) {

            if(rootLayout.isAttachedToWindow()) {
                ViewTreeObserver viewTreeObserver = drawerLayout.getViewTreeObserver();
                if (viewTreeObserver.isAlive()) {
                    viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
                        @Override
                        public void onGlobalLayout() {
                            startRevealAnimation(rootLayout);
                            rootLayout.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                        }
                    });
                }
            }
        }
0
Rehan Sarwar