Android自定义StickinessView粘性滑动效果

发布时间 - 2026-01-10 23:25:35    点击率:

design包的出现,Android界面发生了巨大变化,各种滑动配合的效果,下面我就粘性滑动中的一种进行自定义,效果图如下:


大家看到效果了,这里我是继承了LinerLayout,方便一点,若果是ViewGroup的话,也就复杂一点点。这里分为三部分:

1.head1,顶部可移动的Layout。
2.head2,固定的头部,不会滑动除屏幕外。
3.可滑动的Layout(这里只可以是ListView,不过也可以是任何可滑动的View,只要给出Head可滑动的时机即可)

本StickinessView的难点在于,解决滑动冲突和事件的拦截处理,接下来我一一道来。

一、首先,要确定HeadLayout什么时候可以拦截事件,那么就要确定ListView到达顶部和底部的时机。

 @Override
 public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
  View v = mListView.getChildAt(0);
  //当firstItem的top为0的时候就认为已经到达ListView的顶部了
  if (mListView.getChildCount() > 0 && firstVisibleItem == 0) {
   //滑动到顶部
   if (v.getTop() == 0) {
    //滑动到顶部了
    isListViewTop = true;
   } else {
    isListViewBottom = false;
   }
  }else if (mListView.getChildCount()>0&&firstVisibleItem+visibleItemCount==totalItemCount){
   final View bottomChildView = mListView.getChildAt(mListView.getChildCount()-1);
//当最后一个itemView的bottom>=ListView的高度的时候,那么就认为到达底部了
   if    (mListView.getHeight()>=bottomChildView.getBottom()){
    isListViewBottom = true;
   }else {
    isListViewBottom = false;
   }
  }else {
   isListViewBottom = false;
   isListViewTop = false;
  }

原因很简单,因为View的getTop和getBottom方法是相对父容器的位置,熟悉Layout方法的,想必就会很明白了。

二、知道了HeadView拦截事件的时机,我们就要搞清楚在此基础之上,我们到底啥时候拦击点击事件,进行滑动。

 @Override
 public boolean onInterceptTouchEvent(MotionEvent ev) {
  switch (ev.getAction()) {
   case MotionEvent.ACTION_DOWN:
    touchY = ev.getRawY();
    isIntercept = false;
    break;
   case MotionEvent.ACTION_MOVE:
    float distant = ev.getRawY() - touchY;
    if (isListViewTop) {
     switch (mHeadPosition) {
      case TOP:
       if (distant > 0) isIntercept = true;
       break;
      case CENTER:
       isIntercept = true;
       break;
     }
    }
    if (isListViewBottom){
     switch (mHeadPosition) {
      case CENTER:
       isIntercept = true;
       break;
      case BOTTOM:
       if (distant < 0) isIntercept = true;
       break;
     }
    }

    break;
   case MotionEvent.ACTION_UP:
    isIntercept = true;
    break;
  }
  return isIntercept;
 }

跟大家讲解一下onInterceptTouchEvent(MotionEvent ev),这个方法会最先调用,当一个事件序列拦截一次后,那么这个事件的后续事件动作就不会再调用该方法,也就是说,当该ViewGroup决定拦截某个事件后,那么它注定要消费后续的事件动作。这里贴出HeadView的位置状态

public static final int TOP = 0;//收缩状态
public static final int CENTER = 1;//中间状态
public static final int BOTTOM = 2;//展开状态

关于细节,想必大家画个图就可以知道了,注意一点:在拦截事件序列的时候,一般ACTION_DOWN事件不可以被拦截,因为拦截的话,没得意义了,后续事件就无法控制了,不可能继续往ChildView传递事件序列。

三、移动HeadView。

@Override
public boolean onTouchEvent(MotionEvent event) {
 switch (event.getAction()) {
  case MotionEvent.ACTION_DOWN:
   //获取不到的
   break;
  case MotionEvent.ACTION_MOVE:
   int distant = (int) (touchY - event.getRawY());
   if (getScrollY() + distant-1 < MAXY && getScrollY() + distant > 0) {
    scrollTo(0, getScrollY() + distant);
   }
   break;
  case MotionEvent.ACTION_UP:
   if (getScrollY() == 0) mHeadPosition = BOTTOM;
   if (getScrollY() == MAXY) mHeadPosition = TOP;
   if (getScrollY() > 0 && getScrollY() < MAXY) mHeadPosition = CENTER;
   if (getScrollY() > MAXY / 2) {
    mScroll.startScroll(0, getScrollY(), 0, MAXY-getScrollY(),100);
    invalidate();
    mHeadPosition = TOP;
   }
   if (getScrollY() < MAXY / 2) {
    mScroll.startScroll(0, getScrollY(),0,-getScrollY(),100);
    invalidate();
    mHeadPosition = BOTTOM;
   }
   break;
 }
 return super.onTouchEvent(event);
}

