【sgWaterfall】Vue实现图文瀑布流布局模式,图片预加载显示占位区域阴影,加载完成后向上浮动动画出现

news2025/1/22 16:42:51

 

 


Vue实现图文瀑布流布局模式,图片预加载显示占位区域阴影,加载完成后向上浮动动画出现

sgWaterfall.vue源码 

<template>
    <ul :class="$options.name" :style="waterfallStyle">
        <li v-for="(a, i) in  items " :key="i" :style="a.style" :imgLoaded="a.imgLoaded">
            <template>
                <div class="cover">
                    <img :src="a.imgURL" @load="a.imgLoaded = true">
                </div>
                <label>{{ a.label }}</label>
            </template>
        </li>
    </ul>
</template>
    
<script>
export default {
    name: 'sgWaterfall',
    data() {
        return {
            imgLoadedStatus: [],
            items: [],
            waterfallStyle: {},
            __var_itemTextHeight: 0,
        }
    },
    props: [
        "data",
        "itemWidth",
        "itemMarginRight",
        "itemMarginBottom",
        "itemTextHeight",
        "webpageMargin",
    ],
    watch: {
        itemTextHeight: {
            handler(d) {
                this.__var_itemTextHeight = parseFloat(d) || 50;
                this.setProperty();
            }, deep: true, immediate: true,
        },
        data: {
            handler(d) {
                d && (this.items = this.calcAnyPhotosPosition(d));
            }, deep: true, immediate: true,
        },
    },
    destroyed() {
        removeEventListener('resize', this.resize);
    },
    mounted() {
        this.setProperty();
        addEventListener('resize', this.resize);
    },
    methods: {
        resize(e) {
            this.items = this.calcAnyPhotosPosition(this.items)
        },
        setProperty() {
            this.$el && this.$el.style.setProperty("--itemTextHeight", this.__var_itemTextHeight + 'px');
        },
        // BEGIN 计算图片的坐标位置_________________________________________________________
        // 计算能显示下多少列
        calcColsCount({ itemWidth, webpageMargin, itemMarginRight }) {
            return Math.floor((window.innerWidth - webpageMargin * 2 + itemMarginRight) / (itemWidth + itemMarginRight));
        },
        // 计算图片的坐标位置
        calcPhotoScaleWidthHeight(d, { itemWidth, itemTextHeight, defaultItemHeight, }) {
            // index是图片索引
            if (d.width && d.height) {
                let originWidth = d.width;//图片原始宽度
                let originHeight = d.height;//图片原始高度
                let proportion = originWidth / originHeight;//图片宽高比
                let scaleWidth = itemWidth;//缩放后的宽度
                let scaleHeight = scaleWidth / proportion;//缩放后的高度
                return {
                    width: scaleWidth,
                    height: scaleHeight + itemTextHeight,
                }
            } else {
                // 后台不提供高度数据的情况下(默认宽度高度,是个正方形)
                return {
                    width: itemWidth,
                    height: defaultItemHeight + itemTextHeight,
                }
            }
        },
        // 批量计算图片的坐标位置
        calcAnyPhotosPosition(arr) {
            arr = JSON.parse(JSON.stringify(arr));
            let itemWidth = parseFloat(this.itemWidth) || 250;//固定宽度
            let defaultItemHeight = itemWidth;//默认高度
            let itemMarginRight = parseFloat(this.itemMarginRight) || 20;//右侧间距
            let itemMarginBottom = parseFloat(this.itemMarginBottom) || 20;//底部间距
            let itemTextHeight = this.__var_itemTextHeight;//文本显示区域高度 
            let webpageMargin = parseFloat(this.webpageMargin) || 0;//网页左右两侧间距

            let colsCount = this.calcColsCount({ itemWidth, webpageMargin, itemMarginRight });//一共有多少列
            let colHeights = [...Array(colsCount)].map(v => 0);//列的高度数组
            arr.forEach(v => {
                let minHeight = Math.min(...colHeights);//找到高度最小的值
                let colIndex = colHeights.indexOf(minHeight);//找到高度最小的那一列
                let { width, height } = this.calcPhotoScaleWidthHeight(v, { itemWidth, itemTextHeight, defaultItemHeight, });
                let left = colIndex * (itemWidth + itemMarginRight);
                let top = minHeight + itemMarginBottom;
                colHeights[colIndex] += (height + itemMarginBottom);
                v.imgLoaded || (v.imgLoaded = false);//预加载图片使用响应式属性(这里不提前声明初始化,动态渲染数据的时候不会生效)
                v.style = {
                    left: left + 'px',
                    top: top + 'px',
                    width: width + 'px',
                    height: height + 'px',
                }
            });
            let waterfallWidth = (itemWidth + itemMarginRight) * colsCount - itemMarginRight;//计算容器宽度
            let waterfallHeight = this.scrollContainerHeight = Math.max(...colHeights);//计算容器高度
            this.waterfallStyle = {
                width: waterfallWidth + 'px',
                height: waterfallHeight + 'px',
            }
            return arr;
        },
        // END _________________________________________________________
    }
};
</script>
    
