Bootstrap

vue2图片预览插件

学习链接

vue插件开发实例-图片预览插件
vue2-pre-img-plugin的gitee代码

准备工作

准备图片与基础的样式

  • 将iconfont下载的字体图标资源放在src/assets/iconfont目录下
  • 将准备预览的图片放到src/static/images目录下

PrevImg.vue

在plugins/PrevImg目录下,创建PrevImg.vue。

直接引入静态图片资源,然后作为vue组件实例的data

<template>
    <div class="prev-container">
        <div class="mask"></div>
        <div class="indicator">
            <i class="iconfont icon-jiahao1"></i>
            <i class="iconfont icon-jian"></i>
            <i class="iconfont icon-reset"></i>
            <i class="iconfont icon-close"></i>
        </div>
        <div class="img-wrapper">
            <img :src="imgPath">
        </div>
    </div>
</template>

<script>
import img1 from '@/static/images/1.jpg'
import img2 from '@/static/images/2.jpg'
import img3 from '@/static/images/3.jpg'
import img4 from '@/static/images/4.png'
import img5 from '@/static/images/5.png'

console.log('PrevImg->img1',img1); // PrevImg->img1 /img/1.1a1b49d1.jpg

export default {
    name: 'PrevImg2',
    data() {
        return {
            imgPath: img1
        }
    },
}
</script>

<style lang="scss">
    .prev-container {
        position: fixed;
        top:0;
        left: 0;
        width: 100%;
        height: 100%;

        .mask {
            position: absolute;
            background-color: rgba(0,0,0,.3);
            width: 100%;
            height: 100%;
            z-index: -1;
        }

        .indicator {
            display: flex;
            justify-content: flex-end;
            padding: 20px;
            z-index: 100;
            i {
                padding: 10px;
                background-color: rgba(255,255,255,.3);
                border-radius: 50%;
                margin: 0 5px;
                cursor: pointer;
            }
        }

        .img-wrapper {
            position: absolute;
            top: 0;
            width: 100%;
            height: 100%;
            z-index: -1;

            display: flex;
            align-items: center;
            justify-content: center;
            img {
                max-height: 100%;
                max-width: 80%;
                transition: all 0.5s;
            }
        }
    }
</style>

App.vue

App.vue中使用改组件

<template>
  <div id="app">
  
     <prev-img/>
     
  </div>
</template>

<script>

import PrevImg from '@/plugins/prevImg/PrevImg'

export default {
  name: 'App',
  components: {
    PrevImg
  }
}
</script>

<style>
* {
  margin: 0;
}

body {
  display: flex;
  flex-direction: column;
}

.btn-box {
  text-align: center;
  margin-top: 20px;
}
</style>

静态效果

在这里插入图片描述

封装插件

修改PrevImg.vue

为按钮绑定相关的方法,图片的路径,将由外部传递过来

<template>
    <div class="prev-container" v-show="isShow">
        <div class="mask"></div>
        <div class="indicator">
            <i @click="operate('zoomIn')" class="iconfont icon-jiahao1"></i>
            <i @click="operate('zoomOut')" class="iconfont icon-jian"></i>
            <i @click="operate('rotate')" class="iconfont icon-reset"></i>
            <i @click="close" class="iconfont icon-close"></i>
        </div>
        <div class="img-wrapper">
            <img :src="imgPath" :style="imgStyle" alt="">
        </div>
    </div>
</template>

<script>

export default {
    name: 'PrevImg',
    components: {
    },
    data() {
        return {
            isShow: false,
            imgPath: '',
            transform: {
                rotate: 0,
                scale: 1
            }
        }
    },
    computed:{
        imgStyle() {
            let {rotate, scale} = this.transform
            return {
                transform: `scale(${scale}) rotate(${rotate}deg)`
            }
        }
    },
    methods: {
        open({url}) {
            this.isShow = true
            this.imgPath = url
        },
        close() {
            console.log('object');
            this.isShow = false
        },
        operate(command) {
            if(command == 'zoomIn') {
                this.transform.scale += 0.04
            } else if(command == 'zoomOut') {
                this.transform.scale -= 0.04
            } else if(command == 'rotate') {
                this.transform.rotate += 90
            } 
        }
    }
}
</script>

