经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » HTML/CSS » HTML5 » 查看文章
轻松实现H5页面下拉刷新:滑动触发、高度提示与数据刷新全攻略
来源:cnblogs  作者:一颗冰淇淋  时间:2024/6/17 15:15:33  对本文有异议

前段时间在做小程序到H5的迁移,其中小程序中下拉刷新的功能引起了产品的注意。他说到,哎,我们迁移后的H5页面怎么没有下拉刷新,于是乎,我就急忙将这部分的内容给填上。

本来是计划使用成熟的组件库来实现,尝试之后发现这些组件和我们H5页面的其他逻辑有冲突(H5还有吸顶、锚点、滑动高亮、横向滚动),小小H5页面上承载了太多的功能,兼容起来非常麻烦,想着下拉刷新功能也不复杂,干脆我自己写一个好了。

流程图示

正常数据展示状态 --> 手指触摸屏幕下拉 --> 手指松开 --> 数据获取 --> 恢复正常数据展示状态

功能梳理

要实现这个功能,主要分为两部分。

监听手指触摸事件

通过监听事件,我们可以得知以下的数据

  • 手指滑动的时机(手指开始触摸,结束触摸时间)
  • 滑动方向(是横向滑动还是纵向滑动)
  • 操作轨迹(手指操作从下往上还是从上往下滑动)
  • 是否首屏(如果非首屏进行滑动时是正常滑动操作)
    只有在向下滑动首屏非加载状态纵向滚动并且有高度时,才能进行上述刷新流程。

css 和 提示文案

  • 手指按住屏幕由上往下滑动未松开时,展示滑动的高度和提示【释放刷新】文案
  • 手指松开后高度回弹,显示【数据更新中】文案
  • 数据请求接口成功后,显示【更新成功】文案,loading 内容和图标缓缓消失

具体实现

触摸的步骤可以分为: 手指按下(开始触摸)、手指移动不离开屏幕(触摸中)、手指离开屏幕(触摸结束),正好对应着三个 js 原生事件,touchstarttouchmovetouchend

触摸事件执行时机

touchstart 和 touchmove 在一次触摸流程只会执行一次,标志着开始和结束,但是 touchmove 不一样,只要你的手指还在屏幕上滑动没有松开,就会一直执行。如下图的输出的执行次数一样。

下拉元素绑定

首先需要给需要设置下拉刷新的区域绑定上这些事件,对于我们业务场景来说,头部区域无论你如何操作,都需要保留展示的,那么我们只需要将事件绑定到下方开始显示下拉刷新的区域。

  1. // html元素
  2. <div className="refreshWrap">
  3. {/* 下拉时文字提示 */}
  4. <div className={`pullDownContent`} style={{ height: pullDownHeight }}>
  5. {loading ? "" : "释放刷新"}
  6. </div>
  7. {/* 加载时动画 */}
  8. <div className={`loadingFlex ${loading ? "" : "loadingHidden"}`}>
  9. <div className="flexCenter">
  10. <div className="loadingRing" />
  11. <div className="loadingText">
  12. {loading ? "数据更新中..." : "更新成功"}
  13. </div>
  14. </div>
  15. </div>
  16. <div className="middleArea">刷新区域下方内容区域</div>
  17. </div>
  18. // js 绑定
  19. const pullDownClassName = ".refreshWrap";
  20. bindPullDown() {
  21. const pulldownElement = document.querySelector(pullDownClassName);
  22. pulldownElement.addEventListener("touchstart", this.bindTouchstart);
  23. pulldownElement.addEventListener("touchmove", this.bindTouchMove);
  24. pulldownElement.addEventListener("touchend", this.bindTouched);
  25. }

触摸开始

手指触摸到屏幕的逻辑非常简单,使用 startTouch 对象来记录触摸的位置,包含 x 、y 轴。

  1. bindTouchstart = (event) => {
  2. this.startTouch = event.touches[0];
  3. };

触摸中

用户触摸中需要给他一个反馈,随着下拉的距离,屏幕上圈出的下拉区域会随之变大(下拉展示的区域会设置一个最大高度,如果能无限扩大展示不好看)

endTouch 来保存触摸中的坐标值,因为触摸中的事件会执行多次,所以 endTouch 也会不断的更新,用来更新下拉时滑动的高度。

  1. bindTouchMove = (event) => {
  2. const { loading } = this.state;
  3. this.endTouch = event.touches[0];
  4. if (!loading && this.isInOneScreenPull() && this.isVerticalSliding()) {
  5. const pullDownHeight = this.getPullDownHeight();
  6. this.setState({
  7. pullDownHeight,
  8. });
  9. }
  10. };

根据 endTouch 的值可以判断出滑动距离、横向还是纵向滑动,滑动的高度、再获取滑动元素是否在首屏。

  1. // 判断滑动的距离
  2. calcDeltaY = () => Math.abs(this.endTouch.pageY - this.startTouch.pageY);
  3. // 判断是否纵向滚动
  4. isVerticalSliding = () => {
  5. const deltaY = this.calcDeltaY();
  6. const deltaX = Math.abs(this.endTouch.pageX - this.startTouch.pageX);
  7. if (deltaY > deltaX && deltaY > 50) return true;
  8. };
  9. // 下拉展示高度最多展示为100,不能让加载区域无限制的扩大
  10. getPullDownHeight = () => {
  11. const deltaY = this.calcDeltaY();
  12. return Math.min(deltaY, 100);
  13. };
  14. // 是否在首屏
  15. isInOneScreenPull() {
  16. const pulldownElement = document.querySelector(pullDownClassName);
  17. return pulldownElement.scrollTop <= 0;
  18. }

触摸结束

触摸结束时,将 pulldownHeight 设置为0,异步加载数据,加载数据时设置变量 loading 表示开始更新、结束更新,防止不停的下拉刷新调用接口。

  1. bindTouched = (e) => {
  2. const { loading, pullDownHeight } = this.state;
  3. // 首屏、非加载状态、纵向滚动有高度时
  4. if (!loading && pullDownHeight) {
  5. this.setState({
  6. pullDownHeight: 0,
  7. });
  8. this.getData();
  9. // 重置触摸Y轴坐标点
  10. this.startTouch = {};
  11. this.endTouch = {};
  12. }
  13. };

平滑过渡动画

当下拉高度发生变化时,直接修改高度效果会比较生硬,使用 css transition 属性进行平滑过渡、animation 设置动画缓慢进入/消失。

  1. .pullDownContent {
  2. display: flex;
  3. align-items: flex-end;
  4. justify-content: center;
  5. font-size: 12px;
  6. color: rgba(0, 0, 0, 0.25);
  7. margin: auto;
  8. transition: height 0.3s ease-out; /* 平滑过渡效果 */
  9. overflow: hidden;
  10. }
  11. .loadingHidden {
  12. animation: shrinkHeight 1s forwards;
  13. }
  14. @keyframes shrinkHeight {
  15. 100% {
  16. height: 0;
  17. opacity: 0;
  18. overflow: hidden;
  19. }
  20. }

完整代码

以上便是滑动触发、高度提示、数据刷新的下拉刷新功能解析,完整代码我放在了 github 上,戳 drop-down-refresh 可查看,欢迎大家点个 star~

原文链接:https://www.cnblogs.com/vigourice/p/18250533

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号