ListView加一个头布局–下拉刷新思路

Android ListView加一个头布局–下拉刷新思路

1.首先是下拉要显示的界面先定义好

2.定义一个类继承ListView并实现里面所有的构造方法 并用此ListView做布局

public RefreshListView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        initView();
    }

    public RefreshListView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    public RefreshListView(Context context) {
        super(context);
        initView();
    }

3.定义一个方法使ListView加一个头布局 this.addHeaderView(下拉界面的view),测量V下拉iew的高度 : .measure(0, 0);
获取高度: getMeasureHeight(); 同时要把获取的高度首先隐藏 即把下拉界面padding一个负值

mHeaderView = View.inflate(getContext(), R.layout.list_refresh_header,
                null);
        this.addHeaderView(mHeaderView);// 添加头布局

        // 隐藏头布局(1, 获取头布局高度, 2.设置负paddingTop,布局就会往上走)
        // int height = mHeaderView.getHeight();//此处无法获取高度,因为布局还没有绘制完成
        // 绘制之前就要获取布局高度
        mHeaderView.measure(0, 0);// 手动测量布局
        mHeaderViewHeight = mHeaderView.getMeasuredHeight();// 测量之后的高度
        // 隐藏头布局
        mHeaderView.setPadding(0, -mHeaderViewHeight, 0, 0);

4.此时可以发现下拉刷新功能有三个状态 1.下拉刷新 2.松开刷新 3.正在刷新 默认为下拉刷新状态

private void refreshState() {
        switch (mCurrentState) {
        case STATE_PULL_TO_REFRESH:
            tvTitle.setText("下拉刷新");
            // 箭头向下移动
            ivArrow.startAnimation(animDown);
            // 隐藏进度条
            pbLoading.setVisibility(View.INVISIBLE);
            ivArrow.setVisibility(View.VISIBLE);
            break;
        case STATE_RELEASE_TO_REFRESH:
            tvTitle.setText("松开刷新");
            // 箭头向上移动
            ivArrow.startAnimation(animUp);
            // 隐藏进度条
            pbLoading.setVisibility(View.INVISIBLE);
            ivArrow.setVisibility(View.VISIBLE);
            break;
        case STATE_REFRESHING:
            tvTitle.setText("正在刷新...");
            pbLoading.setVisibility(View.VISIBLE);
            ivArrow.clearAnimation();// 必须清除动画,才能隐藏控件
            ivArrow.setVisibility(View.INVISIBLE);
            break;

        default:
            break;
        }
    }

5.要触碰ListView要使用onTouchEvent方法 当Action.Down时:获取触摸的一开始的高度 Action.Move时可以获取触摸变化时的高度Action.up获取抬起时滑到的结束高度。用滑动的高度减去开始的高度就是下拉刷新界面出现的高度dy 而此时下拉界面隐藏的部分为 dy减去隐藏的高度。 此时当dy大于0而且getFirstVisiblePosition == 0(item显示为第一个的时候)时就要时时监控设置paddingTop隐藏值
而paddingTop隐藏值也有状态 如果paddingTop隐藏值>=0(说明隐藏部分全显示出来了)并且不是松开刷新状态 此刻当前的状态变为松开刷新 再在里面写一个改变的方法; 如果PaddingTop隐藏值<0(说明隐藏部分还没显示出来)并且不是下拉刷新 此刻状态必然是下拉刷新状态并且写入方法。

public boolean onTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
        case MotionEvent.ACTION_DOWN:
            startY = (int) ev.getY();
            break;
        case MotionEvent.ACTION_MOVE:
            if (startY == -1) {// 如果用户按住头条新闻向下滑动, 会导致listview无法拿到ACTION_DOWN,
                                //因为此时事件会被交给父控件处理
                                // 此时要重新获取startY
                startY = (int) ev.getY();
            }

            // 如果当前正在刷新, 什么都不做了
            if (mCurrentState == STATE_REFRESHING) {
                break;
            }

            int endY = (int) ev.getY();
            int dy = endY - startY;

            if (dy > 0 && getFirstVisiblePosition() == 0) {// 向下滑动&当前显示的是第一个item,才允许下拉刷新
                int paddingTop = dy - mHeaderViewHeight;// 计算当前的paddingtop值

                // 根据padding切换状态
                if (paddingTop >= 0
                        && mCurrentState != STATE_RELEASE_TO_REFRESH) {
                    // 切换到松开刷新
                    mCurrentState = STATE_RELEASE_TO_REFRESH;
                    refreshState();
                } else if (paddingTop < 0
                        && mCurrentState != STATE_PULL_TO_REFRESH) {
                    // 切换到下拉刷新
                    mCurrentState = STATE_PULL_TO_REFRESH;
                    refreshState();
                }

                mHeaderView.setPadding(0, paddingTop, 0, 0);// 重新设置头布局padding
                return true;
            }

            break;

