調研:
1、蘑菇街/堆糖/花瓣:absolute
2、手淘:flex,但不算瀑布流,每個card高度一樣
一、使用flex佈局(最妥當)
核心:
1、div分兩列
2、遍歷數據分兩組
3、每次渲染完,記錄height,判斷左右列height高度,再決定遍歷的時每個數據往哪丟
<div>
<div class="left-col">
<ul>...</ul>
</div>
<div class="right-col">
<ul>...</ul>
</div>
</div>
......
divideData () {
const dataScroll = JSON.parse(JSON.stringify(this.scrollData))
let data1 = []
let data2 = []
let leftHeight = 0
let rightHeight = 0
if (dataScroll && dataScroll.length > 0) {
dataScroll.map((item, index) => {
if (leftHeight == rightHeight || leftHeight < rightHeight) {
leftHeight += item.picHeight
data1.push({
renderIndex: index + 1,
...item
})
} else {
rightHeight += item.picHeight
data2.push({
renderIndex: index + 1,
...item
})
}
})
}
this.data1 = JSON.parse(JSON.stringify(data1))
this.data2 = JSON.parse(JSON.stringify(data2))
},
二、使用column(最先進——純css)
1、核心
multi-column實現瀑布流主要依賴以下幾個屬性:
column-count: 設置共有幾列
column-width: 設置每列寬度,列數由總寬度與每列寬度計算得出
column-gap: 設置列與列之間的間距
2、避免斷層
1)表現:
最後一個元素的文本內容被自動斷開,一部分在當前列尾,一部分在下一列的列頭
2)原因:
multi-column佈局會將其內的元素自動進行流動和平衡,儘可能保證每列的高度趨於相同,所以會將其內的文本階段分佈在兩列內
3)解決:
給每個item設置break-inside: avoid;
ul {
columns: 2; //列數
column-gap: 6px; //列兼具
column-fill: aoto; //默認的填充方式
padding: 0;
width: 100%;
font-size: 0;
li {
width: 100%;
height: 100px;
break-inside: avoid; //避免斷層
}
}
利:
1、簡單快速
弊:
1、兼容性一般:column-fill: balance 只在firefox下支持,大部分瀏覽器只支持默認的column-fill: fill,就只能先渲染完第一列再渲染第二列
2、不適合需要順序展示的雙瀑布流
三、使用absolute佈局(最古早)
利:
1、一般不會出錯的方案
弊:
1、代碼繁瑣
參考:
https://juejin.cn/post/6844904142058553358
https://juejin.cn/post/6844904004720263176