npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

vue-virtual-scroller-classic

v1.1.5

Published

A Vue 3 implementation of the original VirtualScroller from vue-virtual-scroller

Downloads

1,476

Readme

vue-virtual-scroller-classic

This is a vue 3 implementation of the original VirtualScroller from version v0.12.2 of vue-virtual-scroller.

This removes the RecycleScroller and uses the original approach, which means you're safe to use local component instance data inside the scroller.

The end result is slower rendering, but less implementation specific code, and no need to use idState

The use case this is for is where virtualisation is required to speed up initial page renders on lists where each component entry has significant local state. But a little bit of loading time is acceptable when components are mounted/unmounted as the list is scrolled.

For vue 2 feel free to use the original v0.12.2, in this package all vue 2 -> 3 deprecations are migrated.

Table of contents

Installation

npm install --save vue-virtual-scroller-classic
import { createApp } from 'vue'
import { VirtualScroller } from 'vue-virtual-scroller-classic'
import { ObserveVisibility } from 'vue-observe-visibility';

const app = createApp({})

// latest release of dependency vue-observe-visibility uses a vue 2 API, this
// ensures compatibility until next release
app.directive('observe-visibility', {
  beforeMount: (el, binding, vnode) => {
    vnode.context = binding.instance;
    ObserveVisibility.bind(el, binding, vnode);
  },
  update: ObserveVisibility.update,
  unmounted: ObserveVisibility.unbind,
});

app.component('VirtualScroller', VirtualScroller)

⚠️ A css file is included when importing the package:

import 'vue-virtual-scroller-classic/dist/vue-virtual-scroller-classic.css'

Browser

<link rel="stylesheet" href="vue-virtual-scroller-classic/dist/vue-virtual-scroller-classic.css"/>

<script src="vue.js"></script>
<script src="vue-virtual-scroller-classic/dist/vue-virtual-scroller-classic.min.js"></script>

Usage

The virtual scroller has three main props:

  • items is the list of items you want to display in the scroller. There can be several types of item.
  • itemHeight is the display height of the items in pixels used to calculate the scroll height and position. If it set to null (default value), it will use variable height mode.
  • renderers is a map of component definitions objects or names for each item type (more details). If you don't define renderers, the scroller will use scoped slots (see below).

⚠️ You need to set the size of the virtual-scroller element and the items elements (for example, with CSS). Unless you are using variable height mode, all items should have the same height to prevent display glitches.

It is strongly recommended to use functional components inside virtual-scroller since those are cheap to create and dispose.

The browsers have a height limitation on DOM elements, it means that currently the virtual scroller can't display more than ~500k items depending on the browser.

Renderers

The optional renderers prop is an object containing a component definition for each possible value of the item type. If you don't set this prop, scoped slots will be used instead. The component definition must have an item prop, that will get the item object to render in the scroller. It will also receive an index prop.

There are additional props you can use:

  • typeField to customize which field is used on the items to get their type and use the corresponding definition in the renderers map. The default is 'type'.
  • keyField to customize which field is used on the items to set their key special attribute (see the documentation). The default is 'id'.

For better performance, you should use the keyField prop that will set the key attribute. Warning! You shouldn't expect items to have the key set at all times, since the scroller may disable them depending on the situation.

Scoped slots

Alternatively, you can use scoped slots instead of renderers. This is active when you don't define the renderers prop on the virtual scroller.

The scope will contain the row's item in the item attribute, so you can write scope="props" and then use props.item. It will also have an index attribute.

Here is an example:

<virtual-scroller class="scroller" :items="items" item-height="42" content-tag="table">
  <template slot-scope="props">
    <tr v-if="props.item.type === 'letter'" class="letter" :key="props.itemKey">
      <td>
        {{props.item.value}} Scoped
      </td>
    </tr>

    <tr v-if="props.item.type === 'person'" class="person" :key="props.itemKey">
      <td>
        {{props.item.value.name}}
      </td>
    </tr>
  </template>
</virtual-scroller>

For better performance, you should set the key attribute on direct children using the itemKey field from the scoped slot and set the keyField prop on the virtual scroller.

Page mode

