package com.xmn.util.widget;
import android.annotation.SuppressLint;
import android.content.Context;
import android.os.Handler;
import android.util.AttributeSet;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewTreeObserver;
import android.view.ViewTreeObserver.OnPreDrawListener;
import android.view.animation.Animation;
import android.view.animation.RotateAnimation;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
import com.xmn.R;
/**
* @类名:RefreshLayout
* @功能描述:能下来刷新的ListView<br>
*
* ①把布局作为需要刷新的View的父View<br>
* ②设置监听: setOnRefreshDataListener,在接口方法中开始刷新数据<br>
* ③setFlag()方法开启下拉刷新(默认无法下拉刷新)<br>
* ④刷新完成,调用refreshDataFinish()方法隐藏界面<br>
* @作者:XuanKe'Huang
* @时间:2014-10-15 上午9:55:21
* @Copyright 2014
*/
@SuppressLint({ "InflateParams", "HandlerLeak" })
public class RefreshLayout extends LinearLayout {
@SuppressLint("NewApi")
public RefreshLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init(context);
}
public RefreshLayout(Context context, AttributeSet attrs) {
super(context, attrs);
init(context);
}
public RefreshLayout(Context context) {
super(context);
init(context);
}
private void init(Context context) {
refreshView = LayoutInflater.from(context).inflate(
R.layout.refreshlayout, null);
addView(refreshView);
refreshImageView = (ImageView) refreshView
.findViewById(R.id.imageview_refresh);
refreshTextView = (TextView) refreshView
.findViewById(R.id.textview_refresh);
/**
* 设置布局监听,当布局完完成了以后,首先要隐藏头部
*/
ViewTreeObserver observer = getViewTreeObserver();
observer.addOnPreDrawListener(new OnPreDrawListener() {
@Override
public boolean onPreDraw() {
if (refreshView.getHeight() > -tmpHeight) {
tmpHeight = -refreshView.getHeight();
bottonPadding = getPaddingBottom();
setPadding(0, tmpHeight, 0, 0);
}
return true;
}
});
}
private View refreshView;// 刷新的Layout
private ImageView refreshImageView;// 刷新图片
private TextView refreshTextView;// 刷新提示
private boolean flag = false;// 是否可以下拉刷新表示为,默认无法下拉刷新,需要设置
private int startY = 0;// 开始点
private int moveDistance = 0;// 移动的距离
private int tmpHeight = 0;// 顶部内边距
private int bottonPadding = 0;// 底部内边距
private boolean ifMove = false;// 是否有移动
private RefreshDataListener listener;
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
if (flag) {// 只有设置为true的时候才可以下拉刷新
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:// 手指按下
tmpHeight = getPaddingTop();
bottonPadding = getPaddingBottom();
startY = (int) ev.getRawY();
break;
case MotionEvent.ACTION_MOVE:// 手指移动
moveDistance = (int) (ev.getRawY() - startY);
ifMove = true;// 手指有移动
Log.e("moveDistance", moveDistance + " " + getScrollY() + " "
+ refreshView.getHeight());
if (moveDistance > 50) {// 只有Layout滚到到了头部,才可以下拉刷新
if (moveDistance > refreshView.getHeight()) {// 如果下拉的高度超过的了下拉刷新Layout的高度,就提示可以放手了
refreshImageView.setImageResource(R.drawable.finger_up);
refreshTextView.setText("松手刷新");
}
setPadding(getPaddingLeft(), tmpHeight + moveDistance,
getPaddingRight(), bottonPadding - moveDistance);
}
break;
case MotionEvent.ACTION_UP:// 手指离开
if (ifMove) {// 必须移动
ifMove = false;
Log.e("up", moveDistance + " " + refreshView.getHeight());
if (moveDistance > refreshView.getHeight()) {// 如果下拉的高度超过了Layout的高度,就显示正在刷新,并开始执行刷新
refreshImageView
.setImageResource(R.drawable.progress_wait);
refreshTextView.setText("正在刷新...");
setPadding(getPaddingLeft(), 0, getPaddingRight(),
bottonPadding - refreshView.getHeight());
listener.startReLoadData();
handler.sendEmptyMessage(10087);
} else {// 下载高度不足,重新隐藏界面
setPadding(getPaddingLeft(), -refreshView.getHeight(),
getPaddingRight(), getPaddingBottom());
}
if (moveDistance > 50) {// 有过下拉的手势操作,需要吸收手势
Log.e("手势被吸收", "true");
return true;// 吸收掉手势
}
}
moveDistance = 0;// 把距离重新设置为0
startY = 0;// 把接触点重置为0
default:
break;
}
}
return super.dispatchTouchEvent(ev);
}
/**
* 方法名: setOnRefreshDataListener
*
* 功能描述:设置刷新数据监听
*
* @param listener
* 监听回调
* @param childView
* 子布局View
* @return void
*
* </br>throws
*/
public void setOnRefreshDataListener(RefreshDataListener listener,
View childView) {
this.listener = listener;
childView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
}
});
}
/**
* @类名:RefreshDataListener
* @功能描述:刷新数据监听接口
* @作者:XuanKe'Huang
* @时间:2014-10-15 下午2:42:20
* @Copyright 2014
*/
public interface RefreshDataListener {
/**
* 方法名: startReLoadData
*
* 功能描述:开始刷新数据
*
* @return void
*
* </br>throws
*/
public void startReLoadData();
}
/**
* 方法名: refreshDataFinish
*
* 功能描述:数据完成,可调用次方法,隐藏刷新Layout
*
* @return void
*
* </br>throws
*/
public void refreshDataFinish() {
handler.sendEmptyMessage(10088);// 停止动画
refreshImageView.setImageResource(R.drawable.enter);
refreshTextView.setText("刷新成功");
handler.sendEmptyMessageDelayed(10086, 1500);
}
private Handler handler = new Handler() {
private boolean stop = false;// 停止动画
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case 10088:// 刷新完成,停止动画
stop = true;
case 10087:// 正在刷新,启动动画
if (!stop) {
Animation animation = new RotateAnimation(0, 360,
Animation.RELATIVE_TO_SELF, 0.5f,
Animation.RELATIVE_TO_SELF, 0.5f);
animation.setDuration(300);
refreshImageView.startAnimation(animation);
handler.sendEmptyMessageDelayed(10087, 280);
}
break;
case 10086:// 隐藏界面
stop = false;
refreshImageView.setImageResource(R.drawable.finger_move);
refreshTextView.setText("下拉刷新");
setPadding(getPaddingLeft(), -refreshView.getHeight(),
getPaddingRight(), getPaddingBottom());// 重新隐藏下拉刷新界面
break;
default:
break;
}
};
};
/**
* 方法名: setFlag
*
* 功能描述:设置是否禁止下拉刷新
*
* @param flag
* @return void
*
* </br>throws
*/
public void setFlag(boolean flag) {
this.flag = flag;
}
}
xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/black" >
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="70dp"
android:layout_centerInParent="true"
android:orientation="horizontal"
android:padding="10dp" >
<ImageView
android:id="@+id/imageview_refresh"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:scaleType="fitXY"
android:src="@drawable/finger_move" />
<TextView
android:id="@+id/textview_refresh"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:singleLine="true"
android:text="下拉可刷新"
android:textColor="@color/white" />
</LinearLayout>
</RelativeLayout>