28

当新项目添加到列表中时,我希望我的 RecyclerView 滚动到底部。下面是我的代码:

RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(getActivity());
recyclerView.setLayoutManager(layoutManager);

adapter = new RecyclerViewAdapter(data, recyclerView);
recyclerView.setAdapter(adapter);

当我单击 下方的按钮时RecyclerView,数据将添加到列表中:

public void addNewCard() {
    data.add("New");
    adapter.notifyItemInserted(data.size() - 1);
    recyclerView.scrollToPosition(data.size() - 1);
}

每当我添加一个项目时,它总是会向上滚动到最高位置。我尝试使用smoothScrollToPosition()并尝试scrollToPosition()使用LinearLayoutManager。我什至尝试使用ViewTreeObserver. 也尝试更改RecyclerView依赖项的版本。我在 Stack Overflow 上进行了彻底的搜索,没有一个解决方案适合我。

关于可能是什么问题的任何想法?我在RecyclerView片段内使用。这可能是问题的原因吗?

我的 .xml 文件RecyclerView

<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.NestedScrollView 
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/scroll_view_recycler"
android:layout_width="match_parent"
android:layout_height="match_parent">

<RelativeLayout
    android:id="@+id/rel_layout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/listView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="6dp" />

    <Button
        android:id="@+id/button_add_new_card"
        style="@style/Widget.AppCompat.Button.Borderless"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/listView"
        android:layout_marginBottom="16dp"
        android:layout_marginTop="8dp"
        android:padding="20dp"
        android:text="+ Add new Item" />
</RelativeLayout>
</android.support.v4.widget.NestedScrollView>
4

8 回答 8

25

如果要将数据添加到列表底部,则需要setStackFromEnd()在 RecyclerView 布局管理器中使用。

但首先,您需要修复您的适配器。您不得将 RecylerView 传递给您的适配器。所以下面的代码是错误的:

...
// This is wrong!!
adapter = new RecyclerViewAdapter(data, recyclerView);

recyclerView.setAdapter(adapter);

您需要将 Adapter 构造函数更改为仅接收数据作为其参数。像这样的东西:

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder> {

  private List<Data> mData;

  public RecyclerViewAdapter(List<Data> data) {
    this.mData = data;
  }

  ...
}

然后您可以使用以下代码将数据设置为始终添加到最后一个底部:

LinearLayoutManager linearLayoutManager = new LinearLayoutManager(this);
linearLayoutManager.setStackFromEnd(true);
recyclerView.setLayoutManager(linearLayoutManager);

adapter = new RecyclerViewAdapter(data);
recyclerView.setAdapter(adapter);

要添加新数据,最好在适配器中创建一个新方法。像这样的东西:

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder> {

  private List<Data> mData;
  ...

  public void addItem(Data datum) {
    mData.add(datum);
    notifyItemInserted(mData.size());
  }
}

每当您添加新数据时,您都需要使用方法滚动到底部scrollToPosition。像这样的东西:

adapter.addItem(newData);
recyclerView.scrollToPosition(adapter.getItemCount() - 1);

请记住,您需要getItemCount()在适配器中覆盖。它应该是这样的:

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder> {

  private List<Data> mData;
  public RecyclerViewAdapter(List<Data> data) {
    this.mData = data;
  }

  // Return the total count of items
  @Override
  public int getItemCount() {
    return mData.size();
  }

  ...
}

请注意,我在这里使用 Data pojo 作为示例。您需要根据您的数据类型进行更改。

于 2017-11-13T05:33:35.947 回答
9
adapter.registerAdapterDataObserver(object : RecyclerView.AdapterDataObserver() {
    override fun onItemRangeInserted(positionStart: Int, itemCount: Int) {
        super.onItemRangeInserted(positionStart, itemCount)

        val msgCount = adapter.getItemCount()
        val lastVisiblePosition = 
            linearLayoutManager.findLastCompletelyVisibleItemPosition()

        if (lastVisiblePosition == -1 || positionStart >= msgCount - 1 && 
            lastVisiblePosition == positionStart - 1) {
            recyclerView.scrollToPosition(positionStart)
        } else {
            recyclerView.scrollToPosition(adapter.getItemCount() - 1);
        }
    }
})
于 2018-10-11T07:50:11.640 回答
7

这是我想出的解决方案,对于基础来说已经足够好了,并且不需要任何自定义适配器。

myAdapter.registerAdapterDataObserver(object :
        RecyclerView.AdapterDataObserver() {
        override fun onItemRangeInserted(positionStart: Int, itemCount: Int) {
            myRecyclerView.scrollToPosition(positionStart)
        }
    })
于 2020-06-12T20:17:16.320 回答
3

也许这会帮助某人

@Override
public void onChildAdded(@NonNull DataSnapshot dataSnapshot, @Nullable String s) {
    if (pingAdapter != null) {

        Ping ping = dataSnapshot.getValue(Ping.class);
        pingAdapter.pings.add(ping);
        recyclerView.smoothScrollToPosition(pingAdapter.getItemCount());

        pingAdapter.notifyDataSetChanged();
    }
}
recyclerView.smoothScrollToPosition(pingAdapter.pings.size());

甚至更快

于 2019-04-20T19:19:41.517 回答
1

ישו אוהב אותך 的解决方案适用于其他项目。但是设置recyclerView.scrollToPosition()对我不起作用。这就是最终完成的工作:

public void addNewCard() {
    data.add("New");
    adapter.notifyItemInserted(data.size());

    scrollView.post(new Runnable() {
        @Override
        public void run() {
            scrollView.scrollTo(0,buttonAddNewItem.getTop());
        }
    });
}
于 2017-11-13T09:24:42.210 回答
0

这对我有用。

adapter.registerAdapterDataObserver(object : RecyclerView.AdapterDataObserver() {
        override fun onChanged() {
            super.onChanged()
            if (canShowLive()) wvl_log_rv.scrollToPosition(adapter.itemCount - 1)
        }
 })

经过测试和验证

于 2020-10-02T15:13:52.707 回答
0

或者,如果您将 ListAdapter 与 DiffUtill 一起使用并通过 提交新列表submitList(),则可以使用 ListAdapter 的onCurrentListChanged. 例如:

override fun onCurrentListChanged(previousList: MutableList<Item>, currentList: MutableList<Item>) {
    super.onCurrentListChanged(previousList, currentList)
     //E.g. check if new item has been added
    if (currentList.size == previousList.size + 1) {
        recyclerView.scrollToPosition(currentList.size - 1)
    }
}
于 2021-07-05T15:41:33.127 回答
0

发生这种情况是因为您的

recyclerView.scrollToPosition(data.size() - 1);

首先被调用,然后列表被初始化。您需要在初始化回收站视图视图后添加此行。您可以在处理程序中添加此行。

于 2019-07-19T05:28:15.027 回答