vue封装公用弹出框方法,实现点击出现操作弹出框

如上图所示,这次要实现一个点击出现操作弹框的效果;并将这个功能封装成一个函数,便于在项目的多个地方使用。

具体思路是:

    封装一个组件,组件保护一个插槽,我们可以根据不同的场景,利用插槽随意在这个弹框里插入任何元素,这个弹框显示时根据我鼠标的点击位置,定位弹窗的位置,并在组件里面监听鼠标抬起事件,触发事件时将弹窗隐藏;

接着在函数中利用createElement和appendChild方法将弹出框创建并插入到页面中;


具体实现:

首先,我们先写一个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>

接着,开始着手写弹框组件

  • 弹框组件的显示隐藏用v-show控制,为什么不用v-if ?因为这里我监听了mouseup事件来让弹框隐藏,如果在插槽里的元素绑定事件,比如点击事件,用v-if 的话,点击插槽里的元素时,弹框先消失,插槽里的点击事件就不会生效了。
  • 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?: string;
    right?: string;
    top?: string;
    bottom?: string;
}
@Component
export default class PublicModel extends Vue {
    showModel:boolean = false;
    style:IStyle = {};

    // 组件显示时
    handleOpen($event:any){
        const { clientWidth, clientHeight, scrollWidth, scrollHeight } = document.body || document.documentElement;
        const { pageX, pageY, clientX, clientY } = $event;
        let style:IStyle = {} 
        if(clientX > (clientWidth * 2)/3 ){
            style.right = scrollWidth - pageX + 10 + 'px';
        }else{
            style.left = pageX+10+'px'
        }
        if(clientY > (clientHeight * 2) / 3 ){
            style.bottom = scrollHeight - pageY + 10 + 'px';
        }else{
            style.top = pageY + 10 + "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组件

利用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>

效果:

posted @ 2020-10-25 18:53  陈山豆  阅读(3247)  评论(0编辑  收藏  举报