来实现一个右键菜单吧_右键菜单栏怎么设置
zhezhongyun 2025-09-19 06:24 1 浏览
由于公司项目的需要,需要在目前视图库上加一个右键菜单,我开始觉得这还挺好搞得,因为我在这个项目得其他地方看到过类似的东西,我开始以为是一个组件来着,后面找到对应的页面一看,得,不是组件,本来想直接复制的,看到上面还绑定了一大堆事件啥的,而且点击后出现的位置还是固定的,就想着算了,还是自己写一个吧,后边用着也方便,就随便写了下。有不同的想法欢迎交流~
废话不多说,俺这就开始。首先准备好基础环境,由于公司项目是vue2的,因此我这里使用的也是vue2 + element ui。相信各位都是使用vue的高手了,因此其他方面就不赘述了。最后就是像下面这个样子。
以上这个组件总共包含了两部分,rightClickMenu 和 rightMenuItem,其中rightClickMenu是整个组件,其中有方法和属性控制整个组件的显示位置、隐藏等。rightMenuItem为放入组件的内容,相当于element的el-form-item。使用方式如下
<rightMenu v-model="showMenu" :position="menuPosition">
<rightMenuItem label="1"></rightMenuItem>
<rightMenuItem label="2" child>
<div>
<rightMenuItem label="2-1"></rightMenuItem>
<rightMenuItem label="2-2"></rightMenuItem>
</div>
</rightMenuItem>
<rightMenuItem label="3"></rightMenuItem>
<rightMenuItem label="4"></rightMenuItem>
</rightMenu>
一、创建rightClickMenu容器
新建一个组件rightClickMenu.vue,增加基本的布局代码,并且增加插槽,用来放置我们要展现的内容
<template>
<div class="right-menu">
<slot></slot>
</div>
</template>
增加对于容器的样式设置,这里我设置的定位方式是fixed,因为考虑的后续的复用问题,且使用绝对定位的话还要考虑父级,就直接相对于窗口了。具体的样式大家可以自行更改,这里只是符合了我项目的样式
<style scoped>
.right-menu {
position: fixed;
z-index: 10086;
width: 125px;
height: auto;
left: 0;
top: 0;
font-size: 16px;
text-align: center;
background: #255892;
display: flex;
flex-direction: column;
color: #fff;
justify-content: space-around;
}
基本架构搭好了,下面要让我们的大哥可以移动,那么肯定是要动态传入位置的,所以我们还要改一下,首先模板上要增加样式的绑定。就像这样。
<template>
<div
class="right-menu"
:style="{ left: position.x + 'px', top: position.y + 'px' }"
>
<slot></slot>
</div>
</template>
可以看到,我已经将组件的left 和 top值给绑定了,下面我们只要把鼠标右键点击的坐标传入就好了。那既然已经需要外部传入值了,那么还要一个接收的地方,因此增加下面这段,为组件添加名为position的props用来接收位置值。
<script>
export default {
props: {
position: {
type: Object,
default() {
return {
x: 0,
y: 0,
};
}
}
}
};
</script>
二、获取鼠标点击的位置
上面的步骤中我们的组件的一个最基本的地方就已经创建好了。下面让我们在需要使用的页面中引入它,看看效果。引入后是啥都没有的,因为没有设置组件的高度,那么就先随便的写一些东西,像这样。
<rightMenu> 你过来啊 </rightMenu>
可以看到它出现了,在我们页面的左上角。只不过这会儿没啥反应,因为我们还没有处理鼠标的右键点击事件。下面开始。由于鼠标右键会存在浏览器的默认菜单,因此我们要将这个默认行为给屏蔽掉。
- 找到你页面中的目标元素
- 添加事件监听,取消默认行为
就像这样
上图我要取消右键默认行为的元素是el-col,因此我们在它身上添加 @
contextmenu.prevent.native="rightClick($event, item)", 之后在data中添加menuPosition用来存储获取到的位置,它是一个对象,像这样 menuPosition: { x: 0, y: 0 }。 并且在当前引入的页面的methods中添加rightClick方法。如下
rightClick(e, item) {
e.preventDefault();//取消默认行为
this.menuPosition.x = e.clientX;
this.menuPosition.y = e.clientY;
}
现在我们就可以获取到鼠标右键点击的位置了,下面在组件上绑定好menuPosition。
<rightMenu :position="menuPosition"> 你过来啊 </rightMenu>
现在回到页面中,在目标元素上按下右键,可以看到组件移动了,并且出现在了我们鼠标点击的位置。
三、创建rightMenuItem
上面的步骤中我们的已经搞定了外层组件的移动,这里我们来搞定内容的呈现。同样的,先创建rightMenuItem.vue文件。因为这里公司的业务来说是有些菜单会有子菜单,目前是会有一级的子菜单,因此我这里只考虑了一级的情况。如下图,子菜单这里我使用了element的el-popover组件来实现。
先搭建基本的结构。
<template>
<div class="right-menu-item">
<span>{{ label }}</span>
</div>
</template>
<script>
export default {
props: {
label: {
type: String,
default: "",
},
},
data() {
return {
};
},
methods: {},
};
</script>
<style scoped lang='less'>
.right-menu-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 8px 5px 8px 8px;
position: relative;
cursor: pointer;
text-align: center;
&:hover {
background: #143d6c;
}
}
</style>
基本结构搭建完了,下面引入页面中,放置在rightMenu,多放几个,像这样
<rightMenu :position="menuPosition">
<rightMenuItem label="1"></rightMenuItem>
<rightMenuItem label="2"></rightMenuItem>
<rightMenuItem label="3"></rightMenuItem>
<rightMenuItem label="4"></rightMenuItem>
</rightMenu>
呈现的效果就如下面这样
好,现在一级菜单出来了,那我这里还需要一个二级菜单啊,怎么搞呢。看上面的图可以看到二级菜单的显示效果跟一级菜单其实是差不多的,区别是有子菜单时会在一级菜单后面跟一个箭头。前面说了这里使用的element的el-popover组件来实现的,所以在rightMenuItem中需要改一下。如下。
<template>
<div class="right-menu-item">
<template v-if="!child">
<span>{{ label }}</span>
</template>
<div v-if="child" style="width: 100%">
<el-popover
placement="right"
trigger="hover"
v-model="show"
:visible-arrow="false"
>
<div style="background: #255892">
<slot></slot>
</div>
<div
slot="reference"
style="
width: 100%;
display: flex;
justify-content: space-between;
align-items: center;
"
>
{{ label }} <i class="el-icon-arrow-right"></i>
</div>
</el-popover>
</div>
</div>
</template>
我在props中增加了一个属性child用来标识是否有下级,默认值是false,显示的结构用v-if指令控制。
props: {
label: {
type: String,
default: "",
},
child: {
type: Boolean,
default: false,
},
},
这一步完成后我们只需要这样使用
<rightMenuItem label="2" child>
<div>
<rightMenuItem label="2-1"></rightMenuItem>
<rightMenuItem label="2-2"></rightMenuItem>
</div>
</rightMenuItem>
就能看到在名为2的菜单多出了一个箭头,并且有了下级菜单,如下
到这里这个右键菜单已经完成一大半了。做到这里是否发现了问题呢。对的,这个右键菜单出来后就消失不了了,这很明显不是我想要的效果嘛,那么下面就来添加显示隐藏的逻辑。
四、添加显示、隐藏
最开始说了,显示隐藏是我们的rightMenuItem来进行控制的。所以相关的控制方法都写在它里面,让我们想一下啥时候需要显示隐藏呢。显示的话肯定是在右键点击相关的元素后就要显示,那隐藏呢,有以下几点:
- 点击了菜单后
- 鼠标移开菜单后
所以,就想着如何实现以上几点就好了。
那么如何控制显示和隐藏呢?一提到这个我相信各位脑海中瞬间就能出现两个指令v-if 和 v-show,的确可以用两个指令来实现,但是我就是想用自定义组件的v-model来搞,就像elementel-dialog一样的,这样可以在父页面上省那么点赋值的操作。
让我们修改一下rightMenuItem,像这样
<template>
<div
class="right-menu"
:style="{ left: position.x + 'px', top: position.y + 'px' }"
@click="menuClick"
@mouseleave="onMouseLeave"
@mouseenter="onMouseEnter"
@contextmenu="
(e) => {
e.preventDefault();
}
"
v-show="value"
>
<slot></slot>
</div>
</template>
<script>
export default {
props: {
position: {
type: Object,
default() {
return {
x: 0,
y: 0,
};
},
},
value: {
type: Boolean,
default: false,
},
},
data() {
return {
timer: null,
};
},
methods: {
emitInput(val) {
//抛出input事件用于v-model
this.$emit("input", val);
},
onMouseEnter() {
clearTimeout(this.timer);
//保持显示
this.emitInput(true);
},
onMouseLeave() {
console.log("鼠标离开");
//自动隐藏250毫秒后隐藏组件 一个简陋的防抖
if (this.timer) {
clearTimeout(this.timer);
}
this.timer = setTimeout(() => {
// this.$emit("closeMenu");
//设置为不显示
this.emitInput(false);
}, 250);
},
menuClick(e) {
this.emitInput(false);
},
},
};
上面代码中,添加了鼠标离开、进入、点击的的事件监听,移除了默认的右键菜单。在鼠标点击了或者鼠标离开后250毫秒就会隐藏自身了。 以上对于一级菜单来说是有效的,但是,由于使用了el-popover,因此在我们将鼠标移入二级菜单的时候,会触发鼠标离开事件,导致一级菜单隐藏,这是不愿意看到的。所以,我们修改一下rightMenuItem。如下调整一下
<el-popover
placement="right"
trigger="hover"
v-model="show"
:visible-arrow="false"
>
<div
style="background: #255892"
@mouseenter="$parent.onMouseEnter();"
@mouseleave="$parent.onMouseLeave()"
@contextmenu="
(e) => {
e.preventDefault();
}
"
>
<slot></slot>
</div>
<div
slot="reference"
style="
width: 100%;
display: flex;
justify-content: space-between;
align-items: center;
"
>
{{ label }} <i class="el-icon-arrow-right"></i>
</div>
</el-popover>
在它上面监听鼠标移入和离开事件,在鼠标移入时调用父组件的onMouseEnter()方法保持一级菜单的显示。由于使用了$parent,所以这里只能做到两级,因为这符合我目前的业务需求。具体的可以根据自己的业务做更改。
相关推荐
- Go语言标准库中5个被低估的强大package
-
在Go语言的世界里,开发者们往往对fmt、net/http这些“明星包”耳熟能详,却忽略了标准库里藏着的一批“宝藏工具”。它们功能强大却低调内敛,能解决并发控制、内存优化、日志管理等核心问题。今天就带...
- 作为测试人,如何优雅地查看Log日志?
-
作为一名测试工程师,测试工作中和Linux打交道的地方有很多。比如查看日志、定位Bug、修改文件、部署环境等。项目部署在Linux上,如果某个功能发生错误,就需要我们去排查出错的原因,所以熟练地掌握查...
- Java 从底层与接口实现了解String、StringBuffer、StringBuilder
-
String、StringBuffer和StringBuilder的接口实现关系:String:字符串常量,字符串长度不可变。Java中String是immutable(不可变)的。用于存放字符...
- FluentData 从入门到精通:C#.NET 数据访问最佳实践
-
简介FluentData是一个微型ORM(micro-ORM),主打「FluentAPI」风格,让开发者在保持对原生SQL完全控制的同时,享受链式调用的便捷性。它与Dapper、Massi...
- 团队协作-代码格式化工具clang-format
-
环境:clang-format:10.0.0前言统一的代码规范对于整个团队来说十分重要,通过git/svn在提交前进行统一的ClangFormat格式化,可以有效避免由于人工操作带来的代码格式问题。C...
- C# 数据操作系列 - 15 SqlSugar 增删改查详解(超长篇)
-
0.前言继上一篇,以及上上篇,我们对SqlSugar有了一个大概的认识,但是这并不完美,因为那些都是理论知识,无法描述我们工程开发中实际情况。而这一篇,将带领小伙伴们一起试着写一个能在工程中使用的模...
- Mac OS 下 Unix 使用最多的100条命令(收藏级)
-
MacOS内置基于Unix的强大终端(Terminal),对开发者、运维工程师和日常用户来说,掌握常用的Unix命令是提升效率的关键。本文整理了100条在MacOS下最常用的U...
- C语言字符串操作总结大全(超详细)
-
C语言字符串操作总结大全(超详细)1)字符串操作strcpy(p,p1)复制字符串strncpy(p,p1,n)复制指定长度字符串strcat(p,p1)附加字符串strncat...
- 经常使用到开源的MySQL,今天我们就来系统地认识一下
-
作为程序员,我们在项目中会使用到许多种类的数据库,根据业务类型、并发量和数据要求等选择不同类型的数据库,比如MySQL、Oracle、SQLServer、SQLite、MongoDB和Redis等。今...
- 电脑蓝屏代码大全_电脑蓝屏代码大全及解决方案
-
0X0000000操作完成0X0000001不正确的函数0X0000002系统找不到指定的文件0X0000003系统找不到指定的路径0X0000004系统无法打开文件0X0000005拒绝...
- 8个增强PHP程序安全的函数_php性能优化及安全策略
-
安全是编程非常重要的一个方面。在任何一种编程语言中,都提供了许多的函数或者模块来确保程序的安全性。在现代网站应用中,经常要获取来自世界各地用户的输入,但是,我们都知道“永远不能相信那些用户输入的数据”...
- css优化都有哪些优化方案_css性能优化技巧
-
CSS优化其实可以分成几个层面:性能优化、可维护性优化、兼容性优化以及用户体验优化。这里我帮你梳理一份比较系统的CSS优化方案清单,方便你参考:一、加载性能优化减少CSS文件体积压缩CSS...
- 筹划20年,他终于拍成了这部电影_筹划20年,他终于拍成了这部电影英语
-
如果提名好莱坞最难搞影星,你第一时间会联想到谁?是坏脾气的西恩·潘,还是曾因吸毒锒铛入狱的小罗伯特·唐尼,亦或是沉迷酒精影响工作的罗素·克劳?上述大咖,往往都有着这样或那样的瑕疵。可即便如此,却都仍旧...
- Keycloak Servlet Filter Adapter使用
-
KeycloakClientAdapters简介Keycloakclientadaptersarelibrariesthatmakeitveryeasytosecurea...
- 一些常用的linux常用的命令_linux常用命令有哪些?
-
在Linux的世界里,命令是与系统交互的基础。掌握常用命令不仅能让你高效地管理文件、进程和网络,还能为你进一步学习系统管理和自动化打下坚实的基础。本文将深入探讨一些最常用且功能强大的Linux...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 教程 (33)
- HTML 简介 (35)
- HTML 实例/测验 (32)
- HTML 测验 (32)
- JavaScript 和 HTML DOM 参考手册 (32)
- HTML 拓展阅读 (30)
- HTML文本框样式 (31)
- HTML滚动条样式 (34)
- HTML5 浏览器支持 (33)
- HTML5 新元素 (33)
- HTML5 WebSocket (30)
- HTML5 代码规范 (32)
- HTML5 标签 (717)
- HTML5 标签 (已废弃) (75)
- HTML5电子书 (32)
- HTML5开发工具 (34)
- HTML5小游戏源码 (34)
- HTML5模板下载 (30)
- HTTP 状态消息 (33)
- HTTP 方法:GET 对比 POST (33)
- 键盘快捷键 (35)
- 标签 (226)
- HTML button formtarget 属性 (30)
- opacity 属性 (32)
- transition 属性 (33)