<style lang="scss">
    .prev-container {
        position: fixed;
        top:0;
        left: 0;
        width: 100%;
        height: 100%;

        .mask {
            position: absolute;
            background-color: rgba(0,0,0,.3);
            width: 100%;
            height: 100%;
            z-index: -1;
        }

        .indicator {
            display: flex;
            justify-content: flex-end;
            padding: 20px;
            z-index: 100;
            i {
                padding: 10px;
                background-color: rgba(255,255,255,.3);
                border-radius: 50%;
                margin: 0 5px;
                cursor: pointer;
            }
        }

        .img-wrapper {
            position: absolute;
            top: 0;
            width: 100%;
            height: 100%;
            z-index: -1;

            display: flex;
            align-items: center;
            justify-content: center;
            img {
                max-height: 100%;
                max-width: 80%;
                transition: all 0.5s;
            }
        }
    }
</style>

定义插件逻辑

import PrevImg from '@/plugins/prevImg/PrevImg'

export default {

    install(Vue, options) {

        let PrevImgConstructor = Vue.extend(PrevImg)
        
        let instance = new PrevImgConstructor()
        
        /* 这里会打印dom */
        console.log('instance', instance.$mount().$el);// instance <div class="prev-container" style="display: none;">...</div>
        
        /* 把组件转换成了dom,插入到了body中  */
        document.body.appendChild(instance.$mount().$el) 

        let opts = {
            open(url) {
            	// 调用组件实例上定义的方法
                instance.open({url})
            },
            close() {
                instance.close()
            }
        }

		// 挂载到vue组件实例上
        Vue.prototype.$prevImg = opts
    }
}

main.js中应用此插件

import Vue from 'vue'
import App from './App.vue'

import '@/assets/iconfont/iconfont.css'
import ElementUI from 'element-ui';
import 'element-ui/lib/theme-chalk/index.css';

import prevImg from './plugins/prevImg';

Vue.config.productionTip = false

Vue.use(ElementUI)
Vue.use(prevImg)

new Vue({
  render: h => h(App),
}).$mount('#app')

App.vue中使用此插件

<template>
  <div id="app">

    <div class="btn-box">
      <el-button @click="preview">打开预览</el-button>
    </div>

  </div>
</template>

<script>
import img1 from '@/static/images/1.jpg'
import img2 from '@/static/images/2.jpg'
import img3 from '@/static/images/3.jpg'
import img4 from '@/static/images/4.png'
import img5 from '@/static/images/5.png'

export default {
  name: 'App',
  components: {
  },
  methods: {
    preview() {
      
      // 原本测试的是把PrevImg.vue组件给引进来, 直接调用子组件的方法使用, 但
      // 是没想到插件还能这么玩的, 感觉那种方式会更好
      // this.$refs.prevImgRef.open()

      this.$prevImg.open(img4)
    }
  }
}
</script>

<style>
* {
  margin: 0;
}

body {
  display: flex;
  flex-direction: column;
}

.btn-box {
  text-align: center;
  margin-top: 20px;
}
</style>

在这里插入图片描述

图片预览列表

在这里插入图片描述

修改App.vue

<template>
  <div id="app">

    <div class="btn-box">
      <el-button @click="preview">打开预览</el-button>
    </div>

  </div>
</template>

<script>
import img1 from '@/static/images/1.jpg'
import img2 from '@/static/images/2.jpg'
import img3 from '@/static/images/3.jpg'
import img4 from '@/static/images/4.png'

export default {
  name: 'App',
  components: {
  },
  methods: {
    preview() {

      // this.$refs.prevImgRef.open()

      this.$prevImg.open(0,[img1,img2,img3,img4])
    }
  }
}
</script>

<style>
* {
  margin: 0;
}

body {
  display: flex;
  flex-direction: column;
}

.btn-box {
  text-align: center;
  margin-top: 20px;
}
</style>

修改插件逻辑

import PrevImg from '@/plugins/prevImg/PrevImg'
export default {
    install(Vue, options) {

        let PrevImgConstructor = Vue.extend(PrevImg)
        let instance = new PrevImgConstructor()
        /* 这里会打印dom */
        console.log('instance', instance.$mount().$el);// instance <div class="prev-container" style="display: none;">...</div>
        
        /* 把组件转换成了dom,插入到了body中  */
        document.body.appendChild(instance.$mount().$el) 

        let opts = {
            open(imgIdx,imgPaths) {
                instance.open({imgIdx,imgPaths})
            },
            close() {
                instance.close()
            }
        }

        Vue.prototype.$prevImg = opts
    }
}

