android - 如果只是项目的内容发生更改,PagedListAdapter 不会更新列表
问题描述
我正在使用 Room 和 Paging 库来显示类别。
我的实体:
@Entity(tableName = Database.Table.CATEGORIES)
data class Category(
@PrimaryKey(autoGenerate = true) @ColumnInfo(name = ID) var id: Long = 0,
@ColumnInfo(name = NAME) var name: String = "",
@ColumnInfo(name = ICON_ID) var iconId: Int = 0,
@ColumnInfo(name = COLOR) @ColorInt var color: Int = DEFAULT_COLOR
)
我的道:
@Query("SELECT * FROM $CATEGORIES")
fun getPagedCategories(): DataSource.Factory<Int, Category>
@Update
fun update(category: Category)
我的回购:
val pagedCategoriesList: LiveData<PagedList<Category>> = categoryDao.getPagedCategories().toLiveData(Config(CATEGORIES_LIST_PAGE_SIZE))
我的视图模型:
val pagedCategoriesList: LiveData<PagedList<Category>>
get() = repository.pagedCategoriesList
我的适配器:
class CategoriesAdapter(val context: Context) : PagedListAdapter<Category, CategoriesAdapter.CategoryViewHolder>(CategoriesDiffCallback()) {
//region Adapter
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): CategoryViewHolder {
return CategoryViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.item_category, parent, false))
}
override fun onBindViewHolder(holder: CategoryViewHolder, position: Int) {
holder.bind(getItem(position)!!)
}
//endregion
//region Methods
fun getItemAt(position: Int): Category = getItem(position)!!
//endregion
inner class CategoryViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
private val iconHelper = IconHelper.getInstance(context)
fun bind(category: Category) {
with(itemView) {
txvCategoryItemText.text = category.name
imvCategoryItemIcon.setBackgroundColor(category.color)
iconHelper.addLoadCallback {
imvCategoryItemIcon.setImageDrawable(iconHelper.getIcon(category.iconId).getDrawable(context))
}
}
}
}
class CategoriesDiffCallback : DiffUtil.ItemCallback<Category>() {
override fun areItemsTheSame(oldItem: Category, newItem: Category): Boolean {
return oldItem.id == newItem.id
}
override fun areContentsTheSame(oldItem: Category, newItem: Category): Boolean {
return oldItem == newItem
}
}
}
还有我的片段:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
categoryViewModel = ViewModelProviders.of(this).get(CategoryViewModel::class.java)
adapter = CategoriesAdapter(requireContext())
categoryViewModel.pagedCategoriesList.observe(this, Observer(adapter::submitList))
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
ViewCompat.setTooltipText(fabNewCategory, getString(R.string.NewCategory))
with(mRecyclerView) {
layoutManager = GridLayoutManager(requireContext(), 4)
itemAnimator = DefaultItemAnimator()
addItemDecoration(SpacesItemDecoration(resources.getDimensionPixelSize(R.dimen.card_default_spacing)))
addOnItemTouchListener(OnItemTouchListener(requireContext(), this, this@CategoriesFragment))
}
mRecyclerView.adapter = adapter
fabNewCategory.setOnClickListener(this)
}
插入、删除或加载类别时一切正常。但是当我更新单个实体的颜色或文本时,列表不会更新,尽管提交列表被正确调用。
我调试了整个过程,发现问题:提交列表后,AsyncPagedListDiffer#submitList
被调用。我比较了之前的列表 ( mPagedList
in AsyncPagedListDiffer
) 和新的列表 ( pagedList
in AsyncPagedListDiffer#submitList
)。我在那里编辑的项目是相同的,并且已经保存了新数据。因此,DiffUtil
比较所有内容,尽管显示的列表未更新,但项目已经相等。
如果列表是参考,它会解释为什么适配器列表中的数据已经刷新,但是我该如何解决这个问题呢?
解决方案
我认为问题不在于您加载新数据的方式,而在于更新数据。尽管您没有向我们展示触发项目更新的部分或实际更新是如何发生的,但我猜,如果我错了,抱歉,您可能会像这样直接编辑列表元素:
category = adapter.getItemAt(/*item position*/)
category.name = "a new name"
category.color = 5
categoryViewModel.update(category)
相反,您应该创建一个新Category
对象而不是修改现有对象,如下所示:
prevCategory = adapter.getItemAt(/*put position*/) // Do not edit prevCategory!
newCategory = Category(id=prevCategory.id, name="a new name", color=5, iconId=0)
categoryViewModel.update(newCategory)
每次您想要进行最小的更改时创建一个全新的对象的想法一开始可能并不那么明显,但这种反应式实现依赖于每个事件独立于其他事件的假设。使您的数据类不可变或有效不可变将防止此问题。
我喜欢做的是避免这种错误,我总是将我的数据类中的每个字段都设为 final。
@Entity(tableName = Database.Table.CATEGORIES)
data class Category(
@PrimaryKey(autoGenerate = true) @ColumnInfo(name = ID) val id: Long = 0,
@ColumnInfo(name = NAME) val name: String = "",
@ColumnInfo(name = ICON_ID) val iconId: Int = 0,
@ColumnInfo(name = COLOR) @ColorInt val color: Int = DEFAULT_COLOR
)
推荐阅读
- sql - 为什么我不能在我的存储过程 deleteByMonth 中插入我的表变量,尽管查询会产生结果?
- ios - 将 UIImage 转换为 4 位颜色空间
- jquery - Jquery将超链接添加到字符串值
- python - 日期相同的条件 True with Series 但 False 使用元素
- spring - Spring 配置服务器为客户端应用程序中的属性返回 null
- python - Haystack + Django CMS 搜索错误“reduce() of empty sequence with no initial value”
- java - 枢轴索引和平衡分区(Java)
- ios - NativeScript iOS Delegate Transpilation 错误(找不到变量:__metadata)
- jquery - 如何在 jQuery 手风琴中使用更少的 jQuery
- pug - 在对象上使用 Pug 进行迭代