记录--Vue3基于Grid布局简单实现一个瀑布流组件

这篇具有很好参考价值的文章主要介绍了记录--Vue3基于Grid布局简单实现一个瀑布流组件。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助

记录--Vue3基于Grid布局简单实现一个瀑布流组件

前言

在学习Grid布局之时,我发现其是CSS中的一种强大的布局方案,它将网页划分成一个个网格,可以任意组合不同的网格,做出各种各样的布局,在刷某书和某宝首页时,我们发现其展示方式就是一种瀑布流,是一种流行的网站页面布局,视觉表现为参差不齐的多栏布局,随着页面向下滚动,这种布局会不断加载数据块并附加到当前尾部。采用瀑布流布局的方式可以打破常规网站布局排版,给用户眼前一亮的新鲜感,更好的适应移动端。

因此结合二者,本文将通过grid布局简单实现一个瀑布流组件,该组件已开源上传npm,可以直接安装使用,Git地址在文尾。

实现效果:

记录--Vue3基于Grid布局简单实现一个瀑布流组件

实现原理

1、使用grid布局将页面分为无数个小网格,每个网格高度为1px。

.grid-content {
    display: grid;
    grid-auto-rows: minmax(1px, 1px);
    overflow: auto;
}

2、宽度根据需要自定义的列数自动分配。

   'grid-template-columns': `repeat(${props.columns}, 1fr)`,

3、根据每个卡片窗口的高度计算每个卡片需要跨越几个网格(因为每个网格设置高为1px,所以高度就是需要跨越的网格数)

'grid-row-end': `span ${gridItem.value.clientHeight - 1}`

4、监听瀑布流滚动事件,通过判断滚动条距离底部的高度,在滚动到底部一定距离时加载更多的数据,以实现无限滚动。

主要代码实现

gridContent 组件主要代码,循环展示每个条目,根据自定义的列展示不同的列数量,根据触底数据判断获取最新数据。监听传入的数据进行处理,目前只是做了简单处理,后面将通过虚拟列表的形式,动态处理该数据,以增加性能。

<template>
    <div class="grid-content" ref="gridContent" :style="gridStyle" @scroll="getMoreData">
        <grid-item v-for="item in showDataList" :key="item.dataIndex" :data="item">
            <template #slot-scope="slotProps">
                <slot name="slot-scope" :slotProps="slotProps"></slot>
            </template>
        </grid-item>
    </div>
</template>
<script lang="ts" setup>
import GridItem from './gridItem.vue';
import { ref, watch } from 'vue';

const props = defineProps({
    dataList: {
        type: Array,
        default: []
    },
    columns: {
        type: Number,
        default: 2
    },
    width: {
        type: Number,
        default: 300
    },
    height: {
        type: Number,
        default: 400
    },
    bottom:{
        type: Number,
        default: 50
    },
    loading:{
        type: Boolean,
        default: true
    }

})

const emit=defineEmits(['getMoreData']);

const gridStyle = ref({});
const showDataList = ref<any>([])

watch(() => props.dataList, (newValue) => {
    let tempData: any = [];
    newValue.forEach((item: any, index) => {
        tempData.push({ ...item, dataIndex: index })
    })
    showDataList.value = tempData;

    gridStyle.value = {
        'grid-template-columns': `repeat(${props.columns}, 1fr)`,
        width:props.width + 'px',
        height:props.height + 'px'
    }
}, { immediate: true,deep:true })

const isLoading=ref<boolean>(false);
watch(()=>props.loading,(newValue:boolean)=>{
    isLoading.value=newValue;
})

const gridContent=ref<any>(null);
//根据触底数据判断获取最新数据
const getMoreData=()=>{
    const scrollHeight = gridContent.value.scrollHeight || 0;
    const clientHeight = gridContent.value.clientHeight || 0;
    const scrollTop = gridContent.value.scrollTop || 0;
    if(scrollHeight - clientHeight - scrollTop < props.bottom && !isLoading.value){
        isLoading.value=true;
        emit('getMoreData');
    }
}
</script>
grid-item 组件代码,主要通过获取组件高度设置跨越的网格数,通过插槽展示每个卡片。
<template>
    <div class="grid-item" :style="itemStyle">
        <div ref="gridItem">
            <slot name="slot-scope" :data="data"></slot>
        </div>
    </div>
</template>
<script lang="ts" setup>
import { ref, onMounted } from 'vue';
defineProps({
    data: {
        type: Object,
        default: () => { }
    }
})

const gridItem = ref<any>(null);
const itemStyle = ref({})

onMounted(() => {
    itemStyle.value = { 'grid-row-end': `span ${gridItem.value.clientHeight - 1}` }
})

</script>
<style scoped> 
.grid-item {
    grid-row-end: span 100;
}
</style>

使用示例

npm install @fcli/vue-grid-waterfall --save-dev 来安装

在项目中使用
import VueGridWaterfall from '@fcli/vue-grid-waterfall';
const app=createApp(App)
app.use(VueGridWaterfall);

使用示例:

<template>
  <div class="content">
    <vue-grid-waterfall :data-list="dataList" :columns="3" @getMoreData="getMoreData" :loading="isLoading">
      <template #slot-scope="{ slotProps }">
        <div class="item" :style="{ height: slotProps.data.height, background: slotProps.data.color }">{{ slotProps.data.color
        }}</div>
      </template>
    </vue-grid-waterfall>
  </div>
</template>

<script setup lang="ts">
import vueGridWaterfall from './plugin/index.vue';
import { ref, onMounted } from 'vue'
component: {
  vueGridWaterfall
}

