felixmosh / virtual-scroller

A component for efficiently rendering large lists of variable height items

Home Page:https://catamphetamine.github.io/virtual-scroller/?dynamic=%E2%9C%93

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

VirtualScroller

An open-source implementation of Twitter's VirtualScroller component: a component for efficiently rendering large lists of variable height items. Automatically measures items as they're rendered and supports items of variable height. Also includes a React component for those who're using React.

Demo

DOM (no frameworks):

React:

Rationale

Rendering really long lists in HTML can be performance intensive which sometimes leads to slow page load times and wasting mobile users' battery. For example, consider a chat app rendering a list of a thousand of the most recent messages: when using React the full render cycle can take up to 100 milliseconds or more on a modern PC. If the chat message component is complex enough (rich text formatting, pictures, videos, attachments, buttons) then it could take up to a second or more (on a modern PC). Now imagine users viewing the website on their aged low-tier smartphones and it quickly results in annoying user experience resulting in them closing the website and the website losing its user base.

In 2017 Twitter completely redesigned their website with responsiveness and performance in mind using the latest performance-boosting techniques available at that time. They wrote an article about it where they briefly mentioned this:

Twitter website responsiveness before using the VirtualScroller technique

On slower devices, we noticed that it could take a long time for our main navigation bar to appear to respond to taps, often leading us to tap multiple times, thinking that perhaps the first tap didn’t register. It turns out that mounting and unmounting large trees of components (like timelines of Tweets) is very expensive in React. Over time, we developed a new infinite scrolling component called VirtualScroller. With this new component, we know exactly what slice of Tweets are being rendered into a timeline at any given time, avoiding the need to make expensive calculations as to where we are visually.

However, Twitter didn't share the code for their VirtualScroller component (unlike Facebook, Twitter doesn't share much of their code). This library is an attempt to create an open-source implementation of such VirtualScroller component for anyone to use in their projects.

There's also an "RFC" for a native VirtualScroller component where they try to formulate what is a VirtualScroller component and how it should behave.

How it works

It works by measuring each list item's height as it's being rendered and then as the user scrolls it hides the items which are no longer visible and shows the now-visible items as they're scrolled to. The hidden items on top are compensated by setting padding-top on the container, and the hidden items on the bottom are compensated by setting padding-bottom on the container. The component listens to scroll / resize events and re-renders the currently visible items as the user scrolls (or if the browser window is resized).

Go to the demo page, open Developer Tools ("Elements" tab), find <div id="root"/> element, expand it, see <div id="messages"/> element, expand it and observe the changes to it while scrolling the page.

List items can also have inter-item spacing via margin-top / margin-bottom or border-top / border-bottom, see the Gotchas section for more details on how to do that properly.

Install

npm install virtual-scroller --save

If you're not using a bundler then use a standalone version from a CDN.

Use

The default export is the VirtualScroller utility class. It implements the core logic of a VirtualScroller component and can be used for building a VirtualScroller component for any UI framework. virtual-scroller/dom and virtual-scroller/react are both built upon it so this VirtualScroller utility class is a low-level core and is meant to be used by UI framework library authors and not by the end users — the end users should use high-level components like virtual-scroller/dom (for pure JS) and virtual-scroller/react (for React).

import VirtualScroller from 'virtual-scroller'

new VirtualScroller(
  getContainer,
  items,
  options
)
  • getContainer() function should return the "container" DOM Element where the list items will be rendered to.
  • items is the list of items (is only used to get items count).
  • options is an optional argument.

