摘要:最近是用開發了一套組件庫在開發過程對對于組件化的開發有一些感想,于是開始記錄下這些。彈窗組件一直是開發中必備的,使用頻率相當高,最常見的莫過于,,這些曾經我們都會用來調試程序不同的組件庫對于彈窗的處理也是不一樣的。
最近是用 vue 開發了一套組件庫 vue-carbon , 在開發過程對對于組件化的開發有一些感想,于是開始記錄下這些。
彈窗組件一直是 web 開發中必備的,使用頻率相當高,最常見的莫過于 alert,confirm,prompt .. 這些(曾經我們都會用alert來調試程序), 不同的組件庫對于彈窗的處理也是不一樣的。在開發時需要考慮一下三點:
關于動畫進入和彈出的動畫效果。
z-index 的控制
overlay 遮蓋層
vue 對于動畫的處理相對簡單,給組件加入css transition 動畫即可
外部可以由使用者自行控制,使用 v-if 或是 v-show 控制顯示
z-index 的控制關于z-index的控制,需要完成以下幾點
保證彈出框的 z-index 足夠高能使 其再最外層
后彈出的彈出框的 z-index 要比之前彈出的要高
要滿足以上兩點, 我們需要以下代碼實現
const zIndex = 20141223 // 先預設較高值 const getZIndex = function () { return zIndex++ // 每次獲取之后 zindex 自動增加 }
然后綁定把 z-index 在組件上
overlay 遮蓋層的控制
遮蓋層是彈窗組件中最難處理的部分, 一個完美的遮蓋層的控制需要完成以下幾點:
遮蓋層和彈出層之間的動畫需要并行
遮蓋層的 z-index 要較小與彈出層
遮蓋層的彈出時需要組件頁面滾動
點擊遮蓋層需要給予彈出層反饋
保證整個頁面最多只能有一個遮蓋層(多個疊在一起會使遮蓋層顏色加深)
為了處理這些問題,也保證所有的彈出框組件不用每一個都解決,所以決定利用 vue 的 mixins 機制,將這些彈出層的公共邏輯封裝層一個 mixin ,每個彈出框組件直接引用就好。
vue-popup-mixin明確了上述所有的問題,開始開發 mixin, 首先需要一個 overlay (遮蓋層組件) ;
然后 需要一個 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來加載即可
項目地址 vue-popup-mixin
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/90926.html
摘要:組件結構頭部內容區域尾部操作按鈕模態框結構分為三部分,分別為頭部內部區域和操作區域,都提供了,可以根據需要定制。調用點擊確定按鈕的回調處理點擊取消按鈕的回調處理用創建一個索引就很方便拿到模態框組件內部的方法了。 基本上每個項目都需要用到模態框組件,由于在最近的項目中,alert組件和confirm是兩套完全不一樣的設計,所以我將他們分成了兩個組件,本文主要討論的是confirm組件的實...
摘要:前言最近在使用框架,用到了對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。現在基本上彈窗組件都已實現的差不多了,還差一個彈窗的關閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。然后就想著把這種彈窗組件的實現方式與大家...
摘要:前言最近在使用框架,用到了對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。現在基本上彈窗組件都已實現的差不多了,還差一個彈窗的關閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。然后就想著把這種彈窗組件的實現方式與大家...
摘要:前言最近在使用框架,用到了對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。現在基本上彈窗組件都已實現的差不多了,還差一個彈窗的關閉事件,這里就涉及到子組件往父組件傳參了。 前言 最近在使用element-ui框架,用到了Dialog對話框組件,大致實現的效果,跟我之前自己在移動端項目里面弄的一個彈窗組件差不太多。然后就想著把這種彈窗組件的實現方式與大家...
摘要:仿滴滴出行項目最近,各大社區出現很多小伙伴的項目,趁著這股熱潮我也用擼了一個滴滴出行的項目。可是,后來在手機上發現,輸入的時候居然調不出軟鍵盤,寫項目的時候沒考慮到設備問題,簡直是大大的失誤。也就是說可以在組件內部進行請求,不需要提交。 Vue2.0 仿滴滴出行項目 最近,各大社區出現很多小伙伴的vue項目,趁著這股熱潮我也用vue擼了一個滴滴出行的項目。 效果預覽 showImg(h...
閱讀 2743·2021-10-26 09:50
閱讀 2385·2021-10-11 11:08
閱讀 2128·2019-08-30 15:53
閱讀 1906·2019-08-30 15:44
閱讀 2382·2019-08-28 18:12
閱讀 2519·2019-08-26 13:59
閱讀 2852·2019-08-26 12:19
閱讀 2751·2019-08-26 12:09