6.如果手指要抬起时同时此刻状态为松开刷新状态 那就要改为正在刷新状态 同时下拉刷新界面设置不被隐藏 并加入此刻正在刷新时的方法 并且在此时要用一个回调 如果手松开时的状态为下拉刷新状态 那么 直接隐藏界面不会刷新。

case MotionEvent.ACTION_UP:
            startY = -1;// 起始坐标归零

            if (mCurrentState == STATE_RELEASE_TO_REFRESH) {
                // 如果当前是松开刷新, 就要切换为正在刷新
                mCurrentState = STATE_REFRESHING;
                // 显示头布局
                mHeaderView.setPadding(0, 0, 0, 0);

                refreshState();

                // 下拉刷新回调
                if (mListener != null) {
                    mListener.onRefresh();
                }

            } else if (mCurrentState == STATE_PULL_TO_REFRESH) {
                // 隐藏头布局
                mHeaderView.setPadding(0, -mHeaderViewHeight, 0, 0);
            }

            break;

7定义一个接口 接口里面有个回调方法 然后再写一个方法装入接口的引用 这个方法暴露给注册监听者 同时在正在刷新的时候设置回调方法就可以进行一些网络请求操作 并且在网络请求操作判断是否成功 如果成功就对刷新进行初始化;如果失败的话就不进行初始化。

注:回调接口

public interface OnRefreshListener {
        // 下拉刷新的回调
        public void onRefresh();
    }

注:暴露一个方法供外类使用

private OnRefreshListener mListener;

    public void setOnRefreshListener(OnRefreshListener listener) {
        mListener = listener;
    }

注:正在刷新时的回调

// 下拉刷新回调
                if (mListener != null) {
                    mListener.onRefresh();
                }

当然还有一些其他的方法要绘制画面比如图片动画 文字显示 刷新时间显示 只要在各个状态方法里面实现就可以了!下面附上源码:
可以刷新页面的ListView类RefreshListView:

package com.song.view;

import java.text.SimpleDateFormat;
import java.util.Date;

import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.RotateAnimation;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.ProgressBar;
import android.widget.TextView;

import com.song.R;

/**
 * 下拉刷新的listview
 * 
 * @author song
 */
public class RefreshListView extends ListView {

    private static final int STATE_PULL_TO_REFRESH = 1;// 下拉刷新
    private static final int STATE_RELEASE_TO_REFRESH = 2;// 松开刷新
    private static final int STATE_REFRESHING = 3;// 正在刷新

    // 下拉刷新头布局
    private View mHeaderView;
    // 头布局高度
    private int mHeaderViewHeight;

    private int startY = -1;
    // 当前下拉刷新的状态
    private int mCurrentState = STATE_PULL_TO_REFRESH;// 默认是下拉刷新

    private TextView tvTitle;
    private ImageView ivArrow;
    private ProgressBar pbLoading;
    private TextView tvTime;

    private RotateAnimation animUp;// 箭头向上动画
    private RotateAnimation animDown;// 箭头向下动画

