Heim  >  Artikel  >  Web-Frontend  >  So implementieren Sie das Wasserfall-Flow-Plug-in mit JS

So implementieren Sie das Wasserfall-Flow-Plug-in mit JS

亚连
亚连Original
2018-06-07 14:28:281286Durchsuche

Dieser Artikel enthält eine detaillierte Analyse eines nativen JS-Wasserfall-Plug-Ins und Code-bezogene Erklärungen. Leser, die sich dafür interessieren, können darauf zurückgreifen.

Die Bilder im Wasserfall-Flow-Layout haben ein Kernmerkmal – gleiche Breite und variable Höhe. Das Wasserfall-Flow-Layout wird in gewissem Umfang auf inländischen Websites wie Pinterest, Petals.com usw. verwendet. Basierend auf dieser Funktion beginnen wir dann mit der Erkundungsreise des Wasserfallflusses.

Grundlegende Funktionsimplementierung

Zuerst definieren wir einen Container mit 20 Bildern

<body>
 <style>
  #waterfall {
   position: relative;
  }
  .waterfall-box {
   float: left;
   width: 200px;
  }
 </style>
</body>
<p 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">
  ...
 </p>
由于未知的 css 知识点,丝袜最长的妹子把下面的空间都占用掉了。。。
接着正文,假如如上图,每排有 5 列,那第 6 张图片应该出现前 5 张图片哪张的下面呢?当然是绝对定位到前 5 张图片高度最小的图片下方。
那第 7 张图片呢?这时候把第 6 张图片和在它上面的图片当作是一个整体后,思路和上述是一致的。代码实现如下:
Waterfall.prototype.init = function () {
 ...
 const perNum = this.getPerNum() // 获取每排图片数
 const perList = []       // 存储第一列的各图片的高度
 for (let i = 0; i < perNum; i++) {
  perList.push(imgList[i].offsetHeight)
 }
 let pointer = this.getMinPointer(perList) // 求出当前最小高度的数组下标
 for (let i = perNum; i < imgList.length; i++) {
  imgList[i].style.position = &#39;absolute&#39; // 核心语句
  imgList[i].style.left = `${imgList[pointer].offsetLeft}px`
  imgList[i].style.top = `${perList[pointer]}px`

  perList[pointer] = perList[pointer] + imgList[i].offsetHeight // 数组最小的值加上相应图片的高度
  pointer = this.getMinPointer(perList)
 }
}

Aufmerksame Freunde haben möglicherweise den Code zum Erhalten von Bildern entdeckt verwendet das Attribut offsetHeight und die Summe der Höhen dieses Attributs ist gleich 图片高度 + 内边距 + 边框. Aus diesem Grund verwenden wir Polsterung anstelle von Rand, um den Abstand zwischen Bildern festzulegen. Zusätzlich zum Attribut offsetHeight müssen Sie auch die Unterschiede zwischen offsetHeight, clientHeight, offsetTop, scrollTop und anderen Attributen verstehen, um dieses Projekt besser zu verstehen. Der CSS-Code ist so einfach wie:

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

Implementierung der Scroll- und Größenänderungsereignisüberwachung

Nach der Implementierung der Initialisierungsfunktion init besteht der nächste Schritt darin, die Überwachung des zu implementieren Scroll-Ereignis: Dadurch wird erreicht, dass beim Scrollen zum Ende des übergeordneten Knotens ein stetiger Strom von Bildern geladen wird. Ein zu diesem Zeitpunkt zu berücksichtigender Punkt ist: An welcher Position wird die Ladefunktion beim Scrollen ausgelöst? Das ist von Person zu Person unterschiedlich. Mein Ansatz besteht darin, die Ladefunktion auszulösen, wenn die Bedingung 父容器高度 + 滚动距离 > 最后一张图片的 offsetTop erfüllt ist, also orange Linie + violette Linie > Da der übergeordnete Knoten möglicherweise ein angepasster Knoten ist, bietet er eine Kapselung der Bildlaufüberwachungsfunktion. Der Code lautet wie folgt:

