首页 > 解决方案 > 在 Android ViewPager2 中删除项目

问题描述

我正在将我的代码从 using 更新androidx.viewpagerandroidx.viewpager2。我正在翻阅数量不定的片段,这些片段显示从数据库中检索到的数据记录。加载视图寻呼机并通过我的数据进行分页效果很好,但我在删除项目和更新寻呼机适配器时遇到了一些麻烦。removeItem()我想通过调用适配器上的方法(参见下面的代码)来删除任何给定位置的项目。这应该从我的数据库以及我的片段中删除该项目,然后更新视图。

结果是从数据库中删除了正确的项目。但它不会从我的视图寻呼机中删除预期的片段,而是从下一页删除。当前页面仍然可见。我将位置稍微偏移了正负 1,但没有成功 - 相反:在这些情况下,我的删除例程按最初预期执行。我也尝试了与此处给出的类似考虑。

我想实现以下行为:

  1. 删除任何项目时,应删除该页面并显示列表中的下一个项目。
  2. 删除列表中最后一个/最右边的项目时,应删除该页面并显示上一个(现在是最后一个)页面。
  3. 删除最后一个剩余项目(没有剩余)时,活动应该完成。

我的适配器代码:

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) {
            // ...
        }
    }
}

标签: androidkotlinandroidxandroid-viewpager2

解决方案


更仔细的研究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)
    }
}

推荐阅读