Vue中實現(xiàn)動態(tài)右鍵菜單的示例代碼
在前端開發(fā)中,自定義右鍵菜單是一種常見的交互方式,它能夠為用戶提供更多的功能選項。在本文中,將探討如何在 Vue 中實現(xiàn)一個動態(tài)右鍵菜單,該菜單能夠根據(jù)用戶點擊位置動態(tài)調(diào)整其顯示位置,確保菜單始終在瀏覽器窗口的可視區(qū)域內(nèi)。
實現(xiàn)目標
- 右鍵點擊頁面時顯示自定義菜單。
- 菜單根據(jù)點擊位置動態(tài)定位。
- 確保菜單不會超出瀏覽器窗口的可視區(qū)域。(超出窗口頂部或者底部優(yōu)化)
實現(xiàn)步驟
1. 創(chuàng)建 Vue 組件模板
首先,編寫 Vue 組件的模板部分:
<template> <div v-if="visible"> <a-menu class="contextmenu" :style="style" @click="handleClick"> <a-menu-item v-for="item in list" :key="item.key"> <span>{{ item.text }}</span> </a-menu-item> </a-menu> </div> </template>
在這個模板中,使用 v-if
指令控制菜單的顯示與隱藏,并使用 :style
綁定菜單的動態(tài)樣式。
2. 編寫組件腳本
接下來是組件的腳本部分:
<script> export default { name: "ContextMenu", props: { visible: { type: Boolean, default: false, }, list: { type: Array, required: true, default: () => [], }, }, data() { return { left: 0, top: 0, target: null, }; }, computed: { style() { return { left: this.left + "px", top: this.top + "px", }; }, }, created() { const clickHandler = () => this.closeMenu(); const contextMenuHandler = (e) => { e.preventDefault(); this.setPosition(e); }; window.addEventListener("click", clickHandler); window.addEventListener("contextmenu", contextMenuHandler); this.$emit("hook:beforeDestroy", () => { window.removeEventListener("click", clickHandler); window.removeEventListener("contextmenu", contextMenuHandler); }); }, methods: { closeMenu() { this.$emit("update:visible", false); }, setPosition(e) { // 獲取菜單的寬高 const menu = document.querySelector(".contextmenu"); const menuHeight = menu.offsetHeight; // 獲取窗口的可視高度 const windowHeight = window.innerHeight; this.left = e.clientX; // 計算菜單的上邊位置 if (e.clientY + menuHeight > windowHeight) { const top = e.clientY - menuHeight; if (top < 0) { this.top = 0; // 確保菜單不會超出頂部 } else { // 如果菜單的底部超出了窗口的底部 this.top = top; } } else { // 如果菜單的底部沒有超出窗口的底部 this.top = e.clientY; } this.target = e.target; }, handleClick({ key }) { const _component = this.list.filter((item) => item.key === key)[0] .component; if (_component) { this.$emit("contextMenuClick", _component, key); } this.closeMenu(); }, }, }; </script>
詳細解釋
setPosition 方法
setPosition
方法用于根據(jù)用戶點擊的位置動態(tài)調(diào)整菜單的位置,確保菜單始終在可視區(qū)域內(nèi)。
setPosition(e) { // 獲取菜單的寬高 const menu = document.querySelector(".contextmenu"); const menuHeight = menu.offsetHeight; // 獲取窗口的可視高度 const windowHeight = window.innerHeight; this.left = e.clientX; // 計算菜單的上邊位置 if (e.clientY + menuHeight > windowHeight) { const top = e.clientY - menuHeight; if (top < 0) { this.top = 0; // 確保菜單不會超出頂部 } else { // 如果菜單的底部超出了窗口的底部 this.top = top; } } else { // 如果菜單的底部沒有超出窗口的底部 this.top = e.clientY; } this.target = e.target; }
- 相加:通過
e.clientY + menuHeight
計算菜單底部的位置,如果大于windowHeight
,則表示菜單超出了窗口底部,需要調(diào)整位置。 - 相減:通過
e.clientY - menuHeight
將菜單位置調(diào)整到鼠標點擊位置的上方,確保菜單不會超出窗口底部。 - clientY:鼠標點擊位置的垂直坐標,相對于視口。
- offsetHeight:菜單元素的高度,包括內(nèi)容的高度、內(nèi)邊距和邊框。
事件處理
在 created
生命周期鉤子中,添加了 click
和 contextmenu
事件監(jiān)聽器。
created() { const clickHandler = () => this.closeMenu(); const contextMenuHandler = (e) => { this.setPosition(e); }; window.addEventListener("click", clickHandler); window.addEventListener("contextmenu", contextMenuHandler); this.$emit("hook:beforeDestroy", () => { window.removeEventListener("click", clickHandler); window.removeEventListener("contextmenu", contextMenuHandler); }); }
- clickHandler:點擊頁面時,關閉菜單。
- contextMenuHandler:右鍵點擊時,阻止默認的右鍵菜單行為,并根據(jù)點擊位置設置菜單的位置。
樣式部分
<style lang="scss" scoped> .contextmenu { position: fixed; z-index: 1000; border-radius: 4px; border: 1px lightgrey solid; box-shadow: 4px 4px 10px lightgrey !important; } </style>
總結
通過以上代碼,實現(xiàn)了一個動態(tài)右鍵菜單。這個菜單能夠根據(jù)用戶的點擊位置動態(tài)調(diào)整其顯示位置,確保菜單始終在瀏覽器窗口的可視區(qū)域內(nèi)。這樣的實現(xiàn)可以提升用戶體驗,使應用更加友好和易用。
到此這篇關于Vue中實現(xiàn)動態(tài)右鍵菜單的示例代碼的文章就介紹到這了,更多相關Vue 動態(tài)右鍵菜單內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Vue2?Dialog彈窗函數(shù)式調(diào)用實踐示例
這篇文章主要為大家介紹了Vue2?Dialog彈窗函數(shù)式調(diào)用實踐示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01vue?Router(v3.x)?路由傳參的三種方式場景分析
vue?路由傳參的使用場景一般都是應用在父路由跳轉到子路由時,攜帶參數(shù)跳轉,傳參方式可劃分為?params?傳參和?query?傳參,而?params?傳參又可分為在?url?中顯示參數(shù)和不顯示參數(shù)兩種方式,這就是vue路由傳參的三種方式,感興趣的朋友跟隨小編一起看看吧2023-07-07Vue監(jiān)聽數(shù)據(jù)渲染DOM完以后執(zhí)行某個函數(shù)詳解
今天小編就為大家分享一篇Vue監(jiān)聽數(shù)據(jù)渲染DOM完以后執(zhí)行某個函數(shù)詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09