<style lang="scss" scoped>
.sgWaterfall {
    position: relative;
    overflow: hidden;
    margin: 0 auto;

    li {
        position: absolute;
        display: flex;
        flex-direction: column;
        transition: .382s;

        .cover {
            height: calc(100% - var(--itemTextHeight));
            background-color: #f0f0f0;
            border-radius: 16px;
            overflow: hidden;
            transition: .382s;

            img {
                border-radius: 16px;
                opacity: 0;
                width: 100%;
                height: 100%;
                transform: translateY(100%);
                object-position: center;
                object-fit: cover;
                transition: .8s cubic-bezier(.075, .82, .165, 1); //快速+柔和结束 
            }

        }

        label {
            height: var(--itemTextHeight);
            display: block;
            box-sizing: border-box;
            padding: 10px 0;
            /*多行省略号*/
            overflow: hidden;
            word-break: break-all;
            /*单词分割换行*/
            display: -webkit-box;
            -webkit-box-orient: vertical;
            max-height: min-content;
            -webkit-line-clamp: 2;

            line-height: 1.6;
            font-size: 14px;
        }

        &[imgLoaded] {

            .cover {
                background-color: transparent;

                img {
                    opacity: 1;
                    transform: translateY(0);
                }
            }

        }

    }
}
</style>

 应用组件

<template>
    <div class="sg-body">
        <sgWaterfall :data="data" itemWidth="200" />
    </div>
</template>
    
<script>
import sgWaterfall from "@/vue/components/admin/sgWaterfall";
export default {
    components: { sgWaterfall },
    data: () => ({
        labels: [
            '在新的历史起点上继续推动文化繁荣',
            '江苏一飞机坠落 坠落地有飞机残骸',
            '河南暴雨 宾客坐积水中吃席',
            '多地细化举措护航暑期安全见闻',
            '男子炫耀钓到大鱼挂后备箱示众',
            '男子自称上山捡菌遇老虎 警方:行拘',
            '高温天妈妈将电扇让给孩子被热死',
            '狗狗被电动车踏板打了一路屁股',
            '重度自闭症男孩靠吹气球成网红',
            '男子疑在售水机打水时触电身亡',
            '广州平均月薪10883元',
            '邓伦再成被执行人',
            '9岁男孩被跳楼者砸中昏迷不醒',
            '女子买69平公寓公摊37平',
            '“银行高管”娶四个老婆获刑',
            '封神首映费翔黄渤河南话引爆笑',
            '张一山靠在关晓彤后背哭',
            '马丽的孩子确实叫沈腾舅舅',
            '鹿晗现身医院做康复治疗',
            '男子高空抛砖砸死人 称想跳楼但不敢',
            '一名中国侨胞在巴西遭枪击身亡',
            '山西杀害队长的环卫工已被捕',
            '猴子4小时去超市连偷3次',
            '韩安冉吐槽环球影城3小时花8000',
            '九旬老人逗鹦鹉得了鹦鹉热',
            '二十大以来首个正部级落马',
            '试管失败医院拒绝返还夫妻胚胎',
            '老人怕味大影响乘客高铁上干吃泡面',
            '重庆一消防员考上清华大学研究生',
            '猪场因断电死亡462头猪损失近百万',
            '消防喷水20分钟解救3000只中暑鸭',
        ],
        data: [],
    }),
    created() {
        this.data = this.createData();
    },

    methods: {
        createData(e) {
            return [...Array(50)].map((v, i) => {
                let width = Math.round(Math.random() * (1000 - 500) + 500);
                let height = Math.round(Math.random() * (1000 - 500) + 500);
                return {
                    label: this.labels[Math.round(Math.random() * (this.labels.length - 1))],
                    // imgURL: `static/files/picture/${(Math.random() * (50 - 1) + 1).toFixed(0)}.jpg`,
                    imgURL: `http://via.placeholder.com/${width}x${height}`,
                    width,
                    height,
                }
            }
            )
        },
    },
};
</script> 
<style lang="scss" scoped>
.sg-body {
    width: 100vw;
    height: 100vh;
    overflow: hidden;
    overflow-y: auto;
}
</style>

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

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

