自定义右键菜单项

简介: 自定义右键菜单项


摘要:


  

我们经常在网上能够看到点击鼠标右键弹出自定义的菜单项,自定义菜单可以使菜单项的样式和网站主题相似,开发自己需要的功能。本文分享了自己制作的右键菜单项,其主要特点是,当菜单项超出浏览器的可视区域时,改变菜单项的显示位置。


/*技术难点*/


1、获取文本的高度和宽度


使用window.getComputedStyle()方法获取的是auto值,并不是具体的值,无法实现当超出可视区域换方向


解决方法:


计算每一个li元素的高度和宽度,最终宽度取最大宽度,高度是每个li元素的高度之和加上边距


2、获取浏览器可视区域的高度


使用document.body.clientHeight获取的值为0,因为页面没有内容,理所当然高度为0

解决方法:


使用document.documentElement.clientHeight来计算浏览器的可视区域高度


DEMO:



1 <!DOCTYPE html>
  2 <html>
  3 <head>
  4 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  5 <title>ContextMenu</title>
  6 <style>
  7 #contextMenu {border:1px solid rgb(219,219,219); position:absolute;margin:5px;padding: 0;}
  8 #contextMenu li{list-style:none;margin-bottom: 5px;cursor:pointer;background-color: rgb(255, 174, 0);border-bottom: 1px solid #fff;}
  9 </style>
 10 </head>
 11 <body>
 12 <script>
 13 // 菜单列表
 14 var menuList = [
 15     {
 16         'text': '刷新',
 17         'id': 'one'
 18     },{
 19         'text': '复制',
 20         'id': 'two'
 21     },{
 22         'text': '剪切',
 23         'id': 'three'
 24     },{
 25         'text': '重新加载',
 26         'id': 'four'
 27     }
 28 ];
 29 // 获取菜单的宽和高
 30 function getStyle(list) {
 31     var wid = 0,hei = 0;
 32     for(var i = 0, l = list.length; i < l; i++) {
 33         var res = textSize(list[i].text);
 34         if(res.width > wid) {
 35             wid = res.width
 36         }
 37         hei += res.height;
 38     }
 39     // 加上间距
 40     hei += i*5+20;
 41     wid += 10;
 42     return {
 43         wid: wid,
 44         hei: hei
 45     };
 46 }
 47 // 右键点击显示菜单项
 48 document.oncontextmenu=function(ev){
 49     // 如果菜单项已经存在了,则删除
 50     var ctx=document.getElementById('contextMenu');
 51     if(ctx) {
 52         ctx.parentNode.removeChild(ctx);
 53     }
 54     var menu = document.createElement('ul');
 55     menu.id = 'contextMenu';
 56     menu.style.display = 'none';
 57     for(var i = 0, l = menuList.length; i < l; i++) {
 58         var list=document.createElement("li");
 59         list.id = menuList[i].id;
 60         var node=document.createTextNode(menuList[i].text);
 61         list.appendChild(node);
 62         menu.appendChild(list);
 63     }
 64     document.getElementsByTagName('body')[0].appendChild(menu);
 65     var
 66     l,t,
 67     eve = ev||event,
 68     sty = getStyle(menuList),
 69     wid = parseInt(sty.wid),
 70     hei = parseInt(sty.hei);
 71     l = eve.clientX+wid>document.body.offsetWidth ? eve.clientX - wid : eve.clientX;
 72     t = eve.clientY+hei>document.documentElement.clientHeight ? eve.clientY - hei : eve.clientY;
 73     menu.style.left    = l+'px';    
 74     menu.style.top     = t+'px';
 75     menu.style.display = 'block';
 76     return false;
 77 }
 78 // 当点击页面时,删除菜单项
 79 window.onclick=function (event)
 80 {
 81     var ctx=document.getElementById('contextMenu');
 82     if(ctx) {
 83         if(event.target.parentNode.id === 'contextMenu') {
 84             ctx.style.display = 'none';
 85             alert('你点击了'+event.target.innerText);
 86         };
 87         ctx.parentNode.removeChild(ctx);
 88     }
 89 };
 90 function textSize(text) {
 91     //计算一段文本的真实长度和高度
 92     var span = document.createElement("span");
 93     var result    = {};
 94     result.width  = span.offsetWidth;
 95     result.height = span.offsetHeight;
 96     span.style.visibility = "hidden";
 97     document.getElementsByTagName('body')[0].appendChild(span);
 98     if(typeof span.textContent !== "undefined"){
 99         span.textContent = text;
100     } else {
101         span.innerText = text
102     }
103     result.width  = span.offsetWidth - result.width;
104     result.height = span.offsetHeight - result.height;
105     span.parentNode.removeChild(span);
106     return result;
107 }
108 </script>
109 </body>
110 </html>


