Android下拉刷新效果实现

本文主要包括以下内容

  1. 自定义实现pulltorefreshView
  2. 使用google官方SwipeRefreshLayout

下拉刷新大致原理

判断当前是否在最上面而且是向下滑的,如果是的话,则加载数据,并更新界面。

自定义实现pulltorefreshView

package com.jimstin.pulltorefreshviewdemo.view;import com.jimstin.pulltorefreshviewdemo.R;import android.content.Context;
import android.content.SharedPreferences;
import android.graphics.Canvas;
import android.os.AsyncTask;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewTreeObserver;
import android.view.ViewTreeObserver.OnPreDrawListener;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.RelativeLayout;
import android.widget.ScrollView;
import android.widget.TextView;
import android.view.View.OnTouchListener;
import android.view.animation.Animation;
import android.view.animation.Animation.AnimationListener;
import android.view.animation.LinearInterpolator;
import android.view.animation.RotateAnimation;public class PullToRefreshView  extends LinearLayout implements OnTouchListener {private static PullToRefreshView mPullToRefreshView;private static Context mContext;private AttributeSet mAttrs;private View[] childs;private RelativeLayout mTopView, mBottomView;private MyScrollView mScrollView;private LinearLayout mContentView;private LayoutParams mParams;private static TextView mTopTips, mBottomTips;private static ImageView mTopTipsIcon, mBottomTipsIcon;private RotateAnimation  toAnmt, backAnmt, progressBarAnmt;private boolean isToTop, isToBottom;private boolean isGetChilds;private boolean isRecordY;private boolean isContentNotFull;private boolean isResetLayoutParams;private boolean isActviteScrollEvent;//偏移量达到刷新或加载更多的要求时,为true,否则为falseprivate boolean isTurnUp;//判断手势,向上时为true,向下为falseprivate static boolean isRefresing;//是否正在刷新private static boolean isRefreshed = true;//是否已刷新完毕private static boolean isLoading;//是否正在加载private static boolean isLoaded = true;//是否已加载完毕private static boolean isPulling;//是否正在拖动scrollviewprivate boolean isRotate;//是否已经动画private static int topViewHeight;//topview的高度,和bottomview高度一致private int mHeight;//整个PullToRefreshView的高度private int scrollToEnd;//scrollview滚动到底部的长度private float lastY;//private float guestureLastY;//用于记录手势按下屏幕时的第一个位置private OnRefreshListener mRefreshListener;private OnLoadListener mLoadListener;public PullToRefreshView(Context context, AttributeSet attrs) {super(context, attrs);mContext = context;this.mAttrs = attrs;initView();}private void initView() {mPullToRefreshView = this;toAnmt = new RotateAnimation(0, 180f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);toAnmt.setDuration(300);toAnmt.setFillAfter(true);backAnmt = new RotateAnimation(0, -180f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);backAnmt.setDuration(300);backAnmt.setFillAfter(true);progressBarAnmt = new RotateAnimation(0, 360f, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);progressBarAnmt.setDuration(1800);progressBarAnmt.setRepeatCount(-1);progressBarAnmt.setRepeatMode(RotateAnimation.RESTART);progressBarAnmt.setInterpolator(new LinearInterpolator());toAnmt.setAnimationListener(new AnimationListener() {@Overridepublic void onAnimationStart(Animation animation) {// TODO Auto-generated method stub}@Overridepublic void onAnimationRepeat(Animation animation) {// TODO Auto-generated method stub}@Overridepublic void onAnimationEnd(Animation animation) {// TODO Auto-generated method stubmTopTipsIcon.clearAnimation();mTopTipsIcon.setBackgroundResource(R.drawable.ic_up);mBottomTipsIcon.clearAnimation();mBottomTipsIcon.setBackgroundResource(R.drawable.ic_up);}});backAnmt.setAnimationListener(new AnimationListener() {@Overridepublic void onAnimationStart(Animation animation) {// TODO Auto-generated method stub}@Overridepublic void onAnimationRepeat(Animation animation) {// TODO Auto-generated method stub}@Overridepublic void onAnimationEnd(Animation animation) {// TODO Auto-generated method stubmTopTipsIcon.clearAnimation();mTopTipsIcon.setBackgroundResource(R.drawable.ic_down);mBottomTipsIcon.clearAnimation();mBottomTipsIcon.setBackgroundResource(R.drawable.ic_down);}});ViewTreeObserver ob = getViewTreeObserver();ob.addOnPreDrawListener(new OnPreDrawListener() {@Overridepublic boolean onPreDraw() {if(topViewHeight == 0 || mHeight == 0) {if(topViewHeight == 0) {topViewHeight = mTopView.getHeight();}if(mHeight == 0) {mHeight = getHeight();}if(mHeight > 0 && topViewHeight > 0) {SharedPreferences p = mContext.getSharedPreferences("PullToRefreshView", 0);p.edit().putInt("barHeight", topViewHeight).putInt("height", mHeight).commit();}mParams = new LayoutParams(LayoutParams.MATCH_PARENT, mHeight+2*topViewHeight);setLayoutParams(mParams);setY(-topViewHeight);}return true;}});}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {// TODO Auto-generated method stubsuper.onMeasure(widthMeasureSpec, heightMeasureSpec);}@Overrideprotected void onFinishInflate() {// TODO Auto-generated method stubsuper.onFinishInflate();setLayout();}@Overrideprotected void onLayout(boolean changed, int l, int t, int r, int b) {super.onLayout(changed, l, t, r, b);}public void setLayout() {if(!isGetChilds) {int count = getChildCount();childs = new View[count];for(int i=0; i<count; i++) {childs[i] = getChildAt(i);}for(int i=0; i<count; i++) {removeAllViews();}LayoutInflater inflater = LayoutInflater.from(mContext);mTopView = (RelativeLayout) inflater.inflate(R.layout.layout_top_pull_to_refresh_view, null);mBottomView = (RelativeLayout) inflater.inflate(R.layout.layout_bottom_pull_to_refresh_view, null);mTopTips = (TextView) mTopView.findViewById(R.id.top_tips);mBottomTips = (TextView) mBottomView.findViewById(R.id.bottom_tips);mTopTipsIcon = (ImageView) mTopView.findViewById(R.id.ic_down);mBottomTipsIcon = (ImageView) mBottomView.findViewById(R.id.ic_up);mScrollView = new MyScrollView(mContext, mAttrs);mParams = new LayoutParams(LayoutParams.MATCH_PARENT, 0);mParams.weight = 1;mScrollView.setLayoutParams(mParams);mContentView = new LinearLayout(mContext, mAttrs);mContentView.setLayoutParams(mParams);mContentView.setOrientation(LinearLayout.VERTICAL);for(int i=0; i<count; i++) {mContentView.addView(childs[i]);}mScrollView.addView(mContentView);mScrollView.setOnTouchListener(this);addView(mTopView);addView(mScrollView);addView(mBottomView);isGetChilds = true;}}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);if(mScrollView.getHeight() > mContentView.getHeight()) {isContentNotFull = true;mBottomView.setVisibility(View.GONE);} else {isContentNotFull = false;}}/*** 设置刷新监听器*/public void setOnRefreshListener(OnRefreshListener listener) {mRefreshListener = listener;}/*** 设置加载监听器*/public void setOnLoadListener(OnLoadListener listener) {mLoadListener = listener;}class MyScrollView extends ScrollView {public MyScrollView(Context context, AttributeSet attrs) {super(context, attrs);}@Overrideprotected void onScrollChanged(int l, int t, int oldl, int oldt) {super.onScrollChanged(l, t, oldl, oldt);//MyLog.i("t="+t);if(t == 0) {isToTop = true;} else {if(t == getChildAt(0).getHeight()-getHeight()) {scrollToEnd = t;isToBottom = true;}}}}/*** 刷新监听器*/public static abstract class OnRefreshListener {/*** 正在刷新*/public void onRefresh() {}/*** 停止刷新,刷新完毕时请主动调用*/public void stopRefresh() {isRefresing = false;mTopTips.setText(mContext.getResources().getString(R.string.refreshed));mTopTipsIcon.clearAnimation();mTopTipsIcon.setBackgroundResource(R.drawable.ic_finish);resetYOffset();}}/*** 加载更多监听器**/public static abstract class OnLoadListener {/*** 正在加载*/public void onLoad() {}/*** 停止加载,加载完毕时请主动调用*/public void stopLoad() {isLoading = false;mBottomTips.setText(mContext.getResources().getString(R.string.loaded));mBottomTipsIcon.clearAnimation();mBottomTipsIcon.setBackgroundResource(R.drawable.ic_finish);resetYOffset();}}public static void resetYOffset() {new AsyncTask<Integer, Integer, Integer>() {@Overrideprotected Integer doInBackground(Integer... params) {try {Thread.sleep(1000);} catch (Exception e) {e.printStackTrace();}return null;}protected void onPostExecute(Integer result) {if(!isPulling) {mPullToRefreshView.setY(-topViewHeight);mTopTipsIcon.setBackgroundResource(R.drawable.ic_down);mBottomTipsIcon.setBackgroundResource(R.drawable.ic_up);isRefreshed = true;isLoaded = true;}};}.execute();}@Overridepublic boolean onTouch(View v, MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:guestureLastY = event.getRawY();isPulling = true;break;case MotionEvent.ACTION_MOVE:if(event.getRawY() - guestureLastY > 0) {isTurnUp = false;} else {isTurnUp = true;}if(isContentNotFull && !isResetLayoutParams) {mParams = new LayoutParams(LayoutParams.MATCH_PARENT, getHeight()-topViewHeight);setLayoutParams(mParams);isResetLayoutParams = true;}if(isToTop && !isTurnUp) {if(!isRecordY) {lastY = event.getRawY()/3;isRecordY = true;}mScrollView.setVerticalScrollBarEnabled(false);float y = 0.0f;if(!isRefreshed) {if(!isPulling) {y = event.getRawY()/3 - lastY - topViewHeight;} else {y = event.getRawY()/3 - lastY;}} else {y = event.getRawY()/3 - lastY - topViewHeight;}if(!isRefresing && isRefreshed) {if(Math.abs(event.getRawY()/3 - lastY) >= topViewHeight) {isActviteScrollEvent = true;mTopTips.setText(mContext.getResources().getString(R.string.release_to_refresh));if(!isRotate) {mTopTipsIcon.startAnimation(toAnmt);isRotate = true;}} else {isActviteScrollEvent = false;mTopTips.setText(mContext.getResources().getString(R.string.pull_to_refresh));if(isRotate) {mTopTipsIcon.startAnimation(backAnmt);isRotate = false;}}}setY(y);if(event.getRawY()/3 - lastY < 0) {isToTop = false;isRecordY = false;} else {mScrollView.setScrollY(0);}} else if(isToBottom || isContentNotFull) {if(!isRecordY) {lastY = event.getRawY()/3;isRecordY = true;}mScrollView.setVerticalScrollBarEnabled(false);if(!isLoading && isLoaded) {if(Math.abs(event.getRawY()/3 - lastY) >= topViewHeight) {isActviteScrollEvent = true;mBottomTips.setText(mContext.getResources().getString(R.string.release_to_load));if(!isRotate) {mBottomTipsIcon.startAnimation(backAnmt);isRotate = true;}} else {isActviteScrollEvent = false;mBottomTips.setText(mContext.getResources().getString(R.string.push_to_load));if(isRotate) {mBottomTipsIcon.startAnimation(toAnmt);isRotate = false;}}}float y = 0.0f;if(!isLoaded) {if(isPulling) {y = event.getRawY()/3 - lastY - 2*topViewHeight;} else {y = event.getRawY()/3 - lastY - topViewHeight;}} else {y = event.getRawY()/3 - lastY - topViewHeight;}setY(y);if(event.getRawY()/3 - lastY > 0) {isToBottom = false;isRecordY = false;} else {mScrollView.setScrollY(scrollToEnd);}}break;case MotionEvent.ACTION_UP:mScrollView.setVerticalScrollBarEnabled(true);isRotate = false;isRecordY = false;isPulling = false;if(isActviteScrollEvent) {if(isToTop && !isTurnUp) {if(mRefreshListener != null) {isRefresing = true;isRefreshed = false;setY(0);mTopTips.setText(mContext.getResources().getString(R.string.refreshing));mTopTipsIcon.setBackgroundResource(R.drawable.ic_refresh_progress_bar);mTopTipsIcon.startAnimation(progressBarAnmt);mRefreshListener.onRefresh();}} else if(isToBottom) {if(mLoadListener != null) {setY(-2*topViewHeight);mBottomTips.setText(mContext.getResources().getString(R.string.loading));mBottomTipsIcon.setBackgroundResource(R.drawable.ic_refresh_progress_bar);mBottomTipsIcon.startAnimation(progressBarAnmt);isLoading = true;isLoaded = false;mLoadListener.onLoad();}} else if(isContentNotFull) {setY(-topViewHeight);}isActviteScrollEvent = false;} else {if(isRefresing) {setY(0);} else if(isLoading) {setY(-2*topViewHeight);} else {setY(-topViewHeight);}}if(!isRefresing) {isRefreshed = true;} if(!isLoading) {isLoaded = true;}isToTop = false;isToBottom = false;break;default:break;}return false;}public void initLayoutParams() {SharedPreferences p = mContext.getSharedPreferences("PullToRefreshView", 0);topViewHeight = p.getInt("barHeight", 0);mHeight = p.getInt("height", 0);if(mHeight > 0) {mParams = new LayoutParams(LayoutParams.MATCH_PARENT, mHeight+2*topViewHeight);setLayoutParams(mParams);setY(-topViewHeight);}}
}

