相信大家已经对下拉刷新熟悉得不能再熟悉了,市面上的下拉刷新琳琅满目,然而有很多在我看来略有缺陷,接下来我将说明一下存在的缺陷问题,然后提供一种思路来解决这一缺陷,废话不多说!往下看嘞!

1.市面一些下拉刷新控件普遍缺陷演示

以直播吧APP为例:

第1种情况:

滑动控件在初始的0位置时,手势往下滑动然后再往上滑动,可以看到滑动到初始位置时滑动控件不能滑动。

原因:

下拉刷新控件响应了触摸事件,后续的一系列事件都由它来处理,当滑动控件到顶端的时候,滑动事件都被下拉刷新控件消费掉了,传递不到它的子控件即滑动控件,因此滑动控件不能滑动。

第2种情况:

滑动控件滑动到某个非0位置时,这时下拉回0位置时,可以看到下拉刷新头部没有被拉出来。

原因:

滑动控件响应了触摸事件,后续的一系列事件都由它来处理,当滑动控件到顶端的时候,滑动事件都被滑动控件消费掉了,父控件即下拉刷新控件消费不了滑动事件,因此下拉刷新头部没有被拉出来。

可能大部分人觉得无关痛痒,把手指抬起再下拉就可以了,but对于强迫症的我而言,能提供一个无痕过渡才是最符合操作逻辑的,因此接下来我来讲解下实现的思路。

2.实现的思路讲解

2.1.事件分发机制简介(来源于Android开发艺术探索)

dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent方法的关系伪代码

public boolean dispatchTouchEvent(MotionEvent ev) {

boolean consume = false;

if(onInterceptTouchEvent(ev)) {

consume = onTouchEvent(ev);

} else {

consume = child.dispatchTouchEvent(ev);

}

return consume;

}

1.由代码可知若当前View拦截事件,就交给自己的onTouchEvent去处理,否则就丢给子View继续走相同的流程。

2.事件传递顺序:Activity -> Window -> View,如果View都不处理,最终将由Activity的onTouchEvent

处理,是一种责任链模式的实现。

3.正常情况,一个事件序列只能被一个View拦截且消耗。

4.某个View一旦决定拦截,这一个事件序列只能由它处理,并且它的onInterceptTouchEvent不会再被调用

5.不消耗ACTION_DOWN,则事件序列都会由其父元素处理。

2.2.一般下拉刷新的实现思路猜想

首先,下拉刷新控件作为一个容器,需要重写onInterceptTouchEvent和onTouchEvent这两个方法,然后在onInterceptTouchEvent中判断ACTION_DOWN事件,根据子控件的滑动距离做出判断,若还没滑动过,则onInterceptTouchEvent返回true表示其拦截事件,然后在onTouchEvent中进行下拉刷新的头部显示隐藏的逻辑处理;若子控件滑动过了,不拦截事件,onInterceptTouchEvent返回false,后续其下拉刷新的头部显示隐藏的逻辑处理就无法被调用了。

2.3.无痕过渡下拉刷新控件的实现思路

从2.2中可以看出,要想无痕过渡,下拉刷新控件不能拦截事件,这时候你可能会问,既然把事件给了子控件,后续拉刷新头部逻辑怎么实现呢?

这时候就要用到一般都忽略的事件分发方法dispatchTouchEvent了,此方法在ViewGroup默认返回true表示分发事件,即使子控件拦截了事件,父布局的dispatchTouchEvent仍然会被调用,因为事件是传递下来的,这个方法必定被调用。

所以我们可以在dispatchTouchEvent时对子控件的滑动距离做出判断,在这里把下拉刷新的头部的逻辑处理掉,同时在函数调用return super.dispatchTouchEvent(event) 前把event的action设置为ACTION_CANCEL,这样子子控件就不会响应滑动的操作。

3.代码实现

3.1.确定需求

需要适配任意控件,例如RecyclerView、ListView、ViewPager、WebView以及普通的不能滑动的View

不能影响子控件原来的事件逻辑

暴露方法提供手动调用刷新功能

