如何禁用CoordinatorLayout中的AppBarLayout滚动?

时间:2015-12-05 17:25:22

标签: android android-coordinatorlayout android-collapsingtoolbarlayout supportmapfragment android-appbarlayout

MapFragment内有AppBarLayout视差效果:

enter image description here

我想禁用AppBarLayout上的滚动,因为无法在地图上移动,因为地图上的触摸偶数总是作为滚动事件处理。 我想通过仅在屏幕底部滚动AppBarLayout来处理RecyclerView的折叠。

这是我的xml:

<?xml version="1.0" encoding="utf-8"?>

<android.support.design.widget.AppBarLayout
    android:id="@+id/appbar"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">

    <android.support.design.widget.CollapsingToolbarLayout
        android:id="@+id/collapsing_toolbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:fitsSystemWindows="true"
        app:contentScrim="@color/white"
        app:expandedTitleMarginEnd="64dp"
        app:expandedTitleMarginStart="48dp"
        app:layout_scrollFlags="scroll|exitUntilCollapsed|snap"
        app:titleEnabled="false">

        <FrameLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:layout_collapseMode="parallax"
            android:fitsSystemWindows="true">

            <fragment
                android:id="@+id/map"
                android:name="com.androidmapsextensions.SupportMapFragment"
                android:layout_width="match_parent"
                android:layout_height="match_parent" />
        </FrameLayout>

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="73dp"
            app:contentInsetLeft="0dp"
            app:contentInsetStart="0dp"
            app:layout_collapseMode="pin"
            app:popupTheme="@style/ThemeOverlay.AppCompat.Light">

            <include
                android:id="@+id/search_bar"
                layout="@layout/layout_searchbar" />
        </android.support.v7.widget.Toolbar>

        <View
            android:id="@+id/toolbar_shadow"
            android:layout_width="match_parent"
            android:layout_height="3dp"
            android:layout_below="@id/search_bar"
            android:layout_gravity="bottom"
            android:background="@drawable/toolbar_dropshadow"
            android:visibility="gone" />
    </android.support.design.widget.CollapsingToolbarLayout>
</android.support.design.widget.AppBarLayout>


<RecyclerView
    android:id="@+id/farm_list"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/transparent"
    app:layout_behavior="@string/appbar_scrolling_view_behavior" />

感谢您的回复。

5 个答案:

答案 0 :(得分:49)

我不确定我是否得到了它,但我认为您正在寻找DragCallback

DragCallback界面允许选择是否应通过滚动到AppBarLayout来控制兄弟滚动视图。

您可以通过以下方式定义:

CoordinatorLayout.LayoutParams params = (CoordinatorLayout.LayoutParams) appBarLayout.getLayoutParams();
AppBarLayout.Behavior behavior = (AppBarLayout.Behavior) params.getBehavior();
behavior.setDragCallback(new AppBarLayout.Behavior.DragCallback() {
    @Override
    public boolean canDrag(@NonNull AppBarLayout appBarLayout) {
        return false;
    }
});

通过始终返回false,您的滚动视图将不再受ABL控制。

注意:在调用之前,您应该检查ViewCompat.isLaidOut(appBarLayout),否则params.getBehavior()将返回null。

答案 1 :(得分:27)

<强>问题

  1. 即使滚动内容适合屏幕,AppBarLayout也会滚动。
  2. 这是因为默认情况下我们可以通过触摸&amp;来拖动AppBarLayout。拖动AppBarLayout。
  3. <强>解决方案

    1. 我们将停用&#34;拖动&#34; AppBarLayout的行为。

      // Disable "Drag" for AppBarLayout (i.e. User can't scroll appBarLayout by directly touching appBarLayout - User can only scroll appBarLayout by only using scrollContent)
      if (appBarLayout.getLayoutParams() != null) {
          CoordinatorLayout.LayoutParams layoutParams = (CoordinatorLayout.LayoutParams) appBarLayout.getLayoutParams();
          AppBarLayout.Behavior appBarLayoutBehaviour = new AppBarLayout.Behavior();
          appBarLayoutBehaviour.setDragCallback(new AppBarLayout.Behavior.DragCallback() {
              @Override
              public boolean canDrag(@NonNull AppBarLayout appBarLayout) {
                  return false;
              }
          });
          layoutParams.setBehavior(appBarLayoutBehaviour);
      }
      
    2. <强>参考

      1. 这只是&#34; natario&#34;的扩展。上面接受了答案。

      2. https://developer.android.com/reference/android/support/design/widget/AppBarLayout.Behavior.DragCallback.html

答案 2 :(得分:13)

您可以通过在xml中定义自定义app:layout_behavior来实现此目的。使用这种方法,您不必担心获取对LayoutParams的引用并进行空检查。

  <android.support.design.widget.AppBarLayout
      android:id="@+id/app_bar_layout"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      app:layout_behavior="com.yourcompany.FixedAppBarLayoutBehavior"
      >

然后创建一个从AppBarLayout.Behavior扩展的自定义类。

public class FixedAppBarLayoutBehavior extends AppBarLayout.Behavior {

  public FixedAppBarLayoutBehavior(Context context, AttributeSet attrs) {
    super(context, attrs);

    setDragCallback(new DragCallback() {
      @Override public boolean canDrag(@NonNull AppBarLayout appBarLayout) {
        return false;
      }
    });
  }
}

更新了Kotlin版本:

class FixedAppBarLayoutBehavior(context: Context, attrs: AttributeSet) : AppBarLayout.Behavior(context, attrs) {
  init {
    setDragCallback(object : DragCallback() {
      override fun canDrag(appBarLayout: AppBarLayout): Boolean = false
    })
  }
}

答案 3 :(得分:5)

经过两个小时的尝试后,我找到了一个解决方案,这非常简单。我只需要扩展CoordinatorLayout并覆盖OnInterceptTouchEvent方法,以便类看起来像这样:

public class NonTouchableCoordinatorLayout extends CoordinatorLayout {
public NonTouchableCoordinatorLayout(Context context, AttributeSet attrs) {
    super(context, attrs);
}

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
    return false;
}

}

答案 4 :(得分:0)

您可以使用NestedScrollView.setNestedScrollingEnabled(false)禁用AppBarLayout nestScroll envent