首页 技术 正文
技术 2022年11月18日
0 收藏 343 点赞 4,002 浏览 3714 个字

最近项目新功能需要在垂直方方向可以循环滚动,并且水平方向也可以水平循环滚动,并且可以定位到指定item上。很自然的想到了ViewPager和 VerticalViewPager来解决项目需求,UI的大致结构如下

以下垂直方向滚动的ViewPager所在的Fragment成为A,水平方向滚动的ViewPager所在的Fragment成为B!

1、循环滚动的实现

要实现循环滚动的原理很简单,就是设置item数量为无限大或者为一个很大的数值,然后设置currentItem为该数值的一半这样就可以实现上下(左右)循环滚动了!在PagerAdaper上修改方法:

    @Override
public int getCount() {
return Integer.MAX_VALUE;
}

2、定位到指定item功能实现

从Fragment到Fragment通信,这里选择的是EventBus这个插件!

  1. .在A上监听水平垂直变化的Event,接收到消息后定位到指定行,并且发送水平方向的移动Event
  2. 在B上监听水平移动的Event,接收到消息后定位到指定列

    这样定位到指定item的功能就实现了!

    A:

  @Subscribe(threadMode = ThreadMode.MAIN)
public void onAnimationEvent(RowEvent rowEvent) {
int index = viewPager.getCurrentItem() - viewPager.getCurrentItem() % rowEvent.allRows;
viewPager.setCurrentItem(index + rowEvent.row);
final Intent intent = new Intent(ChallengeItemFragment.COLUMN_ACTION);
intent.putExtra("stageId", rowEvent.stageId);
intent.putExtra("column", rowEvent.column); viewPager.postDelayed(new Runnable() {
@Override
public void run() {
LocalBroadcastManager.getInstance(getActivity()).sendBroadcast(intent);
}
}, 250);
//通知变化
// EventBus.getDefault().post(new ColumnEvent(rowEvent.row, rowEvent.column, rowEvent.stageId));
}

B:

 @Subscribe(threadMode = ThreadMode.MAIN)
public void onAnimationEvent(ColumnEvent columnEvent) {
if (columnEvent.stageId != stage.stageId) {
return;
}
int index = challengeItemViewpager.getCurrentItem() - challengeItemViewpager.getCurrentItem() % stage.challengeList.size();
challengeItemViewpager.setCurrentItem(index + columnEvent.column);
}

以上为项目背景以及基本使用介绍,下面进入主题:

在测试的时候发现一个很奇怪的现象,从菜单进入该页面,并且该页面每次都是重新初始化的,在定位时收到了重复的水平方向定位消息,以下为log截图

并且每从菜单进入一次,重复测试就+1。

经过debug以及log发现,B的实例对象一直存在,就算从菜单进入,并且重新初始化了A也是一样。

作为ViewPager的切入点,当然就是Adapter了,因为项目统一使用的是Fragment而不是v4包的Fragment,所以PagerAdaper是拷贝FragmentStatePagerAdapter的,getItem上的主要方法实现如下

public android.app.Fragment getItem(int position) {
int oriPosition = position;
position = position % data.stageList.size();
B itemFragment = new B();
Bundle param = new Bundle();
param.putParcelable(B.EXTRA_STAGE, stage);
param.putBoolean(B.EXTRA_LOCKED, isLocked);
itemFragment.setArguments(param);
return itemFragment;
}

经过分析发现最终定位到mFragmentManager。在实例化Adapter时,传入的是getFragmentManger(),因为fragmentManager的生命周期是跟随Activity的,所以就算A重新实例化,使用的FragmentManager也是相同的,并且在Adapter上的实现:


public Object instantiateItem(ViewGroup container, int position) {
// If we already have this item instantiated, there is nothing
// to do. This can happen when we are restoring the entire pager
// from its saved state, where the fragment manager has already
// taken care of restoring the fragments we previously had instantiated.
if (mFragments.size() > position) {
Fragment f = mFragments.get(position);
if (f != null) {
return f;
}
} if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
} Fragment fragment = getItem(position);
if (mSavedState.size() > position) {
Fragment.SavedState fss = mSavedState.get(position);
if (fss != null) {
fragment.setInitialSavedState(fss);
}
}
while (mFragments.size() <= position) {
mFragments.add(null);
}
fragment.setMenuVisibility(false);
fragment.setUserVisibleHint(false);
mFragments.set(position, fragment);
mCurTransaction.add(container.getId(), fragment); return fragment;
}

mCurTransaction.add(container.getId(), fragment);实例化后的Fragment加入FragmenManager管理!那么足可以说明每次实例化A后,其实之前已经添加到FragmenManager的B对象时没有销毁的,这就导致了每次从菜单进入A,水平定位上总是收到重复消息数量+1

既然发现了问题,那么就很好解决了,在A destory之前清除已经在FragmenManager上的B对象即可!在Adapter上添加:

  public void clearFragments() {        if (mCurTransaction == null) {
mCurTransaction = mFragmentManager.beginTransaction();
} for (Fragment fragment : mFragments) {
if (fragment != null && fragment.isAdded()) {
mCurTransaction.remove(fragment);
}
} mCurTransaction.commitAllowingStateLoss();
}
  @Override
public void onDestroyView() {
EventBus.getDefault().unregister(this);
challengeAdapter.clearFragments();
super.onDestroyView();
}

这样此问题完美解决了!

在Fragment内使用FragmentManager推荐使用的是 getChildFragmentManager()但是此方法是在API17上添加的,所以还必须使用v4的Fragment。至于在v4.Fragment上是否会出现此问题,等以后遇到了再去研究!!

相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:9,492
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,907
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,740
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,493
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:8,132
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:5,295