在实际开发中,为了节省开发周期,下拉刷新上拉加载通常都会采取使用一些第三方库,典型的就是用PullToRefresh,XListView等等,还有就是谷歌推荐的SwipeRefreshLayout,可惜没有上拉加载功能,需要自己去实现一个上拉加载的脚View,再加上现在代替ListView的RecyclerView+CardView使用的频率也是也来也高,不得不说,CardView效果确实很好看,一个一个的小卡片,用户体验好,I like it!!!废话不说了,奔主题!今天也玩了一下SwipeRefreshLayout+RecyclerView实现下拉刷新上拉自动加载。
我最初的想法就是:先利用RecyclerView添加不同的item布局,上面是一个轮播的ViewPager,下面是一个一个的列表数据,当滑到到底部的时候,可以自动加载更多,当然要在底部添加一个个性化的视图显示正在加载,即脚View。
先看一下RecyclerView如何加载不同的item的,通过getItemViewType实现,注意:继承的是RecyclerView.Adapter。
1.设置3个常量,分别是ViewPager部分,中间的列表数据部分,和底部的正在加载脚部分:
private static final int ITEM_HEADER = 0;
private static final int ITEM_ITEM = 1;
private static final int ITEM_LOAD_FOOTER = 2;
三个上拉加载更多的显示状态:
//上拉加载更多
public static final int PULLUP_LOAD_MORE=0;
//正在加载...
public static final int ISLOADING=1;
//上拉加载的显示状态,初始为0
private int load_more_status=0;
Adapter里设置的一个方法,方便Activity调用改变加载状态来显示不同的加载信息(上拉加载更多,加载中…):
public void changeMoreStatus(int status){
load_more_status=status;
notifyDataSetChanged();
}
2.创建Viewholder:
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
if(viewType==ITEM_HEADER){
View view = LayoutInflater.from(context).inflate(R.layout.item_header,parent,false);
return new HeaderViewHolder(view);
}else if(viewType==ITEM_ITEM){
View view = LayoutInflater.from(context).inflate(R.layout.item_item,parent,false);
return new ItemViewHolder(view);
}else if(viewType==ITEM_LOAD_FOOTER){
View view = LayoutInflater.from(context).inflate(R.layout.refresh_footer,parent,false);
return new LoadFooterViewHolder(view);
}
return null;
}
3.绑定数据:
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, final int position) {
if(holder instanceof ItemViewHolder){
((ItemViewHolder) holder).mTextView.setText(mListData.get(position-1));
}else if(holder instanceof LoadFooterViewHolder){
LoadFooterViewHolder footer = (LoadFooterViewHolder) holder;
switch(load_more_status){
case PULLUP_LOAD_MORE:
footer.progressbar.setVisibility(View.GONE);
footer.loadmore_text.setText("上拉加载更多");
break;
case ISLOADING:
footer.progressbar.setVisibility(View.VISIBLE);
footer.loadmore_text.setText("正在加载...");
break;
}
}
//列表数据的点击事件监听,注意:这里的position是比正常的position+1的,因为多加了一个头部View
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(context, "点击了位置"+position, Toast.LENGTH_SHORT).show();
}
});
}
说明:这里只处理了中间列表的数据,ViewPager部分的数据不在这处理。
4.获取item的数量:
@Override
public int getItemCount() {
return mListData.size()+2;
}
说明:+2是因为除了正常的列表数据外,还多加了一个头和脚View
5.ItemView类型:
@Override
public int getItemViewType(int position) {
if(position==0){
return ITEM_HEADER;
}else if(position==mListData.size()+1){
return ITEM_LOAD_FOOTER;
}else {
return ITEM_ITEM;
}
}
说明:第一项添加一个ViewPager,最后一项添加一个正在加载的脚View,剩下的中间的全部就是正常的列表数据。
下面的三个view内部类:
class HeaderViewHolder extends RecyclerView.ViewHolder{
private ViewPager mViewPager;
private LinearLayout linearlayout;
private ImageView[] points;
private Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
mViewPager.setCurrentItem(mViewPager.getCurrentItem()+1);
handler.sendEmptyMessageDelayed(1,3000);
}
};
public HeaderViewHolder(View itemView) {
super(itemView);
mViewPager = (ViewPager) itemView.findViewById(R.id.viewpager);
linearlayout = (LinearLayout) itemView.findViewById(R.id.linearlayout);
//下面两句要在setAdapter之前调用,否则会出现先加载最后一张图片的情况
mViewPager.setCurrentItem(Integer.MAX_VALUE/2);
handler.sendEmptyMessageDelayed(1,3000);
mViewPager.setAdapter(new MyViewPagerAdapter(context, imagesUrl));
initPoint();
mViewPager.setOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override
public void onPageSelected(int position) {
for(int i=0;i<imagesUrl.length;i++){
if(position%imagesUrl.length==i){
points[i].setImageResource(R.drawable.point_focus);
}else {
points[i].setImageResource(R.drawable.point_normal);
}
}
}
@Override
public void onPageScrollStateChanged(int state) {
}
});
}
private void initPoint() {
points = new ImageView[imagesUrl.length];
for(int i=0;i<points.length;i++){
points[i] = (ImageView) linearlayout.getChildAt(i);
}
points[0].setImageResource(R.drawable.point_focus);
}
}
class ItemViewHolder extends RecyclerView.ViewHolder{
private TextView mTextView;
public ItemViewHolder(View itemView) {
super(itemView);
mTextView = (TextView) itemView.findViewById(R.id.textview);
}
}
class LoadFooterViewHolder extends RecyclerView.ViewHolder{
private ProgressBar loarmore_progressbar;
private TextView loadmore_text;
public LoadFooterViewHolder(View itemView) {
super(itemView);
loarmore_progressbar= (ProgressBar) itemView.findViewById(R.id.loarmore_progressbar);
loadmore_text = (TextView) itemView.findViewById(R.id.loadmore_text);
}
}
说明:可以看出,我头部ViewPager的数据全是在内部类中处理的。
中间是列表数据,第3步绑定数据那里处理的,最下面是加载脚View,在上面的绑定数据那里设置了动态的显示不同的加载状态。
OK 这样数据适配器就搞定了!
下拉刷新有谷歌的SwipeRefreshLayout实现了,该如何检测到何时滑到底了需要加载新数据了呢。方法如下:
//向上滑动,当滑到底的时候,有上滑的趋势,就加载更多
mRecyclerView.setOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if(!mSwipeRefreshLayout.isRefreshing()){
if(newState==RecyclerView.SCROLL_STATE_IDLE&&mAdapter.getItemCount()==lastVisibleItem+1){
//调用Adapter里的changeMoreStatus方法来改变加载脚View的显示状态为:正在加载...
mAdapter.changeMoreStatus(MyRecyclerViewAdapter.ISLOADING);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
mListData.addAll(mListLoadData);
mAdapter.notifyDataSetChanged();
//当加载完数据后,再恢复加载脚View的显示状态为:上拉加载更多
mAdapter.changeMoreStatus(MyRecyclerViewAdapter.PULLUP_LOAD_MORE);
}
},3000);
}
}
}
说明:RecyclerView有一个setOnScrollListener方法,加载更多的逻辑就在这里面实现,具体不说了,看一下就明白了。
MainActivity的完整代码:
package com.example.lenovo.recyclerview;
import android.app.Activity;
import android.graphics.Color;
import android.os.Bundle;
import android.os.Handler;
import android.support.v4.widget.SwipeRefreshLayout;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import java.util.ArrayList;
import java.util.List;
import Adapter.MyRecyclerViewAdapter;
public class MainActivity extends Activity {
private RecyclerView mRecyclerView;
private SwipeRefreshLayout mSwipeRefreshLayout;
private LinearLayoutManager mManager;
private MyRecyclerViewAdapter mAdapter;
private List<String> mListData;
private List<String> mListLoadData;
private String[] imagesUrl = {
"http://bbs.uc.cn/data/attachment/forum/201302/17/163510xsv14x35i9ix41i0.jpg",
"http://p3.so.qhimg.com/t0121ddd5bc66dcc9e8.jpg",
"http://p3.so.qhimg.com/t0181e8d7355386f79d.jpg",
"http://bbs.liebao.cn/data/attachment/forum/201210/25/182447qee2e922myyw8y8m.jpg"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initData();
initView();
//下拉刷新
mSwipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
mSwipeRefreshLayout.setRefreshing(false);
}
},2000);
}
});
//当滑到最后一项,还有上滑的趋势,就加载更多
mRecyclerView.setOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if(!mSwipeRefreshLayout.isRefreshing()){
if(newState==RecyclerView.SCROLL_STATE_IDLE&&mAdapter.getItemCount()==lastVisibleItem+1){
mAdapter.changeMoreStatus(MyRecyclerViewAdapter.LOADING_MORE);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
mListData.addAll(mListLoadData);
mAdapter.notifyDataSetChanged();
mAdapter.changeMoreStatus(MyRecyclerViewAdapter.PULLUP_LOAD_MORE);
}
},3000);
}
}
}
}
});
}
/** * 初始化数据 */
private void initData() {
mListData = new ArrayList<>();
mListLoadData = new ArrayList<>();
for(int i=0;i<30;i++){
mListData.add("数据"+i);
mListLoadData.add("加载的新数据"+i);
}
}
/** * 初始化view */
private void initView() {
mRecyclerView = (RecyclerView) findViewById(R.id.recyclerview);
mManager = new LinearLayoutManager(this);
mRecyclerView.setLayoutManager(mManager);
mAdapter = new MyRecyclerViewAdapter(this,mListData,imagesUrl);
mRecyclerView.setAdapter(mAdapter);
mSwipeRefreshLayout = (SwipeRefreshLayout) findViewById(R.id.swiperefreshlayout);
//最多可以4种颜色
mSwipeRefreshLayout.setColorSchemeColors(Color.BLUE,Color.RED);
}
}
必要的地方都有说明,就不赘述了。搞定!运行!一看效果还是可以的,就下拉刷新 上拉加载玩了一下,当我手动滑动ViewPager的轮播图时候,发现有的不顺畅,下拉刷新有种要被拉下来的感觉,发现有冲突:
百度了一下,发现也有不少人有这个问题,解决办法网上也一大堆:
重写SwipeRefreshLayout。
于是乎,就重写了一下,代码如下:
package View;
import android.content.Context;
import android.support.v4.widget.SwipeRefreshLayout;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.ViewConfiguration;
/** * Created by lenovo on 2016/7/9. */
public class MySwipeRefreshLayout extends SwipeRefreshLayout {
private int mTouchSlop;
// 上一次触摸时的X坐标
private float mPrevX;
public MySwipeRefreshLayout(Context context, AttributeSet attrs) {
super(context, attrs);
// 触发移动事件的最短距离,如果小于这个距离就不触发移动控件
mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
}
@Override
public boolean onInterceptTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
mPrevX = event.getX();
break;
case MotionEvent.ACTION_MOVE:
final float eventX = event.getX();
float xDiff = Math.abs(eventX - mPrevX);
// Log.d("refresh" ,"move----" + eventX + " " + mPrevX + " " + mTouchSlop);
// 增加60的容差,让下拉刷新在竖直滑动时就可以触发
if (xDiff > mTouchSlop + 60) {
return false;
}
}
return super.onInterceptTouchEvent(event);
}
}
再运行!搞定!完美!!