Snackbar与CoordinatorLayout禁用消失功能

10

我正在使用以下支持:

  • FloatingActionButton(浮动操作按钮)
  • Snackbar(消息提示框)
  • CoordinatorLayout(协调布局)

我需要CoordinatorLayout,这样如果SnackBar被显示,FloatingActionButton会向上移动以为SnackBar腾出空间。为了更好地理解,请查看此视频

我正在使用SnackBar来实现双击返回退出应用程序的功能,但是SnackBar可以被取消。

有没有一种方法可以禁用SnackBar上的取消功能?

Snackbar snackbar = Snackbar.make(view, R.string.press_back_again_to_exit, Snackbar.LENGTH_SHORT);
snackbar.setAction(R.string.ok, new View.OnClickListener() {
    @Override
    public void onClick(View v)
    {

    }
});
snackbar.setActionTextColor(getResources().getColor(R.color.white));

View view = snackbar.getView();
view.setBackgroundColor(getResources().getColor(R.color.orange_warning));

snackbar.show();

布局

<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">

    <android.support.v4.widget.DrawerLayout
        android:id="@+id/drawer_layout"
        xmlns:sothree="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@color/white"
        android:fitsSystemWindows="true">

        <com.sothree.slidinguppanel.SlidingUpPanelLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:gravity="bottom"
            sothree:umanoFadeColor="@android:color/transparent"
            sothree:umanoPanelHeight="100dp"
            sothree:umanoShadowHeight="4dp">

            <!-- Toolbar and main content -->
            <LinearLayout
                android:id="@+id/toolbar_and_content"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:orientation="vertical">

                <include layout="@layout/toolbar"/>

                <!-- Your content layout -->
                <FrameLayout
                    android:id="@+id/content_frame"
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:orientation="vertical"/>

            </LinearLayout>

            <!-- Sliding up panel layout -->
            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="@color/darker_grey"
                android:orientation="vertical">

                ...

            </LinearLayout>

        </com.sothree.slidinguppanel.SlidingUpPanelLayout>


        <!-- Navigation drawer -->
        <ExpandableListView
            android:id="@+id/lv_left_drawer"
            android:layout_width="280dp"
            android:layout_height="match_parent"
            android:layout_gravity="start"
            android:background="@color/white"
            android:childDivider="@android:color/transparent"
            android:clickable="true"
            android:divider="@color/divider_color"
            android:dividerHeight="0.6dp"
            android:fadeScrollbars="true"
            android:groupIndicator="@null"
            android:listSelector="@drawable/button_drawer_child_selector"
            android:scrollbarSize="0dp"/>

    </android.support.v4.widget.DrawerLayout>

    <android.support.design.widget.FloatingActionButton
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="right|bottom"
        android:layout_marginBottom="@dimen/floating_action_button_margin"
        android:layout_marginRight="@dimen/floating_action_button_margin"
        android:src="@drawable/ic_add"
        android:visibility="invisible"
        app:backgroundTint="@color/orange"
        app:borderWidth="0dp"
        app:elevation="6dp"
        app:fabSize="normal"/>

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

附言:

我知道 这个 GitHub 库 有这种功能,但是否存在“本地”实现的方法呢?


我认为没有办法。 - Archit Jain
我正在使用SnackBar来实现双击返回退出应用程序的功能 - 你正在违反平台的UI模式。SnackBar仅用于提供小型反馈,而不是用于要求用户执行操作(例如当用户按两次返回键时,我猜测你正在尝试这样做)。 - user
@Luksprog 我知道,使用Toast或Crouton也可以实现同样的效果,但本质上是一样的,SnackBar只是看起来更好看。我个人不太喜欢双击返回退出应用程序的功能,但不幸的是这不是由我决定的。 - Marko
请检查我的答案......https://dev59.com/L-k6XIcBkEYKwwoYEPzw#35576829 - Melbourne Lopes
5个回答

11

Snackbar现在通过使用setBehavior方法实际支持禁用滑动取消功能。这里的好处是以前总会丢失一些行为,而现在这些行为都得以保留。您还需要使用Snackbar.LENGTH_INDEFINITE

请注意,包已经移动,因此您必须导入“snackbar包中的”新版Snackbar

Snackbar.make(view, stringId, Snackbar.LENGTH_INDEFINITE)
    .setBehavior(new NoSwipeBehavior())
    .show();

class NoSwipeBehavior extends BaseTransientBottomBar.Behavior {

    @Override
    public boolean canSwipeDismissView(View child) {
      return false;
    }
}

8
您可以修改 Snackbar 显示的持续时间。这将类似于禁用解除。
int     LENGTH_INDEFINITE   Show the Snackbar indefinitely. 

请查看文档

如果它不起作用 那么只有一种方法,实现自定义Snackbar并重写dismiss()方法,然后什么也不做。:) 因为dismiss()是一个公共API。


1
无效,SnackBar仍然可以被解除,而且永远停留在那里 :) - Marko
3
“dismissable or stay forever”可以翻译为“可解除或永久保存”。这似乎有些矛盾,你能再解释一下吗? - theJango
2
如果我将LENGTH_INDEFINITE设置为SnackBar,那么SnackBar不会在一段时间后消失...(这很有道理),但我仍然可以将其解除(向右滑动即可解除)。 - Marko
@Marko:请检查已编辑的答案。我知道这是一条漫长的路,但它是可行的。 - theJango
@theJango 这并不是...虽然Snackbar不是一个final类,但它的唯一构造函数是private,实际上使它成为了final - Ace

8

仅使用 LENGTH_INDEFINITE 的回答是不够的。

当 Snackbar 不是 CoordinatorLayout子元素时,它才是不可关闭的。当它是一个子元素时,你仍然可以滑动关闭 Snackbar。

在这种情况下,你可以监听关闭滑动事件并重新显示 Snackbar。

public void showAnnoyingSnackBar(View root, String text) {
    Snackbar.make(root, text, Snackbar.LENGTH_INDEFINITE)
        .setCallback(new Snackbar.Callback() {
            @Override public void onDismissed(Snackbar snackbar, int event) {
                // recursively call this method again when the snackbar was dismissed through a swipe
                if (event == DISMISS_EVENT_SWIPE) showAnnoyingSnackBar(root, text);
            }
        })
        .show();
}

6
我注意到当调用show()方法时,SnackbarLayout尚未完全初始化。为了锁定dismiss,我们必须在SnackbarLayout初始化之后清除behavior。最好的地方是在SnackbarLayout的OnPreDrawListener中执行此操作。
final Snackbar snack = Snackbar.make(getView(), "I can't be dissmiss", Snackbar.LENGTH_INDEFINITE);
    snack.show();
    snack.getView().getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
        @Override
        public boolean onPreDraw() {
            snack.getView().getViewTreeObserver().removeOnPreDrawListener(this);
                ((CoordinatorLayout.LayoutParams) snack.getView().getLayoutParams()).setBehavior(null);
                return true;
            }
        });

1
我成功地使用以下方法禁用了滑动侧向关闭小吃店(在调用 snackbar.show() 后)的功能。
((android.support.design.widget.CoordinatorLayout.LayoutParams) snackbar.getView().getLayoutParams()).setBehavior(null);

太棒了,我得试一下。谢谢你的回答! - Marko
1
这对我不起作用。Snackbar snackbar = Snackbar.make(this.rootView, "Blah", Snackbar.LENGTH_SHORT); snackbar.show();((CoordinatorLayout.LayoutParams) snackbar.getView().getLayoutParams()).setBehavior(null); - Jared Burrows

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