一个基于 vue 的图片轮播组件的实现 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
zhangxiaoshang66
V2EX    前端开发

一个基于 vue 的图片轮播组件的实现

  •  
  •   zhangxiaoshang66 2018-05-02 09:02:48 +08:00 3017 次点击
    这是一个创建于 2750 天前的主题,其中的信息可能已经有所发展或是发生改变。

    1. 在线 DEMO

    http://va-carousel.xiaoshang.online

    Github

    2. 首先是一张思维导图

    导图

    3. 然后是以上属性的标注说明

    标注

    4. 代码层

    4.1 除去可从父组件接收的属性,组件自身有以下属性:

    data() { return { list: [], // 当前显示的图片列表 hover: false, // 鼠标是否悬浮在组件上 timer: null, // 自动切换的 setInterval itemWidth: '',// 每项元素的宽度 isReverse: false // 是否是反向切换,触发 prev 时,该属性为 true } } 

    4.2 组件挂载之前

    1. 计算每项元素的宽度,即 itemWidth 的值
    2. 初始化显示的图片列表,即 list。这里存储的数据才是真正会被在页面上渲染的。每次切换,实际上就是修改该 list 中的数据,对应的视图会自动更新,数据驱动视图嘛。
    beforeMount() { this.itemWidth = 100 / this.total + '%' this.list = this.items.slice(0, this.total) }, 

    4.3 组件挂在后,检查 autoplay 属性,若该属性为 true,则产生一个计时器

    mounted() { if (this.autoplay) { this.startTimer() } }, 

    4.4 startTimer 函数很简单,就是间隔一定时间触发一次 next(向后)切换

    // 开始计时器 startTimer() { if (!this.interval || this.interval <= 0) { return } this.timer = setInterval(this.next, this.interval) } 

    4.5 next 函数

     // 下一张 next() { // 如果图片列表小于需要显示的数量,则不进行滚动 if (this.items.length < this.total) { return } // 向后追加一个元素,该元素为: // 显示列表中最后一个元素在原数组中的后一个元素 // 如果已经是最后一个元素,则使用第一个元素 let indexOfItems = this.items.findIndex( item => item.id === this.list[this.list.length - 1].id ) if (indexOfItems === this.items.length - 1) { // 使用第一个元素 this.list.push(this.items[0]) } else { // 使用后一个元素 this.list.push(this.items[indexOfItems + 1]) } // 移除当前显示图片中的第一个 this.list.shift() this.isReverse = false }, 

    4.6 对应的还有一个 prev 函数,与 next 函数逻辑相反,这里就不展示代码了

    4.7 点击图片时,向父组件释放事件 selectedItem,传递两个参数 item 和 index 分别为当前点击的对象,和该对象在 list 中的位置

     // 点击图片 selectedItem(item, index) { this.$emit('selectedItem', item, index) }, 

    4.8 鼠标悬浮在组件上时,停止自动切换(若 autoplay 为 ture ), 鼠标离开时,继续切换

     handleMouseEnter() { this.hover = true this.pauseTimer() }, handleMouseLeave() { this.hover = false if (this.autoplay) { this.startTimer() } }, 

    4.9 然后是过渡效果的实现

    因为 arrow 元素也在 transition-group 中,所以当 arrow=‘ hover ’时,arrow 的显示、隐藏也会触发钩子函数,但是我们的钩子函数是针对 image-item 写的,所以需要在函数中检测是哪个元素触发的,这里通过检查 className 进行判断。 然后针对向前、向后两种情况设置不同的样式

     beforeEnter(el) { // 只对 image-item 使用过渡 let isImageItem = el.className.indexOf('image-item') > -1 if (isImageItem) { el.style.opacity = 0 if (this.isReverse) { el.style.transform = 'translateX(-100%)' } else { el.style.transform = 'translateX(100%)' } } } 

    4.10 这里使用了 Velocity,这是一个实现动画效果的 js 库,之所以使用这个库是因为试了 n 种方案都没能实现预期效果 emm

    enter(el, done) { // 只对 image-item 使用过渡 let isImageItem = el.className.indexOf('image-item') > -1 if (isImageItem) { Velocity(el, { opacity: 1, translateX: '0px' }, { complate: done }) } else { done() } } 

    4.11 然后是对应的 beforeLeave、leave 函数,这里就不展示了

    以上基本就是所有 js 部分,整体感受就是,一旦实现逻辑搞清楚,代码实现起来还是挺容易的,然后就是框架的熟悉程度。

    5. npm 包发布

    这本是工作业务中的一个功能需求,因为没能在网上找到现成的轮子,找个差不多效果的领导不满意,所以只能自己写了,做都做了不发出来感觉白写了. . .

    npm 发布流程简单概括就是

    1.注册 去 npm 官网注册个账号

    2.生成 npm 包
    文件夹中有 package.json 文件就是一个 npm 包

    3.在终端使用 npm publish 发布包,成功之后,该项目文件夹下所有文件都会上传至 npm 官网,当用户使用 npm install 安装后,就会将整个文件夹下载至 node_modules 文件夹中,对于这个项目,本是一个使用 vue-cli 生成的 vue 项目,组件路径src/components/VaCarousel.vue,所以使用 npm install va-carousel 安装之后,只需要在项目中像这样导入即可使用(前提是你的项目也是使用 vue-cli 生成的,对于其他方式搭建的项目可能会出现一些错误):import VaCarousel from 'va-carousel/src/components/VaCarousel.vue'

    以上

    11 条回复    2018-05-02 13:46:30 +08:00
    newbieo0O
        1
    newbieo0O  
       2018-05-02 09:11:17 +08:00
    请使用原生JS写一遍 :D
    aaronlam
        2
    aaronlam  
       2018-05-02 09:21:46 +08:00
    学习了
    GG668v26Fd55CP5W
        3
    GG668v26Fd55CP5W  
       2018-05-02 09:23:01 +08:00 via iPhone
    demo 图片成条状了,我在 iPhone Safari 打开的。
    v21an
        4
    v21an  
       2018-05-02 11:31:54 +08:00 via Android
    zhangxiaoshang66
        5
    zhangxiaoshang66  
    OP
       2018-05-02 12:10:46 +08:00
    @falcon05 针对 PC 端写的 emm
    zhangxiaoshang66
        6
    zhangxiaoshang66  
    OP
       2018-05-02 12:11:06 +08:00
    @newbieo0O 和自己过不去吗
    rabbbit
        7
    rabbbit  
       2018-05-02 12:37:29 +08:00
    跟自己过不去...
    lamada
        8
    lamada  
       2018-05-02 13:28:40 +08:00
    动画优化下,不要做成切换时卡顿的感觉
    zhangxiaoshang66
        9
    zhangxiaoshang66  
    OP
       2018-05-02 13:38:48 +08:00
    @lamada 卡顿使网络加载图片慢的问题,跑一会,图片都缓存下来就不会卡了,或者换成小图片会好一些
    zicla
        10
    zicla  
       2018-05-02 13:41:44 +08:00
    如果能做到 Swiper 丝般润滑就厉害了
    zhangxiaoshang66
        11
    zhangxiaoshang66  
    OP
       2018-05-02 13:46:30 +08:00
    @zhangxiaoshang66 优化的空间还是有的
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     1363 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 26ms UTC 17:05 PVG 01:05 LAX 09:05 JFK 12:05
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86