vue3 无限自滚动列表的实现

最近有个类似虚拟列表的需求,但是要求自己滚动。

实现

我琢磨了一下,用了以下的方法实现。

  • 不管一共有多少数据,每一时刻实际渲染的数据量应该是固定的,我这里用了两屏的数据
  • 这里的滚动没有用滚动条,用的是top的偏移量来实现,因为我这里的需求只需要自己滚动,不用手动滚动,用滚动条的话,相同的原理,应该也可以实现,但是还要把滚动条隐藏了,麻烦。
  • 由于实际的渲染量一定,所以当top 的偏移量等于一屏数据的高度的时候,应该把top 的值重置为零,同时要把已经消失在视野的那一屏数据删除,这样的配合下,top 的改变就不会引起页面的闪动,因为此时的位置下top就是等于零。
  • 在上面删除了已经看不见的一屏数据的时候,就一定要在下面添加上新的一屏数据,这样,就能保证页面上始终有且只有两屏数据,也能确保滚动的这个效果。
  • 最后一屏的数据量可能不够一屏的,这里没有做处理,好处是知道这一轮的数据结束了,当然也可以做个精细化的处理,实现真正的无限滚动,这个看具体的业务需求。

优化项

  • 给页面中元素添加子元素的时候,没有直接添加,而是用的createDocumentFragment,能减少元素的插入次数
  • 这里的滚动走的不是setTimeout ,而是用的requestAnimationFrame

效果

这里可以看见,容器的top值一直在变动,同时里面的两个子元素也是动态变化的。

代码

js 复制代码
<template>
  <div ref="list" class="rollList_content">
    <div ref="vlist" class="vabsolute_content">
      <div class="fragment_content"></div>
      <div class="fragment_content"></div>
    </div>
  </div>
</template>

<script setup>
import { ref, onMounted } from 'vue'

const list = ref()
const vlist = ref()
const containerHeight = ref(600)
let perCount = 0
const lineHight = 20
const count = 40
const data = ref([])
const currentData = ref([])
let startIndex = 0
let endIndex = 0

for (let i = count; i >= 0; i--) {
  data.value.push(i)
}

const fillEle = (ele, startIndex, endIndex) => {
  const fragment = document.createDocumentFragment()
  for (let i = startIndex; i < endIndex; i++) {
    const newLine = document.createElement('div')
    newLine.className = 'roll_line'
    newLine.textContent = data.value[i]
    fragment.appendChild(newLine)
  }
  ele.appendChild(fragment)
}

const loop = () => {
  if (startIndex > data.value.length) {
    startIndex = endIndex = 0
    vlist.value.firstElementChild.innerHTML = ''
    vlist.value.childNodes[1].innerHTML = ''
  }
  if (startIndex === endIndex) {
    let firstFragment = vlist.value.firstElementChild

    endIndex += perCount
    fillEle(firstFragment, startIndex, endIndex)
    const secondChildNode = vlist.value.childNodes[1]
    startIndex = endIndex
    endIndex += perCount
    fillEle(secondChildNode, startIndex, endIndex)
  } else {
    let currentTop = parseInt(vlist.value.style?.top, 10) || 0
    if (Math.abs(currentTop) >= lineHight * perCount) {
      vlist.value.removeChild(vlist.value.firstElementChild)
      vlist.value.style.top = 0
      const newFragment = document.createElement('div')
      newFragment.className = 'fragment_content'
      startIndex = endIndex
      endIndex += perCount
      fillEle(newFragment, startIndex, endIndex)
      vlist.value.appendChild(newFragment)
    } else {
      vlist.value.style.top = `${currentTop - 1}px`
    }
  }

  window.requestAnimationFrame(loop)
}

onMounted(() => {
  containerHeight.value = list.value.offsetHeight
  perCount = ~~(containerHeight.value / lineHight)
  loop()
})
</script>

<style lang="scss" scoped>
.rollList_content {
  height: 100%;
  overflow-y: hidden;
  position: relative;
  .vabsolute_content {
    position: absolute;
    top: 0;
    .roll_line {
      display: flex;
      height: 20px;
      color: #fff;
    }
  }
}
</style>
<style>
.roll_line {
  display: flex;
  height: 20px;
  color: #fff;
}
</style>
相关推荐
赵大仁20 分钟前
深入理解 Vue 3 中的具名插槽
前端·javascript·vue.js·react.js·前端框架·ecmascript·html5
一雨方知深秋25 分钟前
v-bind 操作 class(对象,数组),v-bind 操作 style
前端·css·vue.js·html·style·class·v-bind
安晴晚风2 小时前
从0开始在linux服务器上部署SpringBoot和Vue
linux·运维·前端·数据库·后端·运维开发
前端小小王3 小时前
pnpm、Yarn 和 npm 的区别?
前端·npm·node.js
supermapsupport3 小时前
使用npm包的工程如何引入mapboxgl-enhance/maplibre-gl-enhance扩展包
前端·webpack·npm·supermap·mapboxgl
牛奔3 小时前
windows nvm 切换node版本后,npm找不到
前端·windows·npm·node.js
鱼大大博客3 小时前
Edge SCDN酷盾安全重塑高效安全内容分发新生态
前端·安全·edge
鸭梨山大。3 小时前
NPM组件包 vant部分版本内嵌挖矿代码
前端·安全·npm·node.js·vue
蟾宫曲8 小时前
在 Vue3 项目中实现计时器组件的使用(Vite+Vue3+Node+npm+Element-plus,附测试代码)
前端·npm·vue3·vite·element-plus·计时器
秋雨凉人心8 小时前
简单发布一个npm包
前端·javascript·webpack·npm·node.js