Snackbar 和其他动画在某些 Android 设备上停止工作
Snackbar and other animations stopped working on some Android devices
我有一个非常奇怪的问题,我无法弄清楚。直到最近我才成为问题,但我似乎无法回过头来阻止它。另一件奇怪的事情是它在某些设备上有效,而在其他设备上则无效。
问题是动画。特别是小吃店。小吃店应该上下动画,但事实并非如此。它只是显示然后隐藏。查看下面的视频以查看问题。
这是 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>
支持的设备
- Nexus 9(棉花糖)
- Nexus 4(奇巧)
- 银河 S7(棉花糖)
不工作的设备
- Droid Turbo 2(棉花糖)
- Galaxy S7 (Marshmallow) *我的设备能用,我的测试人员不能用
- Nexus 6p(AndroidN)
其他动画问题与开关有关。我有 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"
我有一个非常奇怪的问题,我无法弄清楚。直到最近我才成为问题,但我似乎无法回过头来阻止它。另一件奇怪的事情是它在某些设备上有效,而在其他设备上则无效。
问题是动画。特别是小吃店。小吃店应该上下动画,但事实并非如此。它只是显示然后隐藏。查看下面的视频以查看问题。
这是 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>
支持的设备
- Nexus 9(棉花糖)
- Nexus 4(奇巧)
- 银河 S7(棉花糖)
不工作的设备
- Droid Turbo 2(棉花糖)
- Galaxy S7 (Marshmallow) *我的设备能用,我的测试人员不能用
- Nexus 6p(AndroidN)
其他动画问题与开关有关。我有 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"