在将适配器提交列表提交到DiffUtil之后,将调用onBindViewHolder,并且由于某种原因,它将始终将列表滚动到顶部。
我在recyclerview中尝试了关于自动滚动更新中SO的每一个答案,但是无法解决问题。
即
RecyclerView: Disable scrolling caused by change in focus
RecyclerView scrolls to top on notifyDataSetChanged in chat screen
android notifyItemRangeInserted disable autoscroll
等
这是我的适配器:
init {
setHasStableIds(true)
}
private val currentDiffer = AsyncListDiffer<Item>(this, MyDiffUtil())
...
override fun setHasStableIds(hasStableIds: Boolean) {
super.setHasStableIds(true)
}
override fun getItemCount(): Int {
return currentDiffer.currentList.size
}
override fun getItemId(position: Int): Long = currentDiffer.currentList[position].name.hashCode().toLong()
override fun getItemViewType(position: Int) = position
override fun onAttachedToRecyclerView(recyclerView: RecyclerView) {
super.onAttachedToRecyclerView(recyclerView)
}
override fun onBindViewHolder(holder: CurrentListViewHolder, position: Int) {
holder.bindView(position, activity, currentDiffer.currentList[position], onCurrencyListener)
}
override fun onViewRecycled(holder: CurrentListViewHolder) {
holder.unbindView()
super.onViewRecycled(holder)
}
fun setData(list: LinkedList<Item>) {
val newList: LinkedList<Item> = LinkedList()
newList.addAll(list)
currentDiffer.submitList(newList.toList())
}
因此,在列表的每个新提交之后,无论当前位置是什么,均将调用onBindViewHolder,并且在执行任何操作之前,列表将被滚动到0位置,如scrollToPosition一样。
编辑:
private fun setView() {
currentListAdapter = CurrentListAdapter()
with(currenciesRv) {
layoutManager = LinearLayoutManager(context)
adapter = currentListAdapter
}
布局:
<RelativeLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/listRv"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</RelativeLayout>
项目:
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/itemLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<de.hdodenhof.circleimageview.CircleImageView
android:id="@+id/itemIvFlag"
android:layout_width="50dp"
android:layout_height="50dp"
android:layout_marginTop="30dp"
android:layout_marginStart="15dp"
android:layout_alignParentStart="true"
android:layout_centerVertical="true"/>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_toEndOf="@id/itemIvFlag"
android:orientation="vertical"
android:layout_centerVertical="true"
android:layout_marginTop="10dp">
<TextView
android:id="@+id/itemTvISO"
android:layout_marginStart="10dp"
android:textStyle="bold"
tools:text="@string/dummy_number"
android:textSize="16sp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<TextView
android:id="@+id/itemTvName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
android:layout_marginStart="10dp"
android:textSize="14sp"
tools:text="@string/dummy_number"/>
</LinearLayout>
<EditText
android:id="@+id/itemEtAmount"
android:layout_alignParentEnd="true"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_marginEnd="15dp"
tools:text="@string/dummy_number"
android:inputType="numberDecimal"
android:enabled="false"
android:textColor="@drawable/et_disabled_color"
android:imeOptions="actionDone"
android:maxLines="1"
android:singleLine="true"/>
</RelativeLayout>
答案 0 :(得分:1)
发布覆盖方法,用于比较DiffUtil提交的列表。我遇到了类似的问题(我的问题是RecycleView专注于list_item_network_state,它正在显示加载视图)。从您的帖子下的注释中:“每隔几秒钟就会从服务器刷新一次列表,并且如果向下滚动列表,则onBindViewHolder会将其滚动回到0” ,因此,当数据正在更新时,它包含相同的项您想集中关注旧列表中的焦点元素?
很难找到问题的解决方案(信息不足),我认为问题不在您的适配器上。
还建议使用ListAdapter
,它本身具有DiffUtil.ItemCallback。
更新 因此,如果onBIndViewHolder中没有任何内容,它仍会滚动到顶部,这意味着问题不存在。问题在于提交新列表(如DiffUtil “它用于计算RecyclerView适配器的更新。” )。与比较期间一样,它决定删除/更新/添加项目的位置。可能存在问题(这是我对您提供的信息的假设,如果可能,请向问题中添加更多代码)。检查方法为areItemsTheSame()(例如,它必须比较项目的ID),并在调用areContentsTheSame()之后检查其更改内容。
答案 1 :(得分:1)
问题出在我的DiffUtil上。我很鲁and,草率,并且在areItemsTheSame
和areContentsTheSame
上都错误/基本上具有相同的值,因此diffUtil看不出它们之间的差异,这意味着每个新内容都与新内容相同。