vue-zscroller
v0.3.23
Published
Virtual scrolling components for Vue 3.5+, rewritten in TypeScript
Readme
vue-zscroller
Virtual scrolling components for Vue 3.5+, written from scratch in TypeScript. Based on the API of vue-virtual-scroller by Guillaume Chau (Akryum).
Not a fork -- every component was rewritten from the ground up using <script setup lang="ts">, @vueuse/core, and Vue 3.5 reactive patterns. The original API is preserved so migration is straightforward.
Maintainer: Zach Handley ([email protected])
Features
- Strict TypeScript with full type definitions and
defineSlotsfor IDE IntelliSense - Global component type augmentation via
app.use() - Uses
@vueuse/coreinstead of rolling custom utilities - SSR-safe -- works with server-side rendering out of the box
- CSS auto-injected, no separate import needed
- Optional
filterprop for filtering items without recreating the scroller - Built-in custom scrollbar with math-based thumb positioning (no jumpy native scrollbar)
- Components: RecycleScroller, DynamicScroller, DynamicScrollerItem, GridScroller
Installation
pnpm add vue-zscrollernpm install vue-zscrolleryarn add vue-zscrollerDefault import
Install all the components:
import VueVirtualScroller from 'vue-zscroller'
app.use(VueVirtualScroller)Use specific components:
import { RecycleScroller } from 'vue-zscroller'
app.component('RecycleScroller', RecycleScroller)Browser
<script src="vue.js"></script>
<script src="vue-zscroller/dist/vue-zscroller.min.js"></script>Install the component:
app.use(VueVirtualScroller)Or register it with a custom name:
app.component('RecycleScroller', VueVirtualScroller.RecycleScroller)Usage
Composables
The library exports composables for advanced use cases:
import {
useCustomScrollbar,
useSlotRefManager,
useSSRSafeEnhanced,
useVirtualScrollCore,
useVirtualScrollPerformance
} from 'vue-zscroller'
// Advanced slot management with automatic cleanup
const slotManager = useSlotRefManager({
enableWeakMap: true,
maxSize: 1000
})
// Performance monitoring and optimization
const perf = useVirtualScrollPerformance({
enableScrollDebounce: true,
scrollDebounceMs: 16,
enableRequestIdleCallback: true
})
// Enhanced SSR utilities
const { useSSRSafeStorage, useSSRSafeViewport } = useSSRSafeEnhanced()Performance monitoring
You can track scroll performance with useVirtualScrollPerformance:
import { useVirtualScrollPerformance } from 'vue-zscroller'
const perf = useVirtualScrollPerformance({
enableScrollDebounce: true,
enableRequestIdleCallback: true
})
// Access performance stats
const stats = perf.getPerformanceStats()
// Reset metrics
perf.resetMetrics()
// Reactive state
console.log(perf.isScrolling.value)
console.log(perf.scrollDirection.value)
console.log(perf.scrollVelocity.value)There are several components provided by vue-zscroller:
RecycleScroller is a component that only renders the visible items in your list. It also re-uses components and dom elements to be as efficient and performant as possible.
DynamicScroller is a component that wraps the RecycleScroller component and extends its features to include dynamic size management. The main use case for this is when you do not know the size of the items in advance. The Dynamic Scroller automatically "discovers" item dimensions as it renders new items during scrolling.
DynamicScrollerItem must wrap each item in a DynamicScroller to handle size computations.
GridScroller is a responsive grid layout that wraps RecycleScroller. You give it itemWidth and itemHeight and it figures out how many columns fit in the container, recalculating on resize.
useIdState is a composable that eases local state management in reused components inside a RecycleScroller.
RecycleScroller
RecycleScroller is a virtual scroller that only renders the visible items. As the user scrolls, RecycleScroller reuses all components and DOM nodes to maintain optimal performance.
Basic usage
Use the scoped slot to render each item in the list:
<template>
<RecycleScroller
class="scroller"
:items="list"
:item-size="32"
key-field="id"
v-slot="{ item }"
>
<div class="user">
{{ item.name }}
</div>
</RecycleScroller>
</template>
<script setup lang="ts">
interface Props {
list: Array<{ id: string; name: string }>
}
defineProps<Props>()
</script>
<style scoped>
.scroller {
height: 100%;
}
.user {
height: 32px;
padding: 0 12px;
display: flex;
align-items: center;
}
</style>Important notes
- You need to set the size of the virtual-scroller element and the items elements (for example, with CSS). Unless you are using variable size mode, all items should have the same height (or width in horizontal mode) to prevent display glitches.
- If the items are objects, the scroller needs to be able to identify them. By default it will look for an
idfield on the items. This can be configured with thekeyFieldprop if you are using another field name. - It is not recommended to use functional components inside RecycleScroller since the components are reused (so it will actually be slower).
- The list item components must be reactive to the
itemprop being updated without being re-created (use computed props or watchers to properly react to props changes!). - You don't need to set
keyon list content (but you should on all nested<img>elements to prevent load glitches). - The browsers have a size limitation on DOM elements, it means that currently the virtual scroller can't display more than ~500k items depending on the browser.
- Since DOM elements are reused for items, it's recommended to define hover styles using the provided
hoverclass instead of the:hoverstate selector (e.g..vue-recycle-scroller__item-view.hoveror.hover .some-element-inside-the-item-view).
How does it work?
- The RecycleScroller creates pools of views to render visible items to the user.
- A view holds a rendered item, and is reused inside its pool.
- For each type of item, a new pool is created so that the same components (and DOM trees) are reused for the same type.
- Views can be deactivated if they go off-screen, and can be reused anytime for a newly visible item.
Here is what the internals of RecycleScroller look like in vertical mode:
<RecycleScroller>
<!-- Wrapper element with a pre-calculated total height -->
<wrapper
:style="{ height: computedTotalHeight + 'px' }"
>
<!-- Each view is translated to the computed position -->
<view
v-for="view of pool"
:style="{ transform: 'translateY(' + view.computedTop + 'px)' }"
>
<!-- Your elements will be rendered here -->
<slot
:item="view.item"
:index="view.nr.index"
:active="view.nr.used"
/>
</view>
</wrapper>
</RecycleScroller>When the user scrolls inside RecycleScroller, the views are mostly just moved around to fill the new visible space, and the default slot properties updated. That way we get the minimum amount of components/elements creation and destruction and we use the full power of Vue virtual-dom diff algorithm to optimize DOM operations!
Props
items: list of items you want to display in the scroller.direction(default:'vertical'): scrolling direction, either'vertical'or'horizontal'.itemSize(default:null): display height (or width in horizontal mode) of the items in pixels used to calculate the scroll size and position. If it is set tonull(the default value), it will use variable size mode.gridItems: display that many items on the same line to create a grid. You must put a value foritemSizeto use this prop (dynamic sizes are not supported).itemSecondarySize: size in pixels (width in vertical mode, height in horizontal mode) of the items in the grid whengridItemsis set. IfitemSecondarySizeis not set, it will use the value ofitemSize.minItemSize: minimum size used if the height (or width in horizontal mode) of a item is unknown.sizeField(default:'size'): field used to get the item's size in variable size mode.typeField(default:'type'): field used to differentiate different kinds of components in the list. For each distinct type, a pool of recycled items will be created.keyField(default:'id'): field used to identify items and optimize managing rendered views.pageMode(default:false): enable Page mode.prerender(default:0): render a fixed number of items for Server-Side Rendering (SSR).buffer(default:200): amount of pixel to add to edges of the scrolling visible area to start rendering items further away.emitUpdate(default:false): emit a'update'event each time the virtual scroller content is updated (can impact performance).updateInterval(default:0): the interval in ms at which the view will be checked for updates after scrolling. When set to0, checked will happen during the next animation frame.listClass(default:''): custom classes added to the item list wrapper.itemClass(default:''): custom classes added to each item.listTag(default:'div'): the element to render as the list's wrapper.itemTag(default:'div'): the element to render as the list item (the direct parent of the default slot content).disableTransform(default:false): disable the use of CSStransformfor positioning items. When enabled, items are positioned usingtop/leftinstead oftranslateX/translateY. Useful when transforms interfere with nested positioned elements.skipHover(default:false): disable the hover class management on item views. Use this to improve performance when hover effects are not needed.startAtBottom(default:false): start the scroller scrolled to the bottom. Useful for chat-like interfaces.initialScrollPercent(default:null): set an initial scroll position as a percentage (0 to 1). For example,0.5starts in the middle of the list.stickToBottom(default:false): automatically scroll to the bottom when new items are appended and the user is already at the bottom. Ideal for chat UIs.stickToBottomThreshold(default:50): how many pixels from the bottom the user can be and still be considered "at the bottom" forstickToBottom.skeletonWhileScrolling(default:false): render lightweight skeleton placeholders instead of full item content during active scrolling. Use theskeletonslot to customize the placeholder.filter: optional function(item) => booleanto filter items before rendering. Items where the function returnsfalseare excluded without recreating the scroller.customScrollbar(default:false): enable the built-in custom scrollbar. Hides the native scrollbar and renders a custom track/thumb overlay. Automatically disabled in page mode.scrollbarOptions: options object to customize the custom scrollbar appearance and behavior. Only used whencustomScrollbaristrue. See Custom Scrollbar for details.
Events
resize: emitted when the size of the scroller changes.visible: emitted when the scroller considers itself to be visible in the page.hidden: emitted when the scroller is hidden in the page.update: emitted each time the views are updated, only ifemitUpdateprop istrue. The handler receives a singleUpdateEventobject:interface UpdateEvent { startIndex: number endIndex: number visibleStartIndex: number visibleEndIndex: number } // Usage: function onUpdate(event: UpdateEvent) { console.log(event.startIndex, event.endIndex) console.log(event.visibleStartIndex, event.visibleEndIndex) }scroll-start: emitted when the first item is rendered.scroll-end: emitted when the last item is rendered.
Methods (via template ref)
scrollToItem(index, alignment?): scroll to the item at the given index.alignmentcan be'start','center','end', or'auto'(default).scrollToPosition(position): scroll to the given pixel position.scrollToBottom(): scroll to the bottom of the list.scrollToPercent(percent): scroll to a position expressed as a percentage (0 to 1). For example,scrollToPercent(0.5)scrolls to the middle.updateVisibleItems(): force an update of the visible items.isAtBottom: reactive ref --truewhen the scroller is scrolled to the bottom (withinstickToBottomThreshold).reset(): reset the scroller state (clears indices, stops scrolling, and recycles all views).
Default scoped slot props
item: item being rendered in a view.index: reflects each item's position in theitemsarrayactive: whether or not the view is active. An active view is considered visible and being positioned byRecycleScroller. An inactive view is not considered visible and is hidden from the user. Any rendering-related computations should be skipped if the view is inactive.
Other Slots
<main>
<slot name="before"></slot>
<wrapper>
<!-- Reused view pools here -->
<slot name="empty"></slot>
<slot name="empty-item" :index="index"></slot>
<slot name="skeleton" :item="item" :index="index"></slot>
</wrapper>
<slot name="after"></slot>
</main>before: content rendered before the item list wrapper.after: content rendered after the item list wrapper.empty: content rendered when the items list is empty.empty-item: skeleton placeholder rendered for items that arenullorundefinedin the items array. Receives an{ index }slot prop. Useful for showing loading placeholders:<RecycleScroller :items="items" :item-size="50"> <template #default="{ item }"> <div>{{ item.name }}</div> </template> <template #empty-item="{ index }"> <div class="skeleton-row">Loading item {{ index }}...</div> </template> </RecycleScroller>skeleton: lightweight placeholder shown in place of full item content during active scrolling whenskeletonWhileScrollingis enabled. Receives{ item, index }slot props.
Example:
<RecycleScroller
class="scroller"
:items="list"
:item-size="32"
>
<template #before>
Hey! I'm a message displayed before the items!
</template>
<template v-slot="{ item }">
<div class="user">
{{ item.name }}
</div>
</template>
</RecycleScroller>Page mode
The page mode expands the virtual-scroller and uses the page viewport to compute which items are visible. That way, you can use it in a big page with HTML elements before or after (like a header and a footer). Set the page-mode prop to true:
<header>
<menu></menu>
</header>
<RecycleScroller page-mode>
<!-- ... -->
</RecycleScroller>
<footer>
Copyright 2017 - Cat
</footer>Variable size mode
This mode can be performance heavy with a lot of items. Use with caution.
If the itemSize prop is not set or is set to null, the virtual scroller will switch to variable size mode. You then need to expose a number field on the item objects with the size of the item element.
You still need to set the size of the items with CSS correctly (with classes for example).
Use the sizeField prop (default is 'size') to set the field used by the scroller to get the size for each item.
Example:
const items = [
{
id: 1,
label: 'Title',
size: 64,
},
{
id: 2,
label: 'Foo',
size: 32,
},
{
id: 3,
label: 'Bar',
size: 32,
},
]Buffer
You can set the buffer prop (in pixels) on the virtual-scroller to extend the viewport considered when determining the visible items. For example, if you set a buffer of 1000 pixels, the virtual-scroller will start rendering items that are 1000 pixels below the bottom of the scroller visible area, and will keep the items that are 1000 pixels above the top of the visible area.
The default value is 200.
<RecycleScroller :buffer="200" />Server-Side Rendering
The prerender props can be set as the number of items to render on the server inside the virtual scroller:
<RecycleScroller
:items="items"
:item-size="42"
:prerender="10"
>Custom Scrollbar
Native browser scrollbars in virtual scrollers have a UX problem: the thumb jumps around as item sizes are measured and the total content height changes. The built-in custom scrollbar solves this by computing thumb position mathematically from the scroller's internal state rather than relying on the browser's scrollHeight.
Enable it with the customScrollbar prop:
<RecycleScroller
:items="items"
:item-size="50"
custom-scrollbar
>
<template #default="{ item }">
<div>{{ item.name }}</div>
</template>
</RecycleScroller>Customize appearance with scrollbarOptions:
<RecycleScroller
:items="items"
:item-size="50"
custom-scrollbar
:scrollbar-options="{
width: 14,
thumbColor: 'rgba(100, 100, 100, 0.6)',
trackColor: 'rgba(0, 0, 0, 0.05)',
thumbBorderRadius: '8px',
autoHideDelay: 2000,
}"
>How it works
- The container keeps
overflow: auto-- native scrolling (wheel, trackpad, touch, keyboard) still works - The native scrollbar is hidden with CSS (
scrollbar-width: none) - A custom track/thumb overlay is rendered on top, positioned absolutely
- Thumb size and position are computed from
totalSize,scrollTop, andclientHeight - The thumb can be dragged or the track clicked to scroll
- Auto-hides after inactivity (configurable)
ScrollbarOptions
| Option | Type | Default | Description |
|--------|------|---------|-------------|
| width | number | 12 | Track width in pixels |
| minThumbSize | number | 30 | Minimum thumb height in pixels |
| autoHide | boolean | true | Auto-hide scrollbar when not scrolling |
| autoHideDelay | number | 1000 | Milliseconds before auto-hiding |
| thumbColor | string | 'rgba(0, 0, 0, 0.4)' | CSS color for the thumb |
| trackColor | string | 'transparent' | CSS color for the track |
| thumbBorderRadius | string | '6px' | CSS border-radius for the thumb |
Notes
- Works with RecycleScroller, DynamicScroller, and GridScroller (they all wrap RecycleScroller)
- Automatically disabled in page mode since the scroll container is the parent/window
- Supports both vertical and horizontal scrolling
- The
useCustomScrollbarcomposable is also exported for advanced use cases
DynamicScroller
This works just like the RecycleScroller, but it can render items with unknown sizes!
Basic usage
<template>
<DynamicScroller
:items="items"
:min-item-size="54"
class="scroller"
>
<template v-slot="{ item, index, active }">
<DynamicScrollerItem
:item="item"
:active="active"
:size-dependencies="[
item.message,
]"
:data-index="index"
>
<div class="avatar">
<img
:src="item.avatar"
:key="item.avatar"
alt="avatar"
class="image"
>
</div>
<div class="text">{{ item.message }}</div>
</DynamicScrollerItem>
</template>
</DynamicScroller>
</template>
<script setup lang="ts">
interface Props {
items: Array<{ id: string; avatar: string; message: string }>
}
defineProps<Props>()
</script>
<style scoped>
.scroller {
height: 100%;
}
</style>Important notes
minItemSizeis required for the initial render of items.DynamicScrollerwon't detect size changes on its own, but you can put values that can affect the item size withsize-dependencieson DynamicScrollerItem.- You don't need to have a
sizefield on the items.
Props
Extends all the RecycleScroller props.
- It's not recommended to change
sizeFieldprop since all the size management is done internally.
Events
Extends all the RecycleScroller events.
Default scoped slot props
Extends all the RecycleScroller scoped slot props.
Other slots
Extends all the RecycleScroller other slots.
Methods (via template ref)
Extends all RecycleScroller methods, plus:
updateItemSize(key, size): manually set the cached size for an item.getItemSize(key): get the cached size for an item.removeItemSize(key): remove the cached size for an item.resetSizes(): clear all cached item sizes (also called byreset()).invalidateItem(key): imperatively remeasure a specific item by key. Waits fornextTickbefore measuring so DOM changes have time to settle.
DynamicScrollerItem
The component that should wrap all the items in a DynamicScroller.
Props
item(required): the item rendered in the scroller.active(required): is the holding view active in RecycleScroller. Will prevent unnecessary size recomputation.sizeDependencies: values that can affect the size of the item. This prop will be watched and if one value changes, the size will be recomputed. Recommended instead ofwatchData.watchData(default:false): deeply watchitemfor changes to re-calculate the size (not recommended, can impact performance).tag(default:'div'): element used to render the component.emitResize(default:false): emit theresizeevent each time the size is recomputed (can impact performance).minItemSize: minimum size hint for initial render before the actual size is measured.dataIndex: the item's index in the list. Used internally for size tracking.
Events
resize: emitted each time the size is recomputed, only ifemitResizeprop istrue.
Scoped slot props
DynamicScrollerItem exposes a scoped slot with a triggerResize function. Call it to imperatively remeasure the item (e.g. after an image loads or async content renders):
<DynamicScrollerItem :item="item" :active="active" :data-index="index">
<template #default="{ triggerResize }">
<img :src="item.avatar" @load="triggerResize" />
<div>{{ item.message }}</div>
</template>
</DynamicScrollerItem>triggerResize waits for nextTick before measuring so DOM changes have time to settle.
useDynamicScrollerItem composable
For deeply nested components inside a DynamicScrollerItem that need to trigger a remeasure without prop drilling, use the useDynamicScrollerItem composable:
<script setup lang="ts">
import { useDynamicScrollerItem } from 'vue-zscroller'
const { triggerResize } = useDynamicScrollerItem()
function onContentReady() {
triggerResize()
}
</script>Must be called within a component that is a descendant of DynamicScrollerItem. Falls back to a no-op with a warning if called outside one.
GridScroller
A responsive grid that wraps RecycleScroller. Give it itemWidth and itemHeight and it calculates how many columns fit, recalculating automatically when the container resizes.
Basic usage
<template>
<GridScroller
class="grid"
:items="items"
:item-width="200"
:item-height="200"
:gap="12"
key-field="id"
v-slot="{ item, column, row, cellWidth, cellHeight }"
>
<div class="card" :style="{ width: cellWidth + 'px', height: cellHeight + 'px' }">
{{ item.name }}
</div>
</GridScroller>
</template>
<style scoped>
.grid {
height: 100%;
}
</style>Props
items: list of items to display.itemWidth(required): base width of each item in pixels. Used to calculate column count.itemHeight(required): base height of each item in pixels.columns(default:null): force a specific column count. When set,itemWidthis only used for aspect ratio.minColumns(default:1): minimum number of columns.maxColumns(default:Infinity): maximum number of columns.gap(default:0): shorthand for bothrowGapandcolumnGap.rowGap(default:0): gap between rows in pixels. Overridesgapfor rows.columnGap(default:0): gap between columns in pixels. Overridesgapfor columns.keyField(default:'id'): field used to identify items.filter: optional function(item) => booleanto filter items before rendering.- Also accepts all RecycleScroller props:
direction,pageMode,prerender,buffer,emitUpdate,updateInterval,listClass,itemClass,listTag,itemTag,disableTransform,skipHover,startAtBottom,initialScrollPercent.
Events
columns-change: emitted when the number of columns changes. Receives the new column count.- Also emits all RecycleScroller events:
resize,visible,hidden,update,scroll-start,scroll-end.
Default scoped slot props
item: the item being rendered.index: index in the items array.active: whether the view is active.column: column index (0-based) of this item in the grid.row: row index (0-based) of this item in the grid.cellWidth: computed width of each cell in pixels.cellHeight: computed height of each cell in pixels.
Other slots
before: content before the grid.after: content after the grid.empty: content when items list is empty.loading: content shown while the container is being measured (before the grid renders).
Methods (via template ref)
scrollToItem(index): scroll to item at the given index.scrollToPosition(position): scroll to a pixel position.scrollToBottom(): scroll to the bottom.scrollToPercent(percent): scroll to a percentage position (0 to 1).scrollToCell(row, col): scroll to a specific grid cell.updateVisibleItems(): force update of visible items.
useIdState
The useIdState composable replaces the legacy IdState mixin for managing per-item local state in reused components inside a RecycleScroller.
Why is this useful?
Since the components in RecycleScroller are reused, you can't directly use Vue's reactive ref or reactive state per-item: otherwise the state will be shared with different items in the list!
useIdState provides a state object that is scoped to each item by its identifier. When the component is reused for a different item, the state automatically switches to that item's stored state.
Example
In this example, we use the id of the item to have a "scoped" state per item:
<template>
<div class="question">
<p>{{ item.question }}</p>
<button @click="idState.replyOpen = !idState.replyOpen">Reply</button>
<textarea
v-if="idState.replyOpen"
v-model="idState.replyText"
placeholder="Type your reply"
/>
</div>
</template>
<script setup lang="ts">
import { getCurrentInstance } from 'vue'
import { useIdState } from 'vue-zscroller'
interface Props {
item: { id: number; question: string }
}
const props = defineProps<Props>()
interface ReplyState {
replyOpen: boolean
replyText: string
}
const { idState, setup } = useIdState<ReplyState>({
idProp: vm => vm.item.id,
initialState: () => ({
replyOpen: false,
replyText: '',
}),
})
setup(getCurrentInstance()?.proxy)
</script>Options
idProp(default:vm => vm.item.id): field name on the component instance (for example:'id') or function returning the id from the component proxy.initialState: factory function returning the default state for a new item.
Migrating from vue-virtual-scroller
Breaking changes
- Package name: Import from
vue-zscrollerinstead ofvue-virtual-scroller - @vueuse/core is now a peer dependency
- IdState mixin removed: Use the
useIdStatecomposable instead (see useIdState) updateevent signature changed: Emits a singleUpdateEventobject instead of 4 positional arguments
What stays the same
All component props, events, and slots are the same. Drop-in replacement aside from the import path and the two breaking changes above.
New stuff
empty-itemslot for skeleton/loading placeholders- New props:
disableTransform,skipHover,startAtBottom,initialScrollPercent,customScrollbar,scrollbarOptions - New methods:
scrollToPercent(percent),reset() useVirtualScrollPerformancecomposable for scroll metrics
Code Examples
Before (vue-virtual-scroller):
<template>
<RecycleScroller
ref="scroller"
:items="items"
:item-size="50"
key-field="id"
>
<template #default="{ item }">
<div>{{ item.name }}</div>
</template>
</RecycleScroller>
</template>
<script>
export default {
data() {
return {
items: []
}
}
}
</script>After (vue-zscroller with TypeScript):
<template>
<RecycleScroller
ref="scrollerRef"
:items="items"
:item-size="50"
key-field="id"
>
<template #default="{ item }">
<div>{{ item.name }}</div>
</template>
</RecycleScroller>
</template>
<script setup lang="ts">
import { ref } from 'vue'
import { RecycleScroller } from 'vue-zscroller'
interface Item {
id: string
name: string
}
const items = ref<Item[]>([])
const scrollerRef = ref<InstanceType<typeof RecycleScroller>>()
</script>Demo Examples
The live demo includes several examples you can reference:
- RecycleScroller -- basic virtual list with fixed-size items
- DynamicScroller -- variable-height items that get measured on render
- Chat -- chat UI using DynamicScroller with
stickToBottom, streaming messages, and thefilterprop for live search - Test Chat -- simpler chat example for testing
stickToBottombehavior - Horizontal -- horizontal scrolling mode
- Grid -- grid layout using RecycleScroller's
gridItemsprop - GridScroller -- responsive grid using the GridScroller component
- Skeleton -- progressive loading with skeleton placeholders
- Simple List -- minimal RecycleScroller setup
Source code for all demos is in packages/demo/src/components/.
Links
- Demo: zachhandley.github.io/vue-zscroller
- GitHub: github.com/ZachHandley/vue-zscroller
- npm: npmjs.com/package/vue-zscroller
- Original project: github.com/Akryum/vue-virtual-scroller
