android - 在 Android ViewPager2 中删除项目
问题描述
我正在将我的代码从 using 更新androidx.viewpager
为androidx.viewpager2
。我正在翻阅数量不定的片段,这些片段显示从数据库中检索到的数据记录。加载视图寻呼机并通过我的数据进行分页效果很好,但我在删除项目和更新寻呼机适配器时遇到了一些麻烦。removeItem()
我想通过调用适配器上的方法(参见下面的代码)来删除任何给定位置的项目。这应该从我的数据库以及我的片段中删除该项目,然后更新视图。
结果是从数据库中删除了正确的项目。但它不会从我的视图寻呼机中删除预期的片段,而是从下一页删除。当前页面仍然可见。我将位置稍微偏移了正负 1,但没有成功 - 相反:在这些情况下,我的删除例程按最初预期执行。我也尝试了与此处给出的类似考虑。
我想实现以下行为:
- 删除任何项目时,应删除该页面并显示列表中的下一个项目。
- 删除列表中最后一个/最右边的项目时,应删除该页面并显示上一个(现在是最后一个)页面。
- 删除最后一个剩余项目(没有剩余)时,活动应该完成。
我的适配器代码:
internal class ShapePagerAdapter(private val activity: AppCompatActivity) : FragmentStateAdapter(activity) {
private val dbManager: DatabaseManager
private var shapeIds: MutableList<String>? = null
init {
dbManager = DatabaseManager(activity)
try {
shapeIds = dbManager.getShapeIds()
} catch (e: DatabaseAccessException) {
// ...
}
}
override fun getItemCount(): Int {
return if (null != shapeIds) shapeIds!!.size else 0
}
override fun createFragment(position: Int): Fragment {
return ShapeFragment.newInstance(shapeIds!![position])
}
fun removeItem(activity: AppCompatActivity, position: Int) {
try {
// Remove from Database.
dbManager.deleteShape(shapeIds!![position])
// Remove from View Pager.
shapeIds!!.removeAt(position)
notifyItemRemoved(position)
notifyItemRangeChanged(position , itemCount)
// Close if nothing to show anymore.
if (itemCount == 0) {
activity.finish()
}
} catch (e: DatabaseAccessException) {
// ...
}
}
}
解决方案
更仔细的研究FragmentStateAdapter
表明,在这种情况下必须重写它的两个方法:
containsItem(long itemId)
和getItemId(int position)
默认实现适用于不添加、移动、删除项目的集合。
搜索我找到了类似问题的答案,为我指明了正确的方向。它不会产生我的问题中给出的确切行为,这就是为什么我要发布一个稍微改编的版本。
关键是这两种方法是在项目顺序可能发生变化的情况下实现的。为了实现这一点,我维护了一个项目和项目 ID 的映射,并在序列发生更改时更新,在本例中是一个已删除的项目。
internal class ShapePagerAdapter(private val activity: AppCompatActivity) : FragmentStateAdapter(activity) {
private val dbManager: DatabaseManager
private lateinit var shapeIds: MutableList<String>
private lateinit var itemIds: List<Long>
init {
dbManager = DatabaseManager(activity)
try {
shapeIds = dbManager.getShapeIds()
updateItemIds()
} catch (e: DatabaseAccessException) {
// ...
}
}
override fun getItemCount(): Int = shapeIds.size
override fun createFragment(position: Int): Fragment = ShapeFragment.newInstance(shapeIds[position])
fun removeItem(activity: AppCompatActivity, position: Int) {
try {
dbManager.deleteShape(shapeIds[position])
shapeIds.removeAt(position)
notifyItemRemoved(position)
notifyItemRangeChanged(position , itemCount)
updateItemIds()
if (itemCount == 0) activity.finish()
} catch (e: DatabaseAccessException) {
// ...
}
}
private fun updateItemIds() {
itemIds = shapeIds.map { it.hashCode().toLong() }
}
override fun getItemId(position: Int): Long = shapeIds[position].hashCode().toLong()
override fun containsItem(itemId: Long): Boolean = itemIds.contains(itemId)
}
}
推荐阅读
- javascript - ckeditor gem custom config.js 未在生产中加载
- python - itertools islice 输出多条txt行
- mysql - 执行 ALTER 时 MySQL InnoDB 行/表锁
- c# - 页面上的组页脚打印时的水晶报表抑制页脚
- node.js - Cloud Foundry - 文件夹结构和相对路径
- css - React Native:使用 AsyncStorage 和 State 动态更改样式
- django - Django Rest Framework 插入和更新可写嵌套序列化器
- sass - scss如何在媒体查询中添加+1px
- json - 嵌套地图返回json数组[]
- java - 用php连接java