我正在创建一个列表,当我进入列表底部时,它将获得更多项目。我也希望此列表也可刷新,因此我需要一个SwipeRefreshLayout
。
假设我有一个像这样的适配器:
class OrdersListAdapter(val selected : (Order) -> Unit) : RecyclerView.Adapter<RecyclerView.ViewHolder>() {
fun addOrders(orders: ArrayList<Order>) {
this.orders.addAll(orders)
notifyItemRangeInserted(this.orders.size - orders.size + FIRST_ITEM_INDEX, orders.size)
}
fun setFirstOrders(orders: ArrayList<Order>) {
this.orders = orders
notifyDataSetChanged()
}
override fun getItemCount(): Int {
return (orders.size + 1 + FIRST_ITEM_INDEX)
}
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
return when (viewType) {
TYPE_ORDERS -> OrdersViewHolder(DataBindingUtil.inflate<OrdersItemBinding>(LayoutInflater.from(parent.context),
R.layout.orders_item, parent, false).apply {
viewModel = OrdersViewModel()
})
else -> LoadingMoreOrdersViewHolder(DataBindingUtil.inflate<LoadingMoreOrdersItemBinding>(LayoutInflater.from(parent.context),
R.layout.loading_more_orders_item, parent, false).apply {
viewModel = LoadingMoreOrdersViewModel()
})
}
}
override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {
when (getItemViewType(position)) {
TYPE_ORDERS -> (holder as? OrdersViewHolder)?.setData(orders[position - FIRST_ITEM_INDEX])
TYPE_LOADING -> (holder as? LoadingMoreOrdersViewHolder)?.setState(loading, error, noMoreItems)
}
}
override fun getItemViewType(position: Int): Int =
if (position == FIRST_ITEM_INDEX - 1) TYPE_HEADER else if (position == itemCount - 1) TYPE_LOADING else TYPE_ORDERS
好吧,如果我将这样的RecyclerView
放在SwipeRefreshLayout
内:
<android.support.v4.widget.SwipeRefreshLayout
android:id="@+id/refresh_layout"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_marginTop="8dp"
android:layout_marginBottom="8dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/toolbar">
<android.support.v7.widget.RecyclerView
android:id="@+id/orders_rv"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:layoutManager="android.support.v7.widget.LinearLayoutManager" />
</android.support.v4.widget.SwipeRefreshLayout>
这将显示所有项目,但是refreshLayout将不起作用:如果我拉开视图,则什么也不会发生。
但是,如果我没有在适配器内设置任何加载布局(列表底部的那个,以加载更多项,从R.layout.loading_more_orders_item
开始膨胀),则SwipeRefreshLayout将正常工作。
如何在适配器中具有2种视图类型,并在SwipeRefreshLayout中设置RecyclerView?
编辑:
这是我的onRefreshListener:
refreshLayout = binding.swipeRefreshLayout
refreshLayout?.setOnRefreshListener { reloadOrders() }
在reloadOrders()中:
private fun reloadOrders() {
viewModel.setLoading()
refreshLayout?.isRefreshing = false
Observable.timer(1, TimeUnit.SECONDS)
.subscribe { getOrders() }
}
答案 0 :(得分:1)
我相信发生的事情是recyclerview接管了swiperefreshlayout中的ontouch事件。您可以尝试覆盖linearlayout及其canScrollVertically()
方法以返回false。干杯!
答案 1 :(得分:0)
回答您的问题,兼容性不是您遇到的问题。不确定布局的大小,因为其高度为0dp。我会考虑视图大小和视图层次。
答案 2 :(得分:0)
好的,感谢@ r2rek,我找到了一个对我有用的解决方案:
在代码中,我们创建一个扩展LinearLayoutManager
并覆盖方法canScrollVertically()
的类,以始终返回false:
inner class MyLinearLayout(context: Context) : LinearLayoutManager(context) {
override fun canScrollVertically(): Boolean {
return (false)
}
}
然后,当我们初始化RecyclerView
时,我们将该类应用于layoutManager
:
ordersRv?.layoutManager = MyLinearLayout(this)
这将允许您刷新,但是您将无法滚动RecyclerView
(duh)。
要解决此问题,请在您的xml中添加一个NestedScrollView
,如下所示:
<android.support.v4.widget.SwipeRefreshLayout
android:id="@+id/refresh_layout"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_marginTop="8dp"
android:layout_marginBottom="8dp"
android:visibility="@{viewModel.ordersVisibility}"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/toolbar">
<android.support.v4.widget.NestedScrollView
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView
android:id="@+id/orders_rv"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</android.support.v4.widget.NestedScrollView>
</android.support.v4.widget.SwipeRefreshLayout>
您现在可以滚动RecyclerView
,然后拉动以刷新SwipeRefreshLayout
。
我认为这种方法有点怪异,因此,如果有人有更好的解决方案,请随时告诉我们!