1、async component render;
2、locate/update/delete/add specify location;
3、scroll loading;
4、chat mode(keep bottom position);
5、loading mode(scroll loading);
1、install;
npm install --save @vb_he/vue-virtual-scroll
2、import;
import VirtualList from '@vb_he/vue-virtual-scroll'
import '@vb_he/vue-virtual-scroll/dist/style.css'
// global registry
createApp(App).use(VirtualList).mount('#app')
3、use in vue component
<div style="width: 800px;height: 1000px;">
    <Virtual-list
    ref="virtualScroll"
    :scrollItem="StaticItem"
    :height="42"
    :perPageItemNum="20"
    ></Virtual-list>
</div>
base props:
| key | default value | required | description | 
|---|---|---|---|
| scrollItem | null | true | scroll item inside component | 
| perPageItemNum | 20 | true | at less display area number, normaly component keep render 2 * perPageItemNum item | 
| height | 100 | true | item size(height),for better performance | 
| direction | 'down' | false | 'up'/'down', need to set 'up' in chat mode | 
| loadingOptions | null | false | see under | 
loadingOptions:
| key | default value | required | description | 
|---|---|---|---|
| loadingFn | null | true | need to return Promise<any[]> | 
| loadingComponent | loading.io style | false | can replace default loading component | 
| nomoreData | false | false | set true if no more data | 
| nomoreDataText | 'no more data' | false | set no more data text | 
you may need to understand expose
| name | arguments | description | 
|---|---|---|
| setSourceData | data: any[],locateIndex?: number | set list data | 
| locate | index: number | locate at specify index | 
| del | index: number | number[] | delete some item | 
| add | index: number, insertData: any[] | add item/items | 
| update | index: number, data: any | update item | 
| getData | null | get list data | 
| getCurrentViewPortData | null | get current viewport data | 
basically, you can only use all this methods to change the view, e.g.
<script setup lang="ts">
import { onMounted, ref, reactive } from 'vue'
import { VirtualExpose } from '@vb_he/vue-virtual-scroll'
import StaticItem from './../components/static.component.vue'
import { getMessage } from "./../mock"
const virtualScroll = ref<VirtualExpose>()
const data = reactive({
    locateNum: 0,
    updateNum: 0,
    delNum: 0,
    addNum: 0,
    size: 100000
})
onMounted(async() => {
    virtualScroll.value!.setSourceData(await loadData())
})
function loadData(): Promise<any[]> {
    return new Promise((resolve) => {
        setTimeout(() => {
            resolve(getMessage(data.size))
        },1000)
    })
}
function locate() {
    virtualScroll.value?.locate(data.locateNum)
}
async function update() {
    const newData = await getMessage(1)
    virtualScroll.value?.update(data.updateNum, newData[0])
}
function del() {
    virtualScroll.value?.del(data.delNum)
}
async function add() {
    virtualScroll.value?.add(data.addNum, await getMessage(1))
}
async function reset() {
    virtualScroll.value!.setSourceData(await loadData())
}
</script>
<template>
    <div style="width: 800px;height: 1000px;">
        <Virtual-list
        ref="virtualScroll"
        :perPageItemNum="40"
        :scrollItem="StaticItem"
        :height="42"
        ></Virtual-list>
    </div>
</template>
1、element must set height value(100px, 100%, flex, etc.) which was wrapping VirtualList;
2、for better performance, emit 'itemLoaded' function to cached offsetHeight when aysnc element(e.g. image) loaded, e.g.
const emits = defineEmits(['itemLoaded'])
function imageLoaded() {
    emits('itemLoaded')
}
<template>
    <img @load="imageLoaded" :src="itemData.imgUrl" />
</template>