滚动时 AppBarLayout 子项不会折叠
AppBarLayout child doesn't collapse when scrolling
我有一个 AppBarLayout
,它的子项中包含一个水平的 RecyclerView
。
AppBarLayout
的兄弟是 ViewPager
(其中包含垂直 RecyclerView
)。
这里是 XML:
<android.support.design.widget.CoordinatorLayout
android:id="@+id/coordinatorLayout"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".ui.search.SearchResultFragment">
<android.support.v4.view.ViewPager
android:id="@+id/view_pager"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:overScrollMode="never"
app:layout_behavior="@string/appbar_scrolling_view_behavior"/>
<android.support.design.widget.AppBarLayout
android:id="@+id/appBarLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:elevation="@dimen/d2"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
<LinearLayout
android:id="@+id/header_container"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:background="@color/white"
android:gravity="center_vertical"
android:orientation="horizontal">
<ImageView
android:id="@+id/back_image_view"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:scaleType="center"
android:src="@drawable/ic_backarrow_gray"/>
<android.support.v7.widget.RecyclerView
android:id="@+id/query_terms_recycler_view"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_gravity="center_vertical"
android:layout_weight="1"/>
<ImageView
android:id="@+id/search_image_view"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:scaleType="center"
android:src="@drawable/ic_search_gray"/>
</LinearLayout>
<android.support.v7.widget.RecyclerView
android:id="@+id/related_terms_recycler_view"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="@color/search_term_bg"
app:layout_scrollFlags="scroll|enterAlways"/>
<android.support.design.widget.TabLayout
android:id="@+id/tab_layout"
android:layout_width="match_parent"
android:layout_height="@dimen/d8"
android:background="@color/tab_and_nav_bar"/>
</android.support.design.widget.AppBarLayout>
</android.support.design.widget.CoordinatorLayout>
我已经有另一个 UI 和 AppBarLayout
,它能够用 Toolbar
成功响应滚动。代码几乎相同,但在这种情况下出现了问题,因为 RecyclerView
没有折叠/隐藏。
AppBarLayout
只是一个垂直的LinearLayout
,所以顺序很重要;
- 让 child 对滚动手势做出反应的原因 实际上是通过
android:layout_scrollFlags
属性定义 scroll
标志。
从那个标志的official documentation,我们可以读到
The view will be scroll in direct relation to scroll events. This flag
needs to be set for any of the other flags to take effect. If any
sibling views before this one do not have this flag, then this value
has no effect.
这意味着您的标志被忽略了,因为其他 child 没有任何标志。我相信这背后的原因是我们只想隐藏 top 视图 - 如果上面还有东西,底部视图就不会消失。
所以在这种情况下(您可能不喜欢)的解决方案是:
- 将该视图移到顶部,这样它将是唯一退出的视图;
为所有视图提供 scroll
标记。他们会一起对滚动手势做出反应,这可能不是您想要的。
<AppBarLayout>
<RecyclerView ...
android:layout_scrollFlags="scroll|enterAlways"/>
<LinearLayout ... >
<TabLayout ... >
</AppBarLayout>
非常感谢@m vai 的提示
我通过自定义 AppBarLayout
的行为解决了这个问题
这是代码:
private void setAppBarLayoutBehaviour() {
AppBarLayout.Behavior behavior = new AppBarLayout.Behavior() {
@Override
public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View directTargetChild, View target, int nestedScrollAxes) {
// Trigger the following events if it is a vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL || super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
// If I slowly reach the top, without fling, show the RecyclerView
int[] firstVisiblePositions = ((StaggeredGridLayoutManager) ((RecyclerView) target).getLayoutManager()).findFirstCompletelyVisibleItemPositions(null);
for (int position : firstVisiblePositions) {
if (position == 0) {
showRelatedTerms();
break;
}
}
}
@Override
public boolean onNestedFling(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, float velocityX, float velocityY, boolean consumed) {
if (velocityY > 500) {
// Hide the RecyclerView
} else if (velocityY < -500) {
// Show the recyclerView
}
return true;
}
};
((CoordinatorLayout.LayoutParams) appBarLayout.getLayoutParams()).setBehavior(behavior);
}
而且我没有在 XML 中应用任何 scrollFlags
。
我有一个 AppBarLayout
,它的子项中包含一个水平的 RecyclerView
。
AppBarLayout
的兄弟是 ViewPager
(其中包含垂直 RecyclerView
)。
这里是 XML:
<android.support.design.widget.CoordinatorLayout
android:id="@+id/coordinatorLayout"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".ui.search.SearchResultFragment">
<android.support.v4.view.ViewPager
android:id="@+id/view_pager"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:overScrollMode="never"
app:layout_behavior="@string/appbar_scrolling_view_behavior"/>
<android.support.design.widget.AppBarLayout
android:id="@+id/appBarLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:elevation="@dimen/d2"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
<LinearLayout
android:id="@+id/header_container"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:background="@color/white"
android:gravity="center_vertical"
android:orientation="horizontal">
<ImageView
android:id="@+id/back_image_view"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:scaleType="center"
android:src="@drawable/ic_backarrow_gray"/>
<android.support.v7.widget.RecyclerView
android:id="@+id/query_terms_recycler_view"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_gravity="center_vertical"
android:layout_weight="1"/>
<ImageView
android:id="@+id/search_image_view"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:scaleType="center"
android:src="@drawable/ic_search_gray"/>
</LinearLayout>
<android.support.v7.widget.RecyclerView
android:id="@+id/related_terms_recycler_view"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="@color/search_term_bg"
app:layout_scrollFlags="scroll|enterAlways"/>
<android.support.design.widget.TabLayout
android:id="@+id/tab_layout"
android:layout_width="match_parent"
android:layout_height="@dimen/d8"
android:background="@color/tab_and_nav_bar"/>
</android.support.design.widget.AppBarLayout>
</android.support.design.widget.CoordinatorLayout>
我已经有另一个 UI 和 AppBarLayout
,它能够用 Toolbar
成功响应滚动。代码几乎相同,但在这种情况下出现了问题,因为 RecyclerView
没有折叠/隐藏。
AppBarLayout
只是一个垂直的LinearLayout
,所以顺序很重要;- 让 child 对滚动手势做出反应的原因 实际上是通过
android:layout_scrollFlags
属性定义scroll
标志。
从那个标志的official documentation,我们可以读到
The view will be scroll in direct relation to scroll events. This flag needs to be set for any of the other flags to take effect. If any sibling views before this one do not have this flag, then this value has no effect.
这意味着您的标志被忽略了,因为其他 child 没有任何标志。我相信这背后的原因是我们只想隐藏 top 视图 - 如果上面还有东西,底部视图就不会消失。
所以在这种情况下(您可能不喜欢)的解决方案是:
- 将该视图移到顶部,这样它将是唯一退出的视图;
为所有视图提供
scroll
标记。他们会一起对滚动手势做出反应,这可能不是您想要的。<AppBarLayout> <RecyclerView ... android:layout_scrollFlags="scroll|enterAlways"/> <LinearLayout ... > <TabLayout ... > </AppBarLayout>
非常感谢@m vai 的提示
我通过自定义 AppBarLayout
的行为解决了这个问题
这是代码:
private void setAppBarLayoutBehaviour() {
AppBarLayout.Behavior behavior = new AppBarLayout.Behavior() {
@Override
public boolean onStartNestedScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View directTargetChild, View target, int nestedScrollAxes) {
// Trigger the following events if it is a vertical scrolling
return nestedScrollAxes == ViewCompat.SCROLL_AXIS_VERTICAL || super.onStartNestedScroll(coordinatorLayout, child, directTargetChild, target, nestedScrollAxes);
}
@Override
public void onNestedScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, int dxConsumed, int dyConsumed, int dxUnconsumed, int dyUnconsumed) {
super.onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed, dxUnconsumed, dyUnconsumed);
// If I slowly reach the top, without fling, show the RecyclerView
int[] firstVisiblePositions = ((StaggeredGridLayoutManager) ((RecyclerView) target).getLayoutManager()).findFirstCompletelyVisibleItemPositions(null);
for (int position : firstVisiblePositions) {
if (position == 0) {
showRelatedTerms();
break;
}
}
}
@Override
public boolean onNestedFling(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, float velocityX, float velocityY, boolean consumed) {
if (velocityY > 500) {
// Hide the RecyclerView
} else if (velocityY < -500) {
// Show the recyclerView
}
return true;
}
};
((CoordinatorLayout.LayoutParams) appBarLayout.getLayoutParams()).setBehavior(behavior);
}
而且我没有在 XML 中应用任何 scrollFlags
。