使用Gird布局实现瀑布流效果

之前在插件里做了一个收藏夹和生词本的需求,大概长下面这样,里面的数据是按照瀑布流去做展示的

一、纯css也很好用

瀑布流布局一般的解决方案都是使用js去做动态计算内容高度,然后把内容分配给最短的'管道',实现调补的效果,但是纯js计算起来太费事了

在我逛了一圈百度后,发现css的gird布局完全可以实现瀑布流效果,可以减少很多计算

二、强大的gird

因为之前也没怎么使用grid,都是现查的,详细的可以看看这篇文章

gird是二维的布局,支持同时设置两个方向的数据,包括内容占比、间距等等,这样就很好办事了。我们可以直接设置'管道'的数量,内容动态去设置格子占比即可

话不多说,先封装容器组件 <WaterFall />

1.模板和参数

模板上,我们只要弄好容器就好了,里面展示什么内容不需要关心,直接用slot就好了

参数不需要支持太多,配置下面的就够了

  • col:管道的数量
  • colGap:管道的列间距
  • girdRowHeight:每一个网格的高度
  • rowGap:内容间行间距
vue 复制代码
<template>
  <div class="waterfall-wrapper" ref="waterfallRef" :style="wrapperStyle">
    <slot></slot>
  </div>
</template>

<script setup lang="ts">
const props = defineProps({
    col: {
      type: Number,
      default: 3
    },
    colGap: {
      type: Number,
      default: 20
    },
    girdRowHeight: {
      type: Number,
      default: 5
    },
    rowGap: {
      type: Number,
      default: 20
    }
})
</script>

<style leng="scss" scoped>
.waterfall-list-wrapper {
  display: grid;
  align-items: start;
}
</style>
4.容器的配置style
vue 复制代码
const wrapperStyle = computed(() => {
    return {
        // 需要多少列,每一列均分空间
        'grid-template-columns': `repeat(${props.col}, 1fr)`,
        
        // 隐式网格的高度
        'grid-auto-rows': `${props.girdRowHeight}px`
    }
})
4.动态计算内容占比
vue 复制代码
const waterfallRef = ref(null)
const updateList = () => {
    const els = Array.from(waterfallRef.value.children)
    els.forEach((el: any) => {
        // 内容占用的网格数
        const contentRows = Math.ceil(el.clientHeight / props.girdRowHeight)
        // 间距占用的网格数
        const gapRows = Math.ceil(props.rowGap / props.girdRowHeight)
    
        const rows = contentRows + gapRows
        
        // 上边框自适应
        el.style.girdRowStart = 'auto'
        
        // 设置下边框为跨度 rows 个网格的大小
        el.style.girdRowEnd = `span ${rows}`
    })
}

执行updateList,就可以实现瀑布流布局,如果有改变内容高度的操作,在操作完成后,再执行一遍updateList即可

三、总结一下

使用gird布局去实现瀑布流的重点在于,预设好隐式网格的高度,计算出内容占用的网格数,再给gird的元素设置上gird-row即可

相关推荐
是梦终空1 天前
vue下载依赖报错npm ERR node-sass@4.14.1 postinstall: `node scripts/build.js`的解决方法
javascript·npm·vue·node-sass·vue依赖
依米_1 天前
一文带你剖析 Promise.then all 实现原理,状态机、发布订阅模式完美实现异步编程
javascript·设计模式
陈陈小白1 天前
npm run dev报错Error: listen EADDRINUSE: address already in use :::8090
前端·npm·node.js·vue
G018_star sky♬1 天前
使用npm运行js脚本覆盖vue依赖包
javascript·vue.js
杂鱼豆腐人1 天前
pnpm环境下防止误使用npm的方法
前端·git·npm·node.js·git bash
我是ed1 天前
# vue2 使用 cesium 展示 TLE 星历数据
前端
宇余1 天前
从 useState 到 URLState:前端状态管理的另一种思路
前端·vue.js
白兰地空瓶1 天前
🚀 10 分钟吃透 CSS position 定位!从底层原理到避坑实战,搞定所有布局难题
前端·css
T___T1 天前
Ajax 数据请求详解与实战
javascript·面试