window.onscroll = function() {
 // ...
 if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) {// 浏览器高度 + 滚动距离 > 最后一张图片的 offsetTop
  const fragment = document.createDocumentFragment()
  for(let i = 0; i < 20; i++) {
   const img = document.createElement(&#39;img&#39;)
   img.setAttribute(&#39;src&#39;, `images/${i+1}.png`)
   img.setAttribute(&#39;class&#39;, &#39;waterfall-box&#39;)
   fragment.appendChild(img)
  }
  $waterfall.appendChild(fragment)
 }
}

Die Größenänderungsereignisüberwachung ähnelt der Bildlaufereignisüberwachung. Rufen Sie zum Zurücksetzen einfach die Init-Funktion auf.

Verwenden Sie das Publish-Subscribe-Modell und die Vererbung, um die Listening-Bindung zu implementieren

Da das Ziel darin besteht, Plug-Ins zu entwickeln, können wir uns nicht nur mit der Realisierung von Funktionen zufrieden geben, sondern auch den entsprechenden Betrieb belassen Raum, mit dem sich Entwickler befassen können. Erinnern wir uns an das Dropdown-Laden von Bildern in Wasserfallflüssen in Geschäftsszenarien: Sie werden im Allgemeinen asynchron über Ajax abgerufen, sodass die geladenen Daten nicht in der Bibliothek fest codiert werden müssen. Es wird erwartet, dass die folgenden Aufrufe implementiert werden können (hier). Wir ziehen Lehren aus der Verwendung von Wasserfall). im asynchronen asynchronen Datensatz von Node.j. Die Kernidee besteht darin, die Funktion über die Abonnementfunktion zum Cache hinzuzufügen und dann den asynchronen Aufruf über die Veröffentlichungsfunktion zu implementieren. Die Codeimplementierung ist unten angegeben:

proto.bind = function () {
  const bindScrollElem = document.getElementById(this.opts.scrollElem)
  util.addEventListener(bindScrollElem || window, &#39;scroll&#39;, scroll.bind(this))
 }
 const util = {
  addEventListener: function (elem, evName, func) {
   elem.addEventListener(evName, func, false)
  },
 }

Als nächstes aktivieren Sie Waterfall, um die Veröffentlichung zu verwenden /subscribe-Modus, lassen Sie Waterfall einfach die eventEmitter-Funktion erben. Die Code-Implementierung lautet wie folgt:

const waterfall = new Waterfall({options})
waterfall.on("load", function () {
 // 此处进行 ajax 同步/异步添加图片
})

Die Vererbungsmethode nutzt die Vorteile der beiden Schreibmethoden basierend auf der Konstruktorvererbung und der Prototypkettenvererbung und verwendet

Um die Unterklasse und die übergeordnete Klasse zu isolieren, können Sie einen weiteren Artikel für weitere Details zur Vererbung schreiben, daher höre ich hier auf.

Kleine OptimierungObject.create

Um zu verhindern, dass das Scroll-Ereignis das mehrfache Laden von Bildern auslöst, können Sie die Funktion Anti-Shake und Throttling verwenden. Basierend auf dem Publish-Subscribe-Modell wird ein isLoading-Parameter definiert, der angibt, ob er geladen wird. Der Code lautet wie folgt:

function eventEmitter() {
 this.sub = {}
}
eventEmitter.prototype.on = function (eventName, func) { // 订阅函数
 if (!this.sub[eventName]) {
  this.sub[eventName] = []
 }
 this.sub[eventName].push(func) // 添加事件监听器
}
eventEmitter.prototype.emit = function (eventName) { // 发布函数
 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) // 调用事件监听器
 }
}
Zu diesem Zeitpunkt Ich muss

hinzufügen, um zu informieren, dass das aktuelle Bild geladen wurde. Der Code lautet wie folgt:

function Waterfall(options = {}) {
 eventEmitter.call(this)
 this.init(options) // 这个 this 是 new 的时候,绑上去的
}
Waterfall.prototype = Object.create(eventEmitter.prototype)
Waterfall.prototype.constructor = Waterfall

Das Obige habe ich für alle zusammengestellt. Ich hoffe, es wird in Zukunft für alle hilfreich sein .

waterfall.doneVerwandte Artikel:

Lebenszyklus von Vue-Komponenten und Route (ausführliches Tutorial)

Verwenden Sie SpringMVC, um domänenübergreifende Vue-Anfragen zu lösen

Neue Funktionen der Webpack 4.0.0-Beta.0-Version (ausführliches Tutorial)

Das obige ist der detaillierte Inhalt vonSo implementieren Sie das Wasserfall-Flow-Plug-in mit JS. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn