全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

Android自定义控件实现下拉刷新效果

app开发中下拉刷新是最常接触到的一个功能,也有很多开源的框架,封装的非常棒。前段时间了解了一下ViewDragHelper,遂用它实现了下拉刷新的功能。

大概和我之前的ViewDragHelper之拖动加载(类似淘宝)这篇代码类似。只是做了相关改动。具体的可以看一下那篇博文了解一下用到的ViewDragHelper的一些知识点。该界面主要是一个LinearLayout,上面的下拉刷新是一个textview(用TV代替),当然这个可以定制,在此只是用一个textview代替,实现简单的功能,下面是一个listview(用LV代替),当然listview也是可以定制的,可以使gridview或者其他你想要的都可以,在此也是只用Listview代替。大概的讲讲吧:

首先,在onLayout中将TV置于屏幕上方,将LV充满屏幕;

上图中蓝色部分是整个手机的屏幕,红色部分是下拉提示TV。TV是置于屏幕之外的是不显示的。

@Override 
 protected void onLayout(boolean changed, int l, int t, int r, int b) { 
  if (pullText.getTop() == 0) { 
   viewHeight = pullText.getMeasuredHeight(); 
   pullText.layout(l, 0, r, b); 
   myList.layout(l, 0, r, b); 
   pullText.offsetTopAndBottom(-viewHeight); 
  } else { 
   pullText.layout(l, pullText.getTop(), r, pullText.getBottom()); 
   myList.layout(l, myList.getTop(), r, myList.getBottom()); 
  } 
 } 

 上面的代码段中,pullText即是TV,myList是LV。这样在下拉LV的时候,TV就会跟着往下走,所以就会出现在屏幕中实现我们想要的效果。

 /** 
  * 这是拖拽效果的主要逻辑 
  */ 
 private class DragHelperCallback extends ViewDragHelper.Callback { 
 
  @Override 
  public void onViewPositionChanged(View changedView, int left, int top, 
    int dx, int dy) { 
   int childIndex = 1; 
   if (changedView == myList) { 
    childIndex = 2; 
   } 
   onViewPosChanged(childIndex, top); 
  } 
 
  @Override 
  public boolean tryCaptureView(View child, int pointerId) { 
   return true; 
  } 
 
  @Override 
  public int getViewVerticalDragRange(View child) { 
   return 1; 
  } 
 
  @Override 
  public void onViewReleased(View releasedChild, float xvel, float yvel) { 
   refreshOrNot(releasedChild, yvel); 
  } 
 
  @Override 
  public int clampViewPositionVertical(View child, int top, int dy) { 
   int finalTop = top; 
   if (child == pullText) { 
    if (top > 0) { 
     finalTop = 0; 
    } 
   } else if (child == myList) { 
    if (top < 0) { 
     finalTop = 0; 
    } 
    if(top >= viewHeight){ 
     pullText.setText("松开刷新"); 
    }else{ 
     pullText.setText("下拉刷新"); 
    } 
   } 
   return child.getTop() + (finalTop - child.getTop()) / 2; 
  } 
 }

上面的代码段中,主要是在clampViewPositionVertical中判断滑动的位置,作用的子view。其他就不多说了,大致和之前的博客相同。主要说说onViewReleased吧。在此函数中是在用户手势抬起时响应的,所以我们在此实现下拉后的刷新。我们先定义一个接口,以便在刷新的时候调用。

public interface pulltorefreshNotifier { 
  public void onPull(); 
 } 
public void setpulltorefreshNotifier(pulltorefreshNotifier pullNotifier) { 
  this.pullNotifier = pullNotifier; 
 } 
private void refreshOrNot(View releasedChild, float yvel) { 
  int finalTop = 0; 
  if (releasedChild == pullText) { 
   // 拖动第一个view松手 
   if (yvel < -50) { 
    finalTop = 0; 
   } else { 
    finalTop = viewHeight; 
   } 
  } else { 
   // 拖动第二个view松手 
   if (yvel > viewHeight - 5 || releasedChild.getTop() >= viewHeight) { 
    finalTop = viewHeight; 
    if (null != pullNotifier) { 
     pullNotifier.onPull(); 
    } 
    pullText.setText("正在刷新"); 
   } 
  } 
 
  if (VDH.smoothSlideViewTo(myList, 0, finalTop)) { 
   ViewCompat.postInvalidateOnAnimation(this); 
  } 
 } 

