preface

Virtual lists, as the name suggests, are not real lists, which generally exist only in the good imagination of human beings…

Off topic. Start again.

Today in the amount of data, most of the time a selector may be a lot of data to be loaded from the server, but the browser on a one-time to render the data in page will appear when the caton or even collapse, so virtualization technology arises at the historic moment, for a better user experience and better use of experience.

That’s a quote from Element-Plus, and I think it makes sense, so I’ll use it a little bit. The appearance of virtual list is to solve the problem that the data content in the list is too long and too much, which leads to slow loading and fast scrolling.

start

First, let’s take a look at an example where we dynamically add 20,000 list items to a Container with a fixed height of 24, and we find that the page loads with significant lag.

<! DOCTYPE html><html lang="en">
<head>
  <meta charset="UTF-8">
  <title>Title</title>
  <style>
    #container {
        width: 240px;
        height: 300px;
        margin-top: 24px;
        overflow-y: auto;
    }
  </style>
</head>
<body>
<div>list-container: </div>
<div id="container"></div>
<script>
  const NUMBER = 20000, HEIGHT = 24;
  const container = document.getElementById('container');
  let str = ` `;
  for (let i = 0; i < NUMBER; i++) {
    let s = ` `;
    s += ` <div style="height: 24px; display: flex; align-items: center; box-sizing: border-box justify-content: space-between; font-size: 14px; border-bottom: 1px solid #ddd;" > <div>list</div> <div>${i}</div>
      </div>
    `
    str += s;
  }
  container.innerHTML = str;
</script>
</body>
</html>
Copy the code

A simple list takes 0.8 seconds, and sometimes there is a brief blank space when you drag the slider quickly.

To solve

For a better customer experience, we can use virtual lists to solve this problem.

The principle is very simple. When we listen for scroll events, we calculate the current list to be displayed to the user by the height of the window and the distance of the scroll bar, and add several before and after as a buffer.

The list is then moved to the current displayed position by positioning or translate offset, depending on the scrolling distance. In the customer’s perception, it looks like a normal list, but in reality, we’ve only rendered these ten or twenty list items all the time.

<! DOCTYPE html><html lang="en">
<head>
  <meta charset="UTF-8">
  <title>Title</title>
  <style>
      #container {
          width: 240px;
          height: 300px;
          margin-top: 24px;
          overflow-y: auto;
      }
      #list-container {
          position: relative;
      }
      #scroll-div {
          width: 100%;
          position: absolute;
          top: 0;
          left: 0;
      }
  </style>
</head>
<body>
<div>list-container: </div>
<div id="container">
  <div id="list-container">
    <div id="scroll-div"></div>
  </div>
</div>
<script>
  const NUMBER = 20000, HEIGHT = 24;
  const container = document.getElementById('container');
  const scrollDiv = document.getElementById('scroll-div');
  const listContainer = document.getElementById('list-container');
  listContainer.style.height = NUMBER * HEIGHT + 'px';
  const height = container.offsetHeight;
  // Take the number over the height
  const num = Math.ceil(height / 24);

  // The callback function for scrolling
  const scrollCallback = function(e) {
    const h = e ? e.target.scrollTop : 0;
    console.log('h', h);
    // Find the number of rows in the display list, then add num, 5 rows up and 5 rows down to get the list to render
    const n = Math.ceil(h / HEIGHT);
    const start = n > 5 ? n - 5 : 0;
    const end = n + (num - 1) > NUMBER - 5 ? NUMBER : n + (num - 1) + 5;
    let str = ` `;
    for (let i = start; i < end; i++) {
      let s = ` `;
      s += ` <div style="height: 24px; display: flex; align-items: center; box-sizing: border-box; justify-content: space-between; font-size: 14px; border-bottom: 1px solid #ddd;" > <div>list</div> <div>${i}</div>
        </div>
      `;
      str += s;
    }
    const top = start * HEIGHT;
    scrollDiv.style.top = top + 'px';
    scrollDiv.innerHTML = str;
  }
  / / initialization
  scrollCallback();
  // Listen for scroll events
  container.onscroll = scrollCallback;
</script>
</body>
</html>
Copy the code

conclusion

Just a simple implementation of the virtual list, because the height is fixed, very limited. Because this is a preliminary study, wide waiting for yourself, for the next time in the exploration, can find a good solution to the adaptive height.

Due to my recent obsession with Hearthstone and the continued “persistent” exploration and learning of Flutter, I finally scored 5500. As a result, this article has been written too little and too shallow. I feel very guilty and urge myself. In the future, I will try to ensure that I write at least one article a week.