ViewPager动态初始化最优方案

ViewPager动态初始化最优方案

前景

默认初始化3个tab, 但是当联网获取到数据之后, 需要再新增一个tab,而且tab的顺序需要调整.

源码分析

定义变量缓存所需数据

//缓存fragment
private val fragments = mutableMapOf<Tab, Fragment>()
//缓存fragment的ItemId
private val keyIdMap = mutableMapOf<Tab, Int>()
//缓存tab的上一次的排序位置
private val prePosList = mutableListOf<Tab>()

初始化的顺序

  1. instantiateItem
  2. getItemId
    //如果没有找到Fragment, 就调用下面getItem获取fragment
  3. getItem

刷新方案

getItemPosition

通过fragments获得key(tab) 
然后key获得新的newIndex
如果旧的index != newIndex,
	清除缓存的fragment,清除id让其重新初始化
	返回POSITION_NONE,让其重新初始化
其余返回 POSITION_UNCHANGED

源码

class MainPagePagerAdapter(fragmentManager: FragmentManager) :
    FragmentPagerAdapter(fragmentManager, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {

    private var tabList = listOf<String>("Chat", "Camera", "Story")

    private val FRAGMENT_SIZE: Int
        get() = tabList.size

    private val fragments = mutableMapOf<String, Fragment>()
    private val keyIdMap = mutableMapOf<String, Int>()
    private val prePosList = mutableListOf<String>()
    private var id = 0

    override fun getItem(position: Int): Fragment {
        return getFragment(position)
    }

    private fun getFragment(position: Int): Fragment {
        val tab = tabList[position]
        var fragment = fragments[tab]
        if (fragment == null) {
            fragment = Fragment()
            fragments[tab] = fragment
        }
        return fragment
    }

    override fun getItemId(position: Int): Long {
        //这个方法是通过itemId来获取fragment,所以要自己管理一份id
        val key = tabList[position]
        var savedId = keyIdMap[key]?.toLong()
        if (savedId == null) {
            savedId = (id).toLong()
            keyIdMap[key] = id++
        }
        return savedId
    }

    override fun getCount(): Int {
        return FRAGMENT_SIZE
    }

    override fun getItemPosition(`object`: Any): Int {
        var key: String? = null
        //找出fragment对应的key
        fragments.forEach loop@{
            if (it.value == `object`) {
                key = it.key
                return@loop
            }
        }
        if (key == null) {
            return POSITION_UNCHANGED
        }
        //取出旧的位置, 对比新的位置, 如果不相等就需要重新创建实例
        val index = prePosList.indexOf(key!!)
        var newIndex = -1
        tabList.forEachIndexed { i, tab ->
            if (tab == key) {
                newIndex = i
            }
        }
        if (index != newIndex) {
            //清除缓存
            fragments.remove(key!!)
            keyIdMap.remove(key!!)
            return POSITION_NONE
        }
        return POSITION_UNCHANGED
    }

    fun update(listOf: List<String>) {
        prePosList.clear()
        prePosList.addAll(tabList.toList())
        tabList = listOf
        notifyDataSetChanged()
    }

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值