可以设置禁止下拉刷新功能

3.2.代码讲解

需要的变量

public class RefreshLayout extends LinearLayout {

// 隐藏的状态

private static final int HIDE = 0;

// 下拉刷新的状态

private static final int PULL_TO_REFRESH = 1;

// 松开刷新的状态

private static final int RELEASE_TO_REFRESH = 2;

// 正在刷新的状态

private static final int REFRESHING = 3;

// 正在隐藏的状态

private static final int HIDING = 4;

// 当前状态

private int mCurrentState = HIDE;

// 头部动画的默认时间(单位:毫秒)

public static final int DEFAULT_DURATION = 200;

// 头部高度

private int mHeaderHeight;

// 内容控件的滑动距离

private int mContentViewOffset;

// 记录上次的Y坐标

private int mLastY;

// 最小滑动响应距离

private int mScaledTouchSlop;

// 滑动的偏移量

private int mTotalDeltaY;

// 是否在处理头部

private boolean mIsHeaderHandling;

// 是否可以下拉刷新

private boolean mIsRefreshable = true;

// 内容控件是否可以滑动,不能滑动的控件会做触摸事件的优化

private boolean mContentViewScrollable = true;

// 头部,为了方便演示选取了TextView

private TextView mHeader;

// 容器要承载的内容控件,在XML里面要放置好

private View mContentView;

// 值动画,由于头部显示隐藏

private ValueAnimator mHeaderAnimator;

// 刷新的监听器

private OnRefreshListener mOnRefreshListener;

初始化时创建头部执行显示隐藏的值动画,添加头部到布局中,并且通过设置paddingTop隐藏头部

public RefreshLayout(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

init();

addHeader(context);

}

private void init() {

mScaledTouchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();

mHeaderAnimator = ValueAnimator.ofInt(0).setDuration(DEFAULT_DURATION);

mHeaderAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public void onAnimationUpdate(ValueAnimator valueAnimator) {

if (getContext() == null) {

// 若是退出Activity了,动画结束不必执行头部动作

return;

}

// 通过设置paddingTop实现显示或者隐藏头部

int offset = (Integer) valueAnimator.getAnimatedValue();

mHeader.setPadding(0, offset, 0, 0);

}

});

mHeaderAnimator.addListener(new AnimatorListenerAdapter() {

@Override

public void onAnimationEnd(Animator animation) {

if (getContext() == null) {

// 若是退出Activity了,动画结束不必执行头部动作

return;

}

if (mCurrentState == RELEASE_TO_REFRESH) {

// 释放刷新状态执行的动画结束,意味接下来就是刷新了,改状态并且调用刷新的监听

mHeader.setText("正在刷新...");

mCurrentState = REFRESHING;

if (mOnRefreshListener != null) {

mOnRefreshListener.onRefresh();

}

} else if (mCurrentState == HIDING) {

// 下拉状态执行的动画结束,隐藏头部,改状态

mHeader.setText("我是头部");

mCurrentState = HIDE;

}

}

});

}

// 头部的创建

private void addHeader(Context context) {

// 强制垂直方法

setOrientation(LinearLayout.VERTICAL);

mHeader = new TextView(context);

mHeader.setBackgroundColor(Color.GRAY);

mHeader.setTextColor(Color.WHITE);

mHeader.setText("我是头部");

mHeader.setTextSize(TypedValue.COMPLEX_UNIT_SP, 25);

mHeader.setGravity(Gravity.CENTER);

addView(mHeader, LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);

mHeader.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {

@Override

public void onGlobalLayout() {

// 算出头部高度

mHeaderHeight = mHeader.getMeasuredHeight();

// 移除监听

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {

mHeader.getViewTreeObserver().removeOnGlobalLayoutListener(this);

} else {

mHeader.getViewTreeObserver().removeGlobalOnLayoutListener(this);

}

// 设置paddingTop为-mHeaderHeight,刚好把头部隐藏掉了

mHeader.setPadding(0, -mHeaderHeight, 0, 0);

}

});

}

在填充完布局后取出内容控件

@Override

protected void onFinishInflate() {

super.onFinishInflate();

// 设置长点击或者短点击都能消耗事件,要不这样做,若孩子都不消耗,最终点击事件会被它的上级消耗掉,后面一系列的事件都只给它的上级处理了

setLongClickable(true);

// 获取内容控件

mContentView = getChildAt(1);

if (mContentView == null) {

// 为空抛异常,强制要求在XML设置内容控件

throw new IllegalArgumentException("You must add a content view!");

}

if (!(mContentView instanceof ScrollingView

|| mContentView instanceof WebView

|| mContentView instanceof ScrollView

|| mContentView instanceof AbsListView)) {

// 不是具有滚动的控件,这里设置标志位

mContentViewScrollable = false;

}

}

重头戏来了,分发对于下拉刷新的特殊处理:

1.mContentViewOffset用于判别内容页的滑动距离,在无偏移值时才去处理下拉刷新的操作;

2.在mContentViewOffset!=0即内容页滑动的第一个瞬间,强制把MOVE事件改为DOWN,是因为之前MOVE都被拦截掉了,若不给个DOWN让内容页重新定下滑动起点,会有一瞬间滑动一大段距离的坑爹效果。

@Override

public boolean dispatchTouchEvent(final MotionEvent event) {

if (!mIsRefreshable) {

// 禁止下拉刷新,直接把事件分发

return super.dispatchTouchEvent(event);

}

if ((mCurrentState == REFRESHING

|| mCurrentState == RELEASE_TO_REFRESH

|| mCurrentState == HIDING)

&& mHeaderAnimator.isRunning()) {

// 正在刷新,正在释放,正在隐藏头部都不处理事件,并且不分发下去

return true;

}

int y = (int) event.getY();

switch (event.getAction()) {

case MotionEvent.ACTION_DOWN:

break;

case MotionEvent.ACTION_MOVE: {

int deltaY = y - mLastY;

if (mContentViewOffset == 0 && (deltaY > 0 || (deltaY < 0 && isHeaderShowing()))) {

// 偏移值为0时,下拉或者在头部还在显示的时候上滑时,交由自己处理滑动事件

mTotalDeltaY += deltaY;

if (mTotalDeltaY > 0

&& mTotalDeltaY <= mScaledTouchSlop

&& !isHeaderShowing()) {

// 优化下拉头部,不要稍微一点位移就响应

mLastY = y;

return super.dispatchTouchEvent(event);

}

// 处理事件

onHandleTouchEvent(event);

// 正在处理事件

mIsHeaderHandling = true;

if (mCurrentState == REFRESHING) {

// 正在刷新,不让contentView响应滑动

event.setAction(MotionEvent.ACTION_CANCEL);

}

} else if (mIsHeaderHandling) {

// 在头部隐藏的那一瞬间的事件特殊处理

if (mContentViewScrollable) {

// 1.可滑动的View,由于之前处理头部,之前的MOVE事件没有传递到内容页,这里

// 需要要ACTION_DOWN来重新告知滑动的起点,不然会瞬间滑动一段距离

// 2.对于不滑动的View设置了点击事件,若这里给它一个ACTION_DOWN事件,在手指

// 抬起时ACTION_UP事件会触发点击,因此这里做了处理

event.setAction(MotionEvent.ACTION_DOWN);

}

mIsHeaderHandling = false;

}

break;

}

case MotionEvent.ACTION_CANCEL:

case MotionEvent.ACTION_UP: {

if (mContentViewOffset == 0 && isHeaderShowing()) {

// 处理手指抬起或取消事件

onHandleTouchEvent(event);

}

mTotalDeltaY = 0;

break;

}

default:

break;

}

mLastY = y;

if (mCurrentState != REFRESHING

&& isHeaderShowing()

&& event.getAction() != MotionEvent.ACTION_UP) {

// 不是在刷新的时候,并且头部在显示, 不让contentView响应事件

event.setAction(MotionEvent.ACTION_CANCEL);

}

return super.dispatchTouchEvent(event);

}

处理事件的逻辑:拿到下拉偏移量,然后动态去设置头部的paddingTop值,即可实现显示隐藏;手指抬起时根据状态决定是显示刷新还是直接隐藏头部

// 自己处理事件

public boolean onHandleTouchEvent(MotionEvent event) {

int y = (int) event.getY();

switch (event.getAction()) {

case MotionEvent.ACTION_MOVE: {

// 拿到Y方向位移

int deltaY = y - mLastY;

// 除以3相当于阻尼值

deltaY /= 3;

// 计算出移动后的头部位置

int top = deltaY + mHeader.getPaddingTop();

// 控制头部位置最大不超过-mHeaderHeight

if (top < -mHeaderHeight) {

mHeader.setPadding(0, -mHeaderHeight, 0, 0);

} else {

mHeader.setPadding(0, top, 0, 0);

}

if (mCurrentState == REFRESHING) {

// 之前还在刷新状态,继续维持刷新状态

mHeader.setText("正在刷新...");

break;

}

if (mHeader.getPaddingTop() > mHeaderHeight / 2) {

// 大于mHeaderHeight / 2时可以刷新了

mHeader.setText("可以释放刷新...");

mCurrentState = RELEASE_TO_REFRESH;

} else {

// 下拉状态

mHeader.setText("正在下拉...");

mCurrentState = PULL_TO_REFRESH;

}

break;

}

case MotionEvent.ACTION_UP: {

if (mCurrentState == RELEASE_TO_REFRESH) {

// 释放刷新状态,手指抬起,通过动画实现头部回到(0,0)位置

mHeaderAnimator.setIntValues(mHeader.getPaddingTop(), 0);

mHeaderAnimator.setDuration(DEFAULT_DURATION);

mHeaderAnimator.start();

mHeader.setText("正在释放...");

} else if (mCurrentState == PULL_TO_REFRESH || mCurrentState == REFRESHING) {

// 下拉状态或者正在刷新状态,通过动画隐藏头部

mHeaderAnimator.setIntValues(mHeader.getPaddingTop(), -mHeaderHeight);

if (mHeader.getPaddingTop() <= 0) {

mHeaderAnimator.setDuration((long) (DEFAULT_DURATION * 1.0 /

mHeaderHeight * (mHeader.getPaddingTop() + mHeaderHeight)));

} else {

mHeaderAnimator.setDuration(DEFAULT_DURATION);

}

mHeaderAnimator.start();

if (mCurrentState == PULL_TO_REFRESH) {

// 下拉状态的话,把状态改为正在隐藏头部状态

mCurrentState = HIDING;

mHeader.setText("收回头部...");

}

}

break;

}

default:

break;

}

mLastY = y;

return super.onTouchEvent(event);

}

你可能会问了,这个mContentViewOffset怎么知道呢?接下来就是处理的方法,我会针对不同的滑动控件,去设置它们的滑动距离的监听,方法各种各样,通过handleTargetOffset去判别View的类型采取不同的策略;然后你可能会觉得要是我那个控件我也要实现监听咋办?这个简单,继承我已经实现的监听器,再补充你想要的功能即可,这个时候就不能再调handleTargetOffset这个方法了呗。

// 设置内容页滑动距离

public void setContentViewOffset(int offset) {

mContentViewOffset = offset;

}

/**

* 根据不同类型的View采取不同类型策略去计算滑动距离

*

* @param view 内容View

*/

public void handleTargetOffset(View view) {

if (view instanceof RecyclerView) {

((RecyclerView) view).addOnScrollListener(new RecyclerViewOnScrollListener());

} else if (view instanceof NestedScrollView) {

((NestedScrollView) view).setOnScrollChangeListener(new NestedScrollViewOnScrollChangeListener());

} else if (view instanceof WebView) {

view.setOnTouchListener(new WebViewOnTouchListener());

} else if (view instanceof ScrollView) {

view.setOnTouchListener(new ScrollViewOnTouchListener());

} else if (view instanceof ListView) {

((ListView) view).setOnScrollListener(new ListViewOnScrollListener());

}

}

/**

* 适用于RecyclerView的滑动距离监听

*/

public class RecyclerViewOnScrollListener extends RecyclerView.OnScrollListener {

int offset = 0;

@Override

public void onScrolled(RecyclerView recyclerView, int dx, int dy) {

super.onScrolled(recyclerView, dx, dy);

offset += dy;

setContentViewOffset(offset);

}

}

/**

* 适用于NestedScrollView的滑动距离监听

*/

public class NestedScrollViewOnScrollChangeListener implements NestedScrollView.OnScrollChangeListener {

@Override

public void onScrollChange(NestedScrollView v, int scrollX, int scrollY, int oldScrollX, int oldScrollY) {

setContentViewOffset(scrollY);

}

}

/**

* 适用于WebView的滑动距离监听

*/

public class WebViewOnTouchListener implements View.OnTouchListener {

@Override

public boolean onTouch(View view, MotionEvent motionEvent) {

setContentViewOffset(view.getScrollY());

return false;

}

}

/**

* 适用于ScrollView的滑动距离监听

*/

public class ScrollViewOnTouchListener extends WebViewOnTouchListener {

}

/**

* 适用于ListView的滑动距离监听

*/

public class ListViewOnScrollListener implements AbsListView.OnScrollListener {

@Override

public void onScrollStateChanged(AbsListView absListView, int i) {

}

@Override

public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {

if (firstVisibleItem == 0) {

View c = view.getChildAt(0);

if (c == null) {

return;

}

int firstVisiblePosition = view.getFirstVisiblePosition();

int top = c.getTop();

int scrolledY = -top + firstVisiblePosition * c.getHeight();

setContentViewOffset(scrolledY);

} else {

setContentViewOffset(1);

}

}

}

最后参考谷歌大大的SwipeRefreshLayout提供setRefreshing来开启或关闭刷新动画,至于openHeader为啥要post(Runnable)呢?相信用过SwipeRefreshLayout在onCreate的时候直接调用setRefreshing(true)没有小圆圈出来的都知道这个坑!

public void setRefreshing(boolean refreshing) {

if (refreshing && mCurrentState != REFRESHING) {

// 强开刷新头部

openHeader();

} else if (!refreshing) {

closeHeader();

}

}

private void openHeader() {

post(new Runnable() {

@Override

public void run() {

mCurrentState = RELEASE_TO_REFRESH;

mHeaderAnimator.setDuration((long) (DEFAULT_DURATION * 2.5));

mHeaderAnimator.setIntValues(mHeader.getPaddingTop(), 0);

mHeaderAnimator.start();

}

});

}

private void closeHeader() {

mHeader.setText("刷新完毕,收回头部...");

mCurrentState = HIDING;

mHeaderAnimator.setIntValues(mHeader.getPaddingTop(), -mHeaderHeight);

// 0~-mHeaderHeight用时DEFAULT_DURATION

mHeaderAnimator.setDuration(DEFAULT_DURATION);

mHeaderAnimator.start();

}

3.3.效果展示

除了以上三个还有在Demo中实现了ListView、ViewPager、ScrollView、NestedScrollView,具体看代码即可

以上所述是小编给大家介绍的Android开发之无痕过渡下拉刷新控件的实现思路详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!