Available options:

  • estimatedItemHeight: number — Is used for the initial render of the list: determines how many list items are rendered initially to cover the screen height plus some extra vertical margin for future scrolling. If not set then the list first renders just the first item, measures it, and then assumes it to be the estimatedItemHeight from which it calculates how many items to show on the second render pass to fill the screen height plus some extra vertical margin for future scrolling. Therefore, this setting is only for the initial render minor optimization and is not required.
  • state: object — The initial state for VirtualScroller. Can be used, for example, to quicky restore the list on "Back" navigation.
  • customState: object — (advanced) The initial "custom" state for VirtualScroller. It can be used to initialize the "custom" part of VirtualScroller state in cases when VirtualScroller state is used to store some "custom" list state.
  • getState(): object — Returns VirtualScroller state. Is used for React VirtualScroller component implementation.
  • setState(newState: object) — Stores VirtualScroller state (including setting the initial state). The state must reflect what's currently rendered on screen. Is used for React VirtualScroller component implementation.
  • onStateChange(newState: object, prevState: object?) — Is called whenever VirtualScroller state is updated (including setting the initial state) if getState() and setState() properties aren't defined.
  • onItemFirstRender(i: number) — Is called for each item the first time it's rendered. Is guaranteed to be called at least one time for each item rendered, though could also be called multiple times. For an example of using such feature, consider a list of items that must be somehow preprocessed (parsed, enhanced, etc) before being rendered, and such preprocessing puts some load on the CPU (and therefore takes some time). In such case, instead of preprocessing the whole list of items upfront, a developer could preprocess the items as they're being rendered thereby eliminating any associated lag/freezing and draining less battery.
  • preserveScrollPositionAtBottomOnMount: boolean — (optional) (advanced) Set to true to preserve scroll position of the bottom of the list when it's first mounted. A possible use case: consider a forum thread only showing unread posts by default. If a user navigates to such thread, it could show "No new posts" message with a "Show previous" button above it. When the user clicks the "Show previous" button, a <VirtualScroller/> is mounted with a list of posts. If preserveScrollPositionAtBottomOnMount: true is set then, after the list of posts is show, page scroll will be restored so that the bottom of the list remain on screen so that the user could scroll up from the most recent posts to the oldest ones.
  • measureItemsBatchSize: number — (optional) (advanced) (experimental) Imagine a situation when a user doesn't gradually scroll through a huge list but instead hits an End key to scroll right to the end of such huge list: this will result in the whole list rendering at once (because an item needs to know the height of all previous items in order to render at correct scroll position) which could be CPU-intensive in some cases (for example, when using React due to its slow performance when initially rendering components on a page). To prevent freezing the UI in the process, a measureItemsBatchSize could be configured, that would limit the maximum count of items that're being rendered in a single pass for measuring their height: if measureItemsBatchSize is configured, then such items will be rendered and measured in batches. By default it's set to 100. This is an experimental feature and could be removed in future non-major versions of this library. For example, the future React 17 will come with Fiber rendering engine that is said to resolve such freezing issues internally. In that case, introducing this option may be reconsidered.
  • shouldUpdateLayoutOnWindowResize(event: Event): boolean — (optional) By default, VirtualScroller always performs a re-layout on window resize event. The resize event is not only triggered when a user resizes the window itself: it's also triggered when the user switches into (and out of) fullscreen mode. By default, VirtualScroller performs a re-layout on all window resize events, except for ones that don't result in actual window width or height change, and except for cases when, for example, a video somewhere in a list is maximized into fullscreen. There still can be other "custom" cases: for example, when an application uses a custom "slideshow" component (rendered outside of the list DOM element) that goes into fullscreen when a user clicks a picture or a video in the list. For such "custom" cases shouldUpdateLayoutOnWindowResize(event) option / property can be specified.

