页面包含一个带有LayoutTransition(或animateLayoutChanges="true")的ViewGroup,这会干扰滚动动画。

20

我使用ViewPager2与碎片,注意到在像小米、裕龙、华硕、vivo等运行Android 8或9的设备上出现两个相关的IllegalStateExceptions(我自己无法复制):

Fatal Exception: java.lang.IllegalStateException: Page can only be offset by a positive amount, not by -758
       at androidx.viewpager2.widget.ScrollEventAdapter.updateScrollEventValues(ScrollEventAdapter.java:280)
       at androidx.viewpager2.widget.ScrollEventAdapter.onScrolled(ScrollEventAdapter.java:178)
       at androidx.recyclerview.widget.RecyclerView.dispatchOnScrolled(RecyclerView.java:5173)
       at androidx.recyclerview.widget.RecyclerView$ViewFlinger.run(RecyclerView.java:5338)
       at android.view.Choreographer$CallbackRecord.run(Choreographer.java:1029)
       at android.view.Choreographer.doCallbacks(Choreographer.java:834)
       at android.view.Choreographer.doFrame(Choreographer.java:760)
       at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:1015)
       at android.os.Handler.handleCallback(Handler.java:873)
       at android.os.Handler.dispatchMessage(Handler.java:99)
       at android.os.Looper.loop(Looper.java:224)
       at android.app.ActivityThread.main(ActivityThread.java:7083)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:537)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:928)

Fatal Exception: java.lang.IllegalStateException: Page(s) contain a ViewGroup with a LayoutTransition (or animateLayoutChanges="true"), which interferes with the scrolling animation. Make sure to call getLayoutTransition().setAnimateParentHierarchy(false) on all ViewGroups with a LayoutTransition before an animation is started.
       at androidx.viewpager2.widget.ScrollEventAdapter.updateScrollEventValues(ScrollEventAdapter.java:272)
       at androidx.viewpager2.widget.ScrollEventAdapter.onScrolled(ScrollEventAdapter.java:178)
       at androidx.recyclerview.widget.RecyclerView.dispatchOnScrolled(RecyclerView.java:5173)
       at androidx.recyclerview.widget.RecyclerView$ViewFlinger.run(RecyclerView.java:5338)
       at android.view.Choreographer$CallbackRecord.run(Choreographer.java:1029)
       at android.view.Choreographer.doCallbacks(Choreographer.java:841)
       at android.view.Choreographer.doFrame(Choreographer.java:769)
       at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:1015)
       at android.os.Handler.handleCallback(Handler.java:794)
       at android.os.Handler.dispatchMessage(Handler.java:99)
       at android.os.Looper.loop(Looper.java:176)
       at android.app.ActivityThread.main(ActivityThread.java:6651)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:547)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:824)

在浏览时,我经过了以下线程:https://issuetracker.google.com/issues/129530305,根据我的理解,似乎总结为在viewPager2的任何父布局上将animateLayoutChanges设置为false,这就是我所做的。不幸的是,那并没有解决我的问题。然后我发现了另一个线程:java.lang.IllegalStateException: Page can only be offset by a positive amount,它似乎没有太大帮助。有什么想法可能会导致这个问题?


你看过源代码的245和254行了吗?源代码链接 - grolschie
4个回答

12
您需要在每个页面/片段上执行此操作。
 View view = layoutInflater.inflate(R.layout.page, parent, false);
 ViewGroup viewGroup = view.findViewById(R.id.animated_viewgroup);
 viewGroup.getLayoutTransition().setAnimateParentHierarchy(false);

在官方文档https://developer.android.com/reference/androidx/viewpager2/widget/ViewPager2中提到:

如果您的页面包含LayoutTransitions,则这些LayoutTransitions必须将animateParentHierarchy设置为false。请注意,如果您在布局xml文件中有一个animateLayoutChanges =“true”的ViewGroup,则会自动向该ViewGroup添加LayoutTransition。您需要在填充xml布局后手动调用getLayoutTransition()。setAnimateParentHierarchy(false)。

这与您错误日志的第一行所说的相同。

