【sgWaterfall】Vue实现图文瀑布流布局模式,图片预加载显示占位区域阴影,加载完成后向上浮动动画出现,支持不同浏览器尺寸宽度下自适应显示列数

news2024/11/24 18:28:49

 


特性:

  1. 自动计算每个图片最佳坐标位置,以达到最新加在图片占据位置尽量让整体更加协调
  2. 图片预加载显示占位区域阴影
  3. 加载完成后向上浮动动画出现
  4. 支持不同浏览器尺寸宽度下自适应显示列数

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/760840.html

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

相关文章

SQL语法与数据库快速入门(2)

目录&#xff1a; 多表简介SQL 约束-外键约束多表关系简介多表查询多表查询-内连接查询多表查询-外连接查询子查询简介子查询实战数据库进阶redis 内存数据库mongodb nosql 数据库neo4j 图数据库 1.多表简介 多表及使用场景介绍&#xff1a; 多表顾名思义就是在数据库设计中…

【NacosSync】注册中心同步组件使用教程ZooKeeper迁移到Nacos

目录 介绍系统模块架构获取安装包数据库配置启动服务器控制台添加注册中心集群信息添加同步任务注意事项 介绍 NacosSync是一个支持多种注册中心的同步组件,基于Spring boot开发框架,数据层采用Spring Data JPA,遵循了标准的JPA访问规范,支持多种数据源存储,默认使用Hibernate…

【论文阅读】《Distilling the Knowledge in a Neural Network》

【论文阅读】《Distilling the Knowledge in a Neural Network》 推荐指数&#xff1a; 1. 动机 &#xff08;1&#xff09;虽然一个ensemble的模型可以提升模型的效果&#xff0c;但是在效率方面实在难以接受&#xff0c;尤其是在每个模型都是一个大型的网络模型的时候。 &…

《TCP IP网络编程》第五章

第5章 基于 TCP 的服务端/客户端&#xff08;2&#xff09; 5.1 回声客户端的完美实现 先回顾一下服务器端的 I/O 相关代码&#xff1a; //持续接收客户端发送的数据&#xff0c;并将数据原样发送回客户端&#xff0c;直到客户端关闭连接。 while ((str_len read(clnt_sock,…

CMS垃圾收集器三色标记-JVM(十二)

上篇文章说了CMS垃圾收集器是赋值清除&#xff0c;所以他不可以碎片整理&#xff0c;于是jvm支持两个参数&#xff0c;几次fullGC之后碎片整理压缩空间。Cms他会抢占cpu资源&#xff0c;因为是并行运行&#xff0c;所以会有浮动垃圾。还有执行不确定性&#xff0c;垃圾收集完&a…

Python爬虫学习笔记(三)————urllib

目录 1.使用urllib来获取百度首页的源码 2.下载网页图片视频 3.总结-1 4.请求对象的定制&#xff08;解决第一种反爬&#xff09; 5.编解码 &#xff08;1&#xff09;get请求方式&#xff1a;urllib.parse.quote&#xff08;&#xff09; &#xff08;2&#xff09;get请求…

深度学习——RNN解决回归问题

详细代码与注释 import torch from torch import nn import numpy as np import matplotlib.pyplot as plt# 有利于复现代码 # torch.manual_seed(1) # reproducible# Hyper Parameters TIME_STEP 10 # rnn time step # 输入sin函数的y值&#xff0c;所以输入尺寸为1 INP…

posix ipc之消息队列

note 1.mq_open函数的参数pathname应以/开始&#xff0c;且最多一个/ 2.mq_receive的参数msg_len应大于等于attr.msgsize 3.消息队列写方写时不要求读方就绪&#xff0c;读方读时不要求写方就绪(和管道不同) code #include <fcntl.h> #include <sys/stat.h> #…

汽车销售数据可视化分析实战

1、任务 市场需求&#xff1a;各年度汽车总销量及环比&#xff0c;各车类、级别车辆销量及环比 消费能力/价位认知&#xff1a;车辆销售规模及环比、不同价位车销量及环比 企业/品牌竞争&#xff1a;各车系、厂商、品牌车销量及环比&#xff0c;市占率及变化趋势 热销车型&…

