vue组件中实现鼠标右键弹出自定义菜单栏

目录

一、背景

二、步骤分析

2.1、监听鼠标右键事件

2.2、自定义弹出框

三、实现效果


一、背景

最近在写一个聊天程序,要实现点击鼠标右键弹出相关的操作菜单栏,以实现相关功能的拓展。这里主要是记录下,如何在vue项目中实现这种功能。

二、步骤分析

2.1、监听鼠标右键事件

首先,我们需要阻止原默认的鼠标右键事件,弹出我们自定义的相关菜单。这里想到可以使用JavaScript的dom监听器来实现此功能。

javascript 复制代码
// 监听快捷键事件,回调参数事件event
onMounted(
  () => {
    document.addEventListener('contextmenu', clickRightMouse)
  }
)

onUnmounted(
  () => {
    document.removeEventListener('contextmenu', clickRightMouse)
  }
)

// 鼠标右键监听
function clickRightMouse(event){
    // 阻止默认事件触发
    event.preventDefault();

    // 编写触发自定义逻辑
}

后面从网上查询了相关资料,发现vue有简写的语法糖,等效使用JavaScript监听器实现的功能实现。

html 复制代码
<!-- @contextmenu vue语法糖等效JavaScript对应的监听器 -->
<!-- .prevent vue事件修饰符,自动阻止默认事件运行 -->
<!-- handleRightClick 自定义事件运行逻辑 -->
<div @contextmenu.prevent="handleRightClick"></div>

2.2、自定义弹出框

点击鼠标右键,在光标所在位置的右侧弹出操作菜单,点击操作按钮,实现对应的操作逻辑。这里重要的是要在弹出框出现前,计算出弹出层的宽高位置。弹出框可以使用position: fiexd; 配合事件回调取出光标位置。

html 复制代码
<template>
    <div class="contactGroupList" @contextmenu.prevent="openMenu">
        <!-- 动态菜单 -->
        <teleport to=".contactGroupList" :defer="true">
          <div
            v-show="isShowMenu"
            class="context-menu"
            :style="{ top: `${mousePosition.y}px`, left: `${mousePosition.x}px` }"
          >
            <ul>
              <li @click="handleAction('addGroup')">
                <el-icon><Plus /></el-icon>添加分组
              </li>
              <li @click="handleAction('deleteGroup')">
                <el-icon><DeleteFilled /></el-icon>删除分组
              </li>
              <li @click="handleAction('manageGroup')">
                <el-icon><EditPen /></el-icon>重命名分组
              </li>
            </ul>
          </div>
        </teleport>
    <div>
</template>


<script setup>
import { ref, reactive, markRaw, onBeforeMount, watch } from 'vue'


// 右键打开自定义菜单
const isShowMenu = ref(false)
const mousePosition = ref({ x: 0, y: 0 })

const openMenu = (e) => {
  mousePosition.value = {
    x: e.clientX,
    y: e.clientY,
  }
  isShowMenu.value = true
  document.addEventListener('click', closeMenu)
}

const closeMenu = () => {
  isShowMenu.value = false
  document.removeEventListener('click', closeMenu)
}

const handleAction = (action) => {
  alert(`执行操作:${action}`)
  closeMenu()
}
</script>

<style lang='scss' scoped>
.contactGroupList {
    width: 100%;
    height: 100%:
    display: flex;
    flex-direction: column;

    // 联系人鼠标右键弹出菜单样式(弹出框挂载在contactList下的)
    .context-menu {
      position: fixed;
      background: white;
      box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
      border-radius: 4px;
      z-index: 1000;
      font-size: 14px;
    }
    .context-menu ul {
      list-style: none;
      padding: 0;
      margin: 0;
      min-width: 120px;
    }
    .context-menu li {
      padding: 8px 12px;
      cursor: pointer;
      line-height: 100%;
      display: flex;
      flex-direction: row;
      gap: 5px;
    }
    .context-menu li:hover {
      background: #f5f5f5;
    }
}
</style>

三、实现效果

相关推荐
2501_920931701 小时前
React Native鸿蒙跨平台实现推箱子游戏,完成玩家移动与箱子推动,当所有箱子都被推到目标位置时,玩家获胜
javascript·react native·react.js·游戏·ecmascript·harmonyos
AI老李2 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
方也_arkling2 小时前
Element Plus主题色定制
javascript·sass
晓晓莺歌2 小时前
vue3某一个路由切换,导致所有路由页面均变成空白页
前端·vue.js
2601_949809593 小时前
flutter_for_openharmony家庭相册app实战+我的Tab实现
java·javascript·flutter
Up九五小庞3 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源
摘星编程3 小时前
React Native + OpenHarmony:UniversalLink通用链接
javascript·react native·react.js
qq_177767373 小时前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos
烬头88214 小时前
React Native鸿蒙跨平台应用实现了onCategoryPress等核心函数,用于处理用户交互和状态更新,通过计算已支出和剩余预算
前端·javascript·react native·react.js·ecmascript·交互·harmonyos
程序员清洒5 小时前
Flutter for OpenHarmony:Text — 文本显示与样式控制
开发语言·javascript·flutter