欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

关于vue.js弹窗组件的知识点总结

程序员文章站 2023-11-13 12:29:52
首先在开发时需要考虑以下三点:      1、进入和弹出的动画效果。     ...

首先在开发时需要考虑以下三点:

     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弹窗组件的一些知识点,希望对大家的学习或者工作带来一定的帮助,如果大家有疑问可以留言交流,谢谢大家对的支持。