Snackbar и другие анимации перестали работать на некоторых устройствах Android

у меня очень странная проблема, которую я не могу понять. До недавнего времени я не был проблемой, но я не могу вернуться назад, чтобы предотвратить это. Также другая странная вещь-она работает на некоторых устройствах, а другие-нет.

проблема в анимации. Один из них-снэк-бар. Закусочная должна анимироваться вверх и вниз, но это не так. он просто показывает, а затем скрывает. проверьте видео ниже, чтобы увидеть проблему.

видео из выпуска

вот код Android чтобы оживить snackbar в

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

его использование ViewCompat для библиотеки v4. У меня есть другие анимации, работающие в другой деятельности, работающей нормально. Также вопрос заключается не только в одном действии его всех. Это заставляет меня думать, что его система как-то широка. Но все они используют разные внутренние темы, но все расширяют тему.Совместимости приложений.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>

устройства он работает on

  • Nexus 9 (Зефир)
  • Nexus 4 (KitKat)
  • Galaxy S7 (Зефир)

устройства не работают

  • Droid Turbo 2 (Зефир)
  • Galaxy S7 (Marshmallow) *мое устройство работает, мои тестеры не
  • Nexus 6p (Android N)

другие проблемы анимации связаны с переключателями. У меня 2 в одном и одна заикается при переключении, а другой просто переключается без анимации.

у меня также есть LayoutTransition для моего AppBarLayout для анимации скрытия / показа моего TabLayout, и он отлично работает и все устройства

3 ответов


Я нашел причину, почему это происходит, но не так, как исправить.

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

который вызывается классом Snackbar и является false на рабочих устройствах и true на устройствах, не работающих. Кто-нибудь знает об этом?

поэтому после того, как я отключил lastpass в моих системных настройках, доступность snackbar теперь анимируется так, как должна. Это безумие, как это работает. Nova launcher имеет тот же эффект. Я предполагаю, что любая служба в доступности, которая включена, вызовет snackbar анимация не работает.


как упоминал Бигнадад, проблема в том, что любая функция доступности, включая такие вещи, как менеджеры паролей, отключает анимацию snackbar. Google, начиная с этого редактирования,исправил это для AndroidX, но не библиотека поддержки дизайна

потому что базовый класс Snackbar, BaseTransientBottomBar, обрабатывает анимацию с помощью пакета private, final methods, у вас есть два варианта, если вы хотите это исправить: сверните свою собственную snackbar с нуля или используйте более хакерский решение с отражением:

Котлин пример:

// 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 не выйдет из бета-версии с правильным исправлением.


Это было исправлено, так как материальные компоненты для Android 1.0.0-alpha3 с этот коммит.

используйте его вместо библиотеки дизайна (что является способом пойти, если вы используете AndroidX):

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