【问题标题】:Cannot start this animator on a detached view! reveal effect无法在分离视图上启动此动画师!显露效果
【发布时间】:2015-01-05 07:38:16
【问题描述】:

我正在尝试在我的应用程序中创建显示效果,但没有成功。我想要的是当我打开一个片段时显示一个卡片视图。到目前为止我尝试的是:

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>

和日志猫:

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。但我不知道这是否是真正的问题。谢谢

【问题讨论】:

  • 你找到解决方案了吗?
  • 不!我试了各种方法都没有成功!

标签: java android android-animation


【解决方案1】:

这就是我解决它的方法,我在 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;
    }

编辑:正如一些人所说,最好在onViewCreated 中调用toggleInformationView 并避免使用View.OnLayoutChangeListener

【讨论】:

  • 嗯,它有效!但我添加了一个按钮,然后 onClick 我开始了过渡。但现在它完美运行!还有一个问题..是否可以决定动画时间?现在对我来说太快了..
  • 你可以修改你的reveal Animator对象并把reveal.setDuration(500);或任何其他持续时间(以毫秒为单位)
  • 也为我工作,谢谢@ferdy182!修复这个 bug 让我想起了 Android 操作系统有多糟糕
  • 它可以,但是API级别低于Lollipop呢?
  • @Anggrian 需要另一种方法,例如这个库 github.com/ozodrukh/CircularReveal
【解决方案2】:

你可以使用 runnable 来做动画。 runnable 将在视图创建后运行。

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

【讨论】:

  • 这种方法更简洁,我更喜欢它而不是接受的方法,尽管它们都正确并提供相同的解决方案。
  • 这比公认的要干净得多。它对我有用,谢谢。
  • 这行得通,但我认为使用view.addOnAttachStateChangeListener() 更清楚
  • 这适用于几乎所有设备,但当我在 onCreate() 中调用 view.post() 时,我的应用程序仍然会在一些设备上崩溃
【解决方案3】:

Hitesh's 答案的小补充。 最好用 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);

【讨论】:

    【解决方案4】:

    简单地把方法放在onViewCreated()方法中怎么样:

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

    有效

    【讨论】:

      【解决方案5】:

      在问题范围之外进行了简化

      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) {
      
                  }
              });
          }
      }
      

      为一个或多个动画调用 startAnimation

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

      【讨论】:

        【解决方案6】:

        啊 !!!,所以最后这里是这个错误的解决方案。实际上,问题在于我们将视图分配给动画的高度和宽度,突然间我们没有注意到可能存在这样一种情况,即当我们将其分配给动画时,视图不会正确加载。因此,为此,我们必须等到视图正确加载。

        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.
                    }
                });
        

        【讨论】:

          【解决方案7】:

          尝试在onViewCreated 方法上调用它

          【讨论】:

          • 所以,不是在 onCreatedView 中,而是在 onViewCreated 中?
          【解决方案8】:

          我使用的是自定义视图,所以我使用isAttachedToWindow() 方法来检查是否附加了视图,并避免在未附加视图时进行显示。

          【讨论】:

            【解决方案9】:

            使用 ViewTreeObserver。

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

            【讨论】:

              【解决方案10】:

              使用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);
                                      }
                                  });
                              }
                          }
                      }
              

              【讨论】:

                【解决方案11】:

                您的视图似乎尚未附加。 view.post() 进入主线程队列并在其他待处理任务完成后执行,这意味着在您的视图被附加之后,这对您有用:

                view.post(new Runnable() {
                        @Override
                        public void run() {
                            //do what you want
                        }
                    });
                

                【讨论】:

                  猜你喜欢
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 2014-09-03
                  • 2017-12-28
                  • 1970-01-01
                  • 1970-01-01
                  相关资源
                  最近更新 更多