vue3实现瀑布流布局组件

先看效果图
在这里插入图片描述

直接上代码
utils.js

// 用于模拟接口请求
export const getRemoteData = (data = '获取数据', time = 2000) => {return new Promise((resolve) => {setTimeout(() => {console.log(`模拟获取接口数据`, data)resolve(data)}, time)})
}// 获取数组随机项
export const getRandomElement = (arr) => {var randomIndex = Math.floor(Math.random() * arr.length);return arr[randomIndex];
}// 指定范围随机数
export const getRandomNumber = (min, max) => {return Math.floor(Math.random() * (max - min + 1) + min);
}// 节流
export const throttle = (fn, time) => {let timer = nullreturn (...args) => {if (!timer) {timer = setTimeout(() => {timer = nullfn.apply(this, args)}, time)}}
}
// 防抖
export const debounce = (fn, time) => {let timer = nullreturn (...args) => {clearTimeout(timer)timer = setTimeout(() => {fn.apply(this, args)}, time)}
}

data.js 模拟后台返回的数据

import { getRandomElement, getRandomNumber } from "./utils.js"const colorList = ['red', 'blue', 'green', 'pink', 'yellow', 'orange', 'purple', 'brown', 'gray', 'skyblue']export const createList = (pageSize) => {let list = Array.from({ length: pageSize }, (v, i) => i)return list.map(x => {return {background: getRandomElement(colorList),width: getRandomNumber(200, 600),height: getRandomNumber(400, 700),x: 0,y: 0}})
}

瀑布流布局组件waterfall.vue

<template><div class="waterfall-container" ref="containerRef" @scroll="handleScroll"><div class="waterfall-list"><divclass="waterfall-item"v-for="(item, index) in resultList":key="index":style="{width: `${item.width}px`,height: `${item.height}px`,transform: `translate3d(${item.x}px, ${item.y}px, 0)`,}"><slot name="item" v-bind="item"></slot></div></div></div>
</template>
<script setup>
import { ref, onMounted, computed, nextTick, onUnmounted } from "vue";
import { createList } from "@/common/data.js";
import { getRemoteData, throttle, debounce } from "@/common/utils.js";
const props = defineProps({// 间距gap: {type: Number,default: 10,},// 列数columns: {type: Number,default: 3,},// 距离底部bottom: {type: Number,default: 0,},// 分页大小pageSize: {type: Number,default: 10,},
});// 容器ref
const containerRef = ref(null);// 卡片宽度
const cardWidth = ref(0);// 列高度
const columnHeight = ref(new Array(props.columns).fill(0));// 数据list
const resultList = ref([]);// 当前页码
const pageNum = ref(1);// 加载状态
const loading = ref(false);// 计算最小列高度及其下标
const minColumn = computed(() => {let minIndex = -1,minHeight = Infinity;columnHeight.value.forEach((item, index) => {if (item < minHeight) {minHeight = item;minIndex = index;}});return {minIndex,minHeight,};
});// 获取接口数据
const getData = async () => {loading.value = true;const list = createList(props.pageSize);const resList = await getRemoteData(list, 300).finally(() => (loading.value = false));pageNum.value++;resultList.value = [...resultList.value, ...getList(resList)];
};// 滚动到底部获取新一页数据-节流
const handleScroll = throttle(() => {const { scrollTop, clientHeight, scrollHeight } = containerRef.value;const bottom = scrollHeight - clientHeight - scrollTop;if (bottom <= props.bottom) {!loading.value && getData();}
});// 拼装数据结构
const getList = (list) => {return list.map((x, index) => {const cardHeight = Math.floor((x.height * cardWidth.value) / x.width);const { minIndex, minHeight } = minColumn.value;const isInit = index < props.columns && resultList.length <= props.pageSize;if (isInit) {columnHeight.value[index] = cardHeight + props.gap;} else {columnHeight.value[minIndex] += cardHeight + props.gap;}return {width: cardWidth.value,height: cardHeight,x: isInit? index % props.columns !== 0? index * (cardWidth.value + props.gap): 0: minIndex % props.columns !== 0? minIndex * (cardWidth.value + props.gap): 0,y: isInit ? 0 : minHeight,background: x.background,};});
};// 监听元素
const resizeObserver = new ResizeObserver(() => {handleResize();
});// 重置计算宽度以及位置
const handleResize = debounce(() => {const containerWidth = containerRef.value.clientWidth;cardWidth.value =(containerWidth - props.gap * (props.columns - 1)) / props.columns;columnHeight.value = new Array(props.columns).fill(0);resultList.value = getList(resultList.value);
});const init = () => {if (containerRef.value) {const containerWidth = containerRef.value.clientWidth;cardWidth.value =(containerWidth - props.gap * (props.columns - 1)) / props.columns;getData();resizeObserver.observe(containerRef.value);}
};onMounted(() => {init();
});
// 取消监听
onUnmounted(() => {containerRef.value && resizeObserver.unobserve(containerRef.value);
});
</script><style lang="scss">
.waterfall {&-container {width: 100%;height: 100%;overflow-y: scroll;overflow-x: hidden;}&-list {width: 100%;position: relative;}&-item {position: absolute;left: 0;top: 0;box-sizing: border-box;transition: all 0.3s;}
}
</style>

