詳解用vue編寫彈出框組件
前言
最近研究了用vue編寫彈出框的組件,發(fā)現(xiàn)其實這里面的門道還是有很多的。這篇文完全是用來記錄總結下最近的學習成果,同時也希望能夠幫得上正在學習糾結的你~ps:本文假設你已經(jīng)了解vue2.0相關框架,因此適合有一定vue2.0基礎的同學閱讀。
設計組件的思考
其實單純的編寫一個彈出框組件并不難,寫一個模板,然后用v-if或者v-show指令還控制組件的出現(xiàn)與消失。真正困擾我的是,這個組件的調用方式,這個問題糾結了我好久。
調研了下資料,有些人建議,直接把組件標簽插進模板中,然后通過直接控制組件的顯示隱藏來控制組件。這樣寫有好處,就是結構清晰,一目了然,人家一看你的代碼就知道你這個頁面可能會有彈出框,并且編寫的組件就更容易,只需關注內部方法就好了,也不存在事件調用的困擾,維護起來也特別容易。但是缺點也很明顯,如果有多個彈窗,并且不知道會有幾個彈窗的情況下,感覺就不太好做,并且這種提前寫模板的形式,難免會在不彈窗的時候要下載一些js文件,有可能會造成性能浪費。
也有些人建議,在寫好的彈出框組件之外再做一層封裝,通過動態(tài)調用的方式來控制彈出框的顯示與隱藏。這樣寫的好處是不用事先在模板里面寫好該組件的標簽,只需要在想調用的地方調用下該組件,就實現(xiàn)了按需使用的目的,符合之前傳統(tǒng)前端框架的編碼習慣。缺點就是感覺代碼寫起來比較復雜,層層嵌套,并且感覺這個與MVVM模式的狀態(tài)驅動界面的思想相違背。
于是我天秤座的糾結病犯了,在選擇哪種技術方案的問題上,思考了很久。但是網(wǎng)上搜了很多,發(fā)現(xiàn)還是后一種實現(xiàn)方法用的人比較多。后來我又研究了了elementUI和iView的彈出框組件,他們也是沿用的后一種方法,想了一下后一種方法雖然代碼易讀性不強,但是它真正模擬了瀏覽器默認的alert事件,在用戶需要的地方來調用,一方面節(jié)省了代碼量,另一方面也很容易解決多個彈窗的情況。最后還是決定用這種模式寫一個簡單的彈出框組件。主要是體會這其中的機理。廢話不多說,來上干貨了。有啥不對的地方還請大家多多指教。(ps:對于天秤座的我,雖然選擇了后一種方法,但是內心還是鐘愛第一種方法,并且后一種方法并沒有足夠的理由說服我呀,不知道哪位有識之士能夠幫忙點醒一下我,晚輩感激不盡)。
alert組件設計
單獨的設計alert彈出框的邏輯是很簡單的,我就直接上代碼了:
<template> <transition name='fade'> <div class="alert" v-if="showAlert"> <div class="wrap"> <div class="head">{{title}}</div> <div class="body"> <slot> <p>{{message}}</p> </slot> </div> <div class="foot"> <div v-if="type === 'confirm'"> <button class="btn-base" @click="sure">確定</button> <button class="btn-gray" @click="cancel">取消</button> </div> <div v-else-if="type === 'inform'"> <button class="btn-base" @click="cancel">知道了</button> </div> </div> </div> </div> </transition> </template> <script> export default { name: 'alert', data() { return { showAlert: false, }; }, props: { title: { type: String, default: '提示', }, message: { type: String, }, type: { // 可以有confirm, 和inform兩個類型 type: String, default: 'confirm', validator(value) { return value === 'confirm' || value === 'inform'; }, }, sureBtn: { type: Function, }, cancelBtn: { type: Function, }, context: { type: Object, }, }, methods: { cancel() { if (this.cancelBtn) { this.cancelBtn.apply(this.context); } this.close(); }, sure() { if (this.sureBtn) { this.sureBtn.apply(this.context); } this.close(); }, show() { this.showAlert = true; }, close() { this.showAlert = false; }, }, }; </script> <!-- Add "scoped" attribute to limit CSS to this component only --> <style scoped lang='scss'> .alert { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0, 0.8); z-index: 1000; transition: all .3s ease-in-out; } .wrap { position: absolute; z-index: 1002; min-width: 400px; background: #fff; left: 50%; top: 50%; transform: translate(-50%, -50%); border-radius: 4px; } .head { height: 40px; line-height: 40px; border-bottom: 1px solid #dedede; padding-left: 10px; color: #333; } .body { padding: 40px 20px; text-align: center; } .foot { height: 50px; text-align: center; button { margin-right: 20px; &:last-child { margin-right: 0; } } } </style>
這里只是寫了簡單的功能,并沒有考慮更復雜的情況,比如按鈕顏色定制,大小定制,z-index層級的考慮,遮罩層的統(tǒng)一管理等等,只是為了掌握編寫彈出框的主要思想,所以沒有寫太多的情況。這里只細分了是確認框還是通知框,可以定制彈出框的內容、標題等一些簡單的常規(guī)操作。
其實這個組件寫好,就可以在頁面用起來了,直接在對應頁面插入這段,可以也可以用:
<!--template--> <button @click="showAlert">點我</button> <alert ref="alert">我是一個確認框</alert> <!--javascript--> ... methods: { showAlert() { this.$refs.alert.show(); } } ...
當然,如果真要這么用,這個組件還是需要修改一些東西的,比如事件拋出,當點擊確定或者取消按鈕的時候,需要emit對應的事件,以提供給父組件捕獲,并做相應的處理。
動態(tài)插入到頁面中
為了能讓組件動態(tài)的插入到頁面中,需要對上面的組件進行封裝,利用Vue.extend機制,可以輕松的做到這種封裝,直接上代碼:
import Vue from 'vue'; import alert from './alert'; const AlertConstructor = Vue.extend(alert); const div = document.createElement('div'); AlertConstructor.show = (options) => { document.body.appendChild(div); options.type = 'inform'; const propsData = Object.assign({}, options); const alertInstance = new AlertConstructor({ propsData, }).$mount(div); alertInstance.show(); }; AlertConstructor.confirm = (options) => { document.body.appendChild(div); options.type = 'confirm'; const propsData = Object.assign({}, options); const alertInstance = new AlertConstructor({ propsData, }).$mount(div); alertInstance.show(); }; export default AlertConstructor;
這里,show對應的是通知框,confirm對應的是確認框。我知道這種封裝有點簡單了,有很多情況沒有考慮,比如有多個彈出框時的處理等。這里只是做了簡單的封裝,為的就是讓大家明白此種封裝主要思路是什么。
總結
這篇文章僅僅是對自己這幾天摸索彈出框組件問題的一個簡短的總結與思考。可能還不是很成熟,當做拋磚引玉吧,歡迎大家多多提意見。希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
如何解決element-ui動態(tài)加載級聯(lián)選擇器默認選中問題
這篇文章主要介紹了如何解決element-ui動態(tài)加載級聯(lián)選擇器默認選中問題,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的朋友可以參考一下2022-09-09Vue中的transition封裝組件的實現(xiàn)方法
這篇文章主要介紹了Vue中的transition封裝組件的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-08-08