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自定义控件
# 在此
# 是一个
# 拖动
# 就会
# 是在
# 第二个
# 的是
# 这是
# 加载
# 也有
# 在他
# 第一个
# 说了
# 和我
# 他就
# 出现在
# 不多
# 好吧
# 重写
# 可以看出
相关文章:
如何彻底卸载建站之星软件?
如何在建站主机中优化服务器配置?
如何选择高效可靠的多用户建站源码资源?
历史网站制作软件,华为如何找回被删除的网站?
Android滚轮选择时间控件使用详解
济南网站制作的价格,历城一职专官方网站?
云南网站制作公司有哪些,云南最好的招聘网站是哪个?
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
如何通过VPS建站无需域名直接访问?
完全自定义免费建站平台:主题模板在线生成一站式服务
北京网页设计制作网站有哪些,继续教育自动播放怎么设置?
网站微信制作软件,如何制作微信链接?
如何破解联通资金短缺导致的基站建设难题?
如何在腾讯云服务器上快速搭建个人网站?
北京制作网站的公司,北京铁路集团官方网站?
常州企业网站制作公司,全国继续教育网怎么登录?
如何快速搭建二级域名独立网站?
网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?
手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?
已有域名建站全流程解析:网站搭建步骤与建站工具选择
建站主机类型有哪些?如何正确选型
宝盒自助建站智能生成技巧:SEO优化与关键词设置指南
如何通过老薛主机一键快速建站?
如何设计高效校园网站?
免费制作统计图的网站有哪些,如何看待现如今年轻人买房难的情况?
怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?
建站之星如何快速生成多端适配网站?
如何在景安服务器上快速搭建个人网站?
网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
建站之星五站合一营销型网站搭建攻略,流量入口全覆盖优化指南
网站制作需要会哪些技术,建立一个网站要花费多少?
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
在线制作视频的网站有哪些,电脑如何制作视频短片?
公司网站制作费用多少,为公司建立一个网站需要哪些费用?
广州网站设计制作一条龙,广州巨网网络科技有限公司是干什么的?
简易网站制作视频教程,使用记事本编写一个简单的网页html文件?
外贸公司网站制作哪家好,maersk船公司官网?
建站之星安装失败:服务器环境不兼容?
javascript中的try catch异常捕获机制用法分析
如何通过西部数码建站助手快速创建专业网站?
如何用PHP快速搭建高效网站?分步指南
如何通过远程VPS快速搭建个人网站?
免费ppt制作网站,有没有值得推荐的免费PPT网站?
如何快速查询网址的建站时间与历史轨迹?
如何通过免费商城建站系统源码自定义网站主题与功能?
为什么Go需要go mod文件_Go go mod文件作用说明
我的世界制作壁纸网站下载,手机怎么换我的世界壁纸?
如何用PHP快速搭建CMS系统?
如何在服务器上三步完成建站并提升流量?
*请认真填写需求信息,我们会在24小时内与您取得联系。