修改PrevImg.vue

<template>
    <div class="prev-container" v-show="isShow">
        <div class="mask"></div>
        <div class="indicator">
            <i @click="operate('zoomIn')" class="iconfont icon-jiahao1"></i>
            <i @click="operate('zoomOut')" class="iconfont icon-jian"></i>
            <i @click="operate('rotate')" class="iconfont icon-reset"></i>
            <i @click="close" class="iconfont icon-close"></i>
        </div>
        <div :class="['prev-next', 'left', { 'pointer-allowd': isPrev }]" @click="operate('left')">
            <i class="iconfont icon-houtuishangyige"></i>
        </div>
        <div :class="['prev-next', 'right', { 'pointer-allowd': isNext }]" @click="operate('right')">
            <i class="iconfont icon-qianjinxiayige"></i>
        </div>
        <div ref="imgWrapper" :class="['img-wrapper']">
            <img :src="imgPaths[imgIdx]" :style="imgStyle" alt="">
        </div>
    </div>
</template>

<script>



export default {
    name: 'PrevImg',
    components: {
    },
    data() {
        return {
            isShow: false,
            imgIdx: 0,
            imgPaths: [],
            transform: {
                rotate: 0,
                scale: 1
            },
            enterAniClass: '',
        }
    },
    computed: {
        imgStyle() {
            let { rotate, scale } = this.transform
            return {
                transform: `scale(${scale}) rotate(${rotate}deg)`
            }
        },
        isNext() {
            return this.imgIdx != this.imgPaths.length - 1
        },
        isPrev() {
            return this.imgIdx != 0
        },
    },
    mounted() {
        this.$refs['imgWrapper'].addEventListener('webkitAnimationEnd', (e) => {
            console.log('动画停止了');
            this.enterAniClass = ''
        })
    },
    methods: {
        open({ imgIdx, imgPaths }) {
            this.isShow = true
            this.imgIdx = imgIdx
            this.imgPaths = imgPaths
        },
        close() {
            console.log('object');
            this.isShow = false
            this.transform = {
                rotate: 0,
                scale: 1
            }
        },
        operate(command) {
            if (command == 'zoomIn') {
                this.transform.scale += 0.04
            } else if (command == 'zoomOut') {
                this.transform.scale -= 0.04
            } else if (command == 'rotate') {
                this.transform.rotate += 90
            } else {
                this.transform = {
                    rotate: 0,
                    scale: 1
                }
                if (command == 'left') {
                if (this.imgIdx == 0) {
                    return
                }
                this.imgIdx = this.imgIdx - 1
                } else if (command == 'right') {
                    if (this.imgIdx == this.imgPaths.length - 1) {
                        return
                    }
                    this.imgIdx = this.imgIdx + 1
                }
                
            }
        }
    }
}
</script>

<style lang="scss">
.prev-container {
    position: fixed;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;

    .mask {
        position: absolute;
        background-color: rgba(0, 0, 0, .3);
        width: 100%;
        height: 100%;
        z-index: -1;
    }

    .pointer-allowd {
        cursor: pointer !important;
    }

    .prev-next {
        width: 36px;
        height: 36px;
        border-radius: 50%;
        background-color: rgba(255, 255, 255, .3);
        text-align: center;
        line-height: 36px;
        cursor: not-allowed;

        position: absolute;
        top: 0;
        bottom: 0;
        margin: auto;

        z-index: 2000;
    }

    .left {
        left: 30px;
    }

    .right {
        right: 30px;
    }

    .indicator {
        display: flex;
        justify-content: flex-end;
        padding: 20px;
        z-index: 100;

        i {
            padding: 10px;
            background-color: rgba(255, 255, 255, .3);
            border-radius: 50%;
            margin: 0 5px;
            cursor: pointer;
        }
    }

    .img-wrapper {
        position: absolute;
        top: 0;
        width: 100%;
        height: 100%;
        z-index: -1;

        display: flex;
        align-items: center;
        justify-content: center;

        img {
            max-height: 100%;
            max-width: 80%;
            transition: all 0.5s;
        }
    }
}

</style>
;