Android 设计支持库的菜单和自动隐藏 FloatingActionButton
Menu and Autohide FloatingActionButton of Android Design Support Library
我正在使用 Android 设计支持库,我想要一个具有滚动自动隐藏功能的 FloatingActionButton,
我的布局是:
<android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ScrollView
android:id="@+id/scrollView"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/LargeText" />
</ScrollView>
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:layout_anchorGravity="bottom|right|end"
app:layout_anchor="@id/scrollView"
android:src="@drawable/abc_btn_rating_star_off_mtrl_alpha" />
</android.support.design.widget.CoordinatorLayout>
FloatingActionButton 在滚动文本时总是显示,我想在滚动文本时自动隐藏它。
而且,我想通过单击 FloatingActionButton 来获得一个 FloatingActionButton 菜单,如下所示:
FloatingActionButton that have AutoHide by Scrolling,
您必须使用 android.support.v4.widget.NestedScrollView
而不是 ScrollView
。您不能使用滚动视图。您必须使用 NestedScrollView 或实现 NestedScrollingChild 接口的视图,例如 RecyclerView。
要实现这种模式,您必须实现您的自定义 Behavior
。 Googler 发布了一个很好的代码,它在用户向下滚动时隐藏 FAB,并在用户向上滚动时显示它。重用 FloatingActionButton.Behavior 用于 hiding/showing FAB 的相同动画以响应 AppBarLayout exiting/entering。
2015 年 7 月 18 日更新
在 22.2.1 中,您可以使用预先构建的动画简单地添加下面发布的代码。
只需像这样使用 class:(原始来源 here)
public class ScrollAwareFABBehavior extends FloatingActionButton.Behavior {
public ScrollAwareFABBehavior(Context context, AttributeSet attrs) {
super();
}
@Override
public boolean onStartNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View directTargetChild, final View target, final int nestedScrollAxes) {
// Ensure we react to vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL
|| super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View target, final int dxConsumed, final int dyConsumed,
final int dxUnconsumed, final int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
if (dyConsumed > 0 && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
child.hide();
} else if (dyConsumed < 0 && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
child.show();
}
}
}
然后您可以使用以下方法将此行为应用于您的 FAB:
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
app:layout_behavior="com.support.android.designlibdemo.ScrollAwareFABBehavior" />
使用 设计 22.2.0:
您必须像这样使用 class:(原始来源 here)
public class ScrollAwareFABBehavior extends FloatingActionButton.Behavior {
private static final Interpolator INTERPOLATOR = new FastOutSlowInInterpolator();
private boolean mIsAnimatingOut = false;
public ScrollAwareFABBehavior(Context context, AttributeSet attrs) {
super();
}
@Override
public boolean onStartNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View directTargetChild, final View target, final int nestedScrollAxes) {
// Ensure we react to vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL
|| super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View target, final int dxConsumed, final int dyConsumed,
final int dxUnconsumed, final int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
if (dyConsumed > 0 && !this.mIsAnimatingOut && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
animateOut(child);
} else if (dyConsumed < 0 && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
animateIn(child);
}
}
// Same animation that FloatingActionButton.Behavior uses to hide the FAB when the AppBarLayout exits
private void animateOut(final FloatingActionButton button) {
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).scaleX(0.0F).scaleY(0.0F).alpha(0.0F).setInterpolator(INTERPOLATOR).withLayer()
.setListener(new ViewPropertyAnimatorListener() {
public void onAnimationStart(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = true;
}
public void onAnimationCancel(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
}
public void onAnimationEnd(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
view.setVisibility(View.GONE);
}
}).start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.fab_out);
anim.setInterpolator(INTERPOLATOR);
anim.setDuration(200L);
anim.setAnimationListener(new Animation.AnimationListener() {
public void onAnimationStart(Animation animation) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = true;
}
public void onAnimationEnd(Animation animation) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
button.setVisibility(View.GONE);
}
@Override
public void onAnimationRepeat(final Animation animation) {
}
});
button.startAnimation(anim);
}
}
// Same animation that FloatingActionButton.Behavior uses to show the FAB when the AppBarLayout enters
private void animateIn(FloatingActionButton button) {
button.setVisibility(View.VISIBLE);
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).scaleX(1.0F).scaleY(1.0F).alpha(1.0F)
.setInterpolator(INTERPOLATOR).withLayer().setListener(null)
.start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.fab_in);
anim.setDuration(200L);
anim.setInterpolator(INTERPOLATOR);
button.startAnimation(anim);
}
}
}
然后您可以使用以下方法将此行为应用于您的 FAB:
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
app:layout_behavior="com.support.android.designlibdemo.ScrollAwareFABBehavior" />
当然您可以更改此代码以获得您喜欢的图案。
And, I want to have a FloatingActionButton Menu by clicking on FloatingActionButton, like this:
目前原FAB不支持该模式。您必须实现自定义代码才能实现它。
你能做到
这是我的代码。
第 1 步:
首先制作 Getter FloatingActionMenu 以便您可以从另一个 activity 或从使用您的 recycleview 的片段调用它
public FloatingActionMenu getFloatingActionMenu() {
return fabMenu;
}
第 2 步:
从另一个 activity 或片段
调用下面的行
FloatingActionMenu fabMenu=((MainActivity)getActivity()).getFloatingActionMenu();
第 3 步:
现在检查 recycleview 是否滚动取决于 "dy" 位置
这里我使用了动画 fabMenu
Animation FabMenu_fadOut = AnimationUtils.loadAnimation(getActivity(),
R.anim.fade_out);
Animation FabMenu_fadIn = AnimationUtils.loadAnimation(getActivity(),
R.anim.abc_grow_fade_in);
recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (dy > 0 && floatingActionButton.isShown()) {
//fabMenu.startAnimation(FabMenu_fadIn);
fabMenu.setVisibility(View.GONE);
}
if (dy < 0 && !floatingActionButton.isShown()) {
// fabMenu.startAnimation(FabMenu_fadOut);
fabMenu.setVisibility(View.VISIBLE);
}
}
});
注意:如果您想在滚动时隐藏 FloatingActionButton,请使用与 FloatingActionMenu 中相同的代码。
谢谢。
您可以使用 this 实现浮动操作菜单。
关于浮动操作菜单的动画,您可以创建一个名为 ScrollAwareFloatingActionMenuBehaviour 的class:
public class ScrollAwareFloatingActionMenuBehaviour extends CoordinatorLayout.Behavior<FloatingActionsMenu> {
private static final Interpolator INTERPOLATOR = new FastOutSlowInInterpolator();
private boolean mIsAnimatingOut = false;
private boolean mIsAnimatingIn = false;
public ScrollAwareFloatingActionMenuBehaviour(Context context, AttributeSet attrs) {
}
@Override
public boolean layoutDependsOn(CoordinatorLayout parent, FloatingActionsMenu child, View dependency) {
return dependency instanceof Snackbar.SnackbarLayout;
}
@Override
public boolean onDependentViewChanged(CoordinatorLayout parent, FloatingActionsMenu child, View dependency) {
float translationY = Math.min(0, dependency.getTranslationY() - dependency.getHeight());
child.setTranslationY(translationY);
return true;
}
@Override
public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout,
FloatingActionsMenu child, View directTargetChild, View target, int nestedScrollAxes) {
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL ||
super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target,
nestedScrollAxes);
}
@Override
public void onNestedScroll(CoordinatorLayout coordinatorLayout, FloatingActionsMenu child,
View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed,
dyUnconsumed);
if (dyConsumed > 10 && !this.mIsAnimatingOut && !this.mIsAnimatingIn && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
animateOut(child);
} else if (dyConsumed < -10 && !this.mIsAnimatingOut && !this.mIsAnimatingIn && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
animateIn(child);
}
}
// Same animation that FloatingActionButton.Behavior uses to hide the FAB when the AppBarLayout exits
private void animateOut(final FloatingActionsMenu button) {
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).translationYBy(200F).alpha(0.0F).setInterpolator(INTERPOLATOR).withLayer()
.setListener(new ViewPropertyAnimatorListener() {
public void onAnimationStart(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = true;
}
public void onAnimationCancel(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
}
public void onAnimationEnd(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
view.setVisibility(View.GONE);
}
}).start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.design_fab_out);
anim.setInterpolator(INTERPOLATOR);
anim.setDuration(200L);
anim.setAnimationListener(new Animation.AnimationListener() {
public void onAnimationStart(Animation animation) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = true;
}
public void onAnimationEnd(Animation animation) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
button.setVisibility(View.GONE);
}
@Override
public void onAnimationRepeat(final Animation animation) {
}
});
button.startAnimation(anim);
}
}
// Same animation that FloatingActionButton.Behavior uses to show the FAB when the AppBarLayout enters
private void animateIn(FloatingActionsMenu button) {
button.setVisibility(View.VISIBLE);
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).translationYBy(-200F).alpha(1.0F)
.setInterpolator(INTERPOLATOR).withLayer().setListener(new ViewPropertyAnimatorListener() {
@Override
public void onAnimationStart(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = true;
}
@Override
public void onAnimationEnd(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = false;
}
@Override
public void onAnimationCancel(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = false;
}
})
.start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), android.support.design.R.anim.design_fab_in);
anim.setDuration(200L);
anim.setInterpolator(INTERPOLATOR);
button.startAnimation(anim);
}
}
}
请注意,R.anim.fab_in 和 R.anim.fab_out 分别被 R.anim.design_fab_in 和 R.anim.design_fab_out 替换。
在xml中使用它:
<com.getbase.floatingactionbutton.FloatingActionsMenu
xmlns:fab="http://schemas.android.com/apk/res-auto"
android:id="@+id/fab_menu"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="end|bottom"
android:layout_margin="@dimen/fab_margin"
fab:fab_addButtonColorNormal="@color/colorAccent"
fab:fab_addButtonColorPressed="@color/colorAccentLight"
fab:fab_addButtonPlusIconColor="@android:color/white"
app:layout_behavior="com.example.widgets.behaviour.ScrollAwareFloatingActionMenuBehaviour"
fab:fab_labelStyle="@style/menu_labels_style"
fab:fab_labelsPosition="left">
<com.getbase.floatingactionbutton.FloatingActionButton
android:id="@+id/fab_share"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
fab:fab_colorNormal="@color/fab_normal_blue"
fab:fab_colorPressed="@color/fab_normal_blue_pressed"
fab:fab_icon="@drawable/ic_social_share"
fab:fab_title="@string/fab_menu_group_chat" />
</com.getbase.floatingactionbutton.FloatingActionsMenu>
我正在使用 Android 设计支持库,我想要一个具有滚动自动隐藏功能的 FloatingActionButton,
我的布局是:
<android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ScrollView
android:id="@+id/scrollView"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/LargeText" />
</ScrollView>
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:layout_anchorGravity="bottom|right|end"
app:layout_anchor="@id/scrollView"
android:src="@drawable/abc_btn_rating_star_off_mtrl_alpha" />
</android.support.design.widget.CoordinatorLayout>
FloatingActionButton 在滚动文本时总是显示,我想在滚动文本时自动隐藏它。
而且,我想通过单击 FloatingActionButton 来获得一个 FloatingActionButton 菜单,如下所示:
FloatingActionButton that have AutoHide by Scrolling,
您必须使用 android.support.v4.widget.NestedScrollView
而不是 ScrollView
。您不能使用滚动视图。您必须使用 NestedScrollView 或实现 NestedScrollingChild 接口的视图,例如 RecyclerView。
要实现这种模式,您必须实现您的自定义 Behavior
。 Googler 发布了一个很好的代码,它在用户向下滚动时隐藏 FAB,并在用户向上滚动时显示它。重用 FloatingActionButton.Behavior 用于 hiding/showing FAB 的相同动画以响应 AppBarLayout exiting/entering。
2015 年 7 月 18 日更新
在 22.2.1 中,您可以使用预先构建的动画简单地添加下面发布的代码。 只需像这样使用 class:(原始来源 here)
public class ScrollAwareFABBehavior extends FloatingActionButton.Behavior {
public ScrollAwareFABBehavior(Context context, AttributeSet attrs) {
super();
}
@Override
public boolean onStartNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View directTargetChild, final View target, final int nestedScrollAxes) {
// Ensure we react to vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL
|| super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View target, final int dxConsumed, final int dyConsumed,
final int dxUnconsumed, final int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
if (dyConsumed > 0 && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
child.hide();
} else if (dyConsumed < 0 && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
child.show();
}
}
}
然后您可以使用以下方法将此行为应用于您的 FAB:
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
app:layout_behavior="com.support.android.designlibdemo.ScrollAwareFABBehavior" />
使用 设计 22.2.0: 您必须像这样使用 class:(原始来源 here)
public class ScrollAwareFABBehavior extends FloatingActionButton.Behavior {
private static final Interpolator INTERPOLATOR = new FastOutSlowInInterpolator();
private boolean mIsAnimatingOut = false;
public ScrollAwareFABBehavior(Context context, AttributeSet attrs) {
super();
}
@Override
public boolean onStartNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View directTargetChild, final View target, final int nestedScrollAxes) {
// Ensure we react to vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL
|| super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(final CoordinatorLayout coordinatorLayout, final FloatingActionButton child,
final View target, final int dxConsumed, final int dyConsumed,
final int dxUnconsumed, final int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
if (dyConsumed > 0 && !this.mIsAnimatingOut && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
animateOut(child);
} else if (dyConsumed < 0 && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
animateIn(child);
}
}
// Same animation that FloatingActionButton.Behavior uses to hide the FAB when the AppBarLayout exits
private void animateOut(final FloatingActionButton button) {
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).scaleX(0.0F).scaleY(0.0F).alpha(0.0F).setInterpolator(INTERPOLATOR).withLayer()
.setListener(new ViewPropertyAnimatorListener() {
public void onAnimationStart(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = true;
}
public void onAnimationCancel(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
}
public void onAnimationEnd(View view) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
view.setVisibility(View.GONE);
}
}).start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.fab_out);
anim.setInterpolator(INTERPOLATOR);
anim.setDuration(200L);
anim.setAnimationListener(new Animation.AnimationListener() {
public void onAnimationStart(Animation animation) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = true;
}
public void onAnimationEnd(Animation animation) {
ScrollAwareFABBehavior.this.mIsAnimatingOut = false;
button.setVisibility(View.GONE);
}
@Override
public void onAnimationRepeat(final Animation animation) {
}
});
button.startAnimation(anim);
}
}
// Same animation that FloatingActionButton.Behavior uses to show the FAB when the AppBarLayout enters
private void animateIn(FloatingActionButton button) {
button.setVisibility(View.VISIBLE);
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).scaleX(1.0F).scaleY(1.0F).alpha(1.0F)
.setInterpolator(INTERPOLATOR).withLayer().setListener(null)
.start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.fab_in);
anim.setDuration(200L);
anim.setInterpolator(INTERPOLATOR);
button.startAnimation(anim);
}
}
}
然后您可以使用以下方法将此行为应用于您的 FAB:
<android.support.design.widget.FloatingActionButton
android:id="@+id/fab"
app:layout_behavior="com.support.android.designlibdemo.ScrollAwareFABBehavior" />
当然您可以更改此代码以获得您喜欢的图案。
And, I want to have a FloatingActionButton Menu by clicking on FloatingActionButton, like this:
目前原FAB不支持该模式。您必须实现自定义代码才能实现它。
你能做到
这是我的代码。
第 1 步:
首先制作 Getter FloatingActionMenu 以便您可以从另一个 activity 或从使用您的 recycleview 的片段调用它
public FloatingActionMenu getFloatingActionMenu() {
return fabMenu;
}
第 2 步:
从另一个 activity 或片段
调用下面的行 FloatingActionMenu fabMenu=((MainActivity)getActivity()).getFloatingActionMenu();
第 3 步:
现在检查 recycleview 是否滚动取决于 "dy" 位置 这里我使用了动画 fabMenu
Animation FabMenu_fadOut = AnimationUtils.loadAnimation(getActivity(),
R.anim.fade_out);
Animation FabMenu_fadIn = AnimationUtils.loadAnimation(getActivity(),
R.anim.abc_grow_fade_in);
recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (dy > 0 && floatingActionButton.isShown()) {
//fabMenu.startAnimation(FabMenu_fadIn);
fabMenu.setVisibility(View.GONE);
}
if (dy < 0 && !floatingActionButton.isShown()) {
// fabMenu.startAnimation(FabMenu_fadOut);
fabMenu.setVisibility(View.VISIBLE);
}
}
});
注意:如果您想在滚动时隐藏 FloatingActionButton,请使用与 FloatingActionMenu 中相同的代码。
谢谢。
您可以使用 this 实现浮动操作菜单。
关于浮动操作菜单的动画,您可以创建一个名为 ScrollAwareFloatingActionMenuBehaviour 的class:
public class ScrollAwareFloatingActionMenuBehaviour extends CoordinatorLayout.Behavior<FloatingActionsMenu> {
private static final Interpolator INTERPOLATOR = new FastOutSlowInInterpolator();
private boolean mIsAnimatingOut = false;
private boolean mIsAnimatingIn = false;
public ScrollAwareFloatingActionMenuBehaviour(Context context, AttributeSet attrs) {
}
@Override
public boolean layoutDependsOn(CoordinatorLayout parent, FloatingActionsMenu child, View dependency) {
return dependency instanceof Snackbar.SnackbarLayout;
}
@Override
public boolean onDependentViewChanged(CoordinatorLayout parent, FloatingActionsMenu child, View dependency) {
float translationY = Math.min(0, dependency.getTranslationY() - dependency.getHeight());
child.setTranslationY(translationY);
return true;
}
@Override
public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout,
FloatingActionsMenu child, View directTargetChild, View target, int nestedScrollAxes) {
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL ||
super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target,
nestedScrollAxes);
}
@Override
public void onNestedScroll(CoordinatorLayout coordinatorLayout, FloatingActionsMenu child,
View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed,
dyUnconsumed);
if (dyConsumed > 10 && !this.mIsAnimatingOut && !this.mIsAnimatingIn && child.getVisibility() == View.VISIBLE) {
// User scrolled down and the FAB is currently visible -> hide the FAB
animateOut(child);
} else if (dyConsumed < -10 && !this.mIsAnimatingOut && !this.mIsAnimatingIn && child.getVisibility() != View.VISIBLE) {
// User scrolled up and the FAB is currently not visible -> show the FAB
animateIn(child);
}
}
// Same animation that FloatingActionButton.Behavior uses to hide the FAB when the AppBarLayout exits
private void animateOut(final FloatingActionsMenu button) {
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).translationYBy(200F).alpha(0.0F).setInterpolator(INTERPOLATOR).withLayer()
.setListener(new ViewPropertyAnimatorListener() {
public void onAnimationStart(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = true;
}
public void onAnimationCancel(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
}
public void onAnimationEnd(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
view.setVisibility(View.GONE);
}
}).start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), R.anim.design_fab_out);
anim.setInterpolator(INTERPOLATOR);
anim.setDuration(200L);
anim.setAnimationListener(new Animation.AnimationListener() {
public void onAnimationStart(Animation animation) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = true;
}
public void onAnimationEnd(Animation animation) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingOut = false;
button.setVisibility(View.GONE);
}
@Override
public void onAnimationRepeat(final Animation animation) {
}
});
button.startAnimation(anim);
}
}
// Same animation that FloatingActionButton.Behavior uses to show the FAB when the AppBarLayout enters
private void animateIn(FloatingActionsMenu button) {
button.setVisibility(View.VISIBLE);
if (Build.VERSION.SDK_INT >= 14) {
ViewCompat.animate(button).translationYBy(-200F).alpha(1.0F)
.setInterpolator(INTERPOLATOR).withLayer().setListener(new ViewPropertyAnimatorListener() {
@Override
public void onAnimationStart(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = true;
}
@Override
public void onAnimationEnd(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = false;
}
@Override
public void onAnimationCancel(View view) {
ScrollAwareFloatingActionMenuBehaviour.this.mIsAnimatingIn = false;
}
})
.start();
} else {
Animation anim = AnimationUtils.loadAnimation(button.getContext(), android.support.design.R.anim.design_fab_in);
anim.setDuration(200L);
anim.setInterpolator(INTERPOLATOR);
button.startAnimation(anim);
}
}
}
请注意,R.anim.fab_in 和 R.anim.fab_out 分别被 R.anim.design_fab_in 和 R.anim.design_fab_out 替换。
在xml中使用它:
<com.getbase.floatingactionbutton.FloatingActionsMenu
xmlns:fab="http://schemas.android.com/apk/res-auto"
android:id="@+id/fab_menu"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="end|bottom"
android:layout_margin="@dimen/fab_margin"
fab:fab_addButtonColorNormal="@color/colorAccent"
fab:fab_addButtonColorPressed="@color/colorAccentLight"
fab:fab_addButtonPlusIconColor="@android:color/white"
app:layout_behavior="com.example.widgets.behaviour.ScrollAwareFloatingActionMenuBehaviour"
fab:fab_labelStyle="@style/menu_labels_style"
fab:fab_labelsPosition="left">
<com.getbase.floatingactionbutton.FloatingActionButton
android:id="@+id/fab_share"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
fab:fab_colorNormal="@color/fab_normal_blue"
fab:fab_colorPressed="@color/fab_normal_blue_pressed"
fab:fab_icon="@drawable/ic_social_share"
fab:fab_title="@string/fab_menu_group_chat" />
</com.getbase.floatingactionbutton.FloatingActionsMenu>