成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

關(guān)于 vue 彈窗組件的一些感想

idealcn / 658人閱讀

摘要:最近是用開發(fā)了一套組件庫(kù)在開發(fā)過(guò)程對(duì)對(duì)于組件化的開發(fā)有一些感想,于是開始記錄下這些。彈窗組件一直是開發(fā)中必備的,使用頻率相當(dāng)高,最常見的莫過(guò)于,,這些曾經(jīng)我們都會(huì)用來(lái)調(diào)試程序不同的組件庫(kù)對(duì)于彈窗的處理也是不一樣的。

最近是用 vue 開發(fā)了一套組件庫(kù) vue-carbon , 在開發(fā)過(guò)程對(duì)對(duì)于組件化的開發(fā)有一些感想,于是開始記錄下這些。

彈窗組件一直是 web 開發(fā)中必備的,使用頻率相當(dāng)高,最常見的莫過(guò)于 alert,confirm,prompt .. 這些(曾經(jīng)我們都會(huì)用alert來(lái)調(diào)試程序), 不同的組件庫(kù)對(duì)于彈窗的處理也是不一樣的。在開發(fā)時(shí)需要考慮一下三點(diǎn):

進(jìn)入和彈出的動(dòng)畫效果。

z-index 的控制

overlay 遮蓋層

關(guān)于動(dòng)畫

vue 對(duì)于動(dòng)畫的處理相對(duì)簡(jiǎn)單,給組件加入css transition 動(dòng)畫即可



外部可以由使用者自行控制,使用 v-if 或是 v-show 控制顯示

z-index 的控制

關(guān)于z-index的控制,需要完成以下幾點(diǎn)

保證彈出框的 z-index 足夠高能使 其再最外層

后彈出的彈出框的 z-index 要比之前彈出的要高

要滿足以上兩點(diǎn), 我們需要以下代碼實(shí)現(xiàn)

const zIndex = 20141223  // 先預(yù)設(shè)較高值

const getZIndex = function () {
    return zIndex++ // 每次獲取之后 zindex 自動(dòng)增加
}

然后綁定把 z-index 在組件上


overlay 遮蓋層的控制

遮蓋層是彈窗組件中最難處理的部分, 一個(gè)完美的遮蓋層的控制需要完成以下幾點(diǎn):

遮蓋層和彈出層之間的動(dòng)畫需要并行

遮蓋層的 z-index 要較小與彈出層

遮蓋層的彈出時(shí)需要組件頁(yè)面滾動(dòng)

點(diǎn)擊遮蓋層需要給予彈出層反饋

保證整個(gè)頁(yè)面最多只能有一個(gè)遮蓋層(多個(gè)疊在一起會(huì)使遮蓋層顏色加深)

為了處理這些問(wèn)題,也保證所有的彈出框組件不用每一個(gè)都解決,所以決定利用 vue 的 mixins 機(jī)制,將這些彈出層的公共邏輯封裝層一個(gè) mixin ,每個(gè)彈出框組件直接引用就好。

vue-popup-mixin

明確了上述所有的問(wèn)題,開始開發(fā) mixin, 首先需要一個(gè) overlay (遮蓋層組件) ;



然后 需要一個(gè) js 來(lái)管理 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: [],  // 用來(lái)儲(chǔ)存所有的彈出層實(shí)例
  overlay: false,
  // 彈窗框打開時(shí) 調(diào)用此方法
  open (instance) {
    if (!instance || this.instances.indexOf(instance) !== -1) return
    
    // 當(dāng)沒(méi)有遮蓋層時(shí),顯示遮蓋層
    if (this.instances.length === 0) {
      this.showOverlay(instance.overlayColor, instance.overlayOpacity)
    }
    this.instances.push(instance) // 儲(chǔ)存打開的彈出框組件
    this.changeOverlayStyle() // 控制不同彈出層 透明度和顏色
    
    // 給彈出層加上z-index
    const dom = getDOM(instance.$el)
    dom.style.zIndex = getZIndex()
  },
  // 彈出框關(guān)閉方法
  close (instance) {
    let index = this.instances.indexOf(instance)
    if (index === -1) return
    
    Vue.nextTick(() => {
      this.instances.splice(index, 1)
      
      // 當(dāng)頁(yè)面上沒(méi)有彈出層了就關(guān)閉遮蓋層
      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)

    // 禁止頁(yè)面滾動(dòng)
    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
  },
  // 遮蓋層點(diǎn)擊處理,會(huì)自動(dòng)調(diào)用 彈出層的 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

最后再封裝成一個(gè) 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"
    }
  },
  // 組件被掛載時(shí)會(huì)判斷show的值開控制打開
  attached () {
    if (this.show && this.overlay) {
      PopupManager.open(this)
    }
  },
  // 組件被移除時(shí)關(guān)閉
  detached () {
    PopupManager.close(this)
  },
  watch: {
    show (val) {
      // 修改 show 值是調(diào)用對(duì)于的打開關(guān)閉方法
      if (val && this.overlay) {
        PopupManager.open(this)
      } else {
        PopupManager.close(this)
      }
    }
  },
  beforeDestroy () {
    PopupManager.close(this)
  }
}
使用

