项目中使用的TabActivity已经被Google弃用,最近尝试着转换 ViewPager+Fragment,解决了以下问题
ViewPager禁止滑动
项目中主界面不要左右滑动,但是ViewPager没有官方api来控制滑动,需要自定义ViewPager,重写几个方法
/**
* 无法左右滑动的ViewPager
*/
public class NoScrollViewPager extends ViewPager {
public NoScrollViewPager(Context context) {
super(context);
}
public NoScrollViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean onTouchEvent(MotionEvent arg0) {
return false;
}
@Override
public boolean onInterceptTouchEvent(MotionEvent arg0) {
return false;
}
}
Fragment是否可见
项目中一个页面里面有轮播图,当界面不可见时,停止轮播,当界面可见时,开始轮播。
注意:当我们使用ViewPager+Fragment时,如果Fragment.onStart()被调用了,只是绑定的Activity可见了,当前Fragment不一定可见。
这里需要借助Fragment.setUserVisibleHint(boolean isVisibleToUser)这个方法。假如ViewPager中有两个Fragment分别为A,B,做简单测试,测试setUserVisibleHint调用时机:
ViewPager初始化
A.setUserVisibleHint(false)
B.setUserVisibleHint(false)
A.onCreate()
A.onAttach()
A.setUserVisibleHint(true)
B.onCreate()
B.onAttach()
B.onCreateView()
A.onCreateView()
切换A到B
A.setUserVisibleHint(false)
B.setUserVisibleHint(true)
绑定Activity变为不可见
未调用fragment.setUserVisibleHint
绑定Activity变为可见
未调用fragment.setUserVisibleHint
总结
- onAttach()之前,ViewPager中的每个Fragment会被调用setUserVisibleHint(false)
- onCreateView()之前,如果fragment为当前页,则再次被调用setUserVisibleHint(true)
- ViewPager的fragment之间进行切换,调用fragment.setUserVisibleHint(true or false)
- 绑定Activity前后台切换时
不调用fragment.setUserVisibleHint
Fragment是否可见还需要结合fragment.onResume()以及fragment.onStop()进行判断,这两个方法的调用跟在Activity相应方法之后,得出代码如下
private boolean isUiPrepared;
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_index, container, false);
isUiPrepared = true;
return view;
}
@Override
public void onResume() {
super.onResume();
setUserVisibleHint(getUserVisibleHint());
}
@Override
public void onStop() {
super.onStop();
// fragment 真的不可见了
}
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isUiPrepared) {//onCreateView调用之后才执行
if (isVisibleToUser) {
// fragment 真的可见了
} else {
// fragment 真的不可见了
}
}
}
懒加载
LazyFragment是实现了懒加载效果的Fragment,@CallSuper用于检测子类是否有调用父类的方法,如果未调用,IDE会给错误提示
/**
* 懒加载fragment<br>
* 作用:如果fragment不可见时,不加载数据;可见时再加载数据<br>
* 使用:<br>
* <ul>
* <li>继承LazyFragment,在{@link LazyFragment#loadData()}加载数据<br>
* <li>onCreateView中只完成ui的操作,不加载数据
* </ul>
*/
public abstract class LazyFragment extends Fragment {
protected Activity activity;
protected LazyFragment fragment;
private boolean isInited;
private boolean isUiPrepared;
@Override
public void onAttach(Context context) {
super.onAttach(context);
this.activity = (Activity) context;
fragment = this;
}
@CallSuper
@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
isUiPrepared = true;
lazyLoad();
}
public void lazyLoad() {
if (getUserVisibleHint() && isUiPrepared && !isInited) {
// 异步初始化,在初始化后显示正常UI
isInited = loadData();
}
}
/**
* 当fragment可见,ui已经完成,数据还未初始化时,回调此方法<br/>
* 可以主动调用{@link LazyFragment#setInited(boolean)}设置
* @return 是否初始化成功,true成功,false失败
*/
public abstract boolean loadData();
/**
* 是否初始化成功,true成功,false失败
* @param inited
*/
public void setInited(boolean inited) {
isInited = inited;
}
public boolean isInited() {
return isInited;
}
@CallSuper
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser) {
lazyLoad();
}
}
}