The pictures in the waterfall layout have a core characteristic — equal width and varying height. The waterfall layout has been used on a certain scale in websites at home and abroad, such as Pinterest and pepetal. The next step is to explore the waterfall flow based on this feature.

To ensure real-time and accurate content, you can follow personal blogs

Basic Function realization

First we define a container with 20 images,

<body>
  <style>
    #waterfall {
      position: relative;
    }
    .waterfall-box {
      float: left;
      width: 200px;
    }
  </style>
</body>
<div id="waterfall">
    <img src="images/1.png" class="waterfall-box">
    <img src="images/2.png" class="waterfall-box">
    <img src="images/3.png" class="waterfall-box">
    <img src="images/4.png" class="waterfall-box">
    <img src="images/5.png" class="waterfall-box">
    <img src="images/6.png" class="waterfall-box">.</div>
Copy the code

Due to unknown CSS knowledge, the girl with the longest stockings took up all the space below…

Then the body, if there are 5 columns in each row as shown above, should the sixth picture appear below which of the first 5? Of course, it is absolutely positioned below the smallest height of the first 5 pictures.

What about picture 7? At this time, the sixth picture and the picture above it are regarded as a whole, and the idea is consistent with the above. The code implementation is as follows:

Waterfall.prototype.init = function () {... const perNum =this.getPerNum() // Get the number of images per row
  const perList = []              // Store the height of each image in the first column
  for (let i = 0; i < perNum; i++) {
    perList.push(imgList[i].offsetHeight)
  }

  let pointer = this.getMinPointer(perList) // Find the current minimum height of the array index

  for (let i = perNum; i < imgList.length; i++) {
    imgList[i].style.position = 'absolute' // Core statement
    imgList[i].style.left = `${imgList[pointer].offsetLeft}px`
    imgList[i].style.top = `${perList[pointer]}px`

    perList[pointer] = perList[pointer] + imgList[i].offsetHeight // The smallest value of the array plus the height of the corresponding image
    pointer = this.getMinPointer(perList)
  }
}
Copy the code

If you are careful, you may have noticed that the property offsetHeight is used in the code to obtain the height of the image. The sum of the height of this property is equal to the height of the image + the inner margin + the border. That is why we use padding instead of margin to set the distance between images. In addition to the offsetHeight attribute, in addition to understand the difference between offsetHeight, clientHeight, offsetTop, scrollTop and other attributes, can better understand this project. The CSS code is simple as follows:

.waterfall-box {
  float: left;
  width: 200px;
  padding-left: 10px;
  padding-bottom: 10px;
}
Copy the code

So far the basic layout of the waterfall flow has been completed, and the renderings are as follows:

Scroll, resize event listening implementation

After the init function is implemented, the next step is to listen for scroll events so that when rolled to the bottom of the parent node a stream of images are loaded. So one of the things to think about is, where do I scroll to when I trigger the load function? This varies from person to person. My method is to trigger the loading function when the condition of parent container height + scrolling distance > offsetTop of the last image is met, that is, orange line + purple line > blue line, and the code is as follows:

window.onscroll = function() {
  // ...
  if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) {// Browser height + scroll distance > offsetTop of the last image
    const fragment = document.createDocumentFragment()
    for(let i = 0; i < 20; i++) {
      const img = document.createElement('img')
      img.setAttribute('src'.`images/${i+1}.png`)
      img.setAttribute('class'.'waterfall-box')
      fragment.appendChild(img)
    }
    $waterfall.appendChild(fragment)
  }
}
Copy the code

Since the parent node may define a custom node, it provides a wrapper for listening scroll function as follows:

  proto.bind = function () {
    const bindScrollElem = document.getElementById(this.opts.scrollElem)
    util.addEventListener(bindScrollElem || window.'scroll', scroll.bind(this))}const util = {
    addEventListener: function (elem, evName, func) {
      elem.addEventListener(evName, func, false)}},Copy the code

The resize event listener is similar to the Scroll event listener. When the resize function is triggered, just call the init function to reset.

Listener bindings are implemented using publish-subscribe patterns and inheritance

Since the development of plug-ins as the goal, can not only be satisfied with the realization of the function, but also set aside the corresponding operation space for developers to deal with. Considering that in the business scenario, the pull-down images loaded in waterfall flow are generally obtained asynchronously from Ajax, so the loaded data must not be written to the database, and it is expected to achieve the following calls (here referring to the use of waterfall),

const waterfall = new Waterfall({options})

waterfall.on("load".function () {
  // Ajax synchronously/asynchronously adds images here
})
Copy the code

Looking at the invocation, it’s not hard to imagine implementing it using the publish/subscribe model, which was described earlier in the Node.js Asynchronous Profile. The core idea is to add functions to the cache through subscription functions, and then implement asynchronous calls through publishing functions. The code implementation is as follows:

function eventEmitter() {
  this.sub = {}
}

eventEmitter.prototype.on = function (eventName, func) { // Subscribe function
  if (!this.sub[eventName]) {
    this.sub[eventName] = []
  }
  this.sub[eventName].push(func) // Add event listeners
}

eventEmitter.prototype.emit = function (eventName) { // Publish functions
  const argsList = Array.prototype.slice.call(arguments.1)
  for (let i = 0, length = this.sub[eventName].length; i < length; i++) {
    this.sub[eventName][i].apply(this, argsList) // Call the event listener}}Copy the code

Then, to enable Waterfall to use publish/subscribe, simply let Waterfall inherit the eventEmitter function, which looks like this:

function Waterfall(options = {}) {
  eventEmitter.call(this)
  this.init(options) // This is attached when new
}

Waterfall.prototype = Object.create(eventEmitter.prototype)
Waterfall.prototype.constructor = Waterfall
Copy the code

The inheritance approach incorporates the advantages of both constructor-based inheritance and archetype-based inheritance, as well as the separation of subclasses and superclasses using Object.create. The more detailed aspects of inheritance can be covered in another article, which ends here.

Small optimization

In order to prevent the scroll event from triggering multiple loading images, we can consider using function anti-shake and throttling. On the basis of publish-subscribe mode, an isLoading parameter is defined to indicate whether it is being loaded or not, and the loading is determined according to its Boolean value. The code is as follows:

let isLoading = false
const scroll = function () {
  if (isLoading) return false // Avoid triggering events more than once
  if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) { // Browser height + scroll distance > offsetTop of the last image
    isLoading = true
    this.emit('load')
  }
}

proto.done = function () {
  this.on('done'.function () {
    isLoading = false. })this.emit('done')}Copy the code

In this case, you need to add waterfall.done to the location of the call to indicate that the current image has been loaded, the code is as follows:

const waterfall = new Waterfall({})
waterfall.on("load".function () {
  // Load images asynchronously/synchronously
  waterfall.done()
})
Copy the code

The project address

The project address

Use of this plugin in the React project

The project is simple, shortcomings are unavoidable, welcome to leave your valuable opinions.