虚拟滚动-数组平级数据结构

2021-08-27  本文已影响0人  有一个程序媛

虚拟滚动平级数据结构

import { onMounted, defineComponent, ref, computed, watch, toRefs } from 'vue'

const Index = defineComponent({

  props: {

    /**

    * 可见区域内显示多少条--区域高度/每一条记录的样式高度

    */

    pageSize: {

      type: Number,

      default: 20

    },

    /**

    * 数组数据

    */

    list: {

      type: Array,

      default: []

    },

    /**

    * 容器高度

    */

    height: {

      type: Number,

      default: 400

    },

    /**

    * 获取接口函数

    */

    getData: {

      type: Function

    },

    /**

    * 获取接口的参数

    */

    params: {

      type: Object,

      default: {}

    }

  },

  name: 'virtualScroll',

  setup(props, ctx) {

    const { pageSize, height, params } = toRefs(props)

    const getData = props.getData

    const list = ref([])

    const startIndex = ref(0) // 当前区域显示的数据起始index

    const endIndex = ref(pageSize.value) // 数据结束index

    const paddingTop = ref(0) // 当前区域下scrollList区域的paddingtop值

    const paddingBottom = ref(0) // 当前区域下scrollList区域的paddingbottom值

    const allHeight = ref(0) // 大量无数条数据的高度

    const itemHeight = ref(22)

    const myRef = ref(null)

    const scrollList = computed(() => list.value.slice(startIndex.value, endIndex.value)) //滚动区域的数据集合

    const defaultStyle = {'overflow-y': 'auto', 'height': height.value + 'px'}

    /**

    * 滚动区域的样式

    */

    const style = computed(() => {

      return {

        paddingTop: paddingTop.value + 'px',

        paddingBottom: paddingBottom.value + 'px'

      }

    })

    /**

    * 监听滚动条的滚动事件

    */

    const scroll = (e) => {

      const container = document.getElementById('container')

      const top = container.scrollTop

      startIndex.value = Math.floor(top / itemHeight.value) // 已经滚走了多少条数据 (当前区域的list start下标)

      endIndex.value = startIndex.value + pageSize.value // 当前区域的list end下标

      paddingTop.value = top

      if (endIndex.value >= list.value.length - 1) { // 最后一条 直接return

        paddingBottom.value = 0

        return

      }

      paddingBottom.value = allHeight.value - height - top // 否则就拿所有数据的高度 - 当前可视区域的高度 - paddingtop值 = paddingbottom值

    }

    /**

    * 获取总高度,padding值

    */

    const getHeightPadding = (val) => {

      list.value = val

      const valLen = val.length

      allHeight.value = valLen * itemHeight.value

      paddingBottom.value = allHeight.value - scrollList.value.length * itemHeight.value

    }

    /**

    * 监听数据的变化,padding值和高度相应变化

    */

    watch(() => list.value, (val) => {

      getHeightPadding(val)

    })

    onMounted(() => {

      getData(params.value).then((res) => {

        getHeightPadding(res)

      })

    })

    return () => (

      <div ref={myRef} id="container" style={defaultStyle} onscroll={(e) => {

        scroll(e)

      }}>

        <div style={style.value}>

          {scrollList.value.map((item) => {

            return <div>{ctx.slots.content?ctx.slots.content(item) : item.name}</div>

          })}

        </div>

      </div>

    )

  },

  // render() {

  // }

})

export default Index

上一篇下一篇

猜你喜欢

热点阅读