VirtualScroller class instance provides methods:

  • onMount() — Should be called when the VirtualScroller component is "mounted" (rendered) on a page.
  • onUpdate() — Is only used when getState() and setState() are supplied: should be called after setState() updates the page. Is used for React VirtualScroller component implementation.
  • onUnmount() — Should be called when the VirtualScroller component is "unmounted" (removed) from the page.
  • getState(): object — Returns VirtualScroller state. Is used for React VirtualScroller component implementation.
  • onItemHeightChange(i: number) — Should be called whenever a list item's height changes: triggers a re-layout of VirtualScroller. This allows VirtualScroller to re-measure the new item's height and re-render correctly. Calling onItemHeightChange() manually could be replaced with detecting item height changes automatically via Resize Observer. For example, when a user clicks an "Expand"/"Collapse" button in a post. Calling onItemHeightChange() is only required when an item shrinks in height. For example, consider a post with an "Expand"/"Collapse" button: when such post is expanded the next posts in the feed might not be visible yet but if a user clicks the "Collapse" button the post is collapsed and the next posts become visible but they're not yet rendered because VirtualScroller didn't render them previously due to them being invisible. Calling onItemHeightChange(i) in such case would make VirtualScroller re-measure the collapsed post height and perform a re-layout.
  • onItemStateChange(i: number, itemState: object) — Can be used to update a list item's state inside VirtualScroller state. For example, storing list item's state inside VirtualScroller state is used in React VirtualScroller component to preserve the state of list items that are unmounted due to being no longer visible: when they're visible again they're re-mounted and their state isn't lost. Calling onItemStateChange() doesn't trigger a re-layout of VirtualScroller because changing a list item's state doesn't necessarily mean a change of its height, so a re-layout isn't necessarily required. If a re-layout is required then call onItemHeightChange(i) manually. For example, consider a social network feed, each post optionally having an attachment. Suppose there's a post in the feed having a YouTube video attachment. The attachment is initially shown as a thumbnail that expands into an embedded YouTube player on click. If a user expands the video, then scrolls down so that the post with the video is unmounted, then scrolls back up so that the post with the video is re-mounted again, then the video should stay expanded (or maybe not, but you get the idea).
  • setItems(newItems: any[], options: object?) — Updates VirtualScroller items. For example, can be used to prepend or append new items to the list. See Dynamically Loaded Lists section for more details. Available options: preserveScrollPositionOnPrependItems: boolean — Set to true to enable "restore scroll position after prepending items" feature (could be useful when implementing "Show previous items" button).
  • getItemCoordinates(i: number): object — Returns coordinates of item with index i relative to the document: top is the top offset of the item relative to the start of the document, bottom is the top offset of the item's bottom edge relative to the start of the document, height is the item's height.
  • updateLayout() — (advanced) Triggers a re-layout of VirtualScroller. It's what's called every time on page scroll or window resize. You most likely won't ever need to call this method manually. Still, it can be called manually when the list's top position changes not as a result of scrolling the page or resizing the window. For example, if some DOM elements above the list are removed (like a closeable "info" panel) or collapsed (like an "accordion" panel) then the list's top position changes which means that now some of the previoulsy shown items might go off screen and the user might be seeing a blank area where items haven't been rendered yet because they were off-screen during the previous VirtualScroller layout. VirtualScroller automatically performs a layout only on page scroll or window resize; in all other cases, when layout needs to be re-run then call it manually via this instance method.

VirtualScroller state provides properties:

  • firstShownItemIndex: number — The index of the first item to render.
  • lastShownItemIndex: number — The index of the last item to render.
  • beforeItemsHeight: number — The padding-top which should be applied to the "container" element.
  • afterItemsHeight: number — The padding-bottom which should be applied to the "container" element.
  • items: any[] — The list of items (can be updated via .setItems()).
  • itemStates: object?[] — The list of item states.
  • itemHeights: number?[] — A list of measured item heights. If an item's height hasn't been measured yet then it's height is undefined.
  • itemSpacing: number? — Inter-item spacing. If it hasn't been measured yet then it's undefined.

DOM

This is an example of using virtual-scroller/dom component. It's the source code of the DOM demo.

import VirtualScroller from 'virtual-scroller/dom'

const messages = [
  {
    username: ...,
    date: ...,
    text: ...
  },
  ...
]