Fatal Exception: java.lang.IllegalStateException: Page(s) contain a ViewGroup with a LayoutTransition (or animateLayoutChanges="true"), which interferes with the scrolling animation. Make sure to call getLayoutTransition().setAnimateParentHierarchy(false) on all ViewGroups with a LayoutTransition before an animation is started.

2
问题是,我在我的片段中找不到任何具有LayoutTransition的ViewGroup。我也循环遍历了子视图,并检查viewGroup.getLayoutTransition(),但它们似乎都为null。因此,我不确定代码在哪里触发异常。 - waseefakhtar
我启动了上述提到的线程。我在Android Vitals中偶尔看到第一个异常,但没有看到第二个异常。我没有使用任何LayoutTransitions,但我正在使用viewpager2.setPageTransformer - grolschie
1
我只遇到过这个异常几次。设备包括:Motorola Moto G4 Play(Android 6.0,en_GB)、Google Pixel 3(Android 9,en_GB)和华为Mate 9(Android 7.0,fr_FR)。似乎没有任何规律可循。这可能是一个时间问题,自动化测试快速操作UI导致的。 - grolschie
1
一样的问题。我曾经试图通过循环遍历所有的ViewGroup并在setAnimateParentHierarchy为true时将其设置为false,以解决每当ViewGroup具有LayoutTransition时可能出现的问题,但是崩溃报告似乎只是不断增加。现在我唯一的选择就是回退到ViewPager。 - waseefakhtar
如果已经明确设置了这个值,请确认它必须设置为OFFSCREEN_PAGE_LIMIT_DEFAULT,就像setOffscreenPageLimit(OFFSCREEN_PAGE_LIMIT_DEFAULT)一样。 - VVB
显示剩余2条评论

4

我遇到的另一个问题是,即使没有出现上述异常,我的viewpager2在转换开始时也会开始闪烁/重新加载页面。 我对这些示例以及如何将setAnimateParentHiearchy设置为false有些困惑,因此我想再添加一个示例来解决这个问题:

ViewPager布局

<layout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">

    <androidx.constraintlayout.widget.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <!-- Toolbar -->
        <include
            android:id="@+id/headline"
            layout="@layout/registration_toolbar"
            app:isBold="@{false}"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toTopOf="parent"
            app:outsideToolbarTitle="@{@string/fragment_user_data_toolbar_title}" />

        <com.google.android.material.tabs.TabLayout
            android:id="@+id/tl_user_data"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/headline" />

        <!-- This Viewpager contains a fragment, which has android:animateLayoutChanges="true" -->
        <androidx.viewpager2.widget.ViewPager2
            android:id="@+id/vp_user_data"
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_marginTop="8dp"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/tl_user_data" />

    </androidx.constraintlayout.widget.ConstraintLayout>
</layout>

片段布局(将位于viewpager内部)

   <androidx.constraintlayout.widget.ConstraintLayout
        android:id="@+id/outerContraintLayout"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:animateLayoutChanges="true" > <-- THIS WILL CAUSE PROBLEMS

       <!-- Deleted unnecessary parts -->


    </androidx.constraintlayout.widget.ConstraintLayout>
</layout>

如果要保留 android:animateLayoutChanges,则需要在片段的 onViewCreated() 方法内调用它。
class MyFragmentInsideViewPagerThatHasProblems : Fragment(R.layout.above_layout) {

     override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        val viewGroup = requireView().findViewById<ConstraintLayout>(R.id.outerContraintLayout) as ViewGroup
        viewGroup.layoutTransition.setAnimateParentHierarchy(false)
     }
}

2

当我使用viewPager2时,出现了第二个错误。 如上所述,在我的fragment布局中,我发现了一个带有animateLayoutChanges="true"属性的viewGroup,然后在OnViewCreated方法中调用了它。

viewGroup.getLayoutTransition().setAnimateParentHierarchy(false);

这解决了我的问题!


0
我之前尝试了一些答案,但都没有起作用,然后我找到了一个解决方案。需要停止平滑动画,我们可以使用TabLayoutMediator。
你需要将这个添加到你的viewpager2中。
 TabLayoutMediator(tabLayout,  binding.viewPager, false , false) { tab,position ->
        tab.text = tabTitle[position]
    }.attach()

如果你不想显示文本,不用担心,只需提交这段代码。 希望这能解决问题。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接