JS 实现图片瀑布流布局

瀑布流 = 定宽 + 不定高 + 列内追加。本文用原生js代码实现完整布局、重排、防抖,不依赖任何库。

一、核心思路

  1. 计算列数 = 容器宽 ÷ 图片宽
  2. 维护数组 colH[] = 每列当前高度
  3. 每张图找 最小高度 那一列塞进去
  4. 更新 colH[col] += 图片高 + 间隙
  5. 窗口 resize → 防抖重新布局

二、核心代码速览

js 复制代码
// ① 定宽
const imgW = 220                                                    
const container = document.getElementById('container')

// ② 列数 & 间隙
function cal() {                                                    
  const cw = container.clientWidth
  const cols = Math.floor(cw / imgW)
  const space = (cw - cols * imgW) / (cols + 1)
  return { cols, space }
}

 // ③ 布局函数
function setPos() {                                                
  const { cols, space } = cal()
  const colH = new Array(cols).fill(0)                              

  // 列高数组
  Array.from(container.children).forEach(img => {
    const minH = Math.min(...colH)
    const col = colH.indexOf(minH)
    img.style.top = minH + 'px'
    img.style.left = (col + 1) * space + col * imgW + 'px'
     // 更新列高
    colH[col] += img.height + space                               
  })

  // 撑开容器
  container.style.height = Math.max(...colH) + 'px'                
}

 // ④ 动态插入
function createImgs() {                                            
  for (let i = 0; i <= 40; i++) {
    const img = new Image()
    img.src = `../img/${i}.jpg`
    img.style.width = imgW + 'px'
    // 每张图加载完就排
    img.onload = setPos                                            
    container.appendChild(img)
  }
}

let timeId = null
// ⑤ 防抖重排
window.onresize = () => {                                           
  clearTimeout(timeId)
  timeId = setTimeout(setPos, 500)
}

// ⑥ 启动
createImgs()                                                        

三、关键点解读

  • 定宽不定高:imgW 固定,height 由图片本身决定
  • 列高数组:colH[col] 永远保存当前列的累计高度
  • DOM 移动:appendChild 不会重复插入,只做位置重排
  • 防抖:500 ms 内多次 resize 只执行最后一次,避免卡顿

四、性能 & 扩展

  • 零依赖:gzip < 1 KB,适用于任何老项目
  • 可异步:把 createImgs 换成 fetch 即可接入后端分页
  • 可响应:把 imgW 换成百分比,cal() 里用 clientWidth * ratio 即可自适应
相关推荐
圆肖1 小时前
[陇剑杯 2021]简单日志分析(问3)
前端·经验分享·github
王嘉俊9251 小时前
Django 入门:快速构建 Python Web 应用的强大框架
前端·后端·python·django·web·开发·入门
IT_陈寒2 小时前
Redis性能翻倍的5个冷门技巧,90%的开发者从不知道第3点!
前端·人工智能·后端
WebGIS开发2 小时前
新中地三维GIS开发智慧城市效果和应用场景
前端·人工智能·gis·智慧城市·webgis
山,离天三尺三3 小时前
线程中互斥锁和读写锁相关区别应用示例
linux·c语言·开发语言·面试·职场和发展
課代表3 小时前
Acrobat DC 文本域表单验证中的 js 使用
javascript·正则表达式·表单验证·数据完整性·字段验证·事件对象·自定义验证
鱼樱前端3 小时前
uni-app快速入门章法(一)
前端·uni-app
uhakadotcom3 小时前
入门教程:常用的 Python 第三方库:python-logstash
后端·面试·github
zhangxuyu11183 小时前
flex布局学习记录
前端·css·学习
掘金一周4 小时前
🍏让前端去做 iPhone 的液态玻璃❓ | 掘金一周 10.2
前端·人工智能·后端