加入收藏 | 设为首页 | 会员中心 | 我要投稿 财气旺网 - 财气网 (https://www.caiqiwang.com/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 综合聚焦 > 编程要点 > 语言 > 正文

用vue怎样完成可操作的弹 出框效果?

发布时间:2022-11-14 11:18:30 所属栏目:语言 来源:
导读:  本文主要给大家分享的是用vue实现可操作的弹出框效果,实现效果如下,下文还有具体的实现思路及实现代码,感兴趣的朋友可以参考了解看看,那么接下来就跟随小编来学习一下吧。

  这次要实现一个点击出现操作
  本文主要给大家分享的是用vue实现可操作的弹出框效果,实现效果如下,下文还有具体的实现思路及实现代码,感兴趣的朋友可以参考了解看看,那么接下来就跟随小编来学习一下吧。

  这次要实现一个点击出现操作弹框的效果;并将这个功能封装成一个函数,便于在项目的多个地方使用。
  
  具体思路是:
  
    封装一个组件,组件保护一个插槽,我们可以根据不同的场景,利用插槽随意在这个弹框里插入任何元素,这个弹框显示时根据我鼠标的点击位置,定位弹窗的位置,并在组件里面监听鼠标抬起事件,触发事件时将弹窗隐藏;
  
  接着在函数中利用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?: 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组件点击时触发这个函数,即在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>
 
 

(编辑:财气旺网 - 财气网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!