原生js如何实现上拉加载下拉刷新?

简介: 原生js如何实现上拉加载下拉刷新?

原生js实现下拉刷新和上拉加载更多

下拉刷新

移动端触屏事件:

  • touchstart(手指按下的一瞬间)
  • touchmove(手指在屏幕上移动时)
  • touchend(手指松开时)

手指在页面上的坐标: pageX,pageY

写之前首先要懂原理,下拉刷新的本质就是用户在页面顶部进行上拉动作时拉到一定的距离触发数据刷新.

大概需要做的几个点:

1.在手指按下时(touchstart)记录手指的按下位置


2.在手指下滑时(touchmove)计算手指的坐标离手指按下时初始位置的差值得出下滑的距离,让容器顺应手指下滑的方向移动(translateY)对应差值的距离,对应的给一个允许用户下滑的最大距离,避免页面下拉过长.


3.在手指松开时(touchend)判断下滑的差值是否达到预期的值来进行对应的刷新数据和回弹loading.


less word,show me the code…

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport"
    content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
<title>Document</title>
<style type="text/css">
    * {
        margin: 0;
        padding: 0;
        list-style: none;
    }
    .container {
        /* 容器原始位置向上移动60px,隐藏掉loading盒子,下拉时才显示出来 */
        position: relative;
        top: -100px;
    }
    .container .loading {
        text-align: center;
        height: 100px;
        line-height: 100px;
    }
    .container .list {
        border: 1px solid #666;
    }
    .container .list li {
        line-height: 80px;
    }
    .container .list li:nth-child(2n) {
        background-color: #ccc;
    }
</style>
</head>
<body>
<section class="container">
    <section class="loading">
        <span>下拉刷新</span>
    </section>
    <section class="list">
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
    </section>
</section>
</body>
<script type="text/javascript">
// 滚动容器
const container = document.querySelector('.container');
// loading文字容器
const span = container.querySelector('span');
let startPosition = 0;// 下拉的开始位置
let distance = 0;// 下拉距离的差值
// 手指按下时
container.addEventListener('touchstart', function (e) {
    // 在回弹后的下一次下拉按下时重置loading文本
    span.textContent = '下拉刷新';
    // 记录开始位置
    startPosition = e.touches[0].pageY;
})
// 手指移动时
container.addEventListener('touchmove', function (e) {
    // 计算下拉差值
    const currentPosition = e.touches[0].pageY;
    // 计算下拉后离开始位置的差值
    distance = currentPosition - startPosition;
    // 如果下拉差值达到,则提示可以松手了 这个达到的具体值这里是取的下拉出来的区域高度
    if (distance > 100) {// 案例以100为临界值,超过了100的距离就提示释放刷新
        span.textContent = '释放刷新';
    }
    // 限制下滑的最大值为120,超过就不再下滑
    if (distance < 120) {
        // 容器的这个下滑是瞬时的 取消过渡效果
        this.style.transition = 'transform 0s';
        this.style.transform = `translateY(${distance}px)`
    }
})
// 手指松开时
container.addEventListener('touchend', function (e) {
    // 回弹的动作可以给个1s的过渡效果
    this.style.transition = 'transform 1s';
    // 如果下拉差值并没有达到 则直接回弹
    if (distance > 0 && distance < 100) {
        this.style.transform = `translateY(0px)`
        return;
    }
    if (distance > 100) {
        // 下拉差值达到了就显示刷新中,并暂时定格在这个位置
        this.style.transform = `translateY(100px)`;
        span.textContent = '刷新中';
        // 等数据回来后显示刷新成功1s然后再回弹 到这里本次整个下拉执行完毕
        setTimeout(() => {// setTimeout模拟异步请求 真实开发这里是一个promise请求
            span.textContent = '刷新成功';
            // 这个setTimeout让刷新成功显示一秒后再回弹
            setTimeout(() => {
                this.style.transform = `translateY(0px)`
            }, 1000)
        }, 2000);
    }
    // 一次下拉结束后重置差值
    distance = 0;
})
</script>
</html>

上拉加载

前置js知识点: 三个dom属性

  • clientHeight: 不包含边框的元素可视区高度
  • scrollTop: 元素滚动时卷上去的距离
  • scrollHeight: 元素实际高度,包含卷上去的高度

知道了上述三个属性后,于是就有了一个公式:

`clientHeight` + `scrollTop` <= `scrollHeight` - 触底的指定距离

然后就是上拉加载的原理:

