原生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>


相关文章
|
25天前
|
JavaScript 前端开发 安全
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
安全开发-JS应用&原生开发&JQuery库&Ajax技术&加密编码库&断点调试&逆向分析&元素属性操作
|
1月前
|
JavaScript
原生JS实现全选、全不选
原生JS实现全选、全不选
|
13天前
|
JavaScript
js 一键复制到剪贴板(原生js实现)
js 一键复制到剪贴板(原生js实现)
10 0
|
17天前
|
前端开发 JavaScript 容器
程序技术好文:纯原生javascript下拉框表单美化实例教程
程序技术好文:纯原生javascript下拉框表单美化实例教程
|
27天前
|
移动开发 JavaScript 前端开发
原生js如何获取dom元素的自定义属性
原生js如何获取dom元素的自定义属性
23 0
|
27天前
|
JavaScript 前端开发
原生JS如何查询元素属性
原生JS如何查询元素属性
18 0
|
2月前
|
前端开发 JavaScript
前端 js 经典:Object 常用原生方法
前端 js 经典:Object 常用原生方法
68 2
|
2月前
|
前端开发 JavaScript
前端 js 经典:array 原生方法
前端 js 经典:array 原生方法
28 1
|
1月前
|
JavaScript 开发者 前端开发
浅谈Vue.js与原生开发
Vue.js 是一款流行的前端框架,以其独特的模板语法简化了动态视图创建,通过指令和表达式便捷处理数据。与原生开发相比,Vue.js 提供了Vue Router进行高效路由管理,Vuex进行状态集中管理,以及丰富的生态系统和工具链如Vue CLI。Vue组件化开发、响应式数据绑定和单文件组件提高了代码复用和可维护性,但原生开发在性能和直接操作DOM方面可能更具优势。
|
2月前
|
JavaScript 索引
jQuery 实现 图片框切换【与原生 JS 对比】
jQuery 实现 图片框切换【与原生 JS 对比】