以上所有的代碼就完成了所有彈出層的共有邏輯, 使用時(shí)只需要當(dāng)做一個(gè)mixin來(lái)加載即可





項(xiàng)目地址 vue-popup-mixin

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/90926.html

相關(guān)文章

  • vue實(shí)現(xiàn)模態(tài)框組件

    摘要:組件結(jié)構(gòu)頭部?jī)?nèi)容區(qū)域尾部操作按鈕模態(tài)框結(jié)構(gòu)分為三部分,分別為頭部?jī)?nèi)部區(qū)域和操作區(qū)域,都提供了,可以根據(jù)需要定制。調(diào)用點(diǎn)擊確定按鈕的回調(diào)處理點(diǎn)擊取消按鈕的回調(diào)處理用創(chuàng)建一個(gè)索引就很方便拿到模態(tài)框組件內(nèi)部的方法了。 基本上每個(gè)項(xiàng)目都需要用到模態(tài)框組件,由于在最近的項(xiàng)目中,alert組件和confirm是兩套完全不一樣的設(shè)計(jì),所以我將他們分成了兩個(gè)組件,本文主要討論的是confirm組件的實(shí)...

    mrcode 評(píng)論0 收藏0
  • 論如何用Vue實(shí)現(xiàn)一個(gè)彈窗-一個(gè)簡(jiǎn)單組件實(shí)現(xiàn)

    摘要:前言最近在使用框架,用到了對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多?,F(xiàn)在基本上彈窗組件都已實(shí)現(xiàn)的差不多了,還差一個(gè)彈窗的關(guān)閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多。然后就想著把這種彈窗組件的實(shí)現(xiàn)方式與大家...

    xialong 評(píng)論0 收藏0
  • 論如何用Vue實(shí)現(xiàn)一個(gè)彈窗-一個(gè)簡(jiǎn)單組件實(shí)現(xiàn)

    摘要:前言最近在使用框架,用到了對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多?,F(xiàn)在基本上彈窗組件都已實(shí)現(xiàn)的差不多了,還差一個(gè)彈窗的關(guān)閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多。然后就想著把這種彈窗組件的實(shí)現(xiàn)方式與大家...

    snowLu 評(píng)論0 收藏0
  • 論如何用Vue實(shí)現(xiàn)一個(gè)彈窗-一個(gè)簡(jiǎn)單組件實(shí)現(xiàn)

    摘要:前言最近在使用框架,用到了對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多?,F(xiàn)在基本上彈窗組件都已實(shí)現(xiàn)的差不多了,還差一個(gè)彈窗的關(guān)閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對(duì)話框組件,大致實(shí)現(xiàn)的效果,跟我之前自己在移動(dòng)端項(xiàng)目里面弄的一個(gè)彈窗組件差不太多。然后就想著把這種彈窗組件的實(shí)現(xiàn)方式與大家...

    Luosunce 評(píng)論0 收藏0
  • Vue2.0 仿滴滴出行項(xiàng)目

    摘要:仿滴滴出行項(xiàng)目最近,各大社區(qū)出現(xiàn)很多小伙伴的項(xiàng)目,趁著這股熱潮我也用擼了一個(gè)滴滴出行的項(xiàng)目??墒?,后來(lái)在手機(jī)上發(fā)現(xiàn),輸入的時(shí)候居然調(diào)不出軟鍵盤,寫項(xiàng)目的時(shí)候沒(méi)考慮到設(shè)備問(wèn)題,簡(jiǎn)直是大大的失誤。也就是說(shuō)可以在組件內(nèi)部進(jìn)行請(qǐng)求,不需要提交。 Vue2.0 仿滴滴出行項(xiàng)目 最近,各大社區(qū)出現(xiàn)很多小伙伴的vue項(xiàng)目,趁著這股熱潮我也用vue擼了一個(gè)滴滴出行的項(xiàng)目。 效果預(yù)覽 showImg(h...

    ShevaKuilin 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<