这里为了使得滑动跟家顺畅我使用了Scroller这个类,该类是专门处理弹性滑动的工具类,先初始化构造器,在调用startScroll()方法(其中四个参数:滑动的x,滑动的y,滑动x的偏移量,滑动y的偏移量),然后刷新视图,最后重写computeScroll()方法,

@Override
public void computeScroll() {
 super.computeScroll();
 if (mScroll.computeScrollOffset()){
  scrollTo(mScroll.getCurrX(),mScroll.getCurrY());
  postInvalidate();
 }
}

好了,基本完成,我们还要第一时间获取HeadView的高度,那么在onMeasure()中获取比较好,并且只获取一次如下

 if (MAXY == -1)
  MAXY = mHeadSecond.getMeasuredHeight();

在onFinishInflate()方法中,该方法的执行标志着所有的View都已经add完毕,这里我们进行初始化是比较妥当的。

 @Override
  protected void onFinishInflate() {
  super.onFinishInflate();
  int count = getChildCount();
  //本粘性布局只支持ListView
  if (count == 3 && getChildAt(2) instanceof ListView)
   init();
 }
 /**
  * 初始化
  */
 private void init() {
  //获得子元素
  mHeadFiest = getChildAt(0);
  mHeadSecond = getChildAt(1);
  mListView = (ListView) getChildAt(2);
  mListView.setOnScrollListener(this);
  mScroll = new Scroller(getContext());
 }

好了,基本就是这些。
GitHub地址:https://github.com/yzzAndroid/LianXinView

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


# Android  # StickinessView  # 粘性滑动  # Android中实现监听ScrollView滑动事件  # android 通过向viewpage中添加listview来完成滑动效果(类似于qq滑动界面)  # 解析Android中实现滑动翻页之ViewFlipper的使用详解  # Android利用ViewPager实现滑动广告板实例源码  # Android中实现水平滑动(横向滑动)ListView示例  # android配合viewpager实现可滑动的标签栏示例分享  # Android App中使用ViewPager+Fragment实现滑动切换效果  # Android编程中ViewPage判断左右滑动方向的方法  # Android ViewPager无限循环实现底部小圆点动态滑动  # Android中RecyclerView实现横向滑动代码  # 好了  # 知道了  # 我是  # 我就  # 偏移量  # 不可能  # 也就  # 在此  # 就不  # 什么时候  # 我一  # 不可以  # 很简单  # 比较好  # 自定义  # 重写  # 第一时间  # 会很  # 会再  # 标志着 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何与Vue.js集成_Laravel + Vue前后端分离项目搭建指南  Laravel如何处理跨站请求伪造(CSRF)保护_Laravel表单安全机制与令牌校验  Laravel如何使用Gate和Policy进行授权?(权限控制)  韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南  如何用y主机助手快速搭建网站?  如何在建站宝盒中设置产品搜索功能?  消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工  Python正则表达式进阶教程_复杂匹配与分组替换解析  香港服务器网站推广:SEO优化与外贸独立站搭建策略  怎么用AI帮你为初创公司进行市场定位分析?  Python结构化数据采集_字段抽取解析【教程】  Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】  打造顶配客厅影院,这份100寸电视推荐名单请查收  网站制作免费,什么网站能看正片电影?  微信推文制作网站有哪些,怎么做微信推文,急?  Laravel Seeder填充数据教程_Laravel模型工厂Factory使用  浅述节点的创建及常见功能的实现  EditPlus中的正则表达式实战(5)  Python进程池调度策略_任务分发说明【指导】  如何在阿里云部署织梦网站?  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  javascript事件捕获机制【深入分析IE和DOM中的事件模型】  如何在阿里云完成域名注册与建站?  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  为什么要用作用域操作符_php中访问类常量与静态属性的优势【解答】  公司网站制作需要多少钱,找人做公司网站需要多少钱?  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  Laravel如何集成Inertia.js与Vue/React?(安装配置)  公司门户网站制作流程,华为官网怎么做?  PHP 500报错的快速解决方法  大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】  网易LOFTER官网链接 老福特网页版登录地址  Linux网络带宽限制_tc配置实践解析【教程】  如何用PHP快速搭建CMS系统?  Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明  如何彻底删除建站之星生成的Banner?  怎么用AI帮你设计一套个性化的手机App图标?  如何将凡科建站内容保存为本地文件?  简单实现Android验证码  Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】  javascript如何操作浏览器历史记录_怎样实现无刷新导航  Gemini怎么用新功能实时问答_Gemini实时问答使用【步骤】  使用PHP下载CSS文件中的所有图片【几行代码即可实现】  Python文件操作最佳实践_稳定性说明【指导】  如何登录建站主机?访问步骤全解析  Laravel怎么在Controller之外的地方验证数据  HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】