SwipeRefresh
基于原生的SwipeRefreshLayout 做了封装处理
此项目中包括种:
1.原生SwipeRefreshLayout(上拉可通过滚动监听实现)
2.自定义支持上拉刷新的组件
3.自定义支持ViewPage的刷新组件VPSwipeRefreshLayout
4.RecyclerView+SwpieRefreshLayout实现下拉刷新效果同时实现上拉功能
主界面
1.原生SwipeRefreshLayout(上拉可通过滚动监听实现)
除了OnRefreshListener接口外,SwipRefreshLayout中还有一些其他重要的方法,具体如下:
1、setonRefreshListener(SwipeRefreshLayout.onRefreshListener listener):设置手势滑动监听器。
2、setProgressBackgroundColor(int colorRes):设置进度圈的背景色(已经弃用)
setProgressBackgroundColorSchemeResource (可以)。
setProgressBackgroundColorSchemeColor(Color c) (可以)
3、setColorSchemeResources(int… colorResIds):设置进度动画的颜色。
4、setRefreshing(Boolean refreshing):设置组件的刷洗状态,显示或者隐藏刷新进度条
5、setSize(int size):设置进度圈的大小,只有两个值:DEFAULT、LARGE
6、postDelayed(new Runable(),long min) 设置刷新延迟时间
7、isRefreshing():检查是否处于刷新状态
下拉刷新
布局,具体内容如下:
Activity核心代码如下:
swipeRefreshLayout = (SwipeRefreshLayout)findViewById(R.id.swipeLayout);
swipeRefreshLayout.setColorSchemeResources(R.color.swipe_color_1,
R.color.swipe_color_2,
R.color.swipe_color_3,
R.color.swipe_color_4);
swipeRefreshLayout.setSize(SwipeRefreshLayout.LARGE);;
swipeRefreshLayout.setProgressBackgroundColor(R.color.swipe_background_color);
//swipeRefreshLayout.setPadding(20, 20, 20, 20);
//swipeRefreshLayout.setProgressViewOffset(true, 100, 200);
//swipeRefreshLayout.setDistanceToTriggerSync(50);
swipeRefreshLayout.setProgressViewEndTarget(true, 100);
swipeRefreshLayout.setonRefreshListener(new onRefreshListener() {
@Override
public void onRefresh() {
new Thread(new Runnable() {
@Override
public void run() {
data.clear();
for(int i=0;i<20;i++){
data.add("SwipeRefreshLayout下拉刷新"+i);
}
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
mHandler.sendEmptyMessage(1);
}
}).start();
}
});
//handler
private Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case 1:
swipeRefreshLayout.setRefreshing(false);
adapter.notifyDataSetChanged();
//swipeRefreshLayout.setEnabled(false);
break;
default:
break;
}
}
};
原生实现上拉效果
通过监听滚动事件,对listview添加底部的组件实现上拉
implements AbsListView.onScrollListener {···
···
footerView = getLayoutInflater().inflate(R.layout.refresh_footview_layout, null);
lv.addFooterView(footerView);
lv.setonScrollListener(this);
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (adapter.getCount() == visibleLastIndex && scrollState == SCROLL_STATE_IDLE) {
Toast.makeText(this, "加载更多完成", Toast.LENGTH_SHORT).show();
footerView.setVisibility(View.GONE);
// new LoadDataThread().start();
}else {
footerView.setVisibility(View.VISIBLE);
// Toast.makeText(this, "加载更多...", Toast.LENGTH_SHORT).show();
}
}
2.自定义支持上拉刷新的组件
上拉刷新
实现下拉和上拉监听
···AppCompatActivity implements SwipeRefreshLayout.OnRefreshListener,
RefreshLayout.onLoadListener {
···
下拉和原先一样用法:
//下拉监听
swipeLayout.setonRefreshListener(this);
//上拉监听
swipeLayout.setonLoadListener(this);
@Override
public void onRefresh() {
swipeLayout.postDelayed(new Runnable() {
@Override
public void run() {
// 更新数据 更新完后调用该方法结束刷新
list.clear();
for (int i = 0; i < 8; i++) {
HashMap map = new HashMap();
map.put("itemImage", i + "刷新");
map.put("itemText", i + "刷新");
list.add(map);
}
adapter.notifyDataSetChanged();
swipeLayout.setRefreshing(false);
}
}, 2000);
}
@Override
public void onLoad() {
swipeLayout.postDelayed(new Runnable() {
@Override
public void run() {
// 更新数据 更新完后调用该方法结束刷新
swipeLayout.setLoading(false);
for (int i = 1; i < 10; i++) {
HashMap map = new HashMap();
map.put("itemImage", i + "更多");
map.put("itemText", i + "更多");
list.add(map);
}
adapter.notifyDataSetChanged();
}
}, 2000);
}
自定义组件如下:
public class RefreshLayout extends SwipeRefreshLayout implements
onScrollListener {
private int mTouchSlop;
private ListView mListView;
private onLoadListener mOnLoadListener;
private View mListViewFooter;
private int mYDown;
private int mLastY;
private boolean isLoading = false;
public RefreshLayout(Context context) {
this(context, null);
}
@SuppressLint("InflateParams")
public RefreshLayout(Context context, AttributeSet attrs) {
super(context, attrs);
mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
mListViewFooter = LayoutInflater.from(context).inflate(
R.layout.listview_footer, null, false);
}
@Override
protected void onLayout(boolean changed, int left, int top, int right,
int bottom) {
super.onLayout(changed, left, top, right, bottom);
// 初始化ListView对象
if (mListView == null) {
getListView();
}
}
private void getListView() {
int childs = getChildCount();
if (childs > 0) {
View childView = getChildAt(0);
if (childView instanceof ListView) {
mListView = (ListView) childView;
// 设置滚动监听器给ListView, 使得滚动的情况下也可以自动加载
mListView.setonScrollListener(this);
Log.d(VIEW_LOG_TAG, "### 找到listview");
}
}
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
final int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
// 按下
mYDown = (int) event.getRawY();
break;
case MotionEvent.ACTION_MOVE:
// 移动
mLastY = (int) event.getRawY();
break;
case MotionEvent.ACTION_UP:
// 抬起
if (canLoad()) {
loadData();
}
break;
default:
break;
}
return super.dispatchTouchEvent(event);
}
private boolean canLoad() {
return isBottom() && !isLoading && isPullUp();
}
private boolean isBottom() {
if (mListView != null && mListView.getAdapter() != null) {
return mListView.getLastVisiblePosition() == (mListView
.getAdapter().getCount() - 1);
}
return false;
}
private boolean isPullUp() {
return (mYDown - mLastY) >= mTouchSlop;
}
private void loadData() {
if (monLoadListener != null) {
// 设置状态
setLoading(true);
//
mOnLoadListener.onLoad();
}
}
public void setLoading(boolean loading) {
isLoading = loading;
if (isLoading) {
mListView.addFooterView(mListViewFooter);
} else {
mListView.removeFooterView(mListViewFooter);
mYDown = 0;
mLastY = 0;
}
}
public void setonLoadListener(onLoadListener loadListener) {
monLoadListener = loadListener;
}
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
}
@Override
public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
// 滚动时到了最底部也可以加载更多
if (canLoad()) {
loadData();
}
}
public static void setRefreshing(SwipeRefreshLayout refreshLayout,
boolean refreshing, boolean notify) {
Class extends SwipeRefreshLayout> refreshLayoutClass = refreshLayout
.getClass();
if (refreshLayoutClass != null) {
try {
Method setRefreshing = refreshLayoutClass.getDeclaredMethod(
"setRefreshing", boolean.class, boolean.class);
setRefreshing.setAccessible(true);
setRefreshing.invoke(refreshLayout, refreshing, notify);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
public static interface onLoadListener {
public void onLoad();
}
}
3.自定义支持ViewPage的刷新组件VPSwipeRefreshLayout
支持viewpager刷新
原生SwipeRefreshLayout会存在以下问题:
1、 SwipeRefreshLayout会吃掉ViewPager的滑动事件。
2、 SwipeRefreshLayout需要套在ScrollView和ListView上的时候才表现的比较友好,在其他ViewGroup上有点问题
重写后的SwipeRefreshLayout,直接复制到项目就可以使用了。
public class VpSwipeRefreshLayout extends SwipeRefreshLayout {
private float startY;
private float startX;
// 记录viewPager是否拖拽的标记
private boolean mIsVpDragger;
private final int mTouchSlop;
public VpSwipeRefreshLayout(Context context, AttributeSet attrs) {
super(context, attrs);
mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
// 记录手指按下的位置
startY = ev.getY();
startX = ev.getX();
// 初始化标记
mIsVpDragger = false;
break;
case MotionEvent.ACTION_MOVE:
// 如果viewpager正在拖拽中,那么不拦截它的事件,直接return false;
if(mIsVpDragger) {
return false;
}
// 获取当前手指位置
float endY = ev.getY();
float endX = ev.getX();
float distanceX = Math.abs(endX - startX);
float distanceY = Math.abs(endY - startY);
// 如果X轴位移大于Y轴位移,那么将事件交给viewPager处理。
if(distanceX > mTouchSlop && distanceX > distanceY) {
mIsVpDragger = true;
return false;
}
break;
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
// 初始化标记
mIsVpDragger = false;
break;
}
// 如果是Y轴位移大于X轴,事件交给swipeRefreshLayout处理。
return super.onInterceptTouchEvent(ev);
}
}
4.RecyclerView+SwpieRefreshLayout实现下拉刷新效果同时实现上拉功能
RecyclerView+SwpieRefreshLayout
RecyclerView实现的列表,默认情况下面是不带下拉刷新和上拉记载更多效果的,但是我在我们的实际项目当中,为了提高用户体验,这种效果一般都需要实现
SwipeRefreshLayout本身自带下拉刷新的效果,那么我们可以选择在RecyclerView布局外部嵌套一层SwipeRefreshLayout布局即可,具体布局文件如下:
在Activity中引用这个布局并初始化
@Override
protected void onCreate(Bundle savedInstanceState) {
//去除系统标题
this.requestWindowFeature(Window.FEATURE_NO_TITLE);
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recycle_view_refresh);
ButterKnife.bind(this);
initView();
}
private LinearLayoutManager linearLayoutManager;
MyRecyclerViewAdapter adapter;
private int lastVisibleItem;//记录滚动位置
private void initView() {
topBarTitle.setText("RecyclerView 刷新");
//设置刷新时动画的颜色,可以设置4个
swiperefreshLayout.setProgressBackgroundColorSchemeResource(android.R.color.white);
swiperefreshLayout.setColorSchemeResources(android.R.color.holo_blue_light,
android.R.color.holo_red_light, android.R.color.holo_orange_light,
android.R.color.holo_green_light);
// 这句话是为了,第一次进入页面的时候显示加载进度条
swiperefreshLayout.setProgressViewOffset(false, 0, (int) TypedValue
.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 24, getResources()
.getDisplayMetrics()));
//设置竖直方向
linearLayoutManager = new LinearLayoutManager(this);
linearLayoutManager.setOrientation(OrientationHelper.VERTICAL);
//设置管理器
recylerview.setLayoutManager(linearLayoutManager);
//添加分隔线
recylerview.addItemDecoration(new AdvanceDecoration(this, OrientationHelper.VERTICAL));
recylerview.setAdapter(adapter = new MyRecyclerViewAdapter(this));
swiperefreshLayout.setonRefreshListener(new SwipeRefreshLayout.onRefreshListener() {
@Override
public void onRefresh() {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
List newDatas = new ArrayList();
for (int i = 0; i < 5; i++) {
int index = i + 1;
newDatas.add("new item" + index);
}
adapter.addItem(newDatas);
swiperefreshLayout.setRefreshing(false);
Toast.makeText(RecycleViewRefreshActivity.this, "更新了五条数据...", Toast.LENGTH_SHORT).show();
}
}, 3000);
}
});
recylerview.setonScrollListener(new RecyclerView.onScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
//当滚动到底部时刷新数据
if (newState == RecyclerView.SCROLL_STATE_IDLE && lastVisibleItem + 1 == adapter.getItemCount()) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
List newDatas = new ArrayList();
for (int i = 0; i < 5; i++) {
int index = i + 1;
newDatas.add("more item" + index);
}
adapter.addMoreItem(newDatas);
swiperefreshLayout.setRefreshing(false);
}
}, 1000);
}
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
//获取滚动的最后位置
lastVisibleItem = linearLayoutManager.findLastVisibleItemPosition();
}
});
}
实现下拉刷新用SwipeRefreshLayout 自带的进度条, 上拉刷新用类似ListView的刷新 提示“加载中”等信息。
我们可以给RecyclerView 也添加一个类似FooterView的item。
我们在Adapter中实现:
class MyRecyclerViewAdapter extends RecyclerView.Adapter{//自定义viewHoder List datas; Context context; private static final int TYPE_ITEM = 0; private static final int TYPE_FOOTER = 1; public MyRecyclerViewAdapter(Context context) { this.context = context; this.datas = new ArrayList (); for (int i = 0; i < 20; i++) { int index = i + 1; datas.add("item" + index); } } //自定义的ViewHolder,持有每个Item的的所有界面元素 public class ViewHolder extends RecyclerView.ViewHolder { public ViewHolder(View view) { super(view); } } //自定义的ViewHolder,持有每个Item的的所有界面元素 public class ItemViewHolder extends ViewHolder { public TextView item_tv; public ItemViewHolder(View view) { super(view); item_tv = (TextView) view.findViewById(R.id.text); } } class FooterViewHolder extends ViewHolder { public FooterViewHolder(View view) { super(view); } } @Override public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { if (viewType == TYPE_FOOTER) { final View view = LayoutInflater.from(context).inflate(R.layout.refresh_footview_layout, parent, false); // view.setLayoutParams(new RecyclerView.LayoutParams(RecyclerView.LayoutParams.MATCH_PARENT, // RecyclerView.LayoutParams.WRAP_CONTENT)); FooterViewHolder viewHolder = new FooterViewHolder(view); return viewHolder; } else if (viewType == TYPE_ITEM) { final View view = LayoutInflater.from(context).inflate(R.layout.item_recycler_layout, parent, false); //这边可以做一些属性设置,甚至事件监听绑定 //view.setBackgroundColor(Color.RED); ItemViewHolder viewHolder = new ItemViewHolder(view); return viewHolder; } return null; } @Override public void onBindViewHolder(ViewHolder holder, int position) { if (holder instanceof ItemViewHolder) { //设置数据 ((ItemViewHolder) holder).item_tv.setText(datas.get(position)); ((ItemViewHolder) holder).item_tv.setTag(position); } } @Override public int getItemViewType(int position) { // 最后一个item设置为footerView if (position + 1 == getItemCount()) { return TYPE_FOOTER; } else { return TYPE_ITEM; } } // RecyclerView的count设置为数据总条数+ 1(footerView) @Override public int getItemCount() { return datas.size() + 1; } //添加数据 public void addItem(List newDatas) { //mTitles.add(position, data); //notifyItemInserted(position); newDatas.addAll(datas); datas.removeAll(datas); datas.addAll(newDatas); notifyDataSetChanged(); } public void addMoreItem(List newDatas) { datas.addAll(newDatas); adapter.notifyDataSetChanged(); } }
refresh_footview_layout
item_recycler_layout
项目源码:SwipeRefresh_jb51.rar
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持考高分网。