const dataList = ref<any>([]);
//获取随机颜色
const getRandomColor = () => {
  const getColor: any = (color: any) => {
    return (color += '0123456789abcdef'[Math.floor(Math.random() * 16)]) && (color.length == 6) ? color : getColor(color);
  };
  return '#' + getColor('')
}

const getMoreData = () => {
  isLoading.value = true;
  getData()
}
const isLoading = ref(true);

//获取数据
const getData = () => {
  for (let i = 0; i < 100; i++) {
    dataList.value.push({ height: 50 + Math.random() * 50 + 'px', color: getRandomColor() })
  }
  setTimeout(()=>{
      isLoading.value = false;
  })
}

onMounted(() => {
  getData()
})
</script>

记录--Vue3基于Grid布局简单实现一个瀑布流组件

 

本文转载于:

https://juejin.cn/post/7280747572695973948

如果对您有所帮助,欢迎您点个关注,我会定时更新技术文档,大家一起讨论学习,一起进步。

 记录--Vue3基于Grid布局简单实现一个瀑布流组件文章来源地址https://www.toymoban.com/news/detail-745925.html

到了这里,关于记录--Vue3基于Grid布局简单实现一个瀑布流组件的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 记录--Vue3自定义一个Hooks,实现一键换肤

    使用CSS变量, 准备两套CSS颜色 , 一套是在 light模式下的颜色,一套是在dark模式下的颜色 dark模式下的 CSS 权重要比 light 模式下的权重高 , 不然当我们给html添加自定义属性 [data-theme=\\\'dark\\\'] 的时候, dark模式权重比light低,会一直不起效果 当我们点击 dark 模式的时候, 给 html 设置自定义

    2024年02月07日
    浏览(39)
  • 《Vue3+Typescript》一个简单的日历组件实现

    这是一个没有套路的前端博主,热衷各种前端向的骚操作,经常想到哪就写到哪,如果有感兴趣的技术和前端效果可以留言~博主看到后会去代替大家踩坑的~ 主页: oliver尹的主页 格言: 跌倒了爬起来就好~ 目录 一、前言 二、效果图 三、核心思路 四、代码实现 4.1 本月日期

    2024年02月14日
    浏览(41)
  • vue3 实现简单计数器示例——一个html文件展示vue3的效果

    目的 :作为一个新手开发,我想使用 Vue 3 将代码封装在 HTML 文件中时,进行界面打开展示。 学了一个简单计数器界面展示,代码如下: 在 上述HTML 文件里,包含了文件头标题 title , 接着定义了一个内容 div id=\\\"app\\\" 而后定义了一个内容,包含标题和数字,以及两个按钮的 templa

    2024年01月18日
    浏览(43)
  • 基于 Vue3 和 WebSocket 实现的简单网页聊天应用

    一个基于Vue3和WebSocket的简易网络聊天室项目,包括服务端和客户端部分。 项目地址 websocket-chat 下面是项目的主要组成部分和功能: 项目结构 功能特性 私聊功能:用户可以选择联系人进行一对一私聊,发送即时消息。 群聊功能:用户可以加入群组,与群组成员进行群聊。

    2024年02月03日
    浏览(54)
  • Vue项目实战——实现一个任务清单【基于 Vue3.x 全家桶(简易版)】

    内容 参考链接 Vue2.x全家桶 Vue2.x 全家桶参考链接 Vue2.x项目(一) Vue2.x 实现一个任务清单 Vue2.x项目(二) Vue2.x 实现GitHub搜索案例 Vue3.x项目(三) Vue3.x 实现一个任务清单 1、前言 如果你对 vue3 的基础知识还很陌生,推荐先去学习一下 vue 基础 内容 参考链接 Vue2.x全家桶 Vu

    2024年02月02日
    浏览(81)
  • 基于Vue3和Element Plus组件库实现一个完整的登录功能

    先看一下最终要实现的效果: 登录页面: 注册页面: (1)引入element-plus组件库 引入组件库的方式有好多种,在这里我就在main.js全局引入了. main.js中代码: 引入之后自己可以用几个按钮测试一下是否引入成功. (2)登录及注册页面 html部分 views/account/Login.vue js部分 css部分(使用了scss) (

    2024年02月05日
    浏览(47)
  • Vue Grid Layout - 适用Vue.js的栅格布局系统(项目实例)

    gitee:grid-project-gitee B站视频:Vue-Grid-Layout 创建vue项目 vue create vite-layout // 使用 vue/cli 创建 vue 项目 下载依赖 npm i vue-grid-layout less less-loader@4 --s // 下载 vue-grid-layout依赖、less预处理器(两者保持版本相兼容即可) 项目样式重置 搭建 vue-grid-layout的环境 文件结构(暂定) - src |

    2024年02月09日
    浏览(35)
  • Grid 拖拽布局实现

    最近有个需求需要实现自定义首页布局,需要将屏幕按照 6 列 4 行进行等分成多个格子,然后将组件可拖拽对应格子进行渲染展示。 示例 对比一些已有的插件,发现想要实现产品的交互效果,没有现成可用的。本身功能并不是太过复杂,于是决定自己基于 vue 手撸一个简易的

    2024年02月05日
    浏览(38)
  • Vue3瀑布流(Waterfall)

    Vue2瀑布流(Waterfall)  瀑布流_百度百科 可自定义设置以下属性: 图片数组(images),类型:Array{title?: string, src: string},默认 [] 要划分的列数(columnCount),类型:number,默认 3 各列之间的间隙(columnGap),类型:number,单位px,默认 30 瀑布流区域的总宽度(width),类型:

    2023年04月22日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包