vue实现列表垂直无缝滚动
更新时间:2022年04月08日 09:51:11 作者:杨阳洋
这篇文章主要为大家详细介绍了vue实现列表垂直无缝滚动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
本文实例为大家分享了vue实现列表垂直无缝滚动的具体代码,供大家参考,具体内容如下
实现新闻列表的轮播(如下图)
上代码
封装的so-marquee.vue
<template> <div class="marquee-wrapper" :style="{ width: realWidth + 'px' }" > <div class="marquee-container" :style="{ height: realHeight + 'px' }" :class="className" > <ul ref="marqueeCon" :id="tooltipId" class="marquee-content" :class="{ anim: animate === true}" @mouseenter="handleStop()" @mouseleave="handleUp()" > <li v-for="(item,index) in realData" :key="`${tooltipId}-${item.id}-${index}`" class="marquee-item" :style="{ height: itemHeigth + 'px' }" @click="handleClick(item)" > <slot name="itemCon" :item="item"></slot> </li> </ul> </div> </div> </template> <script> import { parseToNum, generateId } from '@/utils/util' export default { name: "so-marquee", props: { /* * 可接受传参 * data 列表数据 * className 自定义类名 * width 列表宽度,默认值:400 * height 列表高度,默认值:200 * showNumber 可视的条目数,默认值:5 * speed 轮播速度,默认值:1000 * */ //列表数据 data: { type: Array, default: () => [], }, //自定义类名 className: String, //列表宽度,默认值:400 width: { type: [Number, String], default: 400 }, //列表高度,默认值:200 height: { type: [Number, String], default: 200 }, //可视的条目数,默认值:5 showNumber: { type: [Number, String], default: 5 }, //轮播速度,默认值:1000 speed: { type: [Number, String], default: 1000 } }, data() { return { intnum: undefined, animate: false }; }, computed: { tooltipId() { return `marquee-con-${ generateId() }`; }, realWidth() { return parseToNum(this.width) }, realHeight() { return parseToNum(this.height) }, realShowNumber() { return parseToNum(this.showNumber) }, realSpeed() { return parseToNum(this.speed) < 1000 ? 1000 : parseToNum(this.speed) }, itemHeigth() { return this.realHeight / this.realShowNumber }, realData() { return JSON.parse(JSON.stringify(this.data)) } }, mounted() { if (this.realData.length > this.realShowNumber) { this.scrollUp(); } }, methods: { scrollUp() { // eslint-disable-next-line no-unused-vars this.intnum = setInterval(_ => { this.animate = true; setTimeout(() => { this.realData.push(this.realData[0]); // 将数组的第一个元素添加到数组的 this.realData.shift(); //删除数组的第一个元素 this.animate = false; // margin-top 为0 的时候取消过渡动画,实现无缝滚动 }, this.realSpeed / 2) this.$once('hook:beforeDestroy', () => { this.cleanup() }) }, this.realSpeed); }, handleStop() { this.cleanup() }, handleUp() { this.scrollUp(); }, handleClick(row) { this.$emit('handleClick', row) }, cleanup() { if (this.intnum) { clearInterval(this.intnum); this.intnum = null; } } }, beforeDestroy() { this.cleanup(); }, deactivated() { this.cleanup(); }, watch: { animate(flag) { this.marqueeCon = this.$refs.marqueeCon if (flag) { this.marqueeCon.style.marginTop = `-${ this.itemHeigth }px` } else { this.marqueeCon.style.marginTop = 0 } }, } }; </script> <style scoped lang="scss"> .marquee-container { overflow: hidden; } .marquee-content { position: relative; } .anim { transition: all 0.5s; } .marquee-item { display: flex; align-items: center; justify-content: space-around; } </style>
parseToNum方法
export function parseToNum(value) { if (value !== undefined) { value = parseInt(value, 10) if (isNaN(value)) { value = null; } } return value }
generateId 方法
export const generateId = function() { return Math.floor(Math.random() * 10000); };
父组件调用
<template> <div id="app"> <so-marquee :data="jsonData" :height="200" :showNumber="4" :speed="500" class="my-ui-marquee" @handleClick="handleMarqueeClick" > <template v-slot:itemCon="{item}"> <div>{{ item.id }}</div> <div>{{ item.name }}</div> <div>{{ item.date }}</div> </template> </so-marquee> </div> </template> <script> import soMarquee from './components/so-marquee' export default { name: 'App', data() { return { jsonData: [ { id: 1, name: "开会通知", date: "2020-02-01" }, { id: 2, name: "放假通知", date: "2020-02-02" }, { id: 3, name: "停水通知", date: "2020-02-03" }, { id: 4, name: "停电通知", date: "2020-02-04" }, { id: 5, name: "停车通知", date: "2020-02-05" }, { id: 6, name: "奖励通知", date: "2020-02-06" }, { id: 7, name: "处分通知", date: "2020-02-07" }, { id: 8, name: "处分8通知", date: "2020-02-08" }, { id: 9, name: "处分9通知", date: "2020-02-09" }, { id: 10, name: "处分10通知", date: "2020-02-10" }, ] } }, components: { soMarquee }, methods: { handleMarqueeClick(row) { alert(`当前点击的第${row.id}行`) } } } </script> <style scoped lang="scss"> .my-ui-marquee { ::v-deep.marquee-item { cursor: pointer; } } </style>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
相关文章
electron-vue 运行报错 Object.fromEntries is not a function
Object.fromEntries() 是 ECMAScript 2019 新增的一个静态方法,用于将键值对列表(如数组)转换为对象,如果在当前环境中不支持该方法,可以使用 polyfill 来提供类似功能,接下来通过本文介绍electron-vue 运行报错 Object.fromEntries is not a function的解决方案2023-05-05vue项目启动后,js-base64依赖报错Cannot read properties
这篇文章主要介绍了vue项目启动后,js-base64依赖报错Cannot read properties of null(reading ‘replace’)问题,2024-05-05vue2中Print.js的使用超详细讲解(pdf、html、json、image)
项目中有用到打印功能,网上就找了print.js,下面这篇文章主要给大家介绍了关于vue2中Print.js使用(pdf、html、json、image)的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下2024-03-03vite+vue3项目解决低版本兼容性问题解决方案(Safari白屏)
这篇文章主要介绍了vite+vue3项目解决低版本兼容性问题(Safari白屏),使用官方插件 @vitejs/plugin-legacy 为打包后的文件提供传统浏览器兼容性支持,本文给大家介绍的非常详细,需要的朋友可以参考下2024-03-03关于iview按需引用后使用this.$Modal报错的解决
这篇文章主要介绍了关于iview按需引用后使用this.$Modal报错的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2022-09-09
最新评论