拖动第二个view时,也就是LV时,我们判断一下是否需要刷新,需要刷新则执行onPull();
然后我们来看一下主要的Activity:

package com.maxi.pulltorefreshtest; 
 
import android.annotation.SuppressLint; 
import android.app.Activity; 
import android.os.Bundle; 
import android.os.Handler; 
import android.os.Message; 
 
import com.maxi.pulltorefreshtest.adapter.ProjectAdapter; 
import com.maxi.pulltorefreshtest.widget.MyListView; 
import com.maxi.pulltorefreshtest.widget.PullToRefreshGroup; 
import com.maxi.pulltorefreshtest.widget.PullToRefreshGroup.pulltorefreshNotifier; 
 
public class MainActivity extends Activity { 
 private PullToRefreshGroup pullListgroup; 
 private boolean isDown = false; 
 private MyListView myList; 
 private ProjectAdapter pa; 
 @Override 
 protected void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.activity_main); 
  findView(); 
  init(); 
 } 
 
 private void findView() { 
  pullListgroup = (PullToRefreshGroup) findViewById(R.id.pulltorefresh); 
  myList = pullListgroup.returnMylist(); 
 } 
 
 private void init() { 
  pulltorefreshNotifier pullNotifier = new pulltorefreshNotifier() { 
   @Override 
   public void onPull() { 
    // TODO Auto-generated method stub 
    downLoad(); 
   } 
  }; 
  pullListgroup.setpulltorefreshNotifier(pullNotifier); 
  pa = new ProjectAdapter(this); 
  myList.setAdapter(pa); 
  pa.notifyDataSetChanged(); 
 } 
 
 private void downLoad() { 
  if (!isDown) { 
   isDown = true; 
   new Thread(new Runnable() { 
 
    @Override 
    public void run() { 
     // TODO Auto-generated method stub 
     try { 
      Thread.sleep(2000); 
      handler.sendEmptyMessage(1); 
     } catch (InterruptedException e) { 
//      TODO Auto-generated catch block 
      e.printStackTrace(); 
     } 
    } 
   }).start(); 
  } 
 } 
 
 @SuppressLint("HandlerLeak") 
 private Handler handler = new Handler() { 
 
  @Override 
  public void handleMessage(Message msg) { 
   // TODO Auto-generated method stub 
   super.handleMessage(msg); 
   switch (msg.what) { 
   case 1: 
    pullListgroup.refreshComplete(); 
    isDown = false; 
    break; 
   default: 
    break; 
   } 
  } 
 
 }; 
} 

我们在他刷新的时候执行downLoad();刷新数据。为了达到效果可以看出我让线程暂停2s。然后调用refreshComplete();

public void refreshComplete() { 
 if (VDH.smoothSlideViewTo(myList, 0, 0)) { 
  ViewCompat.postInvalidateOnAnimation(this); 
 } 
} 

实现刷新好后让TV继续返回屏幕上方。

上段代码中我们发现MyListView是重写的ListView,主要是处理手势事件的。

package com.maxi.pulltorefreshtest.widget; 
 
import android.content.Context; 
import android.util.AttributeSet; 
import android.view.MotionEvent; 
import android.view.View; 
import android.widget.ListView; 
 
public class MyListView extends ListView { 
 boolean allowDragBottom = true; 
 float downY = 0; 
 boolean needConsumeTouch = true; 
 public MyListView(Context context){ 
  super(context); 
 } 
 public MyListView(Context context, AttributeSet attrs) { 
  super(context, attrs); 
  // TODO Auto-generated constructor stub 
 } 
 
 @Override 
 public boolean dispatchTouchEvent(MotionEvent ev) { 
  if (ev.getAction() == MotionEvent.ACTION_DOWN) { 
   downY = ev.getRawY(); 
   needConsumeTouch = true; 
   if (getMyScrollY() == 0) { 
    allowDragBottom = true; 
   } else { 
    allowDragBottom = false; 
   } 
  } else if (ev.getAction() == MotionEvent.ACTION_MOVE) { 
   if (!needConsumeTouch) { 
    getParent().requestDisallowInterceptTouchEvent(false); 
    return false; 
   } else if (allowDragBottom) { 
    if (downY - ev.getRawY() < -2) { 
     needConsumeTouch = false; 
     getParent().requestDisallowInterceptTouchEvent(false); 
     return false; 
    } 
   } 
  } 
  getParent().requestDisallowInterceptTouchEvent(needConsumeTouch); 
  return super.dispatchTouchEvent(ev); 
 } 
 
