<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
不知道從哪個版本開始發現,element-plus的message-box在有圖示的時候,錯位比較嚴重,f12跟官網的樣式對比後發現,好傢伙!原來position: absolute被覆蓋了。
錯位效果截圖
ElementPlus官網程式碼截圖
本地專案程式碼截圖
可以看出在本地中el-message-box__status樣式的position並未生效
修改css樣式
.el-message-box__status { position: absolute !important; }
完成效果截圖
Vue 中動態顯示一個元件,無非就是通過 h 函數建立 VNode,並且把這個 VNode 掛載在 DOM 樹上。這裡有兩種掛載的方式:
createApp
在 main.js 中建立 Vue 範例用的就是這種方法,這也是 Vue3 中代替 Vue2 的 Vue.extend 的方法,簡單使用案例如下:
const app = createApp(MessageBox, { message: 'hello?' }) // 建立無父元素的檔案物件,掛載到 DOM 中後直接呼叫 app.unmount() 移除 MessageBox // 和掛載到 div 的區別是 MessageBox 不會作為 div 的子元素 const frg = document.createDocumentFragment() // app.mount 返回元件範例 // 元件範例內包含 expose 出來的方法或者資料 const vm = app.mount(frg) document.body.appendChild(frg)
h + render
和 createApp 方法大同小異
const vn = h(MessageBox, { message: 'vnode' }) render(vn, container) document.body.appendChild(container)
可以看到無論是 createApp 方法還是 h 方法,都可以在第二個引數中傳入元件的 props,於是我們可以封裝一個動態顯示元件的函數,這個函數接受元件的 props。但是在封裝函數之前,讓我們先來實現
MessageBox 這個元件
直接貼程式碼,講下最關鍵的幾點:
進入退出動態效果實現
設定一個 transition flag,初始時為 false,元件 mounted 後 flag 為 true。
全域性遮罩層
一個 fixed 定位寬高為100%的 div。
剩下的主要看 script 部分
<template> <transition name="messagebox-fade" @after-leave="onDestroy"> <div @click="setVisiable(false)" v-show="visiable" class="z-50 flex items-center justify-center fixed top-0 left-0 w-full h-full bg-dark-50/50"> <div @click.stop class="transform -translate-y-1/2 p-2 rounded min-w-3/7 bg-white"> <p class="text-sm text-gray-600 font-light"> {{ title }}</p> <p class="my-4 text-lg"> <content-view :type="type"></content-view> </p> <div class="w-full flex justify-end items-center"> <button @click="okBtnClicked" class="btn btn-primary"> {{ ok }}</button> <button @click="cancelBtnClicked" v-if="cancel" class="ml-2 btn btn-danger"> {{ cancel }}</button> </div> </div> </div> </transition> </template>
<script setup> import { ref, onMounted, h } from 'vue' const { onOK, onCancel, message } = defineProps({ title: { type: String, default: '提示' }, message: { type: String, default: '' }, type: { type: String, validator: (value) => { return ['confirm', 'prompt'].includes(value) } }, ok: { type: String, default: 'OK' }, cancel: { type: String, default: '' }, onDestroy: Function, onOK: Function, onCancel: Function }) const promptContent = ref('') const ContentView = ({ type }) => { switch (type) { case (!type || 'confirm'): return h('p', null, message) case 'prompt': return h('input', { class: 'messagebox-input', onInput: (e) => promptContent.value = e.target.value }) } } const visiable = ref(false); const setVisiable = (vis) => { visiable.value = vis; } const okBtnClicked = () => { setVisiable(false); onOK(promptContent.value) } const cancelBtnClicked = () => { setVisiable(false) onCancel() } onMounted(() => { setVisiable(true); }) </script>
<style scoped> .btn { @apply outline-gray-100 border-none p-1 rounded bg-warm-gray-200 } .btn-primary { @apply bg-sky-300 } .messagebox-input { @apply border rounded border-light-900 outline-none w-full py-1 px-2 text-lg } .messagebox-fade-enter-from, .messagebox-fade-leave-to { @apply opacity-0 } .messagebox-fade-enter-active, .messagebox-fade-leave-active { @apply transition-opacity } </style>
函數式元件
// 第一個引數是 props,第二個引數是 context,類似 setup 的引數 // 返回值為 VNode // 可以匯出或者直接在元件內部使用 const ContentView = ({ type }) => { switch (type) { case (!type || 'confirm'): return h('p', null, message) case 'prompt': return h('input', { class: 'messagebox-input', onInput: (e) => promptContent.value = e.target.value }) } }
import { createApp } from 'vue' import MessageBoxCpn from './MessageBox.vue' const fields = ['confirm', 'prompt'] export default function MessageBox(options) { return new Promise((resolve, reject) => { const app = createApp(MessageBoxCpn, { ...options, onDestroy: () => { app.unmount() }, onOK: (value) => { resolve(value) }, onCancel: () => { reject() } }) const frg = document.createDocumentFragment() app.mount(frg) document.body.appendChild(frg) }) } fields.forEach(field => { MessageBox[field] = (options) => { options.type = field return MessageBox(options) } })
通過元件的 props 傳入回撥,實現按鈕點選事件的傳遞、MessageBox 關閉時取消掛載的操作。
另外可以通過 MessageBox.prompt 等靜態方法直接呼叫對應 type 的 MessageBox,實現方式是在 MessageBox 上掛上對應的靜態方法,並且覆蓋 options 的 type 選項。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45