android编程之下拉刷新实现方法分析

本文实例讲述了android编程之下拉刷新实现方法。分享给大家供大家参考,具体如下:

现在android应用里面下拉刷新比较多 ,今天自己研究一下相关的资料

public class MyListView extends ListView implements OnScrollListener {
 private static final String TAG = "listview";
 private final static int RELEASE_To_REFRESH = 0;
 private final static int PULL_To_REFRESH = 1;
 private final static int REFRESHING = 2;
 private final static int DONE = 3;
 private final static int LOADING = 4;
 // 实际的padding的距离与界面上偏移距离的比例
 private final static int RATIO = 3;
 private LayoutInflater inflater;
 private LinearLayout headView;
 private TextView tipsTextview;
 private TextView lastUpdatedTextView;
 private ImageView arrowImageView;
 private ProgressBar progressBar;
 private RotateAnimation animation;
 private RotateAnimation reverseAnimation;
 // 用于保证startY的值在一个完整的touch事件中只被记录一次
 private boolean isRecored;
 private int headContentWidth;
 private int headContentHeight;
 private int startY;
 private int firstItemIndex;
 private int state;
 private boolean isBack;
 private OnRefreshListener refreshListener;
 private boolean isRefreshable;
 public MyListView(Context context) {
  super(context);
  init(context);
 }
 public MyListView(Context context, AttributeSet attrs) {
  super(context, attrs);
  init(context);
 }
 private void init(Context context) {
  setCacheColorHint(context.getResources().getColor(R.color.transparent));
  inflater = LayoutInflater.from(context);
  headView = (LinearLayout) inflater.inflate(R.layout.head, null);
  arrowImageView = (ImageView) headView
    .findViewById(R.id.head_arrowImageView);
  arrowImageView.setMinimumWidth(70);
  arrowImageView.setMinimumHeight(50);
  progressBar = (ProgressBar) headView
    .findViewById(R.id.head_progressBar);
  tipsTextview = (TextView) headView.findViewById(R.id.head_tipsTextView);
  lastUpdatedTextView = (TextView) headView
    .findViewById(R.id.head_lastUpdatedTextView);
  measureView(headView);
  headContentHeight = headView.getMeasuredHeight();
  headContentWidth = headView.getMeasuredWidth();
  headView.setPadding(0, -1 * headContentHeight, 0, 0);
  headView.invalidate();
  Log.v("size", "width:" + headContentWidth + " height:"
    + headContentHeight);
  addHeaderView(headView, null, false);
  setOnScrollListener(this);
  animation = new RotateAnimation(0, -180,
    RotateAnimation.RELATIVE_TO_SELF, 0.5f,
    RotateAnimation.RELATIVE_TO_SELF, 0.5f);
  animation.setInterpolator(new LinearInterpolator());
  animation.setDuration(250);
  animation.setFillAfter(true);
  reverseAnimation = new RotateAnimation(-180, 0,
    RotateAnimation.RELATIVE_TO_SELF, 0.5f,
    RotateAnimation.RELATIVE_TO_SELF, 0.5f);
  reverseAnimation.setInterpolator(new LinearInterpolator());
  reverseAnimation.setDuration(200);
  reverseAnimation.setFillAfter(true);
  state = DONE;
  isRefreshable = false;
 }
 public void onScroll(AbsListView arg0, int firstVisiableItem, int arg2,
   int arg3) {
  firstItemIndex = firstVisiableItem;
 }
 public void onScrollStateChanged(AbsListView arg0, int arg1) {
 }
 public boolean onTouchEvent(MotionEvent event) {
  if (isRefreshable) {
   switch (event.getAction()) {
   case MotionEvent.ACTION_DOWN:
    if (firstItemIndex == 0 && !isRecored) {
     isRecored = true;
     startY = (int) event.getY();
     Log.v(TAG, "在down时候记录当前位置‘");
    }
    break;
   case MotionEvent.ACTION_UP:
    if (state != REFRESHING && state != LOADING) {
     if (state == DONE) {
      // 什么都不做
     }
     if (state == PULL_To_REFRESH) {
      state = DONE;
      changeHeaderViewByState();
      Log.v(TAG, "由下拉刷新状态,到done状态");
     }
     if (state == RELEASE_To_REFRESH) {
      state = REFRESHING;
      changeHeaderViewByState();
      onRefresh();
      Log.v(TAG, "由松开刷新状态,到done状态");
     }
    }
    isRecored = false;
    isBack = false;
    break;
   case MotionEvent.ACTION_MOVE:
    int tempY = (int) event.getY();
    if (!isRecored && firstItemIndex == 0) {
     Log.v(TAG, "在move时候记录下位置");
     isRecored = true;
     startY = tempY;
    }
    if (state != REFRESHING && isRecored && state != LOADING) {
     // 保证在设置padding的过程中,当前的位置一直是在head,否则如果当列表超出屏幕的话,当在上推的时候,列表会同时进行滚动
     // 可以松手去刷新了
     if (state == RELEASE_To_REFRESH) {
      setSelection(0);
      // 往上推了,推到了屏幕足够掩盖head的程度,但是还没有推到全部掩盖的地步
      if (((tempY - startY) / RATIO < headContentHeight)
        && (tempY - startY) > 0) {
       state = PULL_To_REFRESH;
       changeHeaderViewByState();
       Log.v(TAG, "由松开刷新状态转变到下拉刷新状态");
      }
      // 一下子推到顶了
      else if (tempY - startY <= 0) {
       state = DONE;
       changeHeaderViewByState();
       Log.v(TAG, "由松开刷新状态转变到done状态");
      }
      // 往下拉了,或者还没有上推到屏幕顶部掩盖head的地步
      else {
       // 不用进行特别的操作,只用更新paddingTop的值就行了
      }
     }
     // 还没有到达显示松开刷新的时候,DONE或者是PULL_To_REFRESH状态
     if (state == PULL_To_REFRESH) {
      setSelection(0);
      // 下拉到可以进入RELEASE_TO_REFRESH的状态
      if ((tempY - startY) / RATIO >= headContentHeight) {
       state = RELEASE_To_REFRESH;
       isBack = true;
       changeHeaderViewByState();
       Log.v(TAG, "由done或者下拉刷新状态转变到松开刷新");
      }
      // 上推到顶了
      else if (tempY - startY <= 0) {
       state = DONE;
       changeHeaderViewByState();
       Log.v(TAG, "由DOne或者下拉刷新状态转变到done状态");
      }
     }
     // done状态下
     if (state == DONE) {
      if (tempY - startY > 0) {
       state = PULL_To_REFRESH;
       changeHeaderViewByState();
      }
     }
     // 更新headView的size
     if (state == PULL_To_REFRESH) {
      headView.setPadding(0, -1 * headContentHeight
        + (tempY - startY) / RATIO, 0, 0);
     }
     // 更新headView的paddingTop
     if (state == RELEASE_To_REFRESH) {
      headView.setPadding(0, (tempY - startY) / RATIO
        - headContentHeight, 0, 0);
     }
    }
    break;
   }
  }
  return super.onTouchEvent(event);
 }
 // 当状态改变时候,调用该方法,以更新界面
 private void changeHeaderViewByState() {
  switch (state) {
  case RELEASE_To_REFRESH:
   arrowImageView.setVisibility(View.VISIBLE);
   progressBar.setVisibility(View.GONE);
   tipsTextview.setVisibility(View.VISIBLE);
   lastUpdatedTextView.setVisibility(View.VISIBLE);
   arrowImageView.clearAnimation();
   arrowImageView.startAnimation(animation);
   tipsTextview.setText("松开刷新");
   Log.v(TAG, "当前状态,松开刷新");
   break;
  case PULL_To_REFRESH:
   progressBar.setVisibility(View.GONE);
   tipsTextview.setVisibility(View.VISIBLE);
   lastUpdatedTextView.setVisibility(View.VISIBLE);
   arrowImageView.clearAnimation();
   arrowImageView.setVisibility(View.VISIBLE);
   // 是由RELEASE_To_REFRESH状态转变来的
   if (isBack) {
    isBack = false;
    arrowImageView.clearAnimation();
    arrowImageView.startAnimation(reverseAnimation);
    tipsTextview.setText("下拉刷新");
   } else {
    tipsTextview.setText("下拉刷新");
   }
   Log.v(TAG, "当前状态,下拉刷新");
   break;
  case REFRESHING:
   headView.setPadding(0, 0, 0, 0);
   progressBar.setVisibility(View.VISIBLE);
   arrowImageView.clearAnimation();
   arrowImageView.setVisibility(View.GONE);
   tipsTextview.setText("正在刷新...");
   lastUpdatedTextView.setVisibility(View.VISIBLE);
   Log.v(TAG, "当前状态,正在刷新...");
   break;
  case DONE:
   headView.setPadding(0, -1 * headContentHeight, 0, 0);
   progressBar.setVisibility(View.GONE);
   arrowImageView.clearAnimation();
   arrowImageView.setImageResource(R.drawable.arrow);
   tipsTextview.setText("下拉刷新");
   lastUpdatedTextView.setVisibility(View.VISIBLE);
   Log.v(TAG, "当前状态,done");
   break;
  }
 }
 public void setonRefreshListener(OnRefreshListener refreshListener) {
  this.refreshListener = refreshListener;
  isRefreshable = true;
 }
 public interface OnRefreshListener {
  public void onRefresh();
 }
 public void onRefreshComplete() {
  state = DONE;
  lastUpdatedTextView.setText("最近更新:" + new Date().toLocaleString());
  changeHeaderViewByState();
 }
 private void onRefresh() {
  if (refreshListener != null) {
   refreshListener.onRefresh();
  }
 }
 // 此方法直接照搬自网络上的一个下拉刷新的demo,此处是“估计”headView的width以及height
 private void measureView(View child) {
  ViewGroup.LayoutParams p = child.getLayoutParams();
  if (p == null) {
   p = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,
     ViewGroup.LayoutParams.WRAP_CONTENT);
  }
  int childWidthSpec = ViewGroup.getChildMeasureSpec(0, 0 + 0, p.width);
  int lpHeight = p.height;
  int childHeightSpec;
  if (lpHeight > 0) {
   childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight,
     MeasureSpec.EXACTLY);
  } else {
   childHeightSpec = MeasureSpec.makeMeasureSpec(0,
     MeasureSpec.UNSPECIFIED);
  }
  child.measure(childWidthSpec, childHeightSpec);
 }
 public void setAdapter(BaseAdapter adapter) {
  lastUpdatedTextView.setText("最近更新:" + new Date().toLocaleString());
  super.setAdapter(adapter);
 }
}

