面试必备-Vue3+Ts手写实现无限滚动列表

1,909

先看成果

动画.gif

无限滚动列表

无限滚动列表(Infinite Scroll)是一种在网页或应用程序中加载和显示大量数据的技术。它通过在用户滚动到页面底部时动态加载更多内容,实现无缝的滚动体验,避免一次性加载所有数据而导致性能问题。供更流畅的用户体验。但需要注意在实现时,要考虑合适的加载阈值、数据加载的顺序和流畅度,以及处理加载错误或无更多数据的情况,下面我们用IntersectionObserver来实现无线滚动,并且在vue3+ts中封装成一个可用的hook

IntersectionObserver是什么

IntersectionObserver(交叉观察器)是一个Web API,用于有效地跟踪网页中元素在视口中的可见性。它提供了一种异步观察目标元素与祖先元素或视口之间交叉区域变化的方式。 IntersectionObserver的主要目的是确定一个元素何时进入或离开视口,或者与另一个元素相交。它在各种场景下非常有用,例如延迟加载图片或其他资源,实现无限滚动等。

这里用一个demo来做演示

动画.gif

demo代码如下,其实就是用IntersectionObserver来对某个元素做一个监听,通过siIntersecting属性来判断监听元素的显示和隐藏。

 const observer = new IntersectionObserver((entries) => {
  entries.forEach((entry) => {
    if (entry.isIntersecting) {
        console.log('元素出现');
    } else{
      console.log('元素隐藏');
    }
  });
});
observer.observe(bottom);

无限滚动实现

下面我们开始动手

1.数据模拟

模拟获取数据,比如分页的数据,这里是模拟的表格滚动的数据,每次只加载十条,类似于平时的翻页效果,这里写的比较简单, 在这里给它加了一个最大限度30条,超过30条就不再继续增加了

<template>
  <div ref="container" class="container">
    <div v-for="item in list" class="box">{{ item.id }}</div>
  </div>
</template>
<script setup lang="ts">

const list: any[] = reactive([]);
let idx = 0;

function getList() {
  return new Promise((res) => {
    if(idx<30){
      for (let i = idx; i < idx + 10; i++) {
        list.push({ id: i });
      }
      idx += 10
    }
    res(1);
  });
</script>

2.hook实现

import { createVNode, render, Ref } from 'vue';
/**
 接受一个列表函数、列表容器、底部样式
 */
export function useScroll() {
    // 用ts定义传入的三个参数类型
    async function init(fn:()=>Promise<any[] | unknown>,container:Ref) {
        const res = await fn();
      }
    return { init }
}

执行init就相当于加载了第一次列表 后续通过滚动继续加载列表

import { useScroll } from "../hooks/useScroll.ts";
onMounted(() => {
  const {init} = useScroll()
  //三个参数分别是 加载分页的函数 放置数据的容器 结尾的提示dom
  init(getList,container,bottom)
});

3.监听元素

export function useScroll() {
    // 用ts定义传入的三个参数类型
    async function init(fn:()=>Promise<any[] | unknown>,container:Ref,bottom?:HTMLDivElement) {
        const res = await fn();
        // 使用IntersectionObserver来监听bottom的出现
        const observer = new IntersectionObserver((entries) => {
          entries.forEach((entry) => {
            if (entry.isIntersecting) {
                fn();
                console.log('元素出现');
            } else{
              console.log('元素隐藏');

            }
          });
        });
        observer.observe(bottom);
      }
    return { init }
}

4.hook初始化

获取需要做无限滚动的容器 这里我们用ref的方式来直接获取到dom节点 大家也可以尝试下用getCurrentInstance这个api来获取到

整个实例,其实就是类似于vue2中的this.$refs.container来获取到dom节点容器

根据生命周期我们知道dom节点是在mounted中再挂载的,所以想要拿到dom节点,要在onMounted里面获取到,毕竟没挂载肯定是拿不到的嘛


const container = ref<HTMLElement | null>(null);
onMounted(() => {
  const vnode = createVNode('div', { id: 'bottom',style:"color:#000" }, '到底了~');
  render(vnode, container.value!);
  const bottom = document.getElementById('bottom') as HTMLDivElement;
  // 用到的是createVNode来生成虚拟节点 然后挂载到容器container中
  const {init} = useScroll()
  //三个参数分别是 加载分页的函数 放置数据的容器 结尾的提示dom
  init(getList,container,bottom)
});

这部分代码是生成放到末尾的dom节点 封装的init方法可以自定义传入末尾的提示dom,也可以不传,封装的方法中有默认的dom

优化功能

1.自定义默认底部提示dom

async function init(fn:()=>Promise<any[] | unknown>,container:Ref,bottom?:HTMLDivElement) {
        const res = await fn();
        // 如果没有传入自定义的底部dom 那么就生成一个默认底部节点 
        if(!bottom){
          const vnode = createVNode('div', { id: 'bottom',style:"color:#000" }, '已经到底啦~');
          render(vnode, container.value!);
          bottom = document.getElementById('bottom') as HTMLDivElement;
        }
        // 使用IntersectionObserver来监听bottom的出现
        const observer = new IntersectionObserver((entries) => {
          entries.forEach((entry) => {
            if (entry.isIntersecting) {
                fn();
                console.log('元素出现');
            } else{
              console.log('元素隐藏');

            }
          });
        });
        observer.observe(bottom);
      }

完整代码

import { createVNode, render, Ref } from 'vue';
/**
 接受一个列表函数、列表容器、底部样式
 */
export function useScroll() {
    async function init(fn:()=>Promise<any[] | unknown>,container:Ref,bottom?:HTMLDivElement) {
        const res = await fn();
        // 生成一个默认底部节点
        if(!bottom){
          const vnode = createVNode('div', { id: 'bottom' }, '已经到底啦~');
          render(vnode, container.value!);
          bottom = document.getElementById('bottom') as HTMLDivElement;
        }
        const observer = new IntersectionObserver((entries) => {
          entries.forEach((entry) => {
            if (entry.isIntersecting) {
                fn();
            } 
          });
        });
        observer.observe(bottom);
      }
    return { init }
}

<template>
  <div ref="container" class="container">
    <div v-for="item in list" class="box">{{ item.id }}</div>
  </div>
</template>
<script setup lang="ts">
import { onMounted, createVNode, render, ref, reactive } from 'vue';
import { useScroll } from "../hooks/useScroll.ts";
const list: any[] = reactive([]);
let idx = 0;
function getList() {
  return new Promise((res,rej) => {
    if(idx<=30){
      for (let i = idx; i < idx + 10; i++) {
        list.push({ id: i });
      }
      idx += 10
      res(1);
    }
    rej(0)
  });
}

const container = ref<HTMLElement | null>(null);
onMounted(() => {
  const vnode = createVNode('div', { id: 'bottom' }, '到底了~');
  render(vnode, container.value!);
  const bottom = document.getElementById('bottom') as HTMLDivElement;
  const {init} = useScroll()
  init(getList,container,bottom)
});

</script>
<style scoped>
.container {
  border: 1px solid black;
  width: 200px;
  height: 100px;
  overflow: overlay
}

.box {
  height: 30px;
  width: 100px;
  background: red;
  margin-bottom: 10px
}</style>