Fast, Flexible Virtual Scrolling With Functional Programming

Rate this content
Bookmark

Virtual scrolling is a clever way to reduce rendering overhead, and is especially helpful as the complexity of websites increases. Although there are many plug-and-play libraries which support virtual scrolling in various frameworks, rolling your own is surprisingly easy. In addition, this allows one to maximally optimize each viewport for its content, unlocking additional performance savings.

In this talk, you will learn how to create a flexible virtual scrolling viewport from scratch, and make it performant using concepts from functional programming. We will be following an implementation of a virtual scrolling viewport using web components from start to finish, including variable element heights, skeletons, asynchronous child component rendering. We will then optimize it using memoization, wrapping more-performant imperative constructs in a functional layer, and using element recycling to significantly reduce render churn when using stateless child components.

Finally, we will touch on the state of the art in virtual DOM rendering, and how this approach compares to the use of the new `content-visibility` CSS property.

This talk has been presented at JSNation US 2024, check out the latest edition of this JavaScript Conference.

FAQ

Using a buffer by rendering extra elements before and after the viewport helps absorb latency during fast scrolling, reducing flickering and improving user experience.

CSS content visibility allows elements outside the viewport to be skipped in rendering while still being part of the DOM and accessibility tree, enhancing searchability and accessibility without affecting performance.

The limitation is that un-rendered elements don't appear in browser searches or accessibility tools like screen readers, requiring alternative solutions such as implementing a search bar or using new web standards like CSS content visibility.

Virtual scrolling is beneficial for large lists because it improves initial page load time and scrolling performance by rendering only the elements currently in view, rather than all elements at once.

To implement virtual scrolling, you need to compute the total height of rendered elements, determine which elements should be rendered based on the scroll position, and calculate the alignment of these elements within the viewport.

Virtual scrolling is a technique where only the visible elements in a list are rendered, improving performance by reducing the number of items loaded into the DOM at any given time.

You can use a pool of reusable elements to avoid creating new elements, instead shifting the contents of these elements as needed, which reduces the need for constant DOM updates.

A downside is that all elements must still be constructed in the DOM initially, which can be slow, making it less suitable for lists with many small items compared to virtual scrolling.

Optimizations include memoizing intermediate values, using binary search for index calculations, delaying the rendering of non-essential parts, and reusing a pool of list elements to avoid unnecessary DOM manipulation.

For lists with variable element heights, you compute the cumulative sum of item heights and use it to find the start and end indices for rendering, rather than using basic division.

Adam Niederer
Adam Niederer
16 min
21 Nov, 2024

Comments

Sign in or register to post your comment.
Video Summary and Transcription
Today's Talk focuses on implementing virtual scrolling to improve performance and flexibility of lists. The process involves computing the height of the viewport, scroll position, and individual list item height. By rendering only the elements within the viewport, the initial page load and scrolling performance are enhanced. Various optimization techniques are discussed, including memoization, binary search, and delayed rendering. These techniques significantly improve scrolling performance and provide a better user experience. Additional techniques such as using skeletons, element pooling, and functional purity can further optimize rendering. Advanced techniques, like moving elements within the pool and using content visibility, yield substantial performance improvements. However, content visibility is better suited for large pages with few large sections. Overall, the Talk offers valuable insights into virtual scrolling and its limitations.

1. Introduction to Virtual Scrolling

Short description:

Today we're going to implement a basic virtually scrolled list from scratch, then talk about the improvements we can make to boost the flexibility and usability of the list, and then make the list as fast as it can possibly be. Virtual scrolling helps improve both initial page load and scrolling performance. To compute how tall the elements are, which elements to render, and where to render them, we need the height of the viewport, the scroll position, and the height of an individual item in the list. Let's code it by computing the total height of the elements and figuring out which elements to render based on the viewport and scroll position.

♪♪ Hey, I'm Adam, and welcome to my talk about virtual scrolling. Today we're going to implement a basic virtually scrolled list from scratch, then talk about the improvements we can make to boost the flexibility and usability of the list, and then make the list as fast as it can possibly be. This talk will help you understand how virtual scrolling works and might be able to help you get a few more frames per second out of it, even if you're using a library. So let's get started.

