Snackbar 和其他动画在某些 Android 设备上停止工作

Snackbar and other animations stopped working on some Android devices

我有一个非常奇怪的问题,我无法弄清楚。直到最近我才成为问题,但我似乎无法回过头来阻止它。另一件奇怪的事情是它在某些设备上有效,而在其他设备上则无效。

问题是动画。特别是小吃店。小吃店应该上下动画,但事实并非如此。它只是显示然后隐藏。查看下面的视频以查看问题。

Video of issue

这是 Android 代码,用于在

中为快餐栏设置动画
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
        ViewCompat.setTranslationY(mView, mView.getHeight());
        ViewCompat.animate(mView)
                .translationY(0f)
                .setInterpolator(FAST_OUT_SLOW_IN_INTERPOLATOR)
                .setDuration(ANIMATION_DURATION)
                .setListener(new ViewPropertyAnimatorListenerAdapter() {
                    @Override
                    public void onAnimationStart(View view) {
                        mView.animateChildrenIn(ANIMATION_DURATION - ANIMATION_FADE_DURATION,
                                ANIMATION_FADE_DURATION);
                    }

                    @Override
                    public void onAnimationEnd(View view) {
                        onViewShown();
                    }
                }).start();
    }

它为 v4 库使用 ViewCompat。我有其他动画在另一个 activity 中工作正常。此外,问题不只是一个 activity 而是所有问题。这让我觉得它在某种程度上是全系统的。但它们都使用不同的内部主题,但都扩展了 Theme.AppCompat.NoActionBar.

这是我的主要布局

<android.support.design.widget.AppBarLayout
    android:id="@+id/appbar"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
    app:elevation="4dp">

    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="?attr/actionBarSize"
        android:background="?attr/colorPrimary"
        app:titleTextAppearance="@style/ToolbarTitle"
        app:popupTheme="@style/ThemeOverlay.AppCompat.Light"
        app:layout_scrollFlags="scroll|enterAlways|snap"/>

    <android.support.design.widget.TabLayout
        android:id="@+id/tabs"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:tabTextAppearance="@style/TabText"
        app:tabMinWidth="@dimen/tab_minwidth"
        app:tabMode="fixed"
        app:tabGravity="fill"
        app:layout_scrollFlags="enterAlways"/>


</android.support.design.widget.AppBarLayout>

<FrameLayout
    android:id="@+id/content_frame"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    app:layout_behavior="@string/appbar_scrolling_view_behavior" >

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</FrameLayout>

<fr.castorflex.android.circularprogressbar.CircularProgressBar
    android:id="@+id/base_progressSpinner"
    android:layout_gravity="center"
    android:layout_width="48dp"
    android:layout_height="48dp"
    android:indeterminate="true"
    android:visibility="invisible"
    app:cpb_color="@color/spinner"
    app:cpb_rotation_speed="1.0"
    app:cpb_sweep_speed="1.0"
    app:cpb_stroke_width="4dp"
    app:cpb_min_sweep_angle="10"
    app:cpb_max_sweep_angle="300"/>

<android.support.design.widget.FloatingActionButton
    android:id="@+id/fab_upload"
    android:visibility="gone"
    android:layout_height="wrap_content"
    android:layout_width="wrap_content"
    app:layout_anchor="@id/content_frame"
    app:layout_anchorGravity="bottom|right|end"
    app:borderWidth="0dp"
    android:src="@drawable/app_fab_upload"
    android:layout_margin="@dimen/big_padding"
    android:clickable="true"
    app:backgroundTint="@color/fab_social"/>

<android.support.design.widget.FloatingActionButton
    android:id="@+id/fab_muzei"
    android:visibility="gone"
    android:layout_height="wrap_content"
    android:layout_width="wrap_content"
    app:layout_anchor="@id/content_frame"
    app:layout_anchorGravity="bottom|right|end"
    app:borderWidth="0dp"
    android:src="@drawable/app_fab_muzei"
    android:layout_margin="@dimen/big_padding"
    android:clickable="true"
    app:backgroundTint="@color/fab_social"/>

 </android.support.design.widget.CoordinatorLayout>

支持的设备

不工作的设备

其他动画问题与开关有关。我有 2 个相同的布局,一个在切换时断断续续,另一个只是切换而没有动画。

我还为我的 AppBarLayout 设置了一个 LayoutTransition 来为我的 TabLayout hiding/showing 设置动画,它工作正常并且所有设备

我找到了发生这种情况的原因,但还没有解决方法。

/**
 * Returns true if we should animate the Snackbar view in/out.
 */
private boolean shouldAnimate() {
    return !mAccessibilityManager.isEnabled();
}

由 Snackbar class 调用,在工作设备上为 false,在不工作的设备上为 true。有人知道吗?

因此,在我在系统设置中禁用 lastpass 后,可访问性快餐栏现在可以正常显示动画。这太疯狂了。 Nova 发射器具有相同的效果。我猜任何启用辅助功能的服务都会导致快餐栏动画不工作。

正如 Bignadad 所提到的,问题在于任何可访问性功能(包括密码管理器之类的功能)都会禁用快餐栏动画。 Google,截至此次编辑,has fixed this for AndroidX but not the Design support library

因为 Snackbar 的基础 class,BaseTransientBottomBar,处理动画,包私有的,final 方法,如果你想修复它有两个选择:从头开始滚动你自己的 snackbar,或者使用更 hacky反射解决方案:

Kotlin 示例:

// Only force when necessary, and don't animate when TalkBack or similar services are enabled
val shouldForceAnimate = !accessibilityManager.isEnabled && accessibilityManager.getEnabledAccessibilityServiceList(AccessibilityServiceInfo.FEEDBACK_SPOKEN).isEmpty()

Snackbar.make(coordinatorLayout, text, duration).apply {
    if (shouldForceAnimate) {
        try {
            val accManagerField = BaseTransientBottomBar::class.java.getDeclaredField("mAccessibilityManager")
            accManagerField.isAccessible = true
            val accManager = accManagerField.get(this)
            AccessibilityManager::class.java.getDeclaredField("mIsEnabled").apply {
                isAccessible = true
                setBoolean(accManager, false)
            }
            accManagerField.set(this, accManager)
        } catch (e: Exception) {
            Log.d("Snackbar", "Reflection error: $e")
        }
    }
}.show()

Java 例子:

// Only force when necessary, and don't animate when TalkBack or similar services are enabled
boolean shouldForceAnimate = !accessibilityManager.isEnabled() && accessibilityManager.getEnabledAccessibilityServiceList(AccessibilityServiceInfo.FEEDBACK_SPOKEN).size == 0;

Snackbar snackbar = Snackbar.make(coordinatorLayout, text, duration);
if(shouldForceAnimate){
    try {
        Field accManagerField = BaseTransientBottomBar.class.getDeclaredField("mAccessibilityManager");
        accManagerField.setAccessible(true);
        AccessibilityManager accManager = (AccessibilityManager) accManagerField.get(snackbar);
        Field isEnabledField = AccessibilityManager.class.getDeclaredField("mIsEnabled");
        isEnabledField.setAccessible(true);
        isEnabledField.setBoolean(accManager, false);
        accManagerField.set(snackbar, accManager);
    } catch (Exception e) {
        Log.d("Snackbar", "Reflection error: " + e.toString());
    }
}
snackbar.show();

我喜欢这里的第三个选项,但我不知道有一个,至少在 AndroidX 通过适当的修复退出测试版之前是这样。

自 Material Android 1.0.0-alpha3 的组件 this commit

使用它而不是设计库(如果您使用 AndroidX,这是可行的方法):

implementation "com.google.android.material:material:$material_components_version"