希望本文所述对大家Android程序设计有所帮助。

(0)

相关推荐

  • Android编程之SMS读取短信并保存到SQLite的方法

    本文实例讲述了Android编程之SMS读取短信并保存到SQLite的方法.分享给大家供大家参考,具体如下: Android 之 SMS 短信在Android系统中是保存在SQLite数据库中的,但不让其它程序访问(Android系统的安全机制) 现在我们在读取手机内的SMS短信,先保存在我们自己定义的SQLite数据库中,然后读取SQLite数据库提取短信,并显示 SMS短信SQLite存取代码: package com.homer.sms; import java.sql.Date; imp

  • Android编程实现3D滑动旋转效果的方法

    本文实例讲述了Android编程实现3D滑动旋转效果的方法.分享给大家供大家参考,具体如下: 这里我们通过代码实现一些滑动翻页的动画效果. Animation实现动画有两个方式:帧动画(frame-by-frame animation)和补间动画(tweened animation) 本示例通过继承Animation自定义Rotate3D,实现3D翻页效果.效果图如下: 1.Rotate3D(Animation) 首先,自定义Animation的3D动画类Rotate3D public clas

  • android编程实现对话框的封装实例

    本文实例讲述了android编程实现对话框的封装.分享给大家供大家参考,具体如下: /** * 对话框 */ private static ProgressDialog mProgressDialog; /** * 默认的对话框 * 标题.内容.两个按钮 * @param context * @param title * @param content * @param btnOKStr 是空字符的话 该按钮不显示 (特别注意) * @param btnCancelStr 是空字符的话 该按钮不显

  • Android 定时任务过程详解

    在Android开发中,通过以下三种方法定时执行任务: 一.采用Handler与线程的sleep(long)方法(不建议使用,java的实现方式) 二.采用Handler的postDelayed(Runnable, long)方法(最简单的android实现) 三.采用Handler与timer及TimerTask结合的方法(比较多的任务时建议使用) android里有时需要定时循环执行某段代码,或者需要在某个时间点执行某段代码,这个需求大家第一时间会想到Timer对象,没错,不过我们还有更好的

  • Android编程中Perferences的用法实例分析

    本文实例讲述了Android编程中Perferences的用法.分享给大家供大家参考,具体如下: 浏览手机的/data/data/目录中的各个package的目录,经常会看到有一个shared_prefs文件夹,里面有一个包名_preferences.xml文件,这个文件就是要说的主角,下面简称他为配置文件: 这个文件类似于一个配置文件的角色,记录应用的一些属性值,比如如果你的应用提供了指导用户使用的向导功能,那么估计肯定会提供选项让用户来关闭这个功能,那么你可以将这个开关放置在这个文件里,下次

  • Android 读取Properties配置文件的小例子

    开发应用的时候会有一些有可能会变得值,例如webservice地址 应用的一些ID等等,之前一直都是直接在应用中改代码,不是忘点这忘点那,于是想到了可以用Properties配置文件,我把网址等变量配置的配置文件中,这样之后再改的话就直接改配置文件就行了,就不用改代码了下面给大家说说Properties的用法 复制代码 代码如下: public static String getPropertiesURL(Context c, String s) {  String url = null;  P

  • Android开发之menu菜单

    Android系统里面有四种类型的菜单:options menu(选项菜单),context menu(上下文菜单),sub menu(子菜单),Popup menu(弹出菜单). 首先说 选项菜单(OptionsMenu) 一.方法介绍: public booleanonCreateOptionsMenu(Menu menu):使用此方法调用OptionsMenu . public booleanonOptionsItemSelected(MenuItem item):选中菜单项后发生的动作.

  • android编程实现图片库的封装方法

    本文实例讲述了android编程实现图片库的封装方法.分享给大家供大家参考,具体如下: 大家在做安卓应用的时候 经常要从网络中获取图片 都是通过URL去获取 可是如果本地有图片数据 从本地获取数据不更加快一些 自己在工作中遇到这个问题 所以采用了一个URL和本地图片的一个映射关系 先从本地区获取 假如本地没有再从网络中获取 本方法考虑到多线程问题 欢迎大家一起共同探讨! public class PictureLibrary { /* * 图片库的操作 */ File file; URL url

  • android编程实现系统图片剪裁的方法

    本文实例讲述了android编程实现系统图片剪裁的方法.分享给大家供大家参考,具体如下: package cn.test; import java.io.File; import java.text.SimpleDateFormat; import java.util.Date; import android.app.Activity; import android.content.ContentResolver; import android.content.ContentUris; impo

  • AndroidManifest.xml配置文件解析

    AndroidManifest.xml配置文件对于Android应用开发来说是非常重要的基础知识,本文旨在总结该配置文件中重点的用法,以便日后查阅.下面是一个标准的AndroidManifest.xml文件样例. 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> <manifest> <!-- 基本配置 --> <uses-permission /> <permi

  • android通过配置文件设置应用安装到SD卡上的方法

    安装应用时,有时会出现存储空间不足的错误,这时就只能将应用安装到SD卡,在AndroidManifest.xml文件的manifest里面添加下面一行: 复制代码 代码如下: android:installLocation="preferExternal" 这样应用就会被安装在SD卡,不会出现存储空间不足的错误了.

随机推荐