参考链接

PullToRefreshView下拉刷新上来加载更多,支持任何子view! - zhangjm_123的专栏 - 博客频道 - CSDN.NET

SwipeRefreshLayout 使用

布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:padding="8dp"android:background="#Ffffff"><android.support.v4.widget.SwipeRefreshLayout
        android:id="@+id/swipe_refresh_layout_caselist"android:layout_marginTop="4dp"android:layout_width="wrap_content"android:layout_height="wrap_content"><ListView
        android:id="@+id/list"android:scrollbars="none"android:layout_width="match_parent"android:layout_height="match_parent"android:dividerHeight="0dp"android:divider="#Ffffff"></ListView></android.support.v4.widget.SwipeRefreshLayout>
</LinearLayout>

在Activity中使用

 @Overrideprotected void setUpView() {if (mIsFirstCreated){onRefresh();}listView = $(R.id.list);swipeRefreshLayout=$(R.id.swipe_refresh_layout_caselist);//设置刷新时动画的颜色,可以设置4个swipeRefreshLayout.setColorSchemeResources(android.R.color.holo_blue_light, android.R.color.holo_red_light, android.R.color.holo_orange_light, android.R.color.holo_green_light);swipeRefreshLayout.setOnRefreshListener(this);getData();
//        BaseAdapter arrayAdapter = new ArrayAdapter(getActivity(),R.layout.item_list,R.id.name,
//                Arrays.asList("美食劵", "活动劵", "优惠劵", "团购劵", "外卖劵"));
//        listView.setAdapter(arrayAdapter);}@Overridepublic void onRefresh() {//tv.setText("正在刷新");// TODO Auto-generated method stubmIsFirstCreated = false;getData();}

使用的类要实现SwipeRefreshLayout.OnRefreshListener接口

并且在数据获取成功后发送消息,设置swipeRefreshLayout.setRefreshing(false);

参考链接

Google自己的下拉刷新组件SwipeRefreshLayout - 李金尧 - 博客园

你还在用开源控件的下拉刷新吗?你out了,试一试官方的下拉刷新SwipeRefreshLayout - 推酷

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/494036.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2018及过去20年诺贝尔化学奖获奖者及其贡献!

来源&#xff1a;科学网摘要&#xff1a;2018及过去20年诺贝尔化学奖获奖者及其贡献&#xff01;2018年诺贝尔化学获的获得者为美国科学家Frances H. Arnold, George P. Smith和英国科学家George P.Winter&#xff0c;他们利用进化的力量为人类造福。获奖的内容分别是研究酶的定…

RxJava学习入门

RxJava是什么 一个词&#xff1a;异步。 RxJava 在 GitHub 主页上的自我介绍是 “a library for composing asynchronous and event-based programs using observable sequences for the Java VM”&#xff08;一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序…

解读2018年诺贝尔化学奖成果:用进化的力量解决化学问题

来源&#xff1a;新华网摘要&#xff1a;新华社斯德哥尔摩&#xff11;&#xff10;月&#xff13;日电 科普&#xff1a;用进化的力量解决化学问题——解读&#xff12;&#xff10;&#xff11;&#xff18;年诺贝尔化学奖成果地球的生命经过长期进化最终获得强大的适应力&am…

在win8下安装使用java及在win8下部署java环境变量-图文

为了反编译APK&#xff0c;不得不安装一些Androidfby、apktool1.4.1、dex2jar-0.0.9.9等&#xff0c;甚至连DW也安装了&#xff0c;但是我的电脑是win8X64的&#xff0c;也就是64位的win8系统&#xff0c;这就有点头疼了&#xff0c;出现了&#xff1a; 不是内部或外部命令&…

国际运营商智慧城市探索与实践

来源&#xff1a;中国信息通信研究院CAICT摘要&#xff1a;AT&T、SKT、沃达丰在智慧城市领域探索的经验。2008年11月&#xff0c;IBM提出“智慧地球” 理念引发产业界热议&#xff0c;2010年&#xff0c;该公司进一步提出“智慧城市”愿景作为“智慧地球”在城市运营中的具…

Retrofit学习入门

Retrofit的使用 设置权限与添加依赖 定义请求接口通过创建一个retrofit生成一个接口的实现类(动态代理)调用接口请求数据 设置权限与添加依赖 权限&#xff1a;首先确保在AndroidManifest.xml中请求了网络权限 &#xff1a; <uses-permission android:name"android…

centos安装--两张光盘

不是有两个ISO吗&#xff1f;你在别的电脑上打开第二个iso&#xff0c;就可以看到里面是openoffice的语言包。你安装的时候出现的提示是找不到“openoffice.org-langpack-zh-TW-3.1.1-19.5.el5_5.6.i386.rpm”&#xff0c;这个包就在第二个iso里面。但是&#xff0c;即使安装到…

Retrofit与RXJava整合

Retrofit 除了提供了传统的 Callback 形式的 API&#xff0c;还有 RxJava 版本的 Observable 形式 API。下面我用对比的方式来介绍 Retrofit 的 RxJava 版 API 和传统版本的区别。 以获取一个 User 对象的接口作为例子。使用Retrofit 的传统 API&#xff0c;你可以用这样的方式…

消除疑虑

这要从何说起?有时候我们觉得我们自已很强大,可以学会很多东西,可以做很多事情.无论从逻辑还是环境去考虑,都有可行性. 但是:"我们心中有疑虑" 这不是我们不够自信,而是还没有完全的把握可以战胜敌人,当然这个敌人是假想敌,或者说阻碍我们前进的东西.又或者我们感觉…

autotools入门笔记(一)

GNU autotools作用&#xff1a;收集系统配置信息并自动生成Makefile文件。 GNU autotools主要包括三个工具&#xff1a;autoconf、automake、libtool&#xff0c;还有很多辅助的工具&#xff0c;包括&#xff1a;autoheader、aclocal、autoscan。 ● autoscan检测源文件生成con…

本田、大众宣布智能路口研究新进展 以安全为重点

编译&#xff1a;网易智能摘要&#xff1a;每年&#xff0c;在十字路口发生的交通事故约占交通事故死亡人数的20%&#xff0c;这就是为什么汽车制造商和供应商都下定决心要创造出智能的十字路口&#xff0c;利用尖端技术来减少&#xff08;或者最好是能够避免&#xff09;在十字…

Volley学习总结

本文主要包括以下内容 volly基本操作(String与Json类型) volly图片操作 自定义volly volly源码分析 Volley简单易用&#xff0c;在性能方面也进行了大幅度的调整&#xff0c;它的设计目标就是非常适合去进行数据量不大&#xff0c;但通信频繁的网络操作&#xff0c;而对于大…

vimtutor

在shell命令行下输入vimtutor将出现本教程。 欢 迎 阅 读 《 V I M 教 程 》 —— 版本 1.5 vim 是一个具有很多命令的功能非常强大的编辑器。限于篇幅&#xff0c;在本教程当中 就不详细介绍了。本教程的设计目标是讲述一些必要…

智能连接:5G、AI和IoT的组合如何改变美洲

来源&#xff1a;199IT互联网数据中心摘要&#xff1a;GSMA发布了新报告“智能连接&#xff1a;5G、AI和IoT的组合如何改变美洲”&#xff0c;强调了该地区如何从这些科技中受益。GSMA Intelligence预测&#xff0c;到2025年全球5G连接数量将达到13亿&#xff0c;覆盖全球40%的…

刚刚,生物学横扫诺贝尔3大奖,两名女性获奖!化学奖授予试管中的“进化论”...

来源&#xff1a;Deeptech深科技北京时间 10 月 3 日下午 5 点 45 分&#xff0c;2018 诺贝尔化学奖揭晓——诺贝尔委员会宣布&#xff0c;将此奖项一半颁给女科学家Frances H. Arnold&#xff0c;另一半则由George P. Smith、 Gregory P. Winter两人共享。此次诺贝尔化学奖表彰…

MACIOS Socket编程

转自 https://github.com/kejinlu/objc-doc/blob/master/Socket%E7%BC%96%E7%A8%8B.md 大纲 一.Socket简介二.BSD Socket编程准备 1.地址2.端口3.网络字节序4.半相关与全相关5.网络编程模型三.socket接口编程示例四.使用select五.使用kqueue六.使用流注:文档中设计涉及的代码也…

Android缓存学习入门

本文主要包括以下内容 利用LruCache实现内存缓存 利用DiskLruCache实现磁盘缓存 LruCache与DiskLruCache结合实例 利用了缓存机制的瀑布流实例 内存缓存的实现 public class PhotoWallAdapter extends ArrayAdapter<String> implements OnScrollListener {/*** 记录…

2018年人工智能之自动驾驶研究报告

来源&#xff1a;AMiner未来智能实验室是人工智能学家与科学院相关机构联合成立的人工智能&#xff0c;互联网和脑科学交叉研究机构。未来智能实验室的主要工作包括&#xff1a;建立AI智能系统智商评测体系&#xff0c;开展世界人工智能智商评测&#xff1b;开展互联网&#xf…

MyEclipse+Tomcat 启动时出现 configuration error occured during startup错误的解决方法

配置好Tomcat server&#xff0c;启动Tomcat&#xff0c;报如下错误&#xff1a; 解决方法如下&#xff1a; 选中Tomcat 6中的JDK一项&#xff0c;这里要特别注意&#xff0c;默认的是JRE的运行环境&#xff0c;这里要设定成JDK的&#xff0c;否则&#xff0c;MyEclipse无法正常…

硅谷首场AI硬件峰会干货报告:AI芯片井喷期即将到来

来源&#xff1a;智东西摘要&#xff1a;英特尔和英伟达对AI芯片的角逐&#xff0c;AI芯片行业的创业者机会。AI芯片不仅是未来十年半导体行业中最有希望的增长机会之一&#xff0c;而且还是有可能破坏传统计算市场的力量。现今 99&#xff05;的AI软件尚未编写&#xff0c;只有…