SwipeRefreshLayout是否与具有多个viewType的适配器兼容?

时间:2019-06-11 15:23:51

标签: android kotlin android-recyclerview recycler-adapter swiperefreshlayout

我正在创建一个列表,当我进入列表底部时,它将获得更多项目。我也希望此列表也可刷新,因此我需要一个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() }
}

3 个答案:

答案 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

我认为这种方法有点怪异,因此,如果有人有更好的解决方案,请随时告诉我们!