如上图所示,这次要实现一个点击出现操作弹框的效果;并将这个功能封装成一个函数,便于在项目的多个地方使用。
具体思路是:
封装一个组件,组件保护一个插槽,我们可以根据不同的场景,利用插槽随意在这个弹框里插入任何元素,这个弹框显示时根据我鼠标的点击位置,定位弹窗的位置,并在组件里面监听鼠标抬起事件,触发事件时将弹窗隐藏;
接着在函数中利用createElement和appendChild方法将弹出框创建并插入到页面中;
本次实现基于vuecli3
接下来,具体实现:
首先,我们先写一个demo组件
在点击出现弹出框的元素上把事件对象数据传递一下,以便获取点击时鼠标的数据,以此确定弹出框的位置
// 文件路径参考: src > views > demo> index.vue <template> <div class="demo-wrapper"> <div class="demo-div"> <span>更多功能</span> <i class="xk-icon xk-ellipsis" @click.stop='showMenu($event)'></i> // 为了获取鼠标位置,这里把事件对象数据传递一下 </div> </div> </template> <script lang="ts"> import { Vue, Component, Prop, Watch} from "vue-property-decorator"; @Component({ }) export default class articleView extends Vue { showMenu($event:any){ // 点击时出现弹出框 } }; </script>
接着,我们把弹出框里面的组件也写一下
组件随便命名为ActionList,组件里面把把列表数据及点击事件都基于父组件传递的值而定,由于只是小demo,所以我们传递的menu数据数组只是简单的字符串数组
// 文件路径参考: src > components > ActionList > index.vue<template> <ul class="menu-wrapper"> <li class="menu-item" v-for="item in menu" :key="item" @click="handleClick(item)" > {{ item }} </li> </ul> </template> <script lang="ts"> import { Component, Prop, Vue } from 'vue-property-decorator'; @Component export default class ActionList extends Vue { @Prop() menu: string[]; handleClick(str: string) { this.$emit('click', str); } } </script>
接着,开始着手写弹框组件
1、弹框组件的显示隐藏用v-show控制,为什么不用v-if ?因为这里我监听了mouseup事件来让弹框隐藏,如果在插槽里的元素绑定事件,比如点击事件,用v-if 的话,点击插槽里的元素时,弹框先消失,插槽里的点击事件就不会生效了。
2、handleOpen事件里我们根据鼠标点击位置定位弹框位置。
// 文件路径参考: src > components > PublicModel > index.vue<template> <div class="dropdown-menu" :style="style" v-show='showModel'> <slot></slot> </div> </template> <script lang="ts"> import { Component, Prop, Vue } from 'vue-property-decorator'; interface IStyle { left"px" } this.style = style; this.showModel = true; document.addEventListener('mouseup',this.closeModel) } // 隐藏关闭此组件 closeModel(){ this.showModel = false; document.removeEventListener('mouseup', this.closeModel); } // 组件销毁生命周期 destroyed(){ document.removeEventListener('mouseup', this.closeModel); } } </script>
接着,重点来了,书写公用封装函数
我们要在demo组件点击时触发这个函数,即在demo组件里的showMenu事件触发函数,这个函数要利用createElement和appendChild方法将弹出框创建并插入到页面中。
因为是点击时创建并插入元素,所以为了性能优化,避免有恶意疯狂点击,不断创建和插入元素,我们利用throttle-debounce插件做一个节流。
先直接看代码,其他注释写在了代码里,函数名随意取:ModelFun
// 文件路径参考: src > components > PublicModel > index.ts import Vue from 'vue'; import PublicModel from './index.vue'; // 导入上面所写的弹框组件 const throttleDebounce = require('throttle-debounce'); // throttle-debounce插件 const debounce = throttleDebounce.debounce; const PublicModelConstructor = Vue.extend(PublicModel); let instance:any; const initInstance = () => { instance = new PublicModelConstructor({ el: document.createElement('div'), }); document.body.appendChild(instance.$el); } const insertInstanceSlot = (slotVNode:any, $event:any) => { // 这里两个参数一个是弹框里插槽的组件,还有就是点击的事件对象(方便定位弹框位置) if(!instance){ initInstance() } instance.$slots.default = [slotVNode]; // 将传递过来的插槽组件插入弹框组件中 instance.handleOpen($event) // 触发弹框组件(见上一段代码)的弹框获取定位信息并显示的事件 } const ModelFun = debounce(200, false, insertInstanceSlot) // 使用throttle-debounce里的debounce保证在一系列调用时间中回调函数只执行一次,这里是200毫秒 // 第二个参数为false时,在点击时会在200毫秒后再执行callback(即insertInstanceSlot),但为true时,会立即先执行一次; export default ModelFun
接着,重点来了,书写公用封装函数
我们要在demo组件点击时触发这个函数,即在demo组件里的showMenu事件触发函数,这个函数要利用createElement和appendChild方法将弹出框创建并插入到页面中。
因为是点击时创建并插入元素,所以为了性能优化,避免有恶意疯狂点击,不断创建和插入元素,我们利用throttle-debounce插件做一个节流。
先直接看代码,其他注释写在了代码里,函数名随意取:ModelFun
// 文件路径参考: src > components > PublicModel > index.tsimport Vue from 'vue'; import PublicModel from './index.vue'; // 导入上面所写的弹框组件 const throttleDebounce = require('throttle-debounce'); // throttle-debounce插件 const debounce = throttleDebounce.debounce; const PublicModelConstructor = Vue.extend(PublicModel); let instance:any; const initInstance = () => { instance = new PublicModelConstructor({ el: document.createElement('div'), }); document.body.appendChild(instance.$el); } const insertInstanceSlot = (slotVNode:any, $event:any) => { // 这里两个参数一个是弹框里插槽的组件,还有就是点击的事件对象(方便定位弹框位置) if(!instance){ initInstance() } instance.$slots.default = [slotVNode]; // 将传递过来的插槽组件插入弹框组件中 instance.handleOpen($event) // 触发弹框组件(见上一段代码)的弹框获取定位信息并显示的事件 } const ModelFun = debounce(200, false, insertInstanceSlot) // 使用throttle-debounce里的debounce保证在一系列调用时间中回调函数只执行一次,这里是200毫秒 // 第二个参数为false时,在点击时会在200毫秒后再执行callback(即insertInstanceSlot),但为true时,会立即先执行一次;export default ModelFun
最后,我们回过头来完善一下demo组件
利用vue的 $createElement 将ActionList组件插入弹框中,并将数据和事件传递给ActionList组件,这里我们传递的事件是简单的弹出我们点击的数据
// 文件路径参考: src > views > demo> index.vue<template> <div class="demo-wrapper"> <div class="demo-div"> <span>更多功能</span> <i class="xk-icon xk-ellipsis" @click.stop='showMenu($event)'></i> </div> </div> </template> <script lang="ts"> import { Vue, Component, Prop, Watch} from "vue-property-decorator"; import ActionList from "@/components/ActionList/index.vue"; import modelFun from "@/components/PublicModel/index"; @Component({ }) export default class articleView extends Vue { menuList: string[] = ['菜单1','菜单2','菜单3']; menuClick(name:string){ // 弹框里插槽的点击事件 this.$message({message:name,type:'success'}) } showMenu($event:any){ modelFun( this.$createElement( ActionList, { props: { menu:this.menuList }, on:{ click: this.menuClick, } } ), $event ) } }; </script>
至此,效果如下
最后,我们利用element ui 的 tree 组件结合我们封装的弹框看一下效果
代码:
<template> <div class="demo-wrapper"> <el-tree :data="data" node-key="id" :default-expand-all="true" :expand-on-click-node="false" show-checkbox > <div class="custom-tree-node tree-item" iv slot-scope="{ node }"> <span>{{ node.label }}</span> <span class="action" @click.stop="showMenu($event)" > <i class="el-icon-more"></i> </span> </div> </el-tree> </div> </template> <script lang="ts"> import { Vue, Component, Prop, Watch} from "vue-property-decorator"; import ActionList from "@/components/ActionList/index.vue"; import modelFun from "@/components/PublicModel/index"; @Component({ }) export default class articleView extends Vue { menuList: string[] = ['菜单1','菜单2','菜单3']; data:any[] = [{ id: 1, label: '一级 1', children: [{ id: 4, label: '二级 1-1', children: [{ id: 9, label: '三级 1-1-1' }, { id: 10, label: '三级 1-1-2' }] }] }, { id: 2, label: '一级 2', children: [{ id: 5, label: '二级 2-1' }, { id: 6, label: '二级 2-2' }] }, { id: 3, label: '一级 3', children: [{ id: 7, label: '二级 3-1' }, { id: 8, label: '二级 3-2' }] }]; menuClick(name:string){ console.log(name) this.$message({message:name,type:'success'}) } showMenu($event:any){ modelFun( this.$createElement( ActionList, { props: { menu:this.menuList }, on:{ click: this.menuClick, } } ), $event ) } }; </script>
效果:
以上就是vue实现点击出现操作弹出框的示例的详细内容,更多关于vue 弹出框的资料请关注其它相关文章!
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 【雨果唱片】中国管弦乐《鹿回头》WAV
- APM亚流新世代《一起冒险》[FLAC/分轨][106.77MB]
- 崔健《飞狗》律冻文化[WAV+CUE][1.1G]
- 罗志祥《舞状元 (Explicit)》[320K/MP3][66.77MB]
- 尤雅.1997-幽雅精粹2CD【南方】【WAV+CUE】
- 张惠妹.2007-STAR(引进版)【EMI百代】【WAV+CUE】
- 群星.2008-LOVE情歌集VOL.8【正东】【WAV+CUE】
- 罗志祥《舞状元 (Explicit)》[FLAC/分轨][360.76MB]
- Tank《我不伟大,至少我能改变我。》[320K/MP3][160.41MB]
- Tank《我不伟大,至少我能改变我。》[FLAC/分轨][236.89MB]
- CD圣经推荐-夏韶声《谙2》SACD-ISO
- 钟镇涛-《百分百钟镇涛》首批限量版SACD-ISO
- 群星《继续微笑致敬许冠杰》[低速原抓WAV+CUE]
- 潘秀琼.2003-国语难忘金曲珍藏集【皇星全音】【WAV+CUE】
- 林东松.1997-2039玫瑰事件【宝丽金】【WAV+CUE】