For every scrolled viewport, virtual or not, there are two key parts. An outer viewport element that's one height, and an inner element or list of elements that's a larger height. If it were shorter, there'd be no need to scroll, of course. Traditional scrolled lists render all of their items into the DOM, even those which aren't visible, which can cause lag both when rendering that list initially and while scrolling. Virtual scrolling, on the other hand, only renders the elements that are being viewed at the moment. This helps us improve both initial page load and scrolling performance. Usually when your list size hits triple digits, it's time to start thinking about virtual scrolling, depending on how long it takes to render your individual items. So let's take a look at how to virtually scroll a list.

First, let's make a super basic minimum viable product. For this, I'm going to assume all of the list items will be the same height, but we'll come back to this a little bit later. To virtually scroll something, we actually only need to compute three things. One, how tall is the stuff that we're rendering? Two, which elements in the list should be rendered? And three, how far down inside the inner element should we render it so that it aligns with the viewport? To compute these three things, we're going to need three pieces of data from the DOM. First, the height of the viewport. Self-explanatory. Next, we need the scroll position, which is the distance from the top of the viewport to the top of the inner element. We can get this from the viewport's scroll-top property. Finally, we'll need the height of an individual item in the list, inclusive of borders and margins. They're all the same size in this example, so that's easy enough. Now let's code it.

So the first thing we need to compute is the total height of the stuff we're rendering, which we set as the height of the inner element. The browser can't do it for you like in traditional scrolling because we're not rendering all of the elements at once. So for our basic list, we can set it to the height of an element times the number of elements. Next, we need to figure out which elements we're rendering. For this, we need the height of the viewport and the scroll position within the viewport. From that, the index of the first element we want to render is the floor of the scroll position divided by the height of each element. Similarly, the last is the ceiling of the scroll position plus the height divided by the height of each element.

2. Implementing Virtual Scrolling

Short description:

We only render the elements from our start to end indices and move the list items to the part of the inner element shown in the viewport. To support variable element heights, we compute the cumulative sum of list item heights and find the start and end indices based on the scroll position and viewport height.

So now our code will look a bit like this. We only render the elements from our start to end indices, and to react to scrolling, we can simply use some reactive state. The viewport height can be done with a ref, which I'm not showing here.

Of course, we don't want to render all of these elements at the very beginning of the inner element all of the time. We now need to move the list items down to the part of the inner element that's being shown in the viewport. Thankfully, that's easy. Translate it by the start index times the list items height. Done. That's all you need to have a functional virtually scrolled list. The code fits on a slide.

There are some points we can improve to make the list more flexible to work with, though, and improve the user's experience when scrolling the list. The biggest limitation is that all of the elements have to be of the same height. So let's work around that. To support variable element heights, the main thing we need to change is our method of finding the first and last elements to render. Instead of doing basic division, we first need to compute the cumulative sum of our list items heights. Then to compute our start index, we take our scroll position and find the index of the greatest number that is less than our scroll position in our cumulatively summed heights. The index of that number is our start index. For the end index, we want to do something similar. We want to take our scroll position plus our viewport height, then find the smallest number that is greater than that sum in our cumulatively summed heights. The index of that number is our end index.

Check out more articles and videos

We constantly think of articles and videos that might spark Git people interest / skill us up or help building a stellar career