相关文章

业务变革与架构双驱动的多项目管理︱海康威视流程变革咨询顾问张燕飞

海康威视数字技术股份有限公司流程与变革管理部流程变革咨询顾问张燕飞女士受邀为由PMO评论主办的2023第十二届中国PMO大会演讲嘉宾&#xff0c;演讲议题&#xff1a;业务变革与架构双驱动的多项目管理。大会将于8月12-13日在北京举办&#xff0c;敬请关注&#xff01; 议题简要…

如何在MySQL中安装示例数据库sakila

就像 SQLServer 示例数据库一样,MySQL 也有示例数据库,比如sakila;Sakila 数据库最初由 MySQL AB 文档团队的前成员 Mike Hillyer 开发,旨在提供一个标准模式,可用于书籍、教程、文章、示例等中的示例,它包含示例视图、存储过程和触发器。 以下是在服务器上安装sakila数…

33个字段,我乏了!

家人们&#xff0c;谁懂啊&#xff1f;&#xff1f;最近两天&#xff0c;一直在测试数据报告&#xff0c;一个页面有30多个字段的那种&#xff0c;从基础表写sql语句&#xff0c;执行&#xff0c;与页面显示的字段值进行对比&#xff0c;一个挨着一个&#xff0c;讲真&#xff…

热更新时:app经常因为HBuilder X的更新提示HTML5+ Runtime版本不匹配的问题