function renderMessage(message) {
  // Message element.
  const root = document.createElement('article')
  // Message author.
  const author = document.createElement('a')
  author.setAttribute('href', `/users/${message.username}`)
  author.textContent = `@${message.username}`
  root.appendChild(author)
  // Message date.
  const time = document.createElement('time')
  time.setAttribute('datetime', message.date.toISOString())
  time.textContent = message.date.toString()
  root.appendChild(time)
  // Message text.
  const text = document.createElement('p')
  text.textContent = message.text
  root.appendChild(text)
  // Return message element.
  return root
}

const virtualScroller = new VirtualScroller(
  document.getElementById('messages'),
  messages,
  renderMessage
)

// For "Single Page Apps":
// router.onPageUnload(virtualScroller.onUnmount)

VirtualScroller constructor takes arguments:

  • container — Items list container DOM Element.
  • items — The items list.
  • renderItem(item) — Renders an item as a DOM Element.
  • optionsVirtualScroller options.

VirtualScroller instance provides methods:

  • setItems(items, options) — A proxy for the corresponding VirtualScroller method.
  • onItemHeightChange(i) — A proxy for the corresponding VirtualScroller method.
  • onItemStateChange(i, itemState) — A proxy for the corresponding VirtualScroller method.
  • getItemCoordinates(i) — A proxy for the corresponding VirtualScroller method.

Additional options:

  • onMount — Is called after <VirtualScroller/> component has been mounted and before VirtualScroller.onMount() is called. Can be used in advanced cases: for example, to restore page scroll Y position for the corresponding VirtualScroller state on "Back" navigation.

React

This is an example of using the React virtual-scroller/react component. It's the source code of the React demo.

import React from 'react'
import PropTypes from 'prop-types'
import VirtualScroller from 'virtual-scroller/react'

function Messages({ messages }) {
  return (
    <VirtualScroller
      items={messages}
      itemComponent={Message}
  )
}

const message = PropTypes.shape({
  username: PropTypes.string.isRequired,
  date: PropTypes.instanceOf(Date).isRequired,
  text: PropTypes.string.isRequired
})

Messages.propTypes = {
  messages: PropTypes.arrayOf(message).isRequired
}

function Message({ children: message }) {
  const {
    username,
    date,
    text
  } = message
  return (
    <article>
      <a href={`/users/${username}`}>
        @{username}
      </a>
      <time dateTime={date.toISOString()}>
        {date.toString()}
      </time>
      <p>
        {text}
      </p>
    </article>
  )
}

Message.propTypes = {
  children: message.isRequired
}

<VirtualScroller/> component receives properties:

  • items — The items list.
  • itemComponent — List item component. Receives the list item as the children property. Can optionally receive state and onSaveState() properties for saving list item state before unmounting it and then restoring that state after the item is re-mounted (for example, this supports "Show more" buttons, "Expand YouTube video" buttons, etc). For best performance, make sure it's a React.PureComponent or a React.memo(), otherwise it'll be re-rendering as the user scrolls.
  • itemComponentProps: object — (optional) The props passed to itemComponent.
  • estimatedItemHeight: number — (optional) The estimatedItemHeight option of VirtualScroller class.
  • preserveScrollPositionOnPrependItems: boolean — (optional) The preserveScrollPositionOnPrependItems option of VirtualScroller.setItems() method.
  • preserveScrollPositionAtBottomOnMount: boolean — (optional) The preserveScrollPositionAtBottomOnMount option of VirtualScroller.
  • measureItemsBatchSize: number — (optional) The measureItemsBatchSize option of VirtualScroller.
  • onMount() — (optional) Is called after <VirtualScroller/> component has been mounted and before VirtualScroller.onMount() is called. Can be used in advanced cases: for example, to restore page scroll Y position for the corresponding VirtualScroller state on "Back" navigation.
  • onItemFirstRender(i) — (optional) The onItemFirstRender option of VirtualScroller class.
  • initialState: object — (optional) The initial state for VirtualScroller: the state option of VirtualScroller. For example, can be used to quicky restore the list on "Back" navigation.
  • initialCustomState: object — (advanced) (optional) The initial "custom" state for VirtualScroller: the customState option of VirtualScroller. It can be used to initialize the "custom" part of VirtualScroller state in cases when VirtualScroller state is used to store some "custom" list state.
  • onStateChange(newState: object, prevState: object) — (optional) Can be called when VirtualScroller state is updated (including setting the initial state). For example, can be used to keep VirtualScroller state copy in an instance variable and later in componentWillUnmount() persist it somewhere in global application state for quickly restoring it later on "Back" navigation:
import {
  getVirtualScrollerState,
  setVirtualScrollerState
} from './globalState'

class Example extends React.Component {
  componentWillUnmount() {
    saveVirtualScrollerState(this.virtualScrollerState)
  }
  render() {
    return (
      <VirtualScroller
        items={...}
        itemComponent={...}
        state={hasUserNavigatedBack ? getVirtualScrollerState() : undefined}
        onStateChange={state => this.virtualScrollerState = state}/>
    )
  }
}

itemComponent receives properties:

  • children — The item.
  • state — Item's state. Can be used for storing and then restoring the state of components which are about to be shown again after being previously hidden. For example, consider a post with an "Expand YouTube Video" button. If a user clicks the button the post is expanded and its height changes. Then the user scrolls down until the post is no longer visible and so it's no longer rendered. Then the user scrolls back up and the post is visible again and so it's rendered again. If the "Expand YouTube Video" state wasn't preserved then the list items would "jump" for the expanded minus non-expanded height difference. To avoid that, some { videoExpanded: true } state could be passed to the post component which would make it auto-expand the video on mount and so the list items wouldn't "jump". For that, onStateChange({ videoExpanded: true }) would have to be called when the user clicks the "Expand YouTube Video" button.
  • onStateChange(itemState) — A proxy for VirtualScroller.onItemStateChange(i, itemState). Call this function to update item's state.
  • onHeightChange() — A proxy for VirtualScroller.onItemHeightChange(i). Call this function to re-measure the item if it shrinks in height.

<VirtualScroller/> component instance provides methods:

  • renderItem(i) — Calls .forceUpdate() on the itemComponent instance for item with index i. Does nothing if the item isn't rendered. Is only supported for components that are React.Components.
  • getItemCoordinates(i) — A proxy for the corresponding VirtualScroller method.
  • updateLayout() — A proxy for the corresponding VirtualScroller method.

Dynamically Loaded Lists

The previous examples showcase a static items list. For cases when new items are loaded when the user clicks "Show previous" / "Show next" buttons virtualScroller.setItems(newItems) method can be used where newItems will be previousItems.concat(items) for "Show previous" button and items.concat(nextItems) for "Show next" button. virtual-scroller/react will automatically call .setItems(newItems) when new items property is passed, and virtual-scroller/dom provides a manual .setItems(newItems) method same as VirtualScroller.

Also, one can use on-scroll-to library to render a "Load more items on scroll down" component for "infinite scroll" lists.

Gotchas

Images

VirtualScroller measures item heights as soon as they've rendered and later uses those measurements to determine which items should be rendered when the user scrolls. This means that things like <img/>s require special handling to prevent them from changing their size. For example, when rendering a simple <img src="..."/> first it renders an element with zero width and height and only after the image file header has been parsed does it resize itself to the actual image's width and height. When used inside VirtualScroller items such images would result in scroll position "jumping" as the user scrolls. To avoid that, any <img/>s rendered inside VirtualScroller items must either have their width and height set explicitly or have their aspect ratio set explicitly by making them position: absolute and wrapping them in a parent <div/> having position: relative and padding-bottom: ${100/aspectRatio}%.