android禁止下拉刷新,Android开发之无痕过渡下拉刷新控件的实现思路详解相关推荐

  1. android组件用法说明,Android第三方控件PhotoView使用方法详解

    Android第三方控件PhotoView使用方法详解 发布时间:2020-10-21 15:06:09 来源:脚本之家 阅读:74 作者:zhaihaohao1 PhotoView的简介: 这是一个 ...

  2. 移动应用开发之路 04 Android Studio 5种控件介绍、实战详解

    学校开了一门移动应用开发课程,我一开始兴趣盎然,但是看到使用的环境是 Java 8 的时候心就凉了一半,在询问老师的意见之后决定使用现在比较常用的Android Studio完成学习,特此记录自学之路 ...

  3. android实现左拉菜单,Android原生侧滑控件DrawerLayout使用方法详解

    在android的v4包中有一个控件 Drawerlayout,主要实现了左拉和右拉菜单,类似于之前的"抽屉"功能,此控件使用简单,效果很柔和,操作起来体验非常好,下面是我实现的一 ...

  4. 一种无痕过渡下拉刷新控件的实现思路

    一种无痕过渡下拉刷新控件的实现思路 相信大家已经对下拉刷新熟悉得不能再熟悉了,市面上的下拉刷新琳琅满目,然而有很多在我看来略有缺陷,接下来我将说明一下存在的缺陷问题,然后提供一种思路来解决这一缺陷,废 ...

  5. Android AR开发实践之七:OpenGLES相机预览背景绘制源码详解

    Android AR开发实践之七:OpenGLES相机预览背景绘制源码详解 目录 Android AR开发实践之七:OpenGLES相机预览背景绘制源码详解 一.OpenGL ES渲染管线 1.基本处 ...

  6. android仿知乎按钮动效,Android仿知乎客户端关注和取消关注的按钮点击特效实现思路详解...

    先说明一下,项目代码已上传至github,不想看长篇大论的也可以先去下代码,对照代码,哪里不懂点哪里. 代码在这https://github.com/zgzczzw/ZHFollowButton 前几 ...

  7. COM组件开发实践(八)---多线程ActiveX控件和自动调整ActiveX控件大小(下)

    源代码下载:MyActiveX20081229.rar 声明:本文代码基于CodeProject的文章<A Complete ActiveX Web Control Tutorial>修改 ...

  8. firefox扩展开发(二):用XUL创建窗口控件

    firefox扩展开发(二):用XUL创建窗口控件 2008-06-11 16:57 1.创建一个简单的窗口 <?xml version="1.0"?> <?xm ...

  9. SAP UI5 应用开发教程之六十 - SAP UI5 地图控件的一些高级用法试读版

    一套适合 SAP UI5 初学者循序渐进的学习教程 教程目录 SAP UI5 本地开发环境的搭建 SAP UI5 应用开发教程之一:Hello World SAP UI5 应用开发教程之二:SAP U ...

最新文章

  1. WEB免费打印控件推荐
  2. nginx在windows里面配置
  3. 成功解决xgboost.core.XGBoostError: b‘[14:48:08] 0 feature is supplied. Are you using raw Booster inter
  4. C++11 Lambda表达式(匿名函数)详解
  5. 趣谈设计模式 | 策略模式(Strategy):你还在使用冗长的if-else吗?
  6. telnet到设备里 php_金融行业思科设备典型网络故障案例:76系列典型案例(一)...
  7. python egg
  8. TB6612与电机编码器
  9. Excel如何让单元数据显示单位又能参与运算?
  10. 家用数码相机选购及使用指南
  11. 机场生产运行数据统计指标-第二篇-航班类
  12. 2021年6月大学英语六级作文
  13. IDL简明教程-文件读写
  14. 最详细的Hadoop安装教程
  15. 数学建模Python图论习题
  16. 如何使应用保持后台运行
  17. 佛山法医鉴定中心实验室施工方案
  18. 软件设计7种架构模式
  19. SU2 多段翼型的计算结果
  20. 基于51单片机的简易数字计算器Proteus仿真(源码+仿真+全套资料)

热门文章

  1. 上海快速拿计算机本科文凭,怎么快速拿文凭,急!(建议收藏)
  2. 弱网络模拟测试工具---易测app
  3. Python+OpenCV教程5:颜色空间转换 追踪视频中特定颜色的物体 消除票据中的红色印章
  4. Python入门--爬取淘宝评论并生成词云
  5. RoboMaster电控学习笔记——电机控制(1-CAN)
  6. 在?快来pick你最喜爱的团队!
  7. 高数——无穷小的比较与等价无穷小
  8. 推荐系统-模型(一):召回模型【协同过滤类: ItemCF/UserCF】【Embedding类】【Dssm/双塔/word2vec】【图类召回算法 (Deepwalk、EGES)】
  9. 如何提取fMRI的时间序列,以及构建功能连接矩阵(functional connectivity)
  10. 机器学习-样本不均衡现象