x86架构ubuntu22下运行SFC模拟器zsnet

0. 环境 ubuntu22 1. apt安装 sudo apt install zsnes 2. 运行 zsnet 参考&#xff1a;在Ubuntu上用zsnes玩SFC游戏&#xff0c;https://blog.csdn.net/gqwang2005/article/details/3877121

MyBatis学习笔记之首次开发及文件配置

文章目录 MyBatis概述框架特点 有关resources目录开发步骤从XML中构建SqlSessionFactoryMyBatis中有两个主要的配置文件编写MyBatis程序关于第一个程序的小细节MyBatis的事务管理机制JDBCMANAGED 编写一个较为完整的mybatisjunit测试mybatis集成日志组件 MyBatis概述 框架 在…

win11 系统暂无可用音频设备导致播放失败/音频服务未响应

win11 系统暂无可用音频设备导致播放失败/音频服务未响应 win11再一次更新后音频突然用不了了&#xff0c;驱动和输出设备都显示正常&#xff0c;但每次播放就会出现下面的问题&#xff0c;重启和更新驱动也没用。最后百度了好久终于解决了。 最后发现可能是新的驱动和电脑不兼…

【C++11】function包装器的简单使用

function 1 function包装器使用场景2 包装器3 包装成员函数4 一道例题5 包装器的意义 1 function包装器使用场景 现在有代码如下&#xff1a; 要求声明出这两个函数的类型 int f(int a,int b) {return a b; } struct Functor {int operator(int a,int b){return a b;} }可以…

(atan2)+(最小回文字符统计)

C - Nearest vectors long double eps1e-18 atan2:x正轴旋转的弧度角&#xff0c;y>0为正&#xff0c;y<0为负const long double PIacos(-1.0); const long double eps1e-18; struct node {ll id;long double x,y,z; }t[NN]; bool cmp1(node l,node r) {return l.zep…

布隆过滤器在海量数据去重验证中应用

布隆过滤器在海量数据去重验证中应用 文章目录 布隆过滤器在海量数据去重验证中应用引子面试结束级方案——从数据库中取新手级方案——利用redis的set数据结构专业级方案——利用布隆过滤器 布隆过滤器基本概念优点缺点布隆过滤器的数据结构布隆过滤器的工作流程布隆过滤器的优…

位1的个数:一种高效且优雅的解法

本篇博客会讲解力扣“191. 位1的个数”的解题思路&#xff0c;这是题目链接。 要想求解二进制位中1的个数&#xff0c;首先要了解一个神奇的表达式&#xff1a;n & n - 1。这个表达式的含义是&#xff1a;把n的二进制中最低位的1变成0&#xff0c;也就是消除n的二进制中最低…

day5-环形链表II

142.环形链表II 力扣题目链接 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&am…

Prometheus实现钉钉报警

1、Prometheus实现钉钉报警 1.1 Prometheus环境 # my global config global:scrape_interval: 15s # Set the scrape interval to every 15 seconds. Default is every 1 minute.evaluation_interval: 15s # Evaluate rules every 15 seconds. The default is every 1 minute…

位运算专题

异或:相同为0&#xff0c;相异为1&#xff0c;无进位相加 约定:给定的所有数从左到右依次是从低位到高位&#xff0c;下标从0开始 一)给定一个数&#xff0c;判断它的二进制位表示中的第X位是0还是1 1)(N>>X)&1&#xff0c;先将x位右移动到最低位&#xff0c;然后再将…

warp框架教程5-Filter系统中各个模块

any 模块 any 模块只有一个方法&#xff0c;就是 any 方法&#xff0c;它可以匹配任何路由的过滤器。我们可以使用 any 方法将一些可克隆的资源转换成一个过滤器&#xff0c;从而允许轻松地将它与其他 Filter 结合在一起。当然也可以使用 any 方法创建适用于多个 Filter 的末尾…