https://github.com/googlesamples/android-viewpager2
谷歌官方的示例.
用起来与viewpager类似,但是它是由RecyclerView实现的,略有区别.
我将它应用到AMupdf里面,替换了原来的ViewPager.性能上没有什么差别,使用上倒是修改了一些.把以前的tab也换了.
这次是库的升级,过程中,遇到了一点小麻烦.
比如获取当前的fragment对象.
配置:
api ("androidx.viewpager2:viewpager2:1.0.0-beta04") {
exclude group: 'androidx.recyclerview'
}
api ("com.google.android.material:material:${GOOGLE_METERIAL}") {
exclude group: 'androidx.recyclerview'
}
因为我的RecyclerView是自己有一份源码,不排除,会出现重复的. 这两个都引用了RecyclerView,所以如果你用到了,,情况和我一样,需要指定它不包含RecyclerView.
菜单遇到了一个问题,单个Fragment时,是正常的,只要切换到另一个就崩溃,ActionMenuView,不能转为ViewGroup.所以菜单我也换了.但效果还是要的.
回到:ChooseFileFragmentActivity中
布局:
设置tab
mPagerAdapter = TabsAdapter(this)
mViewPager.adapter = mPagerAdapter
TabLayoutMediator(tabLayout, mViewPager) { tab, position ->
tab.text = mTabs[position].title
}.attach()
// 滑动监听
mViewPager.registerOnPageChangeCallback(object : ViewPager2.OnPageChangeCallback() {
override fun onPageScrollStateChanged(state: Int) {
super.onPageScrollStateChanged(state)
}
override fun onPageScrolled(position: Int, positionOffset: Float, positionOffsetPixels: Int) {
super.onPageScrolled(position, positionOffset, positionOffsetPixels)
}
override fun onPageSelected(position: Int) {
super.onPageSelected(position)
toolbar.menu.clear()
切换不同的子Fragment时,需要切换菜单项,这是重点操作
if (position == 0) {
toolbar.inflateMenu(R.menu.menu_history)
} else {
toolbar.inflateMenu(R.menu.menu_browser)
}
}
});
toolbar的菜单点击:
toolbar.setOnMenuItemClickListener { item ->
val id = item.itemId
when (id) {
R.id.action_about -> startActivity(Intent(this@ChooseFileFragmentActivity, AboutActivity::class.java))
R.id.action_options -> startActivity(Intent(this@ChooseFileFragmentActivity, PdfOptionsActivity::class.java))
R.id.action_search -> {
showSearchDialog()
}
else -> {
val fragment: Fragment? = mPagerAdapter.getItemFragment(mViewPager.currentItem)
Logcat.d("menu:" + id + " fragment:" + fragment + " index:" + mViewPager.currentItem)
fragment?.onOptionsItemSelected(item)
}
}
false
}
对子类的菜单,处理项的点击操作:
还是使用了这个方法.
通过日志可以看到当使用了Toolbar,Activity只有onCreateOptionsMenu会被调用,其它的菜单方法不会被调用.
这是子Fragment的一个方法:
override fun onOptionsItemSelected(menuItem: MenuItem): Boolean {
when (menuItem.itemId) {
R.id.action_backup -> backup()
R.id.action_restore -> restore()
return super.onOptionsItemSelected(menuItem)
}
最后是适配器:
inner class TabsAdapter(activity: AppCompatActivity) : FragmentStateAdapter(activity) {
private val mContext: Context
private val mFragmentArray = SparseArray>()
init {
mContext = activity
}
override fun getItemCount(): Int {
return mTabs.size
}
//这里通过缓存,可以获取当前的Fragment,很多时候,Activity与Fragment需要交互.
fun getItemFragment(index: Int): Fragment? {
return mFragmentArray.get(index)?.get()
}
override fun createFragment(position: Int): Fragment {
val mWeakFragment = mFragmentArray.get(position)
if (mWeakFragment?.get() != null) {
return mWeakFragment.get()!!
}
val info = mTabs[position]
val fragment = Fragment.instantiate(mContext, info.clss.name, info.args)
mFragmentArray.put(position, WeakReference(fragment))
return fragment
}
}
比如我在返回键按下,要一些操作:
override fun onBackPressed() {
val fragment = mPagerAdapter.getItemFragment(mViewPager.currentItem) as BrowserFragment
if (fragment.onBackPressed()) {
return
}
super.onBackPressed()
}
viewpager2默认是不加载当前屏幕外的数据的.viewpager是默认加载当前的,与左右(如果有的话),至少两个 .viewpager2修改了这个规则,可以减少到一个项.这是在没切换到页面时可以减少资源的消耗了.
viewpager2还配置了很多动画,在页面转换过程中可以用的,不过一般数据量大,这种动画还是不要的好,如果只是单个图片画廊这种,动画还挺不错的.
升级的过程,除了菜单的问题,其它都没什么问题.