说明
上篇写了一个demo,这篇来看看使用。
提供了一个获取tab标签页的接口,课程界面
分析
- 因为只提供了一个接口,所以毫无疑问用户的频道喜好是存储在本地的
- 因为我在频道页我的频道点击某个标签回到viewpager页时需对应那个界面,所以可以必须返回一个position
- 还需要频道界面给我我的频道以及其他频道数据
处理
每次进入到课程界面的时候请求到数据,与储存的我的频道数据比较,取交集,作为显示的标签页,剩余标签存做其他频道待进入频道管理界面时传入
代码
进入课程界面,请求网络数据,然后与本地保存做比较
String myItemsStr = (String) get(mContext, SPKey.MyItem, "");
myItems = gson.fromJson(myItemsStr,
new TypeToken<List<DataCommon>>() {
}.getType());
这个用于获取用户退出界面时保存在SharedPreferences里的我的频道数据,因为这是复杂对象,故,使用gson转换。
if (null != myItems) {
List<DataCommon> temp = new ArrayList<>();
for (DataCommon dataCommon : myItems) {
if (mDataCommonList.contains(dataCommon)) {
temp.add(dataCommon);
}
}
这个是取交集的方法,给一个临时变量temp存储我的频道数据
List<DataCommon> tempOther = new ArrayList<>();
for (DataCommon dataCommon : mDataCommonList) {
if (!temp.contains(dataCommon)) {
tempOther.add(dataCommon);
}
}
不在交集里的均为其他频道,给临时变量tempOther
mDataCommonList = temp;
mOtherCommonList = tempOther;
将临时变量赋值给mDataCommonList,mOtherCommonList。这两个数据将作为我的频道以及其他频道数据传给频道界面。
Intent intent = new Intent(mContext, ChannelActivity.class);
Bundle bundle = new Bundle();
bundle.putSerializable(ChannelActivity.TAG_ITEM_MY, (Serializable) mDataCommonList);
bundle.putSerializable(ChannelActivity.TAG_ITEM_OTHER, (Serializable) mOtherCommonList);
intent.putExtras(bundle);
startActivityForResult(intent, CODE_REQUEST_CHANNEL);
这个是跳转到频道界面的操作。这里有一个对象序列化操作,Serializable虽然差点但是方便点暂且先用它里。然后用的startActivityForResult,这个是方便频道选好之后带数据回到这个界面。
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (CODE_REQUEST_CHANNEL == requestCode && data != null && resultCode == RESULT_OK) {
mDataCommonList = (List<DataCommon>) data.getExtras().getSerializable(ChannelActivity.TAG_ITEM_MY);
mOtherCommonList = (List<DataCommon>) data.getExtras().getSerializable(ChannelActivity.TAG_ITEM_OTHER);
mPosition = data.getExtras().getInt(ChannelActivity.TAG_ITEM_POSITION, 0);
initPage();
}
}
选完频道后回到课程界面,带回的数据有mDataCommonList,mOtherCommonList,mPosition。mPosition就是上文所说返回时须为对应标签页所用。
再来看看初始化viewpage操作
private void initPage() {
if (listFragments == null) {
listFragments = new ArrayList<>();
} else {
listFragments.clear();
}
ArrayList<String> list = new ArrayList<>();
for (int i = 0; i < mDataCommonList.size(); i++) {
list.add(mDataCommonList.get(i).name);
CourseListPage courseListFragment = new CourseListPage();
Bundle args = new Bundle();
args.putInt(CourseListPage.TAG_CODE, mDataCommonList.get(i).code);
courseListFragment.setArguments(args);
listFragments.add(courseListFragment);
}
int size = list.size();
mTitles = list.toArray(new String[size]);
// 初始化ViewPager的适配器,并设置给它
mDetailFMViewPagerAdapter = new ViewPagerFMAdapter(getChildFragmentManager(), mTitles, listFragments);
mAllViewpager.setAdapter(mDetailFMViewPagerAdapter);
// 设置ViewPager最大缓存的页面个数
mAllViewpager.setOffscreenPageLimit(3);
// 设置ViewPager标签样式
mAllTablayout.setTabMode(TabLayout.MODE_FIXED);
// 将TabLayout和ViewPager进行关联,让两者联动起来
mAllTablayout.setupWithViewPager(mAllViewpager);
mAllTablayout.setTabMode(TabLayout.MODE_SCROLLABLE);
//跳转目录
mAllViewpager.setCurrentItem(mPosition);
}
listFragments不为null就clear,为null就new一个。这里list转数组是为显示标签头。for循环里创建fragment ,用bundle把code值带入,因为后面fragment数据请求,传入的参数为code。
后面viewpager相关操作就是将TabLayout和ViewPager进行关联,让两者联动起来等等,看注释即可。这里说下ViewPagerFMAdapter继承FragmentStatePagerAdapter,并且复写getItemPosition,并renturn POSITION_NONE,不然频道界面回到课程界面时,fragment没变化。
@Override
public int getItemPosition(Object object) {
return POSITION_NONE;
}
然后用户退出这个界面时,需把用户喜好的频道保存到SharedPreferences
/**
* 把用户喜好的频道保存到sp
*/
private void spSave() {
String curCommons = gson.toJson(mDataCommonList);
String oldCommons = (String) SPUtil.get(mContext, SPKey.MyItem, "");
if (!curCommons.equals(oldCommons)) {
SPUtil.put(mContext, SPKey.MyItem, gson.toJson(mDataCommonList));
}
}
为了避免无用重复保存,我们做个判断,只有当前我的频道和进来时的不一样才做保存,否则不必保存,SharedPreferences不能直接保存list对象,gson.toJson(mDataCommonList)转为字符串保存,回头获取的时候在gson.fromJson(myItemsStr, new TypeToken<List<DataCommon>>() {}.getType());转为list对象即可。
弊端
因为只使用了一个接口,这样用户卸载了app他的喜好就say byebye了,还一种,我在别的地方登录我的账号也是没有我的喜好的频道了。