A Guide to React Rendering Behavior
React Advanced 2022React Advanced 2022
25 min
A Guide to React Rendering Behavior
Top Content
This transcription provides a brief guide to React rendering behavior. It explains the process of rendering, comparing new and old elements, and the importance of pure rendering without side effects. It also covers topics such as batching and double rendering, optimizing rendering and using context and Redux in React. Overall, it offers valuable insights for developers looking to understand and optimize React rendering.
Speeding Up Your React App With Less JavaScript
React Summit 2023React Summit 2023
32 min
Speeding Up Your React App With Less JavaScript
Top Content
Watch video: Speeding Up Your React App With Less JavaScript
Mishko, the creator of Angular and AngularJS, discusses the challenges of website performance and JavaScript hydration. He explains the differences between client-side and server-side rendering and introduces Quik as a solution for efficient component hydration. Mishko demonstrates examples of state management and intercommunication using Quik. He highlights the performance benefits of using Quik with React and emphasizes the importance of reducing JavaScript size for better performance. Finally, he mentions the use of QUIC in both MPA and SPA applications for improved startup performance.
Routing in React 18 and Beyond
React Summit 2022React Summit 2022
20 min
Routing in React 18 and Beyond
Top Content
Routing in React 18 brings a native app-like user experience and allows applications to transition between different environments. React Router and Next.js have different approaches to routing, with React Router using component-based routing and Next.js using file system-based routing. React server components provide the primitives to address the disadvantages of multipage applications while maintaining the same user experience. Improving navigation and routing in React involves including loading UI, pre-rendering parts of the screen, and using server components for more performant experiences. Next.js and Remix are moving towards a converging solution by combining component-based routing with file system routing.
Design Systems: Walking the Line Between Flexibility and Consistency
React Advanced 2021React Advanced 2021
47 min
Design Systems: Walking the Line Between Flexibility and Consistency
Top Content
The Talk discusses the balance between flexibility and consistency in design systems. It explores the API design of the ActionList component and the customization options it offers. The use of component-based APIs and composability is emphasized for flexibility and customization. The Talk also touches on the ActionMenu component and the concept of building for people. The Q&A session covers topics such as component inclusion in design systems, API complexity, and the decision between creating a custom design system or using a component library.
React Concurrency, Explained
React Summit 2023React Summit 2023
23 min
React Concurrency, Explained
Top Content
Watch video: React Concurrency, Explained
React 18's concurrent rendering, specifically the useTransition hook, optimizes app performance by allowing non-urgent updates to be processed without freezing the UI. However, there are drawbacks such as longer processing time for non-urgent updates and increased CPU usage. The useTransition hook works similarly to throttling or bouncing, making it useful for addressing performance issues caused by multiple small components. Libraries like React Query may require the use of alternative APIs to handle urgent and non-urgent updates effectively.
The Future of Performance Tooling
JSNation 2022JSNation 2022
21 min
The Future of Performance Tooling
Top Content
Today's Talk discusses the future of performance tooling, focusing on user-centric, actionable, and contextual approaches. The introduction highlights Adi Osmani's expertise in performance tools and his passion for DevTools features. The Talk explores the integration of user flows into DevTools and Lighthouse, enabling performance measurement and optimization. It also showcases the import/export feature for user flows and the collaboration potential with Lighthouse. The Talk further delves into the use of flows with other tools like web page test and Cypress, offering cross-browser testing capabilities. The actionable aspect emphasizes the importance of metrics like Interaction to Next Paint and Total Blocking Time, as well as the improvements in Lighthouse and performance debugging tools. Lastly, the Talk emphasizes the iterative nature of performance improvement and the user-centric, actionable, and contextual future of performance tooling.

Workshops on related topic