通过监听元素的滚动事件(scroll)判断元素是否滚动到了距离底部指定距离时触发加载数据

知道了原理和三者之间的关系后,我们就知道只需要判断这个公式即可知道滚动条有没有进入触底距离, 话不多说 show code~

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport"
    content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
<title>Document</title>
<script src="https://cdn.bootcdn.net/ajax/libs/lodash.js/4.17.21/lodash.min.js"></script>
<style type="text/css">
    * {
        margin: 0;
        padding: 0;
        list-style: none;
    }
    .container {
        height: 100vh;
        overflow-y: scroll;
    }
    .container .list {
        border: 1px solid #666;
    }
    .container .list li {
        line-height: 80px;
    }
    .container .list li:nth-child(2n) {
        background-color: #ccc;
    }
</style>
</head>
<body>
<section class="container">
    <section class="list">
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
        <li>我是内容</li>
    </section>
</section>
</body>
<script type="text/javascript" defer="defer">
// 滚动容器
const container = document.querySelector('.container');
// 监听滚动事件
container.addEventListener('scroll', _.debounce(function (e) {
    // 当元素的可视高度+滚入的距离>=元素真实高度-触底距离时,触发加载更多
    if ((this.clientHeight + this.scrollTop) >= this.scrollHeight - 50) {
        setTimeout(() => {
            // 这里是一个异步加载数据的操作
            console.log('加载更多')
        }, 1000);
    }
}, 700))
</script>
</html>


相关文章
|
4月前
|
移动开发 前端开发 JavaScript
原生JavaScript+canvas实现五子棋游戏_值得一看
本文介绍了如何使用原生JavaScript和HTML5的Canvas API实现五子棋游戏,包括棋盘的绘制、棋子的生成和落子、以及判断胜负的逻辑,提供了详细的代码和注释。
55 0
原生JavaScript+canvas实现五子棋游戏_值得一看
|
5月前
|
Devops 持续交付 测试技术
JSF遇上DevOps:开发流程将迎巨变?一篇文章带你领略高效协同的魅力!
【8月更文挑战第31天】本文探讨了如何在JavaServer Faces(JSF)开发中融入DevOps文化,通过持续集成与部署、自动化测试、监控与日志记录及反馈机制,提升软件交付速度与质量。文中详细介绍了使用Jenkins进行自动化部署、JUnit与Selenium进行自动化测试、ELK Stack进行日志监控的具体方法,并强调了持续改进的重要性。
45 0
|
5月前
|
JavaScript 前端开发 API
从零开始学表单操作,jQuery 与原生 JavaScript 完全指南,带你轻松掌握网页交互关键!
【8月更文挑战第31天】在网页开发中,表单是实现用户互动的关键元素。无论是收集信息、提交数据还是验证输入,都需要对表单进行有效操作。本文档介绍了如何使用原生 JavaScript 和 jQuery 操作表单,包括获取表单元素、读写表单值、处理表单提交及验证等核心功能。jQuery 提供了更简洁的语法和更好的兼容性,但原生 JavaScript 在性能上有优势。选择合适的方法取决于项目需求和个人偏好。下面通过具体示例展示了两种方式的操作方法。
45 0
|
7月前
|
JavaScript 前端开发 安全
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
|
7月前
|
JavaScript
原生JS实现全选、全不选
原生JS实现全选、全不选
|
6月前
|
JavaScript
js 一键复制到剪贴板(原生js实现)
js 一键复制到剪贴板(原生js实现)
40 0
|
7月前
|
前端开发 JavaScript 容器
程序技术好文:纯原生javascript下拉框表单美化实例教程
程序技术好文:纯原生javascript下拉框表单美化实例教程
123 0
|
7月前
|
移动开发 JavaScript 前端开发
原生js如何获取dom元素的自定义属性
原生js如何获取dom元素的自定义属性
218 0
|
7月前
|
JavaScript 前端开发
原生JS如何查询元素属性
原生JS如何查询元素属性
55 0
|
7月前
|
JavaScript 开发者 前端开发
浅谈Vue.js与原生开发
Vue.js 是一款流行的前端框架,以其独特的模板语法简化了动态视图创建,通过指令和表达式便捷处理数据。与原生开发相比,Vue.js 提供了Vue Router进行高效路由管理,Vuex进行状态集中管理,以及丰富的生态系统和工具链如Vue CLI。Vue组件化开发、响应式数据绑定和单文件组件提高了代码复用和可维护性,但原生开发在性能和直接操作DOM方面可能更具优势。
107 0