Margin collapse

If any vertical margin is set on the list items then this may lead to the list items jumping by the value of that margin when scrolling. The reason is that when the first list item is rendered then there's no padding-top on the containing <div/> so the first item's margin "collapses" with outer margins, but when the first item is no longer rendered the containing <div/> has padding-top to compensate for that which prevents list item margin from "collapsing" with outer margins. To fix that, don't set any margin-top on the first item of the list and don't set any margin-bottom on the last item of the list. An example of fixing margin for the first and the last items of the list:

/* This margin is supposed to "collapse" with the outer ones
   but requires a fix below to work correctly with `VirtualScroller`. */
.list-item {
  margin: 10px;
}
/* Fixes margin "collapse" for the first item. */
.list-item:first-child {
  margin-top: 0;
}
/* Fixes margin "collapse" for the last item. */
.list-item:last-child {
  margin-top: 0;
}

Styling :first-child and :last-child

When styling the first and the last items of the list via :first-child and :last-child one should also check that such styles don't change the item's height which means that one should not add any border or padding styles to :first-child and :last-child otherwise the list items will jump by that extra height while scrolling.

An example of a :first-child/:last-child style that will not work correctly with VirtualScroller:

.list-item {
  border-bottom: 1px solid black;
}
.list-item:first-child {
  border-top: 1px solid black;
}

Search, focus management.

Due to offscreen list items not being rendered native browser features like "Find on page", moving focus through items via Tab key, screen reader announcement and such won't work. A workaround for "search on page" is adding a custom "🔍 Search" input field that would filter items by their content and then call VirtualScroller.setItems().

Only the first item is rendered on page load.

VirtualScroller calculates the shown item indexes when its .onMount() method is called, but if the page styles are applied after VirtualScroller is mounted (for example, if styles are applied via javascript, like Webpack does it in dev mode with its style-loader) then the list might not render correctly and will only show the first item. The reason for that is because calling .getBoundingClientRect() on the list container DOM element on mount returns "incorrect" top position because the styles haven't been applied yet, and so VirtualScroller thinks it's offscreen.

For example, consider a page:

<div class="page">
  <nav class="sidebar">...</nav>
  <main>...</main>
</div>

The sidebar is styled as position: fixed, but until the page styles have been applied it's gonna be a regular <div/> meaning that <main/> will be rendered below the sidebar causing it to be offscreen and so the list will only render the first item. Then, the page styles are loaded and applied and the sidebar is now position: fixed so <main/> is now rendered at the top of the page but VirtualScroller's .onMount() has already been called and it won't re-render until the user scrolls or the window is resized.

This type of a bug doesn't occur in production, but it can appear in development mode when using Webpack. The workaround VirtualScroller implements for such cases is calling .getBoundingClientRect() on the list container DOM element periodically (every second) to check if the top coordinate has changed as a result of CSS being applied: if it has then it recalculates the shown item indexes and re-renders.

Debug

Set window.VirtualScrollerDebug to true to output debug messages to console.

CDN

One can use any npm CDN service, e.g. unpkg.com or jsdelivr.net

<!-- Core. -->
<script src="https://unpkg.com/virtual-scroller@1.x/bundle/virtual-scroller.js"></script>
<script>
  new VirtualScroller(...)
</script>

<!-- DOM component. -->
<script src="https://unpkg.com/virtual-scroller@1.x/bundle/virtual-scroller-dom.js"></script>
<script>
  new VirtualScroller(...)
</script>

<!-- React component. -->
<script src="https://unpkg.com/virtual-scroller@1.x/bundle/virtual-scroller-react.js"></script>
<script>
  <VirtualScroller .../>
</script>

License

MIT

About

A component for efficiently rendering large lists of variable height items

https://catamphetamine.github.io/virtual-scroller/?dynamic=%E2%9C%93

License:MIT License


Languages

Language:JavaScript 100.0%