React Performance Debugging Masterclass
React Summit 2023React Summit 2023
170 min
React Performance Debugging Masterclass
Top Content
Featured WorkshopFree
Ivan Akulov
Ivan Akulov
Ivan’s first attempts at performance debugging were chaotic. He would see a slow interaction, try a random optimization, see that it didn't help, and keep trying other optimizations until he found the right one (or gave up).
Back then, Ivan didn’t know how to use performance devtools well. He would do a recording in Chrome DevTools or React Profiler, poke around it, try clicking random things, and then close it in frustration a few minutes later. Now, Ivan knows exactly where and what to look for. And in this workshop, Ivan will teach you that too.
Here’s how this is going to work. We’ll take a slow app → debug it (using tools like Chrome DevTools, React Profiler, and why-did-you-render) → pinpoint the bottleneck → and then repeat, several times more. We won’t talk about the solutions (in 90% of the cases, it’s just the ol’ regular useMemo() or memo()). But we’ll talk about everything that comes before – and learn how to analyze any React performance problem, step by step.
(Note: This workshop is best suited for engineers who are already familiar with how useMemo() and memo() work – but want to get better at using the performance tools around React. Also, we’ll be covering interaction performance, not load speed, so you won’t hear a word about Lighthouse 🤐)
Master JavaScript Patterns
JSNation 2024JSNation 2024
145 min
Master JavaScript Patterns
Top Content
Featured Workshop
Adrian Hajdin
Adrian Hajdin
During this workshop, participants will review the essential JavaScript patterns that every developer should know. Through hands-on exercises, real-world examples, and interactive discussions, attendees will deepen their understanding of best practices for organizing code, solving common challenges, and designing scalable architectures. By the end of the workshop, participants will gain newfound confidence in their ability to write high-quality JavaScript code that stands the test of time.
Points Covered:
1. Introduction to JavaScript Patterns2. Foundational Patterns3. Object Creation Patterns4. Behavioral Patterns5. Architectural Patterns6. Hands-On Exercises and Case Studies
How It Will Help Developers:
- Gain a deep understanding of JavaScript patterns and their applications in real-world scenarios- Learn best practices for organizing code, solving common challenges, and designing scalable architectures- Enhance problem-solving skills and code readability- Improve collaboration and communication within development teams- Accelerate career growth and opportunities for advancement in the software industry
React Patterns Made Simple
React Day Berlin 2024React Day Berlin 2024
62 min
React Patterns Made Simple
Featured Workshop
Adrian Hajdin
Adrian Hajdin
Learn widely used React patterns, including HOCs, Compound Components, Provider Patterns, Functions as Child, and Portals, to write cleaner, more efficient code and create scalable, maintainable applications.Overview In this workshop, viewers will learn about key React patterns that can make their code more efficient, readable, and maintainable. We'll introduce each pattern, explain how it works, and demonstrate practical examples. By the end of the session, participants will have a solid understanding of how to use these patterns in their projects.Learning GoalsHOCs Compound Components Provider Patterns Functions as Child Portals Modularity Maintainability Real-world Application.
Building WebApps That Light Up the Internet with QwikCity
JSNation 2023JSNation 2023
170 min
Building WebApps That Light Up the Internet with QwikCity
Featured WorkshopFree
Miško Hevery
Miško Hevery
Building instant-on web applications at scale have been elusive. Real-world sites need tracking, analytics, and complex user interfaces and interactions. We always start with the best intentions but end up with a less-than-ideal site.
QwikCity is a new meta-framework that allows you to build large-scale applications with constant startup-up performance. We will look at how to build a QwikCity application and what makes it unique. The workshop will show you how to set up a QwikCitp project. How routing works with layout. The demo application will fetch data and present it to the user in an editable form. And finally, how one can use authentication. All of the basic parts for any large-scale applications.
Along the way, we will also look at what makes Qwik unique, and how resumability enables constant startup performance no matter the application complexity.
Next.js 13: Data Fetching Strategies
React Day Berlin 2022React Day Berlin 2022
53 min
Next.js 13: Data Fetching Strategies
Top Content
WorkshopFree
Alice De Mauro
Alice De Mauro
- Introduction- Prerequisites for the workshop- Fetching strategies: fundamentals- Fetching strategies – hands-on: fetch API, cache (static VS dynamic), revalidate, suspense (parallel data fetching)- Test your build and serve it on Vercel- Future: Server components VS Client components- Workshop easter egg (unrelated to the topic, calling out accessibility)- Wrapping up
React Performance Debugging
React Advanced 2023React Advanced 2023
148 min
React Performance Debugging
Workshop
Ivan Akulov
Ivan Akulov
Ivan’s first attempts at performance debugging were chaotic. He would see a slow interaction, try a random optimization, see that it didn't help, and keep trying other optimizations until he found the right one (or gave up).
Back then, Ivan didn’t know how to use performance devtools well. He would do a recording in Chrome DevTools or React Profiler, poke around it, try clicking random things, and then close it in frustration a few minutes later. Now, Ivan knows exactly where and what to look for. And in this workshop, Ivan will teach you that too.
Here’s how this is going to work. We’ll take a slow app → debug it (using tools like Chrome DevTools, React Profiler, and why-did-you-render) → pinpoint the bottleneck → and then repeat, several times more. We won’t talk about the solutions (in 90% of the cases, it’s just the ol’ regular useMemo() or memo()). But we’ll talk about everything that comes before – and learn how to analyze any React performance problem, step by step.
(Note: This workshop is best suited for engineers who are already familiar with how useMemo() and memo() work – but want to get better at using the performance tools around React. Also, we’ll be covering interaction performance, not load speed, so you won’t hear a word about Lighthouse 🤐)