关于vue.js弹窗组件的知识点总结
首先在开发时需要考虑以下三点:
1、进入和弹出的动画效果。
2、z-index 的控制
3、overlay 遮盖层
关于动画
vue 对于动画的处理相对简单,给组件加入css transition 动画即可
<template> <div class="modal" transition="modal-scale"> <!--省略其它内容--> </div> </template> <script> // ... </script> <style> .modal-scale-transition{ transition: transform,opacity .3s ease; } .modal-scale-enter, .modal-scale-leave { opacity: 0; } .modal-scale-enter { transform: scale(1.1); } .modal-scale-leave { transform: scale(0.8); } </style>
外部可以由使用者自行控制,使用 v-if
或是 v-show
控制显示
z-index 的控制
关于z-index
的控制,需要完成以下几点
1、保证弹出框的 z-index
足够高能使 其再最外层
2、后弹出的弹出框的 z-index
要比之前弹出的要高
要满足以上两点, 我们需要以下代码实现
const zindex = 20141223 // 先预设较高值 const getzindex = function () { return zindex++ // 每次获取之后 zindex 自动增加 }
然后绑定把 z-index
在组件上
<template> <div class="modal" :style="{'z-index': zindex}" transition="modal-scale"> <!--省略其它内容--> </div> </template> <script> export default { data () { return { zindex: getzindex() } } } </script>
overlay 遮盖层的控制
遮盖层是弹窗组件中最难处理的部分, 一个完美的遮盖层的控制需要完成以下几点:
1、遮盖层和弹出层之间的动画需要并行
2、遮盖层的 z-index
要较小与弹出层
3、遮盖层的弹出时需要组件页面滚动
4、点击遮盖层需要给予弹出层反馈
5、保证整个页面最多只能有一个遮盖层(多个叠在一起会使遮盖层颜色加深)
为了处理这些问题,也保证所有的弹出框组件不用每一个都解决,所以决定利用 vue 的 mixins
机制,将这些弹出层的公共逻辑封装层一个 mixin ,每个弹出框组件直接引用就好。
vue-popup-mixin
明确了上述所有的问题,开始开发 mixin, 首先需要一个 overlay
(遮盖层组件) ;
<template> <div class="overlay" @click="handlerclick" @touchmove="prevent" :style="style" transition="overlay-fade"></div> </template> <script> export default { props: { onclick: { type: function }, opacity: { type: number, default: 0.4 }, color: { type: string, default: '#000' } }, computed: { style () { return { 'opacity': this.opacity, 'background-color': this.color } } }, methods: { prevent (event) { event.preventdefault() event.stoppropagation() }, handlerclick () { if (this.onclick) { this.onclick() } } } } </script> <style lang="less"> .overlay { position: fixed; left: 0; right: 0; top: 0; bottom: 0; background-color: #000; opacity: .4; z-index: 1000; } .overlay-fade-transition { transition: all .3s linear; &.overlay-fade-enter, &.overlay-fade-leave { opacity: 0 !important; } } </style>
然后 需要一个 js 来管理 overlay
的显示和隐藏。
import vue from 'vue' import overlayopt from '../overlay' // 引入 overlay 组件 const overlay = vue.extend(overlayopt) const getdom = function (dom) { if (dom.nodetype === 3) { dom = dom.nextelementsibling || dom.nextsibling getdom(dom) } return dom } // z-index 控制 const zindex = 20141223 const getzindex = function () { return zindex++ } // 管理 const popupmanager = { instances: [], // 用来储存所有的弹出层实例 overlay: false, // 弹窗框打开时 调用此方法 open (instance) { if (!instance || this.instances.indexof(instance) !== -1) return // 当没有遮盖层时,显示遮盖层 if (this.instances.length === 0) { this.showoverlay(instance.overlaycolor, instance.overlayopacity) } this.instances.push(instance) // 储存打开的弹出框组件 this.changeoverlaystyle() // 控制不同弹出层 透明度和颜色 // 给弹出层加上z-index const dom = getdom(instance.$el) dom.style.zindex = getzindex() }, // 弹出框关闭方法 close (instance) { let index = this.instances.indexof(instance) if (index === -1) return vue.nexttick(() => { this.instances.splice(index, 1) // 当页面上没有弹出层了就关闭遮盖层 if (this.instances.length === 0) { this.closeoverlay() } this.changeoverlaystyle() }) }, showoverlay (color, opacity) { let overlay = this.overlay = new overlay({ el: document.createelement('div') }) const dom = getdom(overlay.$el) dom.style.zindex = getzindex() overlay.color = color overlay.opacity = opacity overlay.onclick = this.handleroverlayclick.bind(this) overlay.$appendto(document.body) // 禁止页面滚动 this.bodyoverflow = document.body.style.overflow document.body.style.overflow = 'hidden' }, closeoverlay () { if (!this.overlay) return document.body.style.overflow = this.bodyoverflow let overlay = this.overlay this.overlay = null overlay.$remove(() => { overlay.$destroy() }) }, changeoverlaystyle () { if (!this.overlay || this.instances.length === 0) return const instance = this.instances[this.instances.length - 1] this.overlay.color = instance.overlaycolor this.overlay.opacity = instance.overlayopacity }, // 遮盖层点击处理,会自动调用 弹出层的 overlayclick 方法 handleroverlayclick () { if (this.instances.length === 0) return const instance = this.instances[this.instances.length - 1] if (instance.overlayclick) { instance.overlayclick() } } } window.addeventlistener('keydown', function (event) { if (event.keycode === 27) { // esc if (popupmanager.instances.length > 0) { const topinstance = popupmanager.instances[popupmanager.instances.length - 1] if (!topinstance) return if (topinstance.escpress) { topinstance.escpress() } } } }) export default popupmanager
最后再封装成一个 mixin
import popupmanager from './popup-manager' export default { props: { show: { type: boolean, default: false }, // 是否显示遮盖层 overlay: { type: boolean, default: true }, overlayopacity: { type: number, default: 0.4 }, overlaycolor: { type: string, default: '#000' } }, // 组件被挂载时会判断show的值开控制打开 attached () { if (this.show && this.overlay) { popupmanager.open(this) } }, // 组件被移除时关闭 detached () { popupmanager.close(this) }, watch: { show (val) { // 修改 show 值是调用对于的打开关闭方法 if (val && this.overlay) { popupmanager.open(this) } else { popupmanager.close(this) } } }, beforedestroy () { popupmanager.close(this) } }
使用
以上所有的代码就完成了所有弹出层的共有逻辑, 使用时只需要当做一个mixin来加载即可
<template> <div class="dialog" v-show="show" transition="dialog-fade"> <div class="dialog-content"> <slot></slot> </div> </div> </template> <style> .dialog { left: 50%; top: 50%; transform: translate(-50%, -50%); position: fixed; width: 90%; } .dialog-content { background: #fff; border-radius: 8px; padding: 20px; text-align: center; } .dialog-fade-transition { transition: opacity .3s linear; } .dialog-fade-enter, .dialog-fade-leave { opacity: 0; } </style> <script> import popup from '../src' export default { mixins: [popup], methods: { // 响应 overlay事件 overlayclick () { this.show = false }, // 响应 esc 按键事件 escpress () { this.show = false } } } </script>
总结
以上就是关于vue.js弹窗组件的一些知识点,希望对大家的学习或者工作带来一定的帮助,如果大家有疑问可以留言交流,谢谢大家对的支持。
上一篇: 火狐浏览器怎么样 火狐浏览器简介