    public RefreshListView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        initView();
    }

    public RefreshListView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    public RefreshListView(Context context) {
        super(context);
        initView();
    }

    private void initView() {
        mHeaderView = View.inflate(getContext(), R.layout.list_refresh_header,
                null);
        this.addHeaderView(mHeaderView);// 添加头布局

        // 隐藏头布局(1, 获取头布局高度, 2.设置负paddingTop,布局就会往上走)
        // int height = mHeaderView.getHeight();//此处无法获取高度,因为布局还没有绘制完成
        // 绘制之前就要获取布局高度
        mHeaderView.measure(0, 0);// 手动测量布局
        mHeaderViewHeight = mHeaderView.getMeasuredHeight();// 测量之后的高度
        // 隐藏头布局
        mHeaderView.setPadding(0, -mHeaderViewHeight, 0, 0);

        tvTitle = (TextView) mHeaderView.findViewById(R.id.tv_title);
        ivArrow = (ImageView) mHeaderView.findViewById(R.id.iv_arrow);
        pbLoading = (ProgressBar) mHeaderView.findViewById(R.id.pb_loading);
        tvTime = (TextView) mHeaderView.findViewById(R.id.tv_time);

        initAnim();
        setCurrentTime();// 设置初始时间
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        switch (ev.getAction()) {
        case MotionEvent.ACTION_DOWN:
            startY = (int) ev.getY();
            break;
        case MotionEvent.ACTION_MOVE:
            if (startY == -1) {// 如果用户按住头条新闻向下滑动, 会导致listview无法拿到ACTION_DOWN,
                                //因为此时事件会被交给父控件处理
                                // 此时要重新获取startY
                startY = (int) ev.getY();
            }

            // 如果当前正在刷新, 什么都不做了
            if (mCurrentState == STATE_REFRESHING) {
                break;
            }

            int endY = (int) ev.getY();
            int dy = endY - startY;

            if (dy > 0 && getFirstVisiblePosition() == 0) {// 向下滑动&当前显示的是第一个item,才允许下拉刷新
                int paddingTop = dy - mHeaderViewHeight;// 计算当前的paddingtop值

                // 根据padding切换状态
                if (paddingTop >= 0
                        && mCurrentState != STATE_RELEASE_TO_REFRESH) {
                    // 切换到松开刷新
                    mCurrentState = STATE_RELEASE_TO_REFRESH;
                    refreshState();
                } else if (paddingTop < 0
                        && mCurrentState != STATE_PULL_TO_REFRESH) {
                    // 切换到下拉刷新
                    mCurrentState = STATE_PULL_TO_REFRESH;
                    refreshState();
                }

                mHeaderView.setPadding(0, paddingTop, 0, 0);// 重新设置头布局padding
                return true;
            }

            break;
        case MotionEvent.ACTION_UP:
            startY = -1;// 起始坐标归零

            if (mCurrentState == STATE_RELEASE_TO_REFRESH) {
                // 如果当前是松开刷新, 就要切换为正在刷新
                mCurrentState = STATE_REFRESHING;
                // 显示头布局
                mHeaderView.setPadding(0, 0, 0, 0);

                refreshState();

                // 下拉刷新回调
                if (mListener != null) {
                    mListener.onRefresh();
                }

            } else if (mCurrentState == STATE_PULL_TO_REFRESH) {
                // 隐藏头布局
                mHeaderView.setPadding(0, -mHeaderViewHeight, 0, 0);
            }

            break;

        default:
            break;
        }

        return super.onTouchEvent(ev);
    }

    /**
     * 初始化箭头动画
     */
    private void initAnim() {
        animUp = new RotateAnimation(0, -180, Animation.RELATIVE_TO_SELF, 0.5f,
                Animation.RELATIVE_TO_SELF, 0.5f);
        animUp.setDuration(500);
        animUp.setFillAfter(true);// 保持状态

        animDown = new RotateAnimation(-180, 0, Animation.RELATIVE_TO_SELF,
                0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
        animDown.setDuration(500);
        animDown.setFillAfter(true);// 保持状态
    }

    /**
     * 根据当前状态刷新界面
     */
    private void refreshState() {
        switch (mCurrentState) {
        case STATE_PULL_TO_REFRESH:
            tvTitle.setText("下拉刷新");
            // 箭头向下移动
            ivArrow.startAnimation(animDown);
            // 隐藏进度条
            pbLoading.setVisibility(View.INVISIBLE);
            ivArrow.setVisibility(View.VISIBLE);
            break;
        case STATE_RELEASE_TO_REFRESH:
            tvTitle.setText("松开刷新");
            // 箭头向上移动
            ivArrow.startAnimation(animUp);
            // 隐藏进度条
            pbLoading.setVisibility(View.INVISIBLE);
            ivArrow.setVisibility(View.VISIBLE);
            break;
        case STATE_REFRESHING:
            tvTitle.setText("正在刷新...");
            pbLoading.setVisibility(View.VISIBLE);
            ivArrow.clearAnimation();// 必须清除动画,才能隐藏控件
            ivArrow.setVisibility(View.INVISIBLE);
            break;

        default:
            break;
        }
    }

    private OnRefreshListener mListener;

    public void setOnRefreshListener(OnRefreshListener listener) {
        mListener = listener;
    }

    /**
     * 设置上次刷新时间
     */
    private void setCurrentTime() {
        // 08:10 8:10 1
        SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// HH表示24小时制
        String time = format.format(new Date());
        tvTime.setText(time);
    }

    // 刷新完成
    public void onRefreshComplete(boolean success) {
        // 隐藏头布局
        mHeaderView.setPadding(0, -mHeaderViewHeight, 0, 0);
        mCurrentState = STATE_PULL_TO_REFRESH;
        // 隐藏进度条
        pbLoading.setVisibility(View.INVISIBLE);
        ivArrow.setVisibility(View.VISIBLE);
        tvTitle.setText("下拉刷新");

        // 刷新失败,不需要更新时间
        if (success) {
            setCurrentTime();
        }
    }

    public interface OnRefreshListener {
        // 下拉刷新的回调
        public void onRefresh();
    }
}

操作类中的方法:

lvList.setOnRefreshListener(new OnRefreshListener() {

            @Override
            public void onRefresh() {
                // 从网络加载数据
                getDataFromServer();
            }
        });
private void getDataFromServer() {
        HttpUtils utils = new HttpUtils();
        utils.send(HttpMethod.GET, mUrl, new RequestCallBack<String>() {

            @Override
            public void onSuccess(ResponseInfo<String> responseInfo) {
                String result = responseInfo.result;
                processResult(result);

                System.out.println("访问网络成功!!!");
                CacheUtils.setCache(mUrl, result, mActivity);

                // 收起下拉刷新控件
                lvList.onRefreshComplete(true);
            }

            @Override
            public void onFailure(HttpException error, String msg) {
                // 收起下拉刷新控件
                lvList.onRefreshComplete(false);

                error.printStackTrace();
                Toast.makeText(mActivity, msg, Toast.LENGTH_SHORT).show();
            }
        });

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值