 public int getMyScrollY() { 
  View c = getChildAt(0); 
  if (c == null) { 
   return 0; 
  } 
  int firstVisiblePosition = getFirstVisiblePosition(); 
  int top = c.getTop(); 
  return -top + firstVisiblePosition * c.getHeight(); 
 } 
} 

ok。先这样吧。像上拉加载更多,我感觉也可以这么实现。有时间试试吧,大家有时间也可以动动手试试。

好吧。大致就这些,有疑问或建议请留言,共同进步,谢谢!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # 下拉刷新  # android自定义控件ImageView实现圆形图片  # Android自定义控件ImageView实现点击之后出现阴影效果  # Android自定义控件ViewFipper实现竖直跑马灯效果  # Android自定义控件打造绚丽平行空间引导页  # Android自定义控件EditText实现清除和抖动功能  # Android自定义控件EditText使用详解  # 基于Android自定义控件实现雷达效果  # Android编程实现自定义控件的方法示例  # Android自定义控件之日期选择控件使用详解  # Android自定义控件实现九宫格解锁功能  # 实例讲解Android自定义控件  # 在此  # 是一个  # 拖动  # 就会  # 是在  # 第二个  # 的是  # 这是  # 加载  # 也有  # 在他  # 第一个  # 说了  # 和我  # 他就  # 出现在  # 不多  # 好吧  # 重写  # 可以看出 


相关文章: 建站VPS推荐:2025年高性能服务器配置指南  做企业网站制作流程,企业网站制作基本流程有哪些?  h5在线制作网站电脑版下载,h5网页制作软件?  网站制作需要会哪些技术,建立一个网站要花费多少?  怎么将XML数据可视化 D3.js加载XML  武清网站制作公司,天津武清个人营业执照注销查询系统网站?  如何在VPS电脑上快速搭建网站?  如何在云服务器上快速搭建个人网站?  个人摄影网站制作流程,摄影爱好者都去什么网站?  Python如何创建带属性的XML节点  定制建站策划方案_专业建站与网站建设方案一站式指南  建站之星如何优化SEO以实现高效排名?  网站制作模板下载什么软件,ppt模板免费下载网站?  常州企业建站如何选择最佳模板?  如何用PHP工具快速搭建高效网站?  大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?  儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?  广州美橙建站如何快速搭建多端合一网站?  金*站制作公司有哪些,金华教育集团官网?  建站之星代理如何获取技术支持?  韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐  网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?  建站之星下载版如何获取与安装?  微网站制作教程,不会写代码,不会编程,怎么样建自己的网站?  如何选择香港主机高效搭建外贸独立站?  b2c电商网站制作流程,b2c水平综合的电商平台?  如何通过虚拟主机快速搭建个人网站?  如何在云指建站中生成FTP站点?  高防服务器租用首荐平台,企业级优惠套餐快速部署  代刷网站制作软件,别人代刷火车票靠谱吗?  宁波免费建站如何选择可靠模板与平台?  建站上传速度慢?如何优化加速网站加载效率?  如何基于PHP生成高效IDC网络公司建站源码?  贸易公司网站制作流程,出口贸易网站设计怎么做?  ,网页ppt怎么弄成自己的ppt?  如何用美橙互联一键搭建多站合一网站?  家庭服务器如何搭建个人网站?  c++23 std::expected怎么用 c++优雅处理函数错误返回【详解】  较简单的网站制作软件有哪些,手机版网页制作用什么软件?  网站制作企业,网站的banner和导航栏是指什么?  如何规划企业建站流程的关键步骤?  Android使用GridView实现日历的简单功能  免费网站制作模板下载,除了易企秀之外还有什么H5平台可以制作H5长页面,最好是免费的?  魔方云NAT建站如何实现端口转发?  高防网站服务器:DDoS防御与BGP线路的AI智能防护方案  广州建站公司哪家好?十大优质服务商推荐  实例解析Array和String方法  电商网站制作公司有哪些,1688网是什么意思?  如何在自有机房高效搭建专业网站?  logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。