使用该组件(这里columns写死了3列)

<template><div class="container"><WaterFall :columns="3" :gap="10"><template #item="{ background }"><div class="card-box" :style="{ background }"></div></template></WaterFall></div>
</template><script setup>
import WaterFall from "@/components/waterfall.vue";
</script><style scoped lang="scss">
.container {width: 700px;  /* 一般业务场景不是固定宽度 */height: 800px;border: 2px solid #000;margin-top: 10px;margin-left: auto;
}
.card-box {position: relative;width: 100%;height: 100%;border-radius: 4px;
}
</style>

若要响应式调整列数,可参考以下代码


const fContainerRef = ref(null);
const columns = ref(3);
const fContainerObserver = new ResizeObserver((entries) => {changeColumn(entries[0].target.clientWidth);
});// 根据宽度,改变columns列数
const changeColumn = (width) => {if (width > 1200) {columns.value = 5;} else if (width >= 768 && width < 1200) {columns.value = 4;} else if (width >= 520 && width < 768) {columns.value = 3;} else {columns.value = 2;}
};onMounted(() => {fContainerRef.value && fContainerObserver.observe(fContainerRef.value);
});onUnmounted(() => {fContainerRef.value && fContainerObserver.unobserve(fContainerRef.value);
});

瀑布流布局组件监听columns变化

watch(() => props.columns,() => {handleResize();}
);

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/691614.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

给label-studio 配置sam(segment anything)ml 记录

给label-studio 配置sam&#xff08;segment anything&#xff09;ml 后端记录 配置ml后台下载代码下载模型文件创建环境模型转换后端服务启动 配置label-studio 前端配置模型后端连接配置标注模板标注界面使用 参考链接 配置ml后台 下载代码 git clone https://github.com/H…

AT24C02(I2C总线)通信的学习

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、存储器介绍二、AT24C02芯片二、I2C总线I2C电路规范I2C时序结构I2C数据帧AT24C02数据帧 总结 前言 学习AT24C02(I2C总线)芯片 一、存储器介绍 RAM&#xf…

QT-地形3D

QT-地形3D 一、 演示效果二、关键程序三、下载链接 一、 演示效果 二、关键程序 #include "ShaderProgram.h"namespace t3d::core {void ShaderProgram::init() {initializeOpenGLFunctions();loadShaders(); }void ShaderProgram::addShader(const QString &fil…

计算机二级C语言的注意事项及相应真题-5-程序修改

目录 41.累加链表结点数据域中的数据作为函数值返回42.根据整型形参m&#xff0c;计算如下公式的值43.删除数列中值为x的元素44.从N个字符串中找出最长的那个串&#xff0c;并将其地址作为函数值返回45.将两个长度相等的纯数字字符串当作两个加数&#xff0c;求其代表的数值之和…

欢迎来到IT时代----盘点曾经爆火全网的计算机电影

计算机专业必看的几部电影 计算机专业必看的几部电影&#xff0c;就像一场精彩的编程盛宴&#xff01;《黑客帝国》让你穿越虚拟世界&#xff0c;感受高科技的魅力&#xff1b;《社交网络》揭示了互联网巨头的创业之路&#xff0c;《源代码》带你穿越时间解救世界&#xff0c;这…

【动态规划】【字符串】2167移除所有载有违禁货物车厢所需的最少时间

作者推荐 【深度优先搜索】【树】【有向图】【推荐】685. 冗余连接 II 本文涉及知识点 动态规划汇总 LeetCode2167移除所有载有违禁货物车厢所需的最少时间 给你一个下标从 0 开始的二进制字符串 s &#xff0c;表示一个列车车厢序列。s[i] ‘0’ 表示第 i 节车厢 不 含违…

第五次作业(防御安全)

需求: 1.办公区设备可以通过电信链路和移动链路上网&#xff08;多对多的NAT&#xff0c;并且需要保留一个公网IP 不能用来转换&#xff09; 2.分公司设备可以通过总公司的移动链路和电信链路访问到DMZ区的http服务器 3.分公司内部的客户端可以通过公网地址访问到内部的服务…

Linux:grep进阶(11)

Linux&#xff1a;shell脚本&#xff1a;基础使用&#xff08;4&#xff09;《正则表达式-grep工具》_shell grep 全角字符串-CSDN博客https://blog.csdn.net/w14768855/article/details/132338954?ops_request_misc%257B%2522request%255Fid%2522%253A%252217083360171680022…

c#,dotnet, DataMatrix 类型二维码深度识别,OCR,(基于 Halcon)

代码中部分调用的 c 函数参数&#xff0c;具体说明自行研究~&#xff08;我也是参考的其他资源&#xff0c;还没研究透彻&#xff09; 例如&#xff1a;HOperatorSet.GenRectangle2() &#xff0c; 2000, 2000, 0, 2000, 2000 这些数字应该是选取的图片解析范围、尺寸&#xff…

Linux环境变量配置文件--《一图胜千言》

这张图是一个关于Linux系统中shell启动时配置文件加载顺序的流程图。图中分为登录shell和非登录shell两种情况&#xff0c;来描述不同配置文件的读取过程。 登录shell&#xff1a; 当用户登录时&#xff0c;会首先检查是否存在/etc/profile文件&#xff0c;如果存在&#xff0c…

酷开科技 | 酷开系统壁纸模式,让过年更有氛围感!

在阵阵爆竹声中&#xff0c;家家户户都沉浸在浓浓的年味中。过年&#xff0c;是团圆&#xff0c;是温暖。团团圆圆的日子里&#xff0c;仪式感不可少&#xff0c;换上一张喜气洋洋的电视壁纸吧&#xff0c;寓意幸福一年又一年。打开酷开系统壁纸模式挑选一张年味十足的壁纸&…

【深度学习】神经网络的建立与推理

文章目录 神经网络&#xff08;neural network&#xff09;的结构神经元中常用的激活函数&#xff08;activation function&#xff09;神经网络的表示神经网络的代码实现使用已学习完毕的神经网络进行推理&#xff08;inference&#xff09; 源代码文件请点击此处&#xff01;…

毕设(二)——NB-IOT通信模块(nb卡通信测试)+gps定位

文章目录 一、关于接线2月1日记录2月4日记录 二、网络连接测试三、HTTP通信3.1 网络调试3.2 nb-lot的连接测试 一、关于接线 如果pico的供电能力不行&#xff0c;可能会直接用4.2V的锂电池对右下引脚进行供电 这个模块只支持nb卡&#xff0c;我哭死&#xff0c;20块钱&#xff…

01 Qt自定义风格控件的基本原则

目录 1.继承原生控件 2.组合原生控件 3.仿写原生控件 PS:后续将继续分享开发实践中各类自定义控件的方法、思路以及组件库 1.继承原生控件 关键字&#xff1a;继承、paintEvent 这里想说的是&#xff0c;Qt的Gui框架在封装原生控件的同时&#xff0c; 也为开发者提供了各…

每日一题——LeetCode1464.数组中两元素的最大乘积

这题就是找数组里的最大值和次大值 方法一 排序 var maxProduct function(nums) {nums.sort((a,b)>b-a)return (nums[0] - 1) * (nums[1] - 1); }; 消耗时间和内存情况&#xff1a; 方法二 一次遍历&#xff1a; var maxProduct function(nums) {let first-1,second-…

MySQL的备份与恢复案例

新建数据库 数据库备份&#xff0c;数据库为school&#xff0c;素材如下1.创建student和score表CREATE TABLE student ( id INT(10) NOT NULL UNIQUE PRIMARY KEY , name VARCHAR(20) NOT NULL , sex VARCHAR(4) , birth YEAR, department VARCHAR(20) , address…

打码半年,开源一款自定义大屏设计软件!

hi&#xff0c;大家好&#xff0c;我是Tduck马马。 最近我们开源了一款大屏软件-TReport&#xff0c;与大家分享。 TReport是一款基于Vue3技术栈的数据可视化系统&#xff0c;支持静态、动态api等数据源&#xff1b;可用于数据可视化分析、报表分析、海报设计使用。 提供自定…

leetcode hot100 分割等和子集

在本题中&#xff0c;我们是要把一个数组&#xff0c;分割成两个子集&#xff0c;并且两个子集的元素和相等。那么也就是说&#xff0c;两个子集的和是相等的&#xff0c;并且都是整个数组的一半。那我们考虑这是一个01背包问题&#xff0c;物品的价值和物品的质量一样&#xf…

linux 10 定时任务

作用: 计划任务主要是做一些周期性的任务&#xff0c; 目前最主要的用途是定期备份数据。 at命令的时间格式&#xff1a; 例子&#xff1a; crontab有系统级别的任务&#xff0c;用户的不放在这里 查看用户任务 或者用

VSCODE使用Django

https://code.visualstudio.com/docs/python/tutorial-django#_use-a-template-to-render-a-page 通过模板渲染页面 HTML文件 实现步骤 1&#xff0c; 修改代码&#xff0c;hello的App名字增加到installed_apps表中。 2&#xff0c; hello子目录下&#xff0c;创建 .\templat…