相关文章
|
10月前
|
域名解析 人工智能 弹性计算
DeepSeek服务器繁忙解决方法:使用阿里云一键部署DeepSeek个人网站!
通过阿里云一键部署DeepSeek个人网站,解决服务器繁忙问题。学生用户可领取300元代金券实现0成本部署,普通用户则可用99元/年的服务器。教程涵盖从选择套餐、设置密码到获取百炼API-KEY的全流程,助您快速搭建专属大模型主页,体验DeepSeek、Qwen-max、Llama等多款模型,无需代码,最快5分钟完成部署。支持绑定个人域名,共享亲友使用,日均成本仅约1元。
964 10
|
程序员 测试技术 项目管理
项目管理中分工问题如何用RACI责任矩阵解决?
RACI责任矩阵是团队协作中的高效工具,通过明确任务中每个人的角色(负责、批准、咨询、知会),避免责任不清、重复工作和信息不对称等问题。
1041 10
项目管理中分工问题如何用RACI责任矩阵解决?
|
机器学习/深度学习 数据采集 人工智能
《人工智能如何加速药物研发进程:从新药发现到临床试验的突破》
在医药领域,药物研发复杂且成本高昂,新药推出面临诸多挑战。人工智能(AI)通过分析海量生物数据,加速靶点识别、药物分子设计及临床试验设计与数据分析,显著提升研发效率和质量。AI利用机器学习算法和大数据分析,优化试验方案,预测药物疗效与安全性,助力智能化药物研发平台的建设。尽管存在数据质量和隐私等挑战,AI仍为药物研发带来巨大潜力与创新机遇。
706 10
|
移动开发 前端开发 API
React 拖拽上传组件 Drag & Drop Upload
拖拽上传(Drag & Drop Upload)是现代文件上传方式,允许用户通过简单拖拽操作将文件上传至网页。本文介绍如何在React应用中实现拖拽上传组件,涵盖HTML5拖放API、React状态管理、组件构建及常见问题解决。包括视觉反馈、文件类型和大小限制等优化措施,确保组件的用户体验和稳定性。
530 27
|
缓存 运维 Linux
深入探索Linux内核:CPU拓扑结构探测
【10月更文挑战第18天】在现代计算机系统中,CPU的拓扑结构对性能优化和资源管理至关重要。了解CPU的核心、线程、NUMA节点等信息,可以帮助开发者和系统管理员更好地调优应用程序和系统配置。本文将深入探讨如何在Linux内核中探测CPU拓扑结构,介绍相关工具和方法。
351 0
|
JavaScript 前端开发
`addEventListener` 方法的第三个参数有什么作用?
【10月更文挑战第29天】`addEventListener` 方法的第三个参数提供了对事件传播阶段的精细控制,使开发人员能够根据具体的业务场景和需求,灵活地处理事件的触发顺序和执行逻辑,从而实现更加丰富和精确的交互效果。
|
Ubuntu Linux 数据安全/隐私保护
使用Cython库包对python的py文件(源码)进行加密,把python的.py文件生成.so文件并调用
本文介绍了在Linux系统(Ubuntu 18.04)下将Python源代码(`.py文件`)加密为`.so文件`的方法。首先安装必要的工具如`python3-dev`、`gcc`和`Cython`。然后通过`setup.py`脚本使用Cython将`.py文件`转化为`.so文件`,从而实现源代码的加密保护。文中详细描述了从编写源代码到生成及调用`.so文件`的具体步骤。此方法相较于转化为`.pyc文件`提供了更高的安全性。
1861 2
|
Linux 调度 数据库
|
JavaScript
vue3 中借助 tsx 使用 JSX (以实现字体下拉选择为例)
vue3 中借助 tsx 使用 JSX (以实现字体下拉选择为例)
282 0
|
缓存 安全 应用服务中间件
Nginx的反向代理功能有哪些应用场景呢
【8月更文挑战第22天】Nginx的反向代理功能有哪些应用场景呢
848 0