直接暴力解决问题。 "app-plus" : {"compatible": {"ignoreVersion": false,//可选&#xff0c;Boolean类型&#xff0c;是否忽略版本兼容检查提示}, }APP模块等相关配置改变还是需要整包APK更新。 只有热更新改前端代码&#xff0c;推荐直接配置…

驱动 DAY10

platform驱动实现 match-devicetree.c #include <linux/init.h> #include <linux/module.h> #include <linux/platform_device.h> #include <linux/mod_devicetable.h> #include <linux/gpio.h> #include <linux/of.h> #include <linu…

electron+vue3全家桶+vite项目搭建【21】自定义无边框窗口拖拽移动

文章目录 引入实现思路实现步骤1.主进程监听窗口移动2.通信工具补充ipc调用3.渲染进程封装通用拖拽组件 测试 引入 如果你尝试过透明窗口&#xff0c;并控制透明部分事件击穿&#xff0c;就会发现使用 drag属性样式去控制窗口拖拽会导致点击事件失效&#xff0c;并且带drag属性…

有没有后端程序员想要兼职的?

有一个非常有意思的现象&#xff1a; 后端看不起前端&#xff0c;认为前端是好入门&#xff0c;含金量低&#xff0c;一下就能学会的页面侠&#xff1b; 前端看不起后端&#xff0c;认为后端是每天CRUD、调参、拿来主义的搬砖工&#xff1b; 而实际上&#xff0c;大家都是给老板…

vue生命周期四个阶段(created和mount)

1.四个阶段 1&#xff09;必经阶段 2&#xff09;非必经阶段 提示&#xff1a;主动调用 vm.$destroy() 函数销毁后&#xff0c;可用 vm.$mount("#app") 将断开的 new Vue() 和页面重新建立虚拟 DOM 树&#xff0c;重新绑定起来挂载界面。 2. 生命周期钩子函数&…

【大数据】大数据简介

大数据简介 大数据基础平台架构实际应用关键技术 Hadoop 分布式计算平台Hadoop生态系统Hadoop安装和使用 HDFS分布式文件系统NamenodeSecondary NamenodeDataNodeblock 大数据基础 平台架构 实际应用 关键技术 Hadoop 分布式计算平台 Hadoop生态系统 Hadoop安装和使用 参考htt…

单向链表基本操作

目录 初始化链表 插入 删除 遍历 销毁 清空 初始化链表 代码&#xff1a; struct LinkNode* Init_LinkList() {struct LinkNode* head (struct LinkNode*)malloc(sizeof(struct LinkNode));head->data -1;head->next NULL;// 尾部指针struct LinkNode* pRear …

概率论的学习和整理15: 超几何分布,二项分布,泊松分布是如何趋近收敛的?

目录 1 问题&#xff1a; 2 结论 3 实验1 4 实验2 5 实验3 6 实验4 5 各种规律总结 5.1 1 5.2 2 5.3 3 5.4 4 6 超几何分布&#xff0c;二项分布&#xff0c;泊松分布&#xff0c;三者用EXCEL模拟 6.1 简单的扩展到泊松分布 6.2 比较整体的动态过程&…

在qt界面上内嵌拥有独立句柄的窗口

背景 在qt程序中&#xff0c;如果数据刷新频率过高&#xff0c;容易造成窗口卡顿&#xff0c;因为qt程序是整个窗口刷新&#xff0c;在此种背景下可以在qt程序的主程序上内嵌一个拥有独立句柄的窗口&#xff0c;两个窗口刷新就互不干扰。 案例 以下例子&#xff0c;在主窗口…

【技术指南】3D转换工具HOOPS Exchange的功能特征和典型使用场景全解析(一)

一、什么是 HOOPS Exchange&#xff1f; HOOPS Exchange 是一组软件库&#xff0c;可以帮助开发人员在开发应用程序时读取和写入主流的 2D 和 3D 格式。HOOPS Exchange 支持 在主流的3D 文件格式中读取 CAD 数据&#xff0c;并支持将 3D 数据转换为 PRC 数据格式&#xff0c;…

2.2 顺序表与链表特性对比

1. 插入删除操作对比 1. 顺序表插入删除元素 插入策略: 在某个位置插入元素时, 把从该位置开始的所有元素都往后挪一个位置, 规定顺序表最后一个元素后面的位置也是一个可插入位置. 后面的元素先往后挪动位置. 删除策略: 在某位置删除元素, 把从该位置之后的所有元素都往前…

【技术篇】• 饮用水除硝酸盐的技术解析

​​​​​​​ 近年来由于农业活动及排污物的影响&#xff0c;部分地表水源水中硝酸盐含量呈现明显的增加趋势&#xff0c;硝酸盐污染成为地下水和饮用水领域关注的热点问题之一。 硝酸盐是有氧环境中稳定的含氮化合物形式&#xff0c;也是含氮有机物通过无机化分解的产物&am…

搭建archetype骨架工程

搭建archetype骨架工程 一、archetype概念1、archetype简介2、archetype组成结构3、archetype生命周期4、archetype使用 二、构建我们自定义的骨架工程1、创建一个自定义的项目2、修改pom的build插件3、生成archetype资源文件4、将生成的资源文件制作成archetype jar包5、生成a…

Java中不同变量声明类型

今天在学习分层解耦-三层架构的过程中&#xff0c;具体文章参照&#xff1a;写文章-CSDN创作中心 在Servie层创建Dao对象时&#xff0c;以及在Controller层创建Service对象时&#xff0c;发现与我之前了解的声明变量的方法不一样。具体关键代码如下&#xff1a; 其中EmpServic…

简要介绍 | 边缘计算:原理,研究现状与未来展望

注1&#xff1a;本文系“简要介绍”系列之一&#xff0c;仅从概念上对边缘计算进行非常简要的介绍&#xff0c;不适合用于深入和详细的了解。 边缘计算&#xff1a;原理&#xff0c;研究现状与未来展望 What is Edge Computing? | Moving Intelligence to the Edge 一、背景介…

小平板 大智慧-嵌入式方案满足教育市场多元需求

线上教育观念的深入和技术的更新&#xff0c;直接拉动了教育类硬件及相关终端设备的市场需求。 产品框图 IDO-SBC3566采用瑞芯微RK3566&#xff0c; CPU采用4核A55架构处理器&#xff0c;集成G52图形处理器&#xff0c;内置独立NPU&#xff0c;算力高达1Tops&#xff0c;可满足…

如何在 Windows 中免费合并 PDF 文件 [在线和离线]

PDF是一种广泛使用的文件格式&#xff0c;具有兼容性好、安全性高、易于打印、方便浏览等众多优点。在工作和学习过程中&#xff0c;经常需要将同一类型的PDF文件合并起来&#xff0c;以方便传输和查看&#xff0c;使得合并PDF文件成为一种重要的数据整合方法。 如果您想知道如…