The page mode expand the virtual-scroller and use 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). Just set the page-mode props to true:

<header>
  <menu></menu>
</header>

<virtual-scroller page-mode></virtual-scroller>

<footer>
  Copyright 2017 - Cat
</footer>

Variable height mode

⚠️ This mode can be performance heavy with a lot of items. Use with caution.

If the itemHeight prop is not set or set to null, the virtual scroller will switch to Variable height mode. You then need to expose a number field on the item objects with the height of the item element.

⚠️ You still need to set the height of the items with CSS correctly (with classes for example).

Use the heightField prop (default is 'height') to set the field used by the scroller to get the height for each item.

Example:

const items = [
  {
    id: 1,
    label: 'Title',
    height: 64,
  },
  {
    id: 2,
    label: 'Foo',
    height: 32,
  },
  {
    id: 3,
    label: 'Bar',
    height: 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.

<virtual-scroller buffer="200" />

Pool Size

The poolSize prop (in pixels) is the size in pixels of the viewport pool. The computed 'visible' area can be computed step by step using this pool. This allows creating multiple row at once each in a while. For example, if you set a pool size of 2000 pixels, the rows will be grouped in pools of 2000 pixels height. When the user scrolls too far, the new batch of 2000px height is created, and so on. That way, the DOM isn't updated for each row, but in batches instead.

The default value is 2000.

<virtual-scroller pool-size="2000" />

Update event

Set the emitUpdate boolean prop to true so that the virtual-scroller will emit an update event when the rendered items list is updated. The arguments are startIndex and endIndex.

The default value is false.

<virtual-scroller emit-update @update="(startIndex, endIndex) => ..." />

Customizing the tags

These are optional props you can use to change the DOM tags used in the virtual scroller:

  • mainTag to change the DOM tag of the component root element. The default is 'div'.
  • containerTag to change the DOM tag of the element simulating the height. The default is 'div'.
  • contentTag to change the DOM tag of the element containing the items. The default is 'div'. For example, you can change this to 'table'.

The component template is structured like this:

<main>
  <container>
    <content>
      <!-- Your items here -->
    </content>
  </container>
</main>

If you set contentTag to 'table', the actual result in the DOM will look like the following:

<div>
  <div>
    <table>
      <!-- Your items here -->
    </table>
  </div>
</div>

Customizing the classes

You can use the following props to customize the container and content elements CSS classes:

  • containerClass
  • contentClass

Slots

There are 4 slots you can use to inject things inside the scroller (it may be usefull to add a thead or tbody):

<main>
  <slot name="before-container"></slot>
  <container>
    <slot name="before-content"></slot>
    <content>
      <!-- Your items here -->
    </content>
    <slot name="after-content"></slot>
  </container>
  <slot name="after-container"></slot>
</main>

Server-Side Rendering

The prerender props can be set as the number of items to render on the server inside the virtual scroller:

<virtual-scroller :items="items" item-height="42" page-mode prerender="10">

Example

<template>
  <div class="demo">
    <virtual-scroller
      class="scroller"
      :items="items"
      :renderers="renderers"
      item-height="22"
      type-field="type">
    </virtual-scroller>
  </div>
</template>

<script>
// Data with a type field
const items = [
  { type: 'letter', value: 'A' },
  { type: 'person', value: { name: 'Alan' } },
  { type: 'person', value: { name: 'Alice' } },
]

import Letter from './Letter.vue'
import Item from './Item.vue'

// Bind the components to the item type
const renderers = Object.freeze({
  letter: Letter,
  person: Item,
})

export default {
  data: () => ({
    items,
    renderers,
  }),
}
</script>

<style>
.scroller {
  height: 100%;
}

.scroller .item {
  height: 22px;
}
</style>

Letter.vue source:

<template>
  <div class="letter">({{item.index}}) {{item.value}}</div>
</template>

<script>
export default {
  props: ['item'],
}
</script>

Item.vue source:

<template>
  <div class="person" @click="edit">({{item.index}}) {{item.value.name}}</div>
</template>

<script>
export default {
  props: ['item'],
  methods: {
    edit () {
      this.item.value.